camper/web/static/ckeditor5@39.0.1/ckeditor.js.map

1 line
5.7 MiB
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

{"version":3,"file":"ckeditor.js","mappings":";;;;;AAAA,SAA2CA,EAAMC,GAC1B,iBAAZC,SAA0C,iBAAXC,OACxCA,OAAOD,QAAUD,IACQ,mBAAXG,QAAyBA,OAAOC,IAC9CD,OAAO,GAAIH,GACe,iBAAZC,QACdA,QAAuB,cAAID,IAE3BD,EAAoB,cAAIC,GACzB,CATD,CASGK,MAAM,IACT,M,sBCRA,MAAMC,EAAc,EAAQ,MAMtBC,EAAkB,CAAC,EACzB,IAAK,MAAMC,KAAOC,OAAOC,KAAKJ,GAC7BC,EAAgBD,EAAYE,IAAQA,EAGrC,MAAMG,EAAU,CACfC,IAAK,CAACC,SAAU,EAAGC,OAAQ,OAC3BC,IAAK,CAACF,SAAU,EAAGC,OAAQ,OAC3BE,IAAK,CAACH,SAAU,EAAGC,OAAQ,OAC3BG,IAAK,CAACJ,SAAU,EAAGC,OAAQ,OAC3BI,KAAM,CAACL,SAAU,EAAGC,OAAQ,QAC5BK,IAAK,CAACN,SAAU,EAAGC,OAAQ,OAC3BM,IAAK,CAACP,SAAU,EAAGC,OAAQ,OAC3BO,IAAK,CAACR,SAAU,EAAGC,OAAQ,OAC3BQ,IAAK,CAACT,SAAU,EAAGC,OAAQ,CAAC,QAC5BS,QAAS,CAACV,SAAU,EAAGC,OAAQ,CAAC,YAChCU,OAAQ,CAACX,SAAU,EAAGC,OAAQ,CAAC,WAC/BW,QAAS,CAACZ,SAAU,EAAGC,OAAQ,CAAC,YAChCY,IAAK,CAACb,SAAU,EAAGC,OAAQ,CAAC,IAAK,IAAK,MACtCa,MAAO,CAACd,SAAU,EAAGC,OAAQ,CAAC,MAAO,MAAO,QAC5Cc,KAAM,CAACf,SAAU,EAAGC,OAAQ,CAAC,UAG9BZ,EAAOD,QAAUU,EAGjB,IAAK,MAAMkB,KAASpB,OAAOC,KAAKC,GAAU,CACzC,KAAM,aAAcA,EAAQkB,IAC3B,MAAM,IAAIC,MAAM,8BAAgCD,GAGjD,KAAM,WAAYlB,EAAQkB,IACzB,MAAM,IAAIC,MAAM,oCAAsCD,GAGvD,GAAIlB,EAAQkB,GAAOf,OAAOiB,SAAWpB,EAAQkB,GAAOhB,SACnD,MAAM,IAAIiB,MAAM,sCAAwCD,GAGzD,MAAM,SAAChB,EAAQ,OAAEC,GAAUH,EAAQkB,UAC5BlB,EAAQkB,GAAOhB,gBACfF,EAAQkB,GAAOf,OACtBL,OAAOuB,eAAerB,EAAQkB,GAAQ,WAAY,CAACI,MAAOpB,IAC1DJ,OAAOuB,eAAerB,EAAQkB,GAAQ,SAAU,CAACI,MAAOnB,GACzD,CAEAH,EAAQC,IAAIG,IAAM,SAAUH,GAC3B,MAAMsB,EAAItB,EAAI,GAAK,IACbuB,EAAIvB,EAAI,GAAK,IACbwB,EAAIxB,EAAI,GAAK,IACbyB,EAAMC,KAAKD,IAAIH,EAAGC,EAAGC,GACrBG,EAAMD,KAAKC,IAAIL,EAAGC,EAAGC,GACrBI,EAAQD,EAAMF,EACpB,IAAII,EACAC,EAEAH,IAAQF,EACXI,EAAI,EACMP,IAAMK,EAChBE,GAAKN,EAAIC,GAAKI,EACJL,IAAMI,EAChBE,EAAI,GAAKL,EAAIF,GAAKM,EACRJ,IAAMG,IAChBE,EAAI,GAAKP,EAAIC,GAAKK,GAGnBC,EAAIH,KAAKD,IAAQ,GAAJI,EAAQ,KAEjBA,EAAI,IACPA,GAAK,KAGN,MAAME,GAAKN,EAAME,GAAO,EAUxB,OAPCG,EADGH,IAAQF,EACP,EACMM,GAAK,GACXH,GAASD,EAAMF,GAEfG,GAAS,EAAID,EAAMF,GAGjB,CAACI,EAAO,IAAJC,EAAa,IAAJC,EACrB,EAEAhC,EAAQC,IAAII,IAAM,SAAUJ,GAC3B,IAAIgC,EACAC,EACAC,EACAL,EACAC,EAEJ,MAAMR,EAAItB,EAAI,GAAK,IACbuB,EAAIvB,EAAI,GAAK,IACbwB,EAAIxB,EAAI,GAAK,IACbmC,EAAIT,KAAKC,IAAIL,EAAGC,EAAGC,GACnBY,EAAOD,EAAIT,KAAKD,IAAIH,EAAGC,EAAGC,GAC1Ba,EAAQ,SAAUC,GACvB,OAAQH,EAAIG,GAAK,EAAIF,EAAO,EAC7B,EA0BA,OAxBa,IAATA,GACHP,EAAI,EACJC,EAAI,IAEJA,EAAIM,EAAOD,EACXH,EAAOK,EAAMf,GACbW,EAAOI,EAAMd,GACbW,EAAOG,EAAMb,GAETF,IAAMa,EACTN,EAAIK,EAAOD,EACDV,IAAMY,EAChBN,EAAK,EAAI,EAAKG,EAAOE,EACXV,IAAMW,IAChBN,EAAK,EAAI,EAAKI,EAAOD,GAGlBH,EAAI,EACPA,GAAK,EACKA,EAAI,IACdA,GAAK,IAIA,CACF,IAAJA,EACI,IAAJC,EACI,IAAJK,EAEF,EAEApC,EAAQC,IAAIK,IAAM,SAAUL,GAC3B,MAAMsB,EAAItB,EAAI,GACRuB,EAAIvB,EAAI,GACd,IAAIwB,EAAIxB,EAAI,GACZ,MAAM6B,EAAI9B,EAAQC,IAAIG,IAAIH,GAAK,GACzBuC,EAAI,EAAI,IAAMb,KAAKD,IAAIH,EAAGI,KAAKD,IAAIF,EAAGC,IAI5C,OAFAA,EAAI,EAAI,EAAI,IAAME,KAAKC,IAAIL,EAAGI,KAAKC,IAAIJ,EAAGC,IAEnC,CAACK,EAAO,IAAJU,EAAa,IAAJf,EACrB,EAEAzB,EAAQC,IAAIM,KAAO,SAAUN,GAC5B,MAAMsB,EAAItB,EAAI,GAAK,IACbuB,EAAIvB,EAAI,GAAK,IACbwB,EAAIxB,EAAI,GAAK,IAEbwC,EAAId,KAAKD,IAAI,EAAIH,EAAG,EAAIC,EAAG,EAAIC,GAKrC,MAAO,CAAK,MAJD,EAAIF,EAAIkB,IAAM,EAAIA,IAAM,GAId,MAHV,EAAIjB,EAAIiB,IAAM,EAAIA,IAAM,GAGL,MAFnB,EAAIhB,EAAIgB,IAAM,EAAIA,IAAM,GAEI,IAAJA,EACpC,EAaAzC,EAAQC,IAAIW,QAAU,SAAUX,GAC/B,MAAMyC,EAAW9C,EAAgBK,GACjC,GAAIyC,EACH,OAAOA,EAGR,IACIC,EADAC,EAAyBC,IAG7B,IAAK,MAAMjC,KAAWd,OAAOC,KAAKJ,GAAc,CAC/C,MAAM2B,EAAQ3B,EAAYiB,GAGpBkC,GAxBwBC,EAwBYzB,IAxBf0B,EAwBU/C,GAnBjC,GAAK8C,EAAE,KAAO,GAChBC,EAAE,GAAKD,EAAE,KAAO,GAChBC,EAAE,GAAKD,EAAE,KAAO,GAoBdD,EAAWF,IACdA,EAAyBE,EACzBH,EAAwB/B,EAE1B,CA/BD,IAA6BoC,EAAGD,EAiC/B,OAAOJ,CACR,EAEA3C,EAAQY,QAAQX,IAAM,SAAUW,GAC/B,OAAOjB,EAAYiB,EACpB,EAEAZ,EAAQC,IAAIO,IAAM,SAAUP,GAC3B,IAAIsB,EAAItB,EAAI,GAAK,IACbuB,EAAIvB,EAAI,GAAK,IACbwB,EAAIxB,EAAI,GAAK,IAGjBsB,EAAIA,EAAI,SAAaA,EAAI,MAAS,QAAU,IAAQA,EAAI,MACxDC,EAAIA,EAAI,SAAaA,EAAI,MAAS,QAAU,IAAQA,EAAI,MACxDC,EAAIA,EAAI,SAAaA,EAAI,MAAS,QAAU,IAAQA,EAAI,MAMxD,MAAO,CAAK,KAJG,MAAJF,EAAmB,MAAJC,EAAmB,MAAJC,GAIpB,KAHN,MAAJF,EAAmB,MAAJC,EAAmB,MAAJC,GAGX,KAFf,MAAJF,EAAmB,MAAJC,EAAmB,MAAJC,GAG1C,EAEAzB,EAAQC,IAAIQ,IAAM,SAAUR,GAC3B,MAAMO,EAAMR,EAAQC,IAAIO,IAAIP,GAC5B,IAAI+C,EAAIxC,EAAI,GACRuC,EAAIvC,EAAI,GACRyC,EAAIzC,EAAI,GAEZwC,GAAK,OACLD,GAAK,IACLE,GAAK,QAELD,EAAIA,EAAI,QAAYA,IAAM,EAAI,GAAO,MAAQA,EAAM,GAAK,IACxDD,EAAIA,EAAI,QAAYA,IAAM,EAAI,GAAO,MAAQA,EAAM,GAAK,IACxDE,EAAIA,EAAI,QAAYA,IAAM,EAAI,GAAO,MAAQA,EAAM,GAAK,IAMxD,MAAO,CAJI,IAAMF,EAAK,GACZ,KAAOC,EAAID,GACX,KAAOA,EAAIE,GAGtB,EAEAjD,EAAQI,IAAIH,IAAM,SAAUG,GAC3B,MAAM0B,EAAI1B,EAAI,GAAK,IACb2B,EAAI3B,EAAI,GAAK,IACb4B,EAAI5B,EAAI,GAAK,IACnB,IAAI8C,EACAC,EACAC,EAEJ,GAAU,IAANrB,EAEH,OADAqB,EAAU,IAAJpB,EACC,CAACoB,EAAKA,EAAKA,GAIlBF,EADGlB,EAAI,GACFA,GAAK,EAAID,GAETC,EAAID,EAAIC,EAAID,EAGlB,MAAMsB,EAAK,EAAIrB,EAAIkB,EAEbjD,EAAM,CAAC,EAAG,EAAG,GACnB,IAAK,IAAIqD,EAAI,EAAGA,EAAI,EAAGA,IACtBH,EAAKrB,EAAI,EAAI,IAAMwB,EAAI,GACnBH,EAAK,GACRA,IAGGA,EAAK,GACRA,IAIAC,EADG,EAAID,EAAK,EACNE,EAAiB,GAAXH,EAAKG,GAAUF,EACjB,EAAIA,EAAK,EACbD,EACI,EAAIC,EAAK,EACbE,GAAMH,EAAKG,IAAO,EAAI,EAAIF,GAAM,EAEhCE,EAGPpD,EAAIqD,GAAW,IAANF,EAGV,OAAOnD,CACR,EAEAD,EAAQI,IAAIC,IAAM,SAAUD,GAC3B,MAAM0B,EAAI1B,EAAI,GACd,IAAI2B,EAAI3B,EAAI,GAAK,IACb4B,EAAI5B,EAAI,GAAK,IACbmD,EAAOxB,EACX,MAAMyB,EAAO7B,KAAKC,IAAII,EAAG,KAEzBA,GAAK,EACLD,GAAMC,GAAK,EAAKA,EAAI,EAAIA,EACxBuB,GAAQC,GAAQ,EAAIA,EAAO,EAAIA,EAI/B,MAAO,CAAC1B,EAAQ,KAFC,IAANE,EAAW,EAAIuB,GAASC,EAAOD,GAAS,EAAIxB,GAAMC,EAAID,IAExC,MAHdC,EAAID,GAAK,GAIrB,EAEA/B,EAAQK,IAAIJ,IAAM,SAAUI,GAC3B,MAAMyB,EAAIzB,EAAI,GAAK,GACb0B,EAAI1B,EAAI,GAAK,IACnB,IAAI+B,EAAI/B,EAAI,GAAK,IACjB,MAAMoD,EAAK9B,KAAK+B,MAAM5B,GAAK,EAErB6B,EAAI7B,EAAIH,KAAK+B,MAAM5B,GACnB8B,EAAI,IAAMxB,GAAK,EAAIL,GACnB8B,EAAI,IAAMzB,GAAK,EAAKL,EAAI4B,GACxBG,EAAI,IAAM1B,GAAK,EAAKL,GAAK,EAAI4B,IAGnC,OAFAvB,GAAK,IAEGqB,GACP,KAAK,EACJ,MAAO,CAACrB,EAAG0B,EAAGF,GACf,KAAK,EACJ,MAAO,CAACC,EAAGzB,EAAGwB,GACf,KAAK,EACJ,MAAO,CAACA,EAAGxB,EAAG0B,GACf,KAAK,EACJ,MAAO,CAACF,EAAGC,EAAGzB,GACf,KAAK,EACJ,MAAO,CAAC0B,EAAGF,EAAGxB,GACf,KAAK,EACJ,MAAO,CAACA,EAAGwB,EAAGC,GAEjB,EAEA7D,EAAQK,IAAID,IAAM,SAAUC,GAC3B,MAAMyB,EAAIzB,EAAI,GACR0B,EAAI1B,EAAI,GAAK,IACb+B,EAAI/B,EAAI,GAAK,IACb0D,EAAOpC,KAAKC,IAAIQ,EAAG,KACzB,IAAI4B,EACAhC,EAEJA,GAAK,EAAID,GAAKK,EACd,MAAMoB,GAAQ,EAAIzB,GAAKgC,EAMvB,OALAC,EAAKjC,EAAIgC,EACTC,GAAOR,GAAQ,EAAKA,EAAO,EAAIA,EAC/BQ,EAAKA,GAAM,EACXhC,GAAK,EAEE,CAACF,EAAQ,IAALkC,EAAc,IAAJhC,EACtB,EAGAhC,EAAQM,IAAIL,IAAM,SAAUK,GAC3B,MAAMwB,EAAIxB,EAAI,GAAK,IACnB,IAAI2D,EAAK3D,EAAI,GAAK,IACd4D,EAAK5D,EAAI,GAAK,IAClB,MAAM6D,EAAQF,EAAKC,EACnB,IAAIP,EAGAQ,EAAQ,IACXF,GAAME,EACND,GAAMC,GAGP,MAAMb,EAAI3B,KAAK+B,MAAM,EAAI5B,GACnBM,EAAI,EAAI8B,EACdP,EAAI,EAAI7B,EAAIwB,EAEO,IAAV,EAAJA,KACJK,EAAI,EAAIA,GAGT,MAAMS,EAAIH,EAAKN,GAAKvB,EAAI6B,GAExB,IAAI1C,EACAC,EACAC,EAEJ,OAAQ6B,GACP,QACA,KAAK,EACL,KAAK,EAAG/B,EAAIa,EAAIZ,EAAI4C,EAAI3C,EAAIwC,EAAI,MAChC,KAAK,EAAG1C,EAAI6C,EAAI5C,EAAIY,EAAIX,EAAIwC,EAAI,MAChC,KAAK,EAAG1C,EAAI0C,EAAIzC,EAAIY,EAAIX,EAAI2C,EAAG,MAC/B,KAAK,EAAG7C,EAAI0C,EAAIzC,EAAI4C,EAAI3C,EAAIW,EAAG,MAC/B,KAAK,EAAGb,EAAI6C,EAAI5C,EAAIyC,EAAIxC,EAAIW,EAAG,MAC/B,KAAK,EAAGb,EAAIa,EAAIZ,EAAIyC,EAAIxC,EAAI2C,EAI7B,MAAO,CAAK,IAAJ7C,EAAa,IAAJC,EAAa,IAAJC,EAC3B,EAEAzB,EAAQO,KAAKN,IAAM,SAAUM,GAC5B,MAAMgC,EAAIhC,EAAK,GAAK,IACd8D,EAAI9D,EAAK,GAAK,IACdwC,EAAIxC,EAAK,GAAK,IACdkC,EAAIlC,EAAK,GAAK,IAMpB,MAAO,CAAK,KAJF,EAAIoB,KAAKD,IAAI,EAAGa,GAAK,EAAIE,GAAKA,IAInB,KAHX,EAAId,KAAKD,IAAI,EAAG2C,GAAK,EAAI5B,GAAKA,IAGV,KAFpB,EAAId,KAAKD,IAAI,EAAGqB,GAAK,EAAIN,GAAKA,IAGzC,EAEAzC,EAAQQ,IAAIP,IAAM,SAAUO,GAC3B,MAAMwC,EAAIxC,EAAI,GAAK,IACbuC,EAAIvC,EAAI,GAAK,IACbyC,EAAIzC,EAAI,GAAK,IACnB,IAAIe,EACAC,EACAC,EAuBJ,OArBAF,EAAS,OAAJyB,GAAoB,OAALD,GAAqB,MAALE,EACpCzB,GAAU,MAALwB,EAAoB,OAAJD,EAAmB,MAAJE,EACpCxB,EAAS,MAAJuB,GAAoB,KAALD,EAAoB,MAAJE,EAGpC1B,EAAIA,EAAI,SACH,MAASA,IAAM,EAAM,KAAS,KAC5B,MAAJA,EAEHC,EAAIA,EAAI,SACH,MAASA,IAAM,EAAM,KAAS,KAC5B,MAAJA,EAEHC,EAAIA,EAAI,SACH,MAASA,IAAM,EAAM,KAAS,KAC5B,MAAJA,EAEHF,EAAII,KAAKD,IAAIC,KAAKC,IAAI,EAAGL,GAAI,GAC7BC,EAAIG,KAAKD,IAAIC,KAAKC,IAAI,EAAGJ,GAAI,GAC7BC,EAAIE,KAAKD,IAAIC,KAAKC,IAAI,EAAGH,GAAI,GAEtB,CAAK,IAAJF,EAAa,IAAJC,EAAa,IAAJC,EAC3B,EAEAzB,EAAQQ,IAAIC,IAAM,SAAUD,GAC3B,IAAIwC,EAAIxC,EAAI,GACRuC,EAAIvC,EAAI,GACRyC,EAAIzC,EAAI,GAEZwC,GAAK,OACLD,GAAK,IACLE,GAAK,QAELD,EAAIA,EAAI,QAAYA,IAAM,EAAI,GAAO,MAAQA,EAAM,GAAK,IACxDD,EAAIA,EAAI,QAAYA,IAAM,EAAI,GAAO,MAAQA,EAAM,GAAK,IACxDE,EAAIA,EAAI,QAAYA,IAAM,EAAI,GAAO,MAAQA,EAAM,GAAK,IAMxD,MAAO,CAJI,IAAMF,EAAK,GACZ,KAAOC,EAAID,GACX,KAAOA,EAAIE,GAGtB,EAEAjD,EAAQS,IAAID,IAAM,SAAUC,GAI3B,IAAIuC,EACAD,EACAE,EAEJF,GAPUtC,EAAI,GAOL,IAAM,IACfuC,EAPUvC,EAAI,GAON,IAAMsC,EACdE,EAAIF,EAPMtC,EAAI,GAOF,IAEZ,MAAM6D,EAAKvB,GAAK,EACVwB,EAAKvB,GAAK,EACVwB,EAAKvB,GAAK,EAShB,OARAF,EAAIuB,EAAK,QAAWA,GAAMvB,EAAI,GAAK,KAAO,MAC1CC,EAAIuB,EAAK,QAAWA,GAAMvB,EAAI,GAAK,KAAO,MAC1CC,EAAIuB,EAAK,QAAWA,GAAMvB,EAAI,GAAK,KAAO,MAE1CD,GAAK,OACLD,GAAK,IACLE,GAAK,QAEE,CAACD,EAAGD,EAAGE,EACf,EAEAjD,EAAQS,IAAIC,IAAM,SAAUD,GAC3B,MAAMuB,EAAIvB,EAAI,GACRgE,EAAIhE,EAAI,GACRgB,EAAIhB,EAAI,GACd,IAAIqB,EAGJA,EAAS,IADEH,KAAK+C,MAAMjD,EAAGgD,GACV,EAAI9C,KAAKgD,GAEpB7C,EAAI,IACPA,GAAK,KAKN,MAAO,CAACE,EAFEL,KAAKiD,KAAKH,EAAIA,EAAIhD,EAAIA,GAElBK,EACf,EAEA9B,EAAQU,IAAID,IAAM,SAAUC,GAC3B,MAAMsB,EAAItB,EAAI,GACR6B,EAAI7B,EAAI,GAGRmE,EAFInE,EAAI,GAEC,IAAM,EAAIiB,KAAKgD,GAI9B,MAAO,CAAC3C,EAHEO,EAAIZ,KAAKmD,IAAID,GACbtC,EAAIZ,KAAKoD,IAAIF,GAGxB,EAEA7E,EAAQC,IAAIY,OAAS,SAAUmE,EAAMC,EAAa,MACjD,MAAO1D,EAAGC,EAAGC,GAAKuD,EAClB,IAAI1D,EAAuB,OAAf2D,EAAsBjF,EAAQC,IAAII,IAAI2E,GAAM,GAAKC,EAI7D,GAFA3D,EAAQK,KAAKuD,MAAM5D,EAAQ,IAEb,IAAVA,EACH,OAAO,GAGR,IAAI6D,EAAO,IACNxD,KAAKuD,MAAMzD,EAAI,MAAQ,EACxBE,KAAKuD,MAAM1D,EAAI,MAAQ,EACxBG,KAAKuD,MAAM3D,EAAI,MAMlB,OAJc,IAAVD,IACH6D,GAAQ,IAGFA,CACR,EAEAnF,EAAQK,IAAIQ,OAAS,SAAUmE,GAG9B,OAAOhF,EAAQC,IAAIY,OAAOb,EAAQK,IAAIJ,IAAI+E,GAAOA,EAAK,GACvD,EAEAhF,EAAQC,IAAIa,QAAU,SAAUkE,GAC/B,MAAMzD,EAAIyD,EAAK,GACTxD,EAAIwD,EAAK,GACTvD,EAAIuD,EAAK,GAIf,GAAIzD,IAAMC,GAAKA,IAAMC,EACpB,OAAIF,EAAI,EACA,GAGJA,EAAI,IACA,IAGDI,KAAKuD,OAAQ3D,EAAI,GAAK,IAAO,IAAM,IAQ3C,OALa,GACT,GAAKI,KAAKuD,MAAM3D,EAAI,IAAM,GAC1B,EAAII,KAAKuD,MAAM1D,EAAI,IAAM,GAC1BG,KAAKuD,MAAMzD,EAAI,IAAM,EAGzB,EAEAzB,EAAQa,OAAOZ,IAAM,SAAU+E,GAC9B,IAAII,EAAQJ,EAAO,GAGnB,GAAc,IAAVI,GAAyB,IAAVA,EAOlB,OANIJ,EAAO,KACVI,GAAS,KAGVA,EAAQA,EAAQ,KAAO,IAEhB,CAACA,EAAOA,EAAOA,GAGvB,MAAMC,EAA6B,IAAL,KAAbL,EAAO,KAKxB,MAAO,EAJa,EAARI,GAAaC,EAAQ,KACpBD,GAAS,EAAK,GAAKC,EAAQ,KAC3BD,GAAS,EAAK,GAAKC,EAAQ,IAGzC,EAEArF,EAAQc,QAAQb,IAAM,SAAU+E,GAE/B,GAAIA,GAAQ,IAAK,CAChB,MAAMzC,EAAmB,IAAdyC,EAAO,KAAY,EAC9B,MAAO,CAACzC,EAAGA,EAAGA,EACf,CAIA,IAAI+C,EAFJN,GAAQ,GAOR,MAAO,CAJGrD,KAAK+B,MAAMsB,EAAO,IAAM,EAAI,IAC5BrD,KAAK+B,OAAO4B,EAAMN,EAAO,IAAM,GAAK,EAAI,IACvCM,EAAM,EAAK,EAAI,IAG3B,EAEAtF,EAAQC,IAAIU,IAAM,SAAUqE,GAC3B,MAIMO,KAJkC,IAAtB5D,KAAKuD,MAAMF,EAAK,MAAe,MACtB,IAAtBrD,KAAKuD,MAAMF,EAAK,MAAe,IACV,IAAtBrD,KAAKuD,MAAMF,EAAK,MAEGQ,SAAS,IAAIC,cACpC,MAAO,SAASC,UAAUH,EAAOnE,QAAUmE,CAC5C,EAEAvF,EAAQW,IAAIV,IAAM,SAAU+E,GAC3B,MAAMW,EAAQX,EAAKQ,SAAS,IAAIG,MAAM,4BACtC,IAAKA,EACJ,MAAO,CAAC,EAAG,EAAG,GAGf,IAAIC,EAAcD,EAAM,GAEA,IAApBA,EAAM,GAAGvE,SACZwE,EAAcA,EAAYC,MAAM,IAAIC,KAAIC,GAChCA,EAAOA,IACZC,KAAK,KAGT,MAAMC,EAAUC,SAASN,EAAa,IAKtC,MAAO,CAJIK,GAAW,GAAM,IACjBA,GAAW,EAAK,IACP,IAAVA,EAGX,EAEAjG,EAAQC,IAAIc,IAAM,SAAUd,GAC3B,MAAMsB,EAAItB,EAAI,GAAK,IACbuB,EAAIvB,EAAI,GAAK,IACbwB,EAAIxB,EAAI,GAAK,IACb2B,EAAMD,KAAKC,IAAID,KAAKC,IAAIL,EAAGC,GAAIC,GAC/BC,EAAMC,KAAKD,IAAIC,KAAKD,IAAIH,EAAGC,GAAIC,GAC/B0E,EAAUvE,EAAMF,EACtB,IAAI0E,EACAC,EAuBJ,OApBCD,EADGD,EAAS,EACAzE,GAAO,EAAIyE,GAEX,EAIZE,EADGF,GAAU,EACP,EAEHvE,IAAQL,GACHC,EAAIC,GAAK0E,EAAU,EAExBvE,IAAQJ,EACL,GAAKC,EAAIF,GAAK4E,EAEd,GAAK5E,EAAIC,GAAK2E,EAGrBE,GAAO,EACPA,GAAO,EAEA,CAAO,IAANA,EAAoB,IAATF,EAA0B,IAAZC,EAClC,EAEApG,EAAQI,IAAIW,IAAM,SAAUX,GAC3B,MAAM2B,EAAI3B,EAAI,GAAK,IACb4B,EAAI5B,EAAI,GAAK,IAEbmC,EAAIP,EAAI,GAAO,EAAMD,EAAIC,EAAM,EAAMD,GAAK,EAAMC,GAEtD,IAAI2B,EAAI,EAKR,OAJIpB,EAAI,IACPoB,GAAK3B,EAAI,GAAMO,IAAM,EAAMA,IAGrB,CAACnC,EAAI,GAAQ,IAAJmC,EAAa,IAAJoB,EAC1B,EAEA3D,EAAQK,IAAIU,IAAM,SAAUV,GAC3B,MAAM0B,EAAI1B,EAAI,GAAK,IACb+B,EAAI/B,EAAI,GAAK,IAEbkC,EAAIR,EAAIK,EACd,IAAIuB,EAAI,EAMR,OAJIpB,EAAI,IACPoB,GAAKvB,EAAIG,IAAM,EAAIA,IAGb,CAAClC,EAAI,GAAQ,IAAJkC,EAAa,IAAJoB,EAC1B,EAEA3D,EAAQe,IAAId,IAAM,SAAUc,GAC3B,MAAMe,EAAIf,EAAI,GAAK,IACbwB,EAAIxB,EAAI,GAAK,IACbS,EAAIT,EAAI,GAAK,IAEnB,GAAU,IAANwB,EACH,MAAO,CAAK,IAAJf,EAAa,IAAJA,EAAa,IAAJA,GAG3B,MAAM8E,EAAO,CAAC,EAAG,EAAG,GACd7C,EAAM3B,EAAI,EAAK,EACfM,EAAIqB,EAAK,EACTjB,EAAI,EAAIJ,EACd,IAAImE,EAAK,EAGT,OAAQ5E,KAAK+B,MAAMD,IAClB,KAAK,EACJ6C,EAAK,GAAK,EAAGA,EAAK,GAAKlE,EAAGkE,EAAK,GAAK,EAAG,MACxC,KAAK,EACJA,EAAK,GAAK9D,EAAG8D,EAAK,GAAK,EAAGA,EAAK,GAAK,EAAG,MACxC,KAAK,EACJA,EAAK,GAAK,EAAGA,EAAK,GAAK,EAAGA,EAAK,GAAKlE,EAAG,MACxC,KAAK,EACJkE,EAAK,GAAK,EAAGA,EAAK,GAAK9D,EAAG8D,EAAK,GAAK,EAAG,MACxC,KAAK,EACJA,EAAK,GAAKlE,EAAGkE,EAAK,GAAK,EAAGA,EAAK,GAAK,EAAG,MACxC,QACCA,EAAK,GAAK,EAAGA,EAAK,GAAK,EAAGA,EAAK,GAAK9D,EAMtC,OAFA+D,GAAM,EAAMhE,GAAKf,EAEV,CACe,KAApBe,EAAI+D,EAAK,GAAKC,GACM,KAApBhE,EAAI+D,EAAK,GAAKC,GACM,KAApBhE,EAAI+D,EAAK,GAAKC,GAEjB,EAEAvG,EAAQe,IAAIV,IAAM,SAAUU,GAC3B,MAAMwB,EAAIxB,EAAI,GAAK,IAGbqB,EAAIG,EAFAxB,EAAI,GAAK,KAEA,EAAMwB,GACzB,IAAIoB,EAAI,EAMR,OAJIvB,EAAI,IACPuB,EAAIpB,EAAIH,GAGF,CAACrB,EAAI,GAAQ,IAAJ4C,EAAa,IAAJvB,EAC1B,EAEApC,EAAQe,IAAIX,IAAM,SAAUW,GAC3B,MAAMwB,EAAIxB,EAAI,GAAK,IAGbiB,EAFIjB,EAAI,GAAK,KAEJ,EAAMwB,GAAK,GAAMA,EAChC,IAAIR,EAAI,EASR,OAPIC,EAAI,GAAOA,EAAI,GAClBD,EAAIQ,GAAK,EAAIP,GAEVA,GAAK,IAAOA,EAAI,IACnBD,EAAIQ,GAAK,GAAK,EAAIP,KAGZ,CAACjB,EAAI,GAAQ,IAAJgB,EAAa,IAAJC,EAC1B,EAEAhC,EAAQe,IAAIT,IAAM,SAAUS,GAC3B,MAAMwB,EAAIxB,EAAI,GAAK,IAEbqB,EAAIG,EADAxB,EAAI,GAAK,KACA,EAAMwB,GACzB,MAAO,CAACxB,EAAI,GAAc,KAATqB,EAAIG,GAAoB,KAAT,EAAIH,GACrC,EAEApC,EAAQM,IAAIS,IAAM,SAAUT,GAC3B,MAAMkC,EAAIlC,EAAI,GAAK,IAEb8B,EAAI,EADA9B,EAAI,GAAK,IAEbiC,EAAIH,EAAII,EACd,IAAIhB,EAAI,EAMR,OAJIe,EAAI,IACPf,GAAKY,EAAIG,IAAM,EAAIA,IAGb,CAACjC,EAAI,GAAQ,IAAJiC,EAAa,IAAJf,EAC1B,EAEAxB,EAAQgB,MAAMf,IAAM,SAAUe,GAC7B,MAAO,CAAEA,EAAM,GAAK,MAAS,IAAMA,EAAM,GAAK,MAAS,IAAMA,EAAM,GAAK,MAAS,IAClF,EAEAhB,EAAQC,IAAIe,MAAQ,SAAUf,GAC7B,MAAO,CAAEA,EAAI,GAAK,IAAO,MAAQA,EAAI,GAAK,IAAO,MAAQA,EAAI,GAAK,IAAO,MAC1E,EAEAD,EAAQiB,KAAKhB,IAAM,SAAU+E,GAC5B,MAAO,CAACA,EAAK,GAAK,IAAM,IAAKA,EAAK,GAAK,IAAM,IAAKA,EAAK,GAAK,IAAM,IACnE,EAEAhF,EAAQiB,KAAKb,IAAM,SAAU4E,GAC5B,MAAO,CAAC,EAAG,EAAGA,EAAK,GACpB,EAEAhF,EAAQiB,KAAKZ,IAAML,EAAQiB,KAAKb,IAEhCJ,EAAQiB,KAAKX,IAAM,SAAUW,GAC5B,MAAO,CAAC,EAAG,IAAKA,EAAK,GACtB,EAEAjB,EAAQiB,KAAKV,KAAO,SAAUU,GAC7B,MAAO,CAAC,EAAG,EAAG,EAAGA,EAAK,GACvB,EAEAjB,EAAQiB,KAAKR,IAAM,SAAUQ,GAC5B,MAAO,CAACA,EAAK,GAAI,EAAG,EACrB,EAEAjB,EAAQiB,KAAKN,IAAM,SAAUM,GAC5B,MAAMmC,EAAwC,IAAlCzB,KAAKuD,MAAMjE,EAAK,GAAK,IAAM,KAGjCsE,IAFWnC,GAAO,KAAOA,GAAO,GAAKA,GAEpBoC,SAAS,IAAIC,cACpC,MAAO,SAASC,UAAUH,EAAOnE,QAAUmE,CAC5C,EAEAvF,EAAQC,IAAIgB,KAAO,SAAUhB,GAE5B,MAAO,EADMA,EAAI,GAAKA,EAAI,GAAKA,EAAI,IAAM,EAC3B,IAAM,IACrB,C,gBCt0BA,MAAMuG,EAAc,EAAQ,MACtBC,EAAQ,EAAQ,MAEhBzG,EAAU,CAAC,EAEFF,OAAOC,KAAKyG,GA0DpBE,SAAQC,IACd3G,EAAQ2G,GAAa,CAAC,EAEtB7G,OAAOuB,eAAerB,EAAQ2G,GAAY,WAAY,CAACrF,MAAOkF,EAAYG,GAAWzG,WACrFJ,OAAOuB,eAAerB,EAAQ2G,GAAY,SAAU,CAACrF,MAAOkF,EAAYG,GAAWxG,SAEnF,MAAMyG,EAASH,EAAME,GACD7G,OAAOC,KAAK6G,GAEpBF,SAAQG,IACnB,MAAMC,EAAKF,EAAOC,GAElB7G,EAAQ2G,GAAWE,GA9CrB,SAAqBC,GACpB,MAAMC,EAAY,YAAa/B,GAC9B,MAAMgC,EAAOhC,EAAK,GAElB,GAAIgC,QACH,OAAOA,EAGJA,EAAK5F,OAAS,IACjB4D,EAAOgC,GAGR,MAAMC,EAASH,EAAG9B,GAKlB,GAAsB,iBAAXiC,EACV,IAAK,IAAIC,EAAMD,EAAO7F,OAAQkC,EAAI,EAAGA,EAAI4D,EAAK5D,IAC7C2D,EAAO3D,GAAK3B,KAAKuD,MAAM+B,EAAO3D,IAIhC,OAAO2D,CACR,EAOA,MAJI,eAAgBH,IACnBC,EAAUI,WAAaL,EAAGK,YAGpBJ,CACR,CAcgCK,CAAYN,GAC1C9G,EAAQ2G,GAAWE,GAASQ,IArE9B,SAAiBP,GAChB,MAAMC,EAAY,YAAa/B,GAC9B,MAAMgC,EAAOhC,EAAK,GAClB,OAAIgC,QACIA,GAGJA,EAAK5F,OAAS,IACjB4D,EAAOgC,GAGDF,EAAG9B,GACX,EAOA,MAJI,eAAgB8B,IACnBC,EAAUI,WAAaL,EAAGK,YAGpBJ,CACR,CAiDoCO,CAAQR,EAAG,GAC5C,IAGHvH,EAAOD,QAAUU,C,iBChFjB,MAAMwG,EAAc,EAAQ,MA+B5B,SAASe,EAAUZ,GAClB,MAAMa,EAnBP,WACC,MAAMA,EAAQ,CAAC,EAETC,EAAS3H,OAAOC,KAAKyG,GAE3B,IAAK,IAAIU,EAAMO,EAAOrG,OAAQkC,EAAI,EAAGA,EAAI4D,EAAK5D,IAC7CkE,EAAMC,EAAOnE,IAAM,CAGlBR,UAAW,EACX4E,OAAQ,MAIV,OAAOF,CACR,CAIeG,GACRC,EAAQ,CAACjB,GAIf,IAFAa,EAAMb,GAAW7D,SAAW,EAErB8E,EAAMxG,QAAQ,CACpB,MAAMyG,EAAUD,EAAME,MAChBC,EAAYjI,OAAOC,KAAKyG,EAAYqB,IAE1C,IAAK,IAAIX,EAAMa,EAAU3G,OAAQkC,EAAI,EAAGA,EAAI4D,EAAK5D,IAAK,CACrD,MAAM0E,EAAWD,EAAUzE,GACrB2E,EAAOT,EAAMQ,IAEI,IAAnBC,EAAKnF,WACRmF,EAAKnF,SAAW0E,EAAMK,GAAS/E,SAAW,EAC1CmF,EAAKP,OAASG,EACdD,EAAMM,QAAQF,GAEhB,CACD,CAEA,OAAOR,CACR,CAEA,SAASW,EAAKC,EAAMC,GACnB,OAAO,SAAUrD,GAChB,OAAOqD,EAAGD,EAAKpD,GAChB,CACD,CAEA,SAASsD,EAAezB,EAASW,GAChC,MAAMe,EAAO,CAACf,EAAMX,GAASa,OAAQb,GACrC,IAAIC,EAAKN,EAAYgB,EAAMX,GAASa,QAAQb,GAExC2B,EAAMhB,EAAMX,GAASa,OACzB,KAAOF,EAAMgB,GAAKd,QACjBa,EAAKL,QAAQV,EAAMgB,GAAKd,QACxBZ,EAAKqB,EAAK3B,EAAYgB,EAAMgB,GAAKd,QAAQc,GAAM1B,GAC/C0B,EAAMhB,EAAMgB,GAAKd,OAIlB,OADAZ,EAAGK,WAAaoB,EACTzB,CACR,CAEAvH,EAAOD,QAAU,SAAUqH,GAC1B,MAAMa,EAAQD,EAAUZ,GAClBQ,EAAa,CAAC,EAEdM,EAAS3H,OAAOC,KAAKyH,GAC3B,IAAK,IAAIN,EAAMO,EAAOrG,OAAQkC,EAAI,EAAGA,EAAI4D,EAAK5D,IAAK,CAClD,MAAMuD,EAAUY,EAAOnE,GAGH,OAFPkE,EAAMX,GAEVa,SAKTP,EAAWN,GAAWyB,EAAezB,EAASW,GAC/C,CAEA,OAAOL,CACR,C,wBC7FA5H,EAAOD,QAAU,CAChB,UAAa,CAAC,IAAK,IAAK,KACxB,aAAgB,CAAC,IAAK,IAAK,KAC3B,KAAQ,CAAC,EAAG,IAAK,KACjB,WAAc,CAAC,IAAK,IAAK,KACzB,MAAS,CAAC,IAAK,IAAK,KACpB,MAAS,CAAC,IAAK,IAAK,KACpB,OAAU,CAAC,IAAK,IAAK,KACrB,MAAS,CAAC,EAAG,EAAG,GAChB,eAAkB,CAAC,IAAK,IAAK,KAC7B,KAAQ,CAAC,EAAG,EAAG,KACf,WAAc,CAAC,IAAK,GAAI,KACxB,MAAS,CAAC,IAAK,GAAI,IACnB,UAAa,CAAC,IAAK,IAAK,KACxB,UAAa,CAAC,GAAI,IAAK,KACvB,WAAc,CAAC,IAAK,IAAK,GACzB,UAAa,CAAC,IAAK,IAAK,IACxB,MAAS,CAAC,IAAK,IAAK,IACpB,eAAkB,CAAC,IAAK,IAAK,KAC7B,SAAY,CAAC,IAAK,IAAK,KACvB,QAAW,CAAC,IAAK,GAAI,IACrB,KAAQ,CAAC,EAAG,IAAK,KACjB,SAAY,CAAC,EAAG,EAAG,KACnB,SAAY,CAAC,EAAG,IAAK,KACrB,cAAiB,CAAC,IAAK,IAAK,IAC5B,SAAY,CAAC,IAAK,IAAK,KACvB,UAAa,CAAC,EAAG,IAAK,GACtB,SAAY,CAAC,IAAK,IAAK,KACvB,UAAa,CAAC,IAAK,IAAK,KACxB,YAAe,CAAC,IAAK,EAAG,KACxB,eAAkB,CAAC,GAAI,IAAK,IAC5B,WAAc,CAAC,IAAK,IAAK,GACzB,WAAc,CAAC,IAAK,GAAI,KACxB,QAAW,CAAC,IAAK,EAAG,GACpB,WAAc,CAAC,IAAK,IAAK,KACzB,aAAgB,CAAC,IAAK,IAAK,KAC3B,cAAiB,CAAC,GAAI,GAAI,KAC1B,cAAiB,CAAC,GAAI,GAAI,IAC1B,cAAiB,CAAC,GAAI,GAAI,IAC1B,cAAiB,CAAC,EAAG,IAAK,KAC1B,WAAc,CAAC,IAAK,EAAG,KACvB,SAAY,CAAC,IAAK,GAAI,KACtB,YAAe,CAAC,EAAG,IAAK,KACxB,QAAW,CAAC,IAAK,IAAK,KACtB,QAAW,CAAC,IAAK,IAAK,KACtB,WAAc,CAAC,GAAI,IAAK,KACxB,UAAa,CAAC,IAAK,GAAI,IACvB,YAAe,CAAC,IAAK,IAAK,KAC1B,YAAe,CAAC,GAAI,IAAK,IACzB,QAAW,CAAC,IAAK,EAAG,KACpB,UAAa,CAAC,IAAK,IAAK,KACxB,WAAc,CAAC,IAAK,IAAK,KACzB,KAAQ,CAAC,IAAK,IAAK,GACnB,UAAa,CAAC,IAAK,IAAK,IACxB,KAAQ,CAAC,IAAK,IAAK,KACnB,MAAS,CAAC,EAAG,IAAK,GAClB,YAAe,CAAC,IAAK,IAAK,IAC1B,KAAQ,CAAC,IAAK,IAAK,KACnB,SAAY,CAAC,IAAK,IAAK,KACvB,QAAW,CAAC,IAAK,IAAK,KACtB,UAAa,CAAC,IAAK,GAAI,IACvB,OAAU,CAAC,GAAI,EAAG,KAClB,MAAS,CAAC,IAAK,IAAK,KACpB,MAAS,CAAC,IAAK,IAAK,KACpB,SAAY,CAAC,IAAK,IAAK,KACvB,cAAiB,CAAC,IAAK,IAAK,KAC5B,UAAa,CAAC,IAAK,IAAK,GACxB,aAAgB,CAAC,IAAK,IAAK,KAC3B,UAAa,CAAC,IAAK,IAAK,KACxB,WAAc,CAAC,IAAK,IAAK,KACzB,UAAa,CAAC,IAAK,IAAK,KACxB,qBAAwB,CAAC,IAAK,IAAK,KACnC,UAAa,CAAC,IAAK,IAAK,KACxB,WAAc,CAAC,IAAK,IAAK,KACzB,UAAa,CAAC,IAAK,IAAK,KACxB,UAAa,CAAC,IAAK,IAAK,KACxB,YAAe,CAAC,IAAK,IAAK,KAC1B,cAAiB,CAAC,GAAI,IAAK,KAC3B,aAAgB,CAAC,IAAK,IAAK,KAC3B,eAAkB,CAAC,IAAK,IAAK,KAC7B,eAAkB,CAAC,IAAK,IAAK,KAC7B,eAAkB,CAAC,IAAK,IAAK,KAC7B,YAAe,CAAC,IAAK,IAAK,KAC1B,KAAQ,CAAC,EAAG,IAAK,GACjB,UAAa,CAAC,GAAI,IAAK,IACvB,MAAS,CAAC,IAAK,IAAK,KACpB,QAAW,CAAC,IAAK,EAAG,KACpB,OAAU,CAAC,IAAK,EAAG,GACnB,iBAAoB,CAAC,IAAK,IAAK,KAC/B,WAAc,CAAC,EAAG,EAAG,KACrB,aAAgB,CAAC,IAAK,GAAI,KAC1B,aAAgB,CAAC,IAAK,IAAK,KAC3B,eAAkB,CAAC,GAAI,IAAK,KAC5B,gBAAmB,CAAC,IAAK,IAAK,KAC9B,kBAAqB,CAAC,EAAG,IAAK,KAC9B,gBAAmB,CAAC,GAAI,IAAK,KAC7B,gBAAmB,CAAC,IAAK,GAAI,KAC7B,aAAgB,CAAC,GAAI,GAAI,KACzB,UAAa,CAAC,IAAK,IAAK,KACxB,UAAa,CAAC,IAAK,IAAK,KACxB,SAAY,CAAC,IAAK,IAAK,KACvB,YAAe,CAAC,IAAK,IAAK,KAC1B,KAAQ,CAAC,EAAG,EAAG,KACf,QAAW,CAAC,IAAK,IAAK,KACtB,MAAS,CAAC,IAAK,IAAK,GACpB,UAAa,CAAC,IAAK,IAAK,IACxB,OAAU,CAAC,IAAK,IAAK,GACrB,UAAa,CAAC,IAAK,GAAI,GACvB,OAAU,CAAC,IAAK,IAAK,KACrB,cAAiB,CAAC,IAAK,IAAK,KAC5B,UAAa,CAAC,IAAK,IAAK,KACxB,cAAiB,CAAC,IAAK,IAAK,KAC5B,cAAiB,CAAC,IAAK,IAAK,KAC5B,WAAc,CAAC,IAAK,IAAK,KACzB,UAAa,CAAC,IAAK,IAAK,KACxB,KAAQ,CAAC,IAAK,IAAK,IACnB,KAAQ,CAAC,IAAK,IAAK,KACnB,KAAQ,CAAC,IAAK,IAAK,KACnB,WAAc,CAAC,IAAK,IAAK,KACzB,OAAU,CAAC,IAAK,EAAG,KACnB,cAAiB,CAAC,IAAK,GAAI,KAC3B,IAAO,CAAC,IAAK,EAAG,GAChB,UAAa,CAAC,IAAK,IAAK,KACxB,UAAa,CAAC,GAAI,IAAK,KACvB,YAAe,CAAC,IAAK,GAAI,IACzB,OAAU,CAAC,IAAK,IAAK,KACrB,WAAc,CAAC,IAAK,IAAK,IACzB,SAAY,CAAC,GAAI,IAAK,IACtB,SAAY,CAAC,IAAK,IAAK,KACvB,OAAU,CAAC,IAAK,GAAI,IACpB,OAAU,CAAC,IAAK,IAAK,KACrB,QAAW,CAAC,IAAK,IAAK,KACtB,UAAa,CAAC,IAAK,GAAI,KACvB,UAAa,CAAC,IAAK,IAAK,KACxB,UAAa,CAAC,IAAK,IAAK,KACxB,KAAQ,CAAC,IAAK,IAAK,KACnB,YAAe,CAAC,EAAG,IAAK,KACxB,UAAa,CAAC,GAAI,IAAK,KACvB,IAAO,CAAC,IAAK,IAAK,KAClB,KAAQ,CAAC,EAAG,IAAK,KACjB,QAAW,CAAC,IAAK,IAAK,KACtB,OAAU,CAAC,IAAK,GAAI,IACpB,UAAa,CAAC,GAAI,IAAK,KACvB,OAAU,CAAC,IAAK,IAAK,KACrB,MAAS,CAAC,IAAK,IAAK,KACpB,MAAS,CAAC,IAAK,IAAK,KACpB,WAAc,CAAC,IAAK,IAAK,KACzB,OAAU,CAAC,IAAK,IAAK,GACrB,YAAe,CAAC,IAAK,IAAK,I,gFCnJvBmJ,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,oKAAqK,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wDAAwD,MAAQ,GAAG,SAAW,6DAA6D,eAAiB,CAAC,8XAA8X,WAAa,MAEjxB,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,qOAAsO,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,6DAA6D,MAAQ,GAAG,SAAW,qHAAqH,eAAiB,CAAC,ikBAAmkB,WAAa,MAEplC,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,sFAAuF,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,+BAA+B,MAAQ,GAAG,SAAW,mBAAmB,eAAiB,CAAC,yXAAyX,WAAa,MAE3nB,S,+ECJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,6wDAAgxD,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,yDAAyD,+EAA+E,MAAQ,GAAG,SAAW,4cAA4c,eAAiB,CAAC,6wBAA6wB,o8DAAs8D,WAAa,MAEhrK,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,4ZAA6Z,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wDAAwD,8EAA8E,MAAQ,GAAG,SAAW,mJAAmJ,eAAiB,CAAC,k5BAAk5B,+VAA+V,WAAa,MAEhiE,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,qEAAsE,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,sDAAsD,MAAQ,GAAG,SAAW,kBAAkB,eAAiB,CAAC,mVAAmV,WAAa,MAE1lB,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,kVAAmV,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,qDAAqD,2EAA2E,MAAQ,GAAG,SAAW,kGAAkG,eAAiB,CAAC,0YAA4Y,8YAA8Y,WAAa,MAEx8C,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,m3BAAo3B,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,kDAAkD,MAAQ,GAAG,SAAW,gUAAgU,eAAiB,CAAC,i3HAAo3H,WAAa,MAEntK,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,iqBAAkqB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,yDAAyD,MAAQ,GAAG,SAAW,gNAAgN,eAAiB,CAAC,iiCAAiiC,WAAa,MAErkE,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,mTAAoT,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wDAAwD,MAAQ,GAAG,SAAW,wHAAwH,eAAiB,CAAC,2mBAA2mB,WAAa,MAExsC,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,2hBAA4hB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,mEAAmE,MAAQ,GAAG,SAAW,4IAA4I,eAAiB,CAAC,yyBAAyyB,WAAa,MAE7oD,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,2oBAA4oB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wDAAwD,MAAQ,GAAG,SAAW,4IAA4I,eAAiB,CAAC,ggDAAogD,WAAa,MAE78E,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,k/FAAm/F,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,uDAAuD,MAAQ,GAAG,SAAW,uXAAuX,eAAiB,CAAC,qgGAAqgG,WAAa,MAE/hN,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,koDAAqoD,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,2DAA2D,iFAAiF,MAAQ,GAAG,SAAW,gqBAAgqB,eAAiB,CAAC,8mBAAgnB,mnFAAqnF,WAAa,MAE/wL,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,2iCAA8iC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,6DAA6D,mFAAmF,MAAQ,GAAG,SAAW,sVAAsV,eAAiB,CAAC,8YAA8Y,+kDAA+kD,WAAa,MAE1mH,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,2jBAA4jB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,+DAA+D,qFAAqF,MAAQ,GAAG,SAAW,0KAA0K,eAAiB,CAAC,4aAA4a,ynBAAynB,WAAa,MAExhE,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,0ZAA2Z,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,+DAA+D,qDAAqD,MAAQ,GAAG,SAAW,yIAAyI,eAAiB,CAAC,2mBAA6mB,yRAAyR,WAAa,MAEvpD,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,obAAqb,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,qEAAqE,MAAQ,GAAG,SAAW,0GAA0G,eAAiB,CAAC,+iCAAijC,WAAa,MAE9wD,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,41DAA61D,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,sDAAsD,oDAAoD,4EAA4E,MAAQ,GAAG,SAAW,0ZAA0Z,eAAiB,CAAC,wpBAA0pB,wRAAwR,25DAAo6D,WAAa,MAE53K,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,y9CAA09C,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,mDAAmD,oDAAoD,yEAAyE,MAAQ,GAAG,SAAW,4WAA4W,eAAiB,CAAC,k9BAAo9B,wRAAwR,25CAA65C,WAAa,MAExvJ,S,+ECJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,8nDAAmoD,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,oDAAoD,0EAA0E,MAAQ,GAAG,SAAW,0NAA0N,eAAiB,CAAC,iXAAiX,+0EAAm1E,WAAa,MAExxJ,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,y2BAA02B,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,sDAAsD,4EAA4E,MAAQ,GAAG,SAAW,6NAA6N,eAAiB,CAAC,+RAA+R,q/BAAq/B,WAAa,MAEtlF,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,+DAAgE,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wDAAwD,MAAQ,GAAG,SAAW,kBAAkB,eAAiB,CAAC,yPAAyP,WAAa,MAE5f,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,qYAAsY,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,gDAAgD,MAAQ,GAAG,SAAW,yIAAyI,eAAiB,CAAC,wqBAAwqB,WAAa,MAEh2C,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,kwCAAmwC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,+EAA+E,MAAQ,GAAG,SAAW,8OAA8O,eAAiB,CAAC,gkDAAgkD,WAAa,MAEzvG,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,4dAA6d,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,qDAAqD,2EAA2E,MAAQ,GAAG,SAAW,qKAAqK,eAAiB,CAAC,iOAAiO,sqCAAwqC,WAAa,MAEpwE,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,4vDAAiwD,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,oDAAoD,MAAQ,GAAG,SAAW,snBAAsnB,eAAiB,CAAC,unFAAunF,WAAa,MAE3pK,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,4EAA6E,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,6DAA6D,MAAQ,GAAG,SAAW,gDAAgD,eAAiB,CAAC,q7BAAs7B,WAAa,MAEzuC,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,sgZAAujZ,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,mEAAmE,yFAAyF,MAAQ,GAAG,SAAW,kqBAAkqB,eAAiB,CAAC,w/CAAsgD,iibAAijb,WAAa,MAEri5B,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,+WAAgX,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,2DAA2D,qDAAqD,MAAQ,GAAG,SAAW,sJAAsJ,eAAiB,CAAC,4nBAA8nB,yRAAyR,WAAa,MAEtoD,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,2pGAA4pG,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,sDAAsD,2EAA2E,iEAAiE,MAAQ,GAAG,SAAW,2sBAA2sB,eAAiB,CAAC,q6BAAq6B,68EAAi9E,ieAAie,WAAa,MAEz/P,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,0WAA2W,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,sEAAsE,MAAQ,GAAG,SAAW,+GAA+G,eAAiB,CAAC,6oBAA6oB,WAAa,MAEtyC,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,svBAAuvB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,mDAAmD,yEAAyE,MAAQ,GAAG,SAAW,yMAAyM,eAAiB,CAAC,shBAAshB,g0BAAk0B,WAAa,MAE7gF,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,4lCAA6lC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,uDAAuD,6EAA6E,MAAQ,GAAG,SAAW,wRAAwR,eAAiB,CAAC,yRAAyR,y2CAAy2C,WAAa,MAEpvG,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,0gBAA2gB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,kDAAkD,MAAQ,GAAG,SAAW,mOAAmO,eAAiB,CAAC,01EAA61E,WAAa,MAEtvG,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,42BAA62B,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,yDAAyD,MAAQ,GAAG,SAAW,iSAAiS,eAAiB,CAAC,g7CAAg7C,WAAa,MAEhvF,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,s1BAAu1B,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,+DAA+D,qFAAqF,MAAQ,GAAG,SAAW,yJAAyJ,eAAiB,CAAC,8rBAA8rB,2pBAA2pB,WAAa,MAEtlF,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,goCAAioC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,8DAA8D,MAAQ,GAAG,SAAW,8RAA8R,eAAiB,CAAC,2iFAA2iF,WAAa,MAEjoI,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,yeAA0e,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,8EAA8E,MAAQ,GAAG,SAAW,0EAA0E,eAAiB,CAAC,27BAA27B,WAAa,MAEtrD,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,41GAA+1G,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,qDAAqD,0EAA0E,iEAAiE,MAAQ,GAAG,SAAW,qrBAAqrB,eAAiB,CAAC,o7CAAs7C,w1EAA01E,ieAAie,WAAa,MAE9jR,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,msBAAosB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,2DAA2D,iFAAiF,MAAQ,GAAG,SAAW,sJAAsJ,eAAiB,CAAC,wcAAwc,4oBAA4oB,WAAa,MAEnrE,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,8xCAAiyC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,gFAAgF,MAAQ,GAAG,SAAW,yNAAyN,eAAiB,CAAC,+pCAA+pC,WAAa,MAEl2F,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,irKAAkrK,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,iEAAiE,6DAA6D,sFAAsF,4EAA4E,gEAAgE,8DAA8D,+DAA+D,kEAAkE,MAAQ,GAAG,SAAW,ihCAAihC,eAAiB,CAAC,+pBAAiqB,uWAAuW,utIAAouI,gxBAAgxB,geAAge,gWAAgW,seAAse,wUAAwU,WAAa,MAEr8c,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,q2GAAs2G,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,uEAAuE,4FAA4F,gEAAgE,kEAAkE,MAAQ,GAAG,SAAW,ukBAAukB,eAAiB,CAAC,4UAA4U,6qIAAmrI,geAAge,wUAAwU,WAAa,MAElnT,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,8sCAA+sC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,uEAAuE,6FAA6F,MAAQ,GAAG,SAAW,gUAAgU,eAAiB,CAAC,2NAA2N,kvDAAmvD,WAAa,MAE1vH,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,ofAAqf,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,4EAA4E,MAAQ,GAAG,SAAW,gLAAgL,eAAiB,CAAC,4sBAA4sB,WAAa,MAEtjD,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,mjFAAojF,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,+EAA+E,qGAAqG,MAAQ,GAAG,SAAW,kWAAkW,eAAiB,CAAC,k3BAAo3B,iuDAAmuD,WAAa,MAE1xL,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,66HAA86H,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,qEAAqE,0FAA0F,iEAAiE,+DAA+D,iEAAiE,MAAQ,GAAG,SAAW,u0BAAu0B,eAAiB,CAAC,k1EAAk1E,6hGAAoiG,uUAAuU,seAAse,ieAAie,WAAa,MAEl1X,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,8iCAA+iC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,8FAA8F,iEAAiE,MAAQ,GAAG,SAAW,uLAAuL,eAAiB,CAAC,g+BAAk+B,ieAAie,WAAa,MAEj8F,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,yrFAA4rF,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wEAAwE,6FAA6F,iEAAiE,MAAQ,GAAG,SAAW,kcAAkc,eAAiB,CAAC,kYAAkY,o6GAA66G,ieAAie,WAAa,MAE7uP,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,2SAA4S,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,4EAA4E,kGAAkG,MAAQ,GAAG,SAAW,iFAAiF,eAAiB,CAAC,igBAAigB,0OAA0O,WAAa,MAE/4C,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,+pCAAgqC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,0FAA0F,gEAAgE,8DAA8D,gEAAgE,MAAQ,GAAG,SAAW,6PAA6P,eAAiB,CAAC,03DAA44D,geAAge,gWAAgW,ueAAue,WAAa,MAEj+J,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,qaAAsa,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,yEAAyE,+FAA+F,MAAQ,GAAG,SAAW,4JAA4J,eAAiB,CAAC,sUAAsU,ugBAAugB,WAAa,MAEhrD,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,maAAoa,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,6DAA6D,mFAAmF,MAAQ,GAAG,SAAW,4IAA4I,eAAiB,CAAC,8NAA8N,o9BAAw9B,WAAa,MAE/+D,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,qrCAAsrC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,oFAAoF,gEAAgE,8DAA8D,gEAAgE,MAAQ,GAAG,SAAW,4XAA4X,eAAiB,CAAC,00DAAg1D,geAAge,gWAAgW,ueAAue,WAAa,MAEpjK,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,2FAA4F,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,+DAA+D,qFAAqF,MAAQ,GAAG,SAAW,gDAAgD,eAAiB,CAAC,oQAAoQ,2NAA2N,WAAa,MAEx3B,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,oqIAAqqI,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,iFAAiF,sGAAsG,iEAAiE,MAAQ,GAAG,SAAW,qlBAAqlB,eAAiB,CAAC,8XAA8X,ykJAA6kJ,ieAAie,WAAa,MAEvhV,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,+vDAAgwD,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,6DAA6D,6DAA6D,kFAAkF,iEAAiE,MAAQ,GAAG,SAAW,8fAA8f,eAAiB,CAAC,uqBAAyqB,uWAAuW,svEAA2vE,ieAAie,WAAa,MAEh3M,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,+5MAAk6M,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,sEAAsE,2FAA2F,gEAAgE,gEAAgE,MAAQ,GAAG,SAAW,spCAAspC,eAAiB,CAAC,wqCAA8qC,yqLAAqrL,geAAge,ueAAue,WAAa,MAE5vf,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,yrBAA0rB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wEAAwE,8FAA8F,MAAQ,GAAG,SAAW,gMAAgM,eAAiB,CAAC,+fAA+f,+/BAA+/B,WAAa,MAEvpF,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,6qCAA8qC,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,mEAAmE,wFAAwF,gEAAgE,MAAQ,GAAG,SAAW,mTAAmT,eAAiB,CAAC,8fAA8f,k+CAAm+C,ueAAue,WAAa,MAE7vI,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,wWAAyW,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,qEAAqE,0FAA0F,gEAAgE,MAAQ,GAAG,SAAW,4HAA4H,eAAiB,CAAC,4aAA4a,+ZAAia,ueAAue,WAAa,MAEjnE,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,s+DAA2+D,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,kFAAkF,oDAAoD,wGAAwG,MAAQ,GAAG,SAAW,ueAAue,eAAiB,CAAC,6xBAAmyB,wRAAwR,48DAAg9D,WAAa,MAEp0L,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,gTAAiT,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wEAAwE,8FAA8F,MAAQ,GAAG,SAAW,2FAA2F,eAAiB,CAAC,2QAA2Q,ubAAub,WAAa,MAE72C,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,m4IAAo4I,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,mEAAmE,6DAA6D,wFAAwF,iEAAiE,MAAQ,GAAG,SAAW,m5BAAm5B,eAAiB,CAAC,4xCAA8xC,uWAAuW,+qLAAisL,ieAAie,WAAa,MAEh9a,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,mhBAAohB,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,mEAAmE,yFAAyF,MAAQ,GAAG,SAAW,sLAAsL,eAAiB,CAAC,wVAAwV,2xBAA6xB,WAAa,MAEplE,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,8nPAA+nP,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,wDAAwD,uDAAuD,wDAAwD,4DAA4D,2DAA2D,6EAA6E,+EAA+E,4EAA4E,4EAA4E,4EAA4E,8EAA8E,6EAA6E,+EAA+E,MAAQ,GAAG,SAAW,umEAAumE,eAAiB,CAAC,8aAA8a,wVAAwV,uQAAuQ,kWAAkW,izDAAqzD,uwMAAuwM,qRAAqR,+/BAA+/B,+dAA+d,68EAA29E,mQAAmQ,soBAAsoB,koBAAkoB,WAAa,MAE32zB,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,m5MAAo5M,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,mDAAmD,wEAAwE,8DAA8D,gEAAgE,MAAQ,GAAG,SAAW,ymCAAymC,eAAiB,CAAC,8tFAA8tF,i4MAAw4M,gWAAgW,ueAAue,WAAa,MAE5xjB,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,0gDAA2gD,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,yDAAyD,+EAA+E,MAAQ,GAAG,SAAW,4ZAA4Z,eAAiB,CAAC,4jCAA4jC,2sCAA2sC,WAAa,MAE/6I,S,gFCJIF,E,MAA0B,GAA4B,KAE1DA,EAAwBC,KAAK,CAACnJ,EAAOoJ,GAAI,kuPAAquP,GAAG,CAAC,QAAU,EAAE,QAAU,CAAC,6DAA6D,mFAAmF,MAAQ,GAAG,SAAW,s5CAAs5C,eAAiB,CAAC,i2GAA42G,gmRAAinR,WAAa,MAEj2qB,S,wBCCApJ,EAAOD,QAAU,SAAUsJ,GACzB,IAAIC,EAAO,GAuDX,OArDAA,EAAKrD,SAAW,WACd,OAAOsD,KAAKhD,KAAI,SAAUiD,GACxB,IAAIC,EAAUJ,EAAuBG,GAErC,OAAIA,EAAK,GACA,UAAUE,OAAOF,EAAK,GAAI,MAAME,OAAOD,EAAS,KAGlDA,CACT,IAAGhD,KAAK,GACV,EAIA6C,EAAKvF,EAAI,SAAU4F,EAASC,EAAYC,GACf,iBAAZF,IAETA,EAAU,CAAC,CAAC,KAAMA,EAAS,MAG7B,IAAIG,EAAyB,CAAC,EAE9B,GAAID,EACF,IAAK,IAAI9F,EAAI,EAAGA,EAAIwF,KAAK1H,OAAQkC,IAAK,CAEpC,IAAIqF,EAAKG,KAAKxF,GAAG,GAEP,MAANqF,IACFU,EAAuBV,IAAM,EAEjC,CAGF,IAAK,IAAIW,EAAK,EAAGA,EAAKJ,EAAQ9H,OAAQkI,IAAM,CAC1C,IAAIP,EAAO,GAAGE,OAAOC,EAAQI,IAEzBF,GAAUC,EAAuBN,EAAK,MAKtCI,IACGJ,EAAK,GAGRA,EAAK,GAAK,GAAGE,OAAOE,EAAY,SAASF,OAAOF,EAAK,IAFrDA,EAAK,GAAKI,GAMdN,EAAKH,KAAKK,GACZ,CACF,EAEOF,CACT,C,wBC/DA,SAASU,EAAeC,EAAKlG,GAAK,OAUlC,SAAyBkG,GAAO,GAAIC,MAAMC,QAAQF,GAAM,OAAOA,CAAK,CAV3BG,CAAgBH,IAQzD,SAA+BA,EAAKlG,GAAK,IAAIgG,EAAKE,IAA0B,oBAAXI,QAA0BJ,EAAII,OAAOC,WAAaL,EAAI,eAAgB,GAAU,MAANF,EAAY,OAAQ,IAAkDQ,EAAIC,EAAlDC,EAAO,GAAQC,GAAK,EAAUC,GAAK,EAAmB,IAAM,IAAKZ,EAAKA,EAAGa,KAAKX,KAAQS,GAAMH,EAAKR,EAAGc,QAAQC,QAAoBL,EAAKtB,KAAKoB,EAAGxI,QAAYgC,GAAK0G,EAAK5I,SAAWkC,GAA3D2G,GAAK,GAAkE,CAAE,MAAOK,GAAOJ,GAAK,EAAMH,EAAKO,CAAK,CAAE,QAAU,IAAWL,GAAsB,MAAhBX,EAAW,QAAWA,EAAW,QAAK,CAAE,QAAU,GAAIY,EAAI,MAAMH,CAAI,CAAE,CAAE,OAAOC,CAAM,CARnbO,CAAsBf,EAAKlG,IAI5F,SAAqCkH,EAAGC,GAAU,IAAKD,EAAG,OAAQ,GAAiB,iBAANA,EAAgB,OAAOE,EAAkBF,EAAGC,GAAS,IAAIrG,EAAItE,OAAO6K,UAAUnF,SAAS2E,KAAKK,GAAGI,MAAM,GAAI,GAAc,WAANxG,GAAkBoG,EAAEK,cAAazG,EAAIoG,EAAEK,YAAYC,MAAM,GAAU,QAAN1G,GAAqB,QAANA,EAAa,OAAOqF,MAAMrB,KAAKoC,GAAI,GAAU,cAANpG,GAAqB,2CAA2C2G,KAAK3G,GAAI,OAAOsG,EAAkBF,EAAGC,EAAS,CAJ7TO,CAA4BxB,EAAKlG,IAEnI,WAA8B,MAAM,IAAI2H,UAAU,4IAA8I,CAFvDC,EAAoB,CAM7J,SAASR,EAAkBlB,EAAKtC,IAAkB,MAAPA,GAAeA,EAAMsC,EAAIpI,UAAQ8F,EAAMsC,EAAIpI,QAAQ,IAAK,IAAIkC,EAAI,EAAG6H,EAAO,IAAI1B,MAAMvC,GAAM5D,EAAI4D,EAAK5D,IAAO6H,EAAK7H,GAAKkG,EAAIlG,GAAM,OAAO6H,CAAM,CAMtL5L,EAAOD,QAAU,SAAgCyJ,GAC/C,IAAIqC,EAAQ7B,EAAeR,EAAM,GAC7BC,EAAUoC,EAAM,GAChBC,EAAaD,EAAM,GAEvB,IAAKC,EACH,OAAOrC,EAGT,GAAoB,mBAATsC,KAAqB,CAE9B,IAAIC,EAASD,KAAKE,SAASC,mBAAmBC,KAAKC,UAAUN,MACzDO,EAAO,+DAA+D3C,OAAOsC,GAC7EM,EAAgB,OAAO5C,OAAO2C,EAAM,OACpCE,EAAaT,EAAWU,QAAQjG,KAAI,SAAUkG,GAChD,MAAO,iBAAiB/C,OAAOoC,EAAWY,YAAc,IAAIhD,OAAO+C,EAAQ,MAC7E,IACA,MAAO,CAAChD,GAASC,OAAO6C,GAAY7C,OAAO,CAAC4C,IAAgB7F,KAAK,KACnE,CAEA,MAAO,CAACgD,GAAShD,KAAK,KACxB,C,8BCjCA,IACMkG,EADFC,EAEK,WAUL,YAToB,IAATD,IAMTA,EAAOE,QAAQC,QAAUC,UAAYA,SAASC,MAAQF,OAAOG,OAGxDN,CACT,EAGEO,EAAY,WACd,IAAIP,EAAO,CAAC,EACZ,OAAO,SAAkBQ,GACvB,QAA4B,IAAjBR,EAAKQ,GAAyB,CACvC,IAAIC,EAAcL,SAASM,cAAcF,GAEzC,GAAIL,OAAOQ,mBAAqBF,aAAuBN,OAAOQ,kBAC5D,IAGEF,EAAcA,EAAYG,gBAAgBC,IAC5C,CAAE,MAAOC,GAEPL,EAAc,IAChB,CAGFT,EAAKQ,GAAUC,CACjB,CAEA,OAAOT,EAAKQ,EACd,CACF,CAtBgB,GAwBZO,EAAc,GAElB,SAASC,EAAqBC,GAG5B,IAFA,IAAIlG,GAAU,EAEL3D,EAAI,EAAGA,EAAI2J,EAAY7L,OAAQkC,IACtC,GAAI2J,EAAY3J,GAAG6J,aAAeA,EAAY,CAC5ClG,EAAS3D,EACT,KACF,CAGF,OAAO2D,CACT,CAEA,SAASmG,EAAavE,EAAMwE,GAI1B,IAHA,IAAIC,EAAa,CAAC,EACdC,EAAc,GAETjK,EAAI,EAAGA,EAAIuF,EAAKzH,OAAQkC,IAAK,CACpC,IAAIyF,EAAOF,EAAKvF,GACZqF,EAAK0E,EAAQG,KAAOzE,EAAK,GAAKsE,EAAQG,KAAOzE,EAAK,GAClD0E,EAAQH,EAAW3E,IAAO,EAC1BwE,EAAa,GAAGlE,OAAON,EAAI,KAAKM,OAAOwE,GAC3CH,EAAW3E,GAAM8E,EAAQ,EACzB,IAAIC,EAAQR,EAAqBC,GAC7BQ,EAAM,CACRC,IAAK7E,EAAK,GACV8E,MAAO9E,EAAK,GACZ+E,UAAW/E,EAAK,KAGH,IAAX2E,GACFT,EAAYS,GAAOK,aACnBd,EAAYS,GAAOM,QAAQL,IAE3BV,EAAYvE,KAAK,CACfyE,WAAYA,EACZa,QAASC,EAASN,EAAKN,GACvBU,WAAY,IAIhBR,EAAY7E,KAAKyE,EACnB,CAEA,OAAOI,CACT,CAEA,SAASW,EAAmBb,GAC1B,IAAIc,EAAQ7B,SAAS8B,cAAc,SAC/BC,EAAahB,EAAQgB,YAAc,CAAC,EAExC,QAAgC,IAArBA,EAAWC,MAAuB,CAC3C,IAAIA,EAAmD,KAEnDA,IACFD,EAAWC,MAAQA,EAEvB,CAMA,GAJAxO,OAAOC,KAAKsO,GAAY3H,SAAQ,SAAU7G,GACxCsO,EAAMI,aAAa1O,EAAKwO,EAAWxO,GACrC,IAE8B,mBAAnBwN,EAAQmB,OACjBnB,EAAQmB,OAAOL,OACV,CACL,IAAIzB,EAASD,EAAUY,EAAQmB,QAAU,QAEzC,IAAK9B,EACH,MAAM,IAAIvL,MAAM,2GAGlBuL,EAAO+B,YAAYN,EACrB,CAEA,OAAOA,CACT,CAaA,IACMO,EADFC,GACED,EAAY,GACT,SAAiBhB,EAAOkB,GAE7B,OADAF,EAAUhB,GAASkB,EACZF,EAAUG,OAAOzC,SAASpG,KAAK,KACxC,GAGF,SAAS8I,EAAoBX,EAAOT,EAAOqB,EAAQpB,GACjD,IAAIC,EAAMmB,EAAS,GAAKpB,EAAIE,MAAQ,UAAU5E,OAAO0E,EAAIE,MAAO,MAAM5E,OAAO0E,EAAIC,IAAK,KAAOD,EAAIC,IAIjG,GAAIO,EAAMa,WACRb,EAAMa,WAAWC,QAAUN,EAAYjB,EAAOE,OACzC,CACL,IAAIsB,EAAU5C,SAAS6C,eAAevB,GAClCwB,EAAajB,EAAMiB,WAEnBA,EAAW1B,IACbS,EAAMkB,YAAYD,EAAW1B,IAG3B0B,EAAWhO,OACb+M,EAAMmB,aAAaJ,EAASE,EAAW1B,IAEvCS,EAAMM,YAAYS,EAEtB,CACF,CAEA,SAASK,EAAWpB,EAAOd,EAASM,GAClC,IAAIC,EAAMD,EAAIC,IACVC,EAAQF,EAAIE,MACZC,EAAYH,EAAIG,UAepB,GAbID,EACFM,EAAMI,aAAa,QAASV,GAE5BM,EAAMqB,gBAAgB,SAGpB1B,GAA6B,oBAATxC,OACtBsC,GAAO,uDAAuD3E,OAAOqC,KAAKE,SAASC,mBAAmBC,KAAKC,UAAUmC,MAAe,QAMlIK,EAAMa,WACRb,EAAMa,WAAWC,QAAUrB,MACtB,CACL,KAAOO,EAAMsB,YACXtB,EAAMkB,YAAYlB,EAAMsB,YAG1BtB,EAAMM,YAAYnC,SAAS6C,eAAevB,GAC5C,CACF,CAEA,IAAI8B,EAAY,KACZC,EAAmB,EAEvB,SAAS1B,EAASN,EAAKN,GACrB,IAAIc,EACAyB,EACAb,EAEJ,GAAI1B,EAAQqC,UAAW,CACrB,IAAIG,EAAaF,IACjBxB,EAAQuB,IAAcA,EAAYxB,EAAmBb,IACrDuC,EAASd,EAAoBgB,KAAK,KAAM3B,EAAO0B,GAAY,GAC3Dd,EAASD,EAAoBgB,KAAK,KAAM3B,EAAO0B,GAAY,EAC7D,MACE1B,EAAQD,EAAmBb,GAC3BuC,EAASL,EAAWO,KAAK,KAAM3B,EAAOd,GAEtC0B,EAAS,YAxFb,SAA4BZ,GAE1B,GAAyB,OAArBA,EAAM4B,WACR,OAAO,EAGT5B,EAAM4B,WAAWV,YAAYlB,EAC/B,CAkFM6B,CAAmB7B,EACrB,EAIF,OADAyB,EAAOjC,GACA,SAAqBsC,GAC1B,GAAIA,EAAQ,CACV,GAAIA,EAAOrC,MAAQD,EAAIC,KAAOqC,EAAOpC,QAAUF,EAAIE,OAASoC,EAAOnC,YAAcH,EAAIG,UACnF,OAGF8B,EAAOjC,EAAMsC,EACf,MACElB,GAEJ,CACF,CAEAxP,EAAOD,QAAU,SAAUuJ,EAAMwE,IAC/BA,EAAUA,GAAW,CAAC,GAGTqC,WAA0C,kBAAtBrC,EAAQqC,YACvCrC,EAAQqC,UAAYvD,KAItB,IAAI+D,EAAkB9C,EADtBvE,EAAOA,GAAQ,GAC0BwE,GACzC,OAAO,SAAgB8C,GAGrB,GAFAA,EAAUA,GAAW,GAE2B,mBAA5CrQ,OAAO6K,UAAUnF,SAAS2E,KAAKgG,GAAnC,CAIA,IAAK,IAAI7M,EAAI,EAAGA,EAAI4M,EAAgB9O,OAAQkC,IAAK,CAC/C,IACIoK,EAAQR,EADKgD,EAAgB5M,IAEjC2J,EAAYS,GAAOK,YACrB,CAIA,IAFA,IAAIqC,EAAqBhD,EAAa+C,EAAS9C,GAEtC/D,EAAK,EAAGA,EAAK4G,EAAgB9O,OAAQkI,IAAM,CAClD,IAEI+G,EAASnD,EAFKgD,EAAgB5G,IAIK,IAAnC2D,EAAYoD,GAAQtC,aACtBd,EAAYoD,GAAQrC,UAEpBf,EAAYqD,OAAOD,EAAQ,GAE/B,CAEAH,EAAkBE,CAtBlB,CAuBF,CACF,C,GC3QIG,EAA2B,CAAC,EAGhC,SAASC,EAAoBC,GAE5B,IAAIC,EAAeH,EAAyBE,GAC5C,QAAqBE,IAAjBD,EACH,OAAOA,EAAapR,QAGrB,IAAIC,EAASgR,EAAyBE,GAAY,CACjD9H,GAAI8H,EAEJnR,QAAS,CAAC,GAOX,OAHAsR,EAAoBH,GAAUlR,EAAQA,EAAOD,QAASkR,GAG/CjR,EAAOD,OACf,CCrBAkR,EAAoBpM,EAAK7E,IACxB,IAAIsR,EAAStR,GAAUA,EAAOuR,WAC7B,IAAOvR,EAAiB,QACxB,IAAM,EAEP,OADAiR,EAAoBO,EAAEF,EAAQ,CAAEpM,EAAGoM,IAC5BA,CAAM,ECLdL,EAAoBO,EAAI,CAACzR,EAAS0R,KACjC,IAAI,IAAInR,KAAOmR,EACXR,EAAoBhG,EAAEwG,EAAYnR,KAAS2Q,EAAoBhG,EAAElL,EAASO,IAC5EC,OAAOuB,eAAe/B,EAASO,EAAK,CAAEoR,YAAY,EAAMC,IAAKF,EAAWnR,IAE1E,ECND2Q,EAAoBhP,EAAI,WACvB,GAA0B,iBAAf2P,WAAyB,OAAOA,WAC3C,IACC,OAAOrI,MAAQ,IAAIsI,SAAS,cAAb,EAChB,CAAE,MAAOpE,GACR,GAAsB,iBAAXX,OAAqB,OAAOA,MACxC,CACA,CAPuB,GCAxBmE,EAAoBhG,EAAI,CAACmD,EAAK0D,IAAUvR,OAAO6K,UAAU2G,eAAenH,KAAKwD,EAAK0D,GCAlFb,EAAoBe,QAAKZ,E,0DCwBzB,MAAMa,EATC,WAEN,IACC,OAAOC,UAAUD,UAAUE,aAC5B,CAAE,MAAQ1E,GACT,MAAO,EACR,CACD,CAEkB2E,GAoGX,IAAoBH,EAlB3B,QApBqB,CACpBI,MAAOA,EAAOJ,GAEdK,WAmC0BL,EAnCJA,EAoCfA,EAAUM,QAAS,YAAe,GAlCzCC,QA2CM,SAAkBP,GACxB,QAASA,EAAU7L,MAAO,aAC3B,CA7CUoM,CAASP,GAElBQ,SAmDM,SAAmBR,GACzB,OAAOA,EAAUM,QAAS,kBAAqB,IAAwC,IAAnCN,EAAUM,QAAS,SACxE,CArDWE,CAAUR,GAEpBS,MA2DM,SAAgBT,GAEtB,QAASA,EAAU7L,MAAO,iBAAsBiM,EAAOJ,IAAeC,UAAUS,eAAiB,CAClG,CA9DQD,CAAOT,GAEdW,UAoEM,SAAoBX,GAC1B,OAAOA,EAAUM,QAAS,YAAe,CAC1C,CAtEYK,CAAWX,GAEtBY,QA4EM,SAAkBZ,GAGxB,OAAOA,EAAUM,QAAS,YAAe,GAAKN,EAAUM,QAAS,SAAY,CAC9E,CAhFUM,CAASZ,GAElBa,SAAU,CACTC,iCAoFK,WACN,IAAIC,GAAc,EAKlB,IAECA,EAA8D,IAAhD,IAAIC,OAAQ,IAAIC,OAAQ,WAAY,KACnD,CAAE,MAAQC,GAEV,CAEA,OAAOH,CACR,CAlGoCD,KAY7B,SAASV,EAAOJ,GACtB,OAAOA,EAAUM,QAAS,cAAiB,CAC5C,CCFe,SAASa,EACvBlO,EACAhD,EACAmR,EACAC,GAGAD,EAAMA,GAAO,SAAUnO,EAAGhD,GACzB,OAAOgD,IAAMhD,CACd,EAQA,MAAMqR,EAAmBrJ,MAAMC,QAASjF,GAAMA,EAAIgF,MAAMkB,UAAUC,MAAMT,KAAM1F,GACxEsO,EAAmBtJ,MAAMC,QAASjI,GAAMA,EAAIgI,MAAMkB,UAAUC,MAAMT,KAAM1I,GAGxEuR,EAsBP,SAAuCC,EAAwB9H,EAAwByH,GAEtF,MAAMM,EAAaC,EAA0BF,EAAM9H,EAAMyH,GAGzD,IAAqB,IAAhBM,EACJ,MAAO,CAAEA,YAAa,EAAGE,cAAe,EAAGC,cAAe,GAI3D,MAAMC,EAAmBC,EAAeN,EAAMC,GACxCM,EAAmBD,EAAepI,EAAM+H,GAaxCO,EAAYN,EAA0BG,EAAkBE,EAAkBZ,GAG1EQ,EAAeH,EAAK7R,OAASqS,EAC7BJ,EAAelI,EAAK/J,OAASqS,EAEnC,MAAO,CAAEP,aAAYE,eAAcC,eACpC,CArDuBK,CAA2BZ,EAAQC,EAAQH,GAG3D3L,EAAS4L,EAqHhB,SAAuCG,EAA8BW,GACpE,MAAM,WAAET,EAAU,aAAEE,EAAY,aAAEC,GAAiBL,EAGnD,IAAqB,IAAhBE,EACJ,OAAOzJ,MAAOkK,GAAYC,KAAM,SAGjC,IAAI3M,EAA4B,GAE3BiM,EAAa,IACjBjM,EAASA,EAAOgC,OAAQQ,MAAOyJ,GAAaU,KAAM,WAG9CP,EAAeH,EAAa,IAChCjM,EAASA,EAAOgC,OAAQQ,MAAO4J,EAAeH,GAAaU,KAAM,YAG7DR,EAAeF,EAAa,IAChCjM,EAASA,EAAOgC,OAAQQ,MAAO2J,EAAeF,GAAaU,KAAM,YAG7DP,EAAeM,IACnB1M,EAASA,EAAOgC,OAAQQ,MAAOkK,EAAYN,GAAeO,KAAM,WAGjE,OAAO3M,CACR,CA/IE4M,CAA8Bb,EAAeD,EAAO3R,QAmFtD,SAAoC0S,EAA4Bd,GAC/D,MAAM/L,EAA2B,IAC3B,WAAEiM,EAAU,aAAEE,EAAY,aAAEC,GAAiBL,EAK9CK,EAAeH,EAAa,GAChCjM,EAAOyB,KAAM,CACZgF,MAAOwF,EACPa,KAAM,SACNC,OAAQF,EAASlJ,MAAOsI,EAAYG,KAIjCD,EAAeF,EAAa,GAChCjM,EAAOyB,KAAM,CACZgF,MAAOwF,GAAeG,EAAeH,GACrCa,KAAM,SACNE,QAASb,EAAeF,IAI1B,OAAOjM,CACR,CA1GEiN,CAAwBnB,EAAQC,GAEjC,OAAO/L,CACR,CAkDA,SAASkM,EAA6BF,EAAwB9H,EAAwByH,GACrF,QAAUtP,EAAI,EAAGA,EAAI3B,KAAKC,IAAKqR,EAAK7R,OAAQ+J,EAAK/J,QAAUkC,IAC1D,QAAmB,IAAd2P,EAAM3P,SAAmC,IAAd6H,EAAM7H,KAAsBsP,EAAKK,EAAM3P,GAAK6H,EAAM7H,IACjF,OAAOA,EAIT,OAAQ,CACT,CASA,SAASiQ,EAAkB/J,EAAuByK,GACjD,OAAOzK,EAAIoB,MAAOqJ,GAAUE,SAC7B,CCpLe,SAAS9R,EACvBoC,EACAhD,EACAmR,GAGAA,EAAMA,GAAO,SAAUnO,EAAGhD,GACzB,OAAOgD,IAAMhD,CACd,EAEA,MAAM2S,EAAU3P,EAAErD,OACZiT,EAAU5S,EAAEL,OAGlB,GAAKgT,EAAU,KAAOC,EAAU,KAAOD,EAAUC,EAAU,IAC1D,OAAOhS,EAAKsQ,SAAUlO,EAAGhD,EAAGmR,GAAK,GAIlC,IAAI0B,EAAqBC,EAGzB,GAAKF,EAAUD,EAAU,CACxB,MAAMI,EAAM/P,EAEZA,EAAIhD,EACJA,EAAI+S,EAGJF,EAAU,SACVC,EAAU,QACX,MACCD,EAAU,SACVC,EAAU,SAGX,MAAMlQ,EAAII,EAAErD,OACNgD,EAAI3C,EAAEL,OACNS,EAAQuC,EAAIC,EAGZoQ,EAA2C,CAAC,EAE5CC,EAA+B,CAAC,EAEtC,SAASC,EAAOlS,GAGf,MAAMmS,QAAuB,IAAhBF,EAAIjS,EAAI,GAAoBiS,EAAIjS,EAAI,IAAO,GAAM,EAExD6B,OAAqB,IAAhBoQ,EAAIjS,EAAI,GAAoBiS,EAAIjS,EAAI,IAAO,EAEhDoS,EAAMD,EAAKtQ,GAAM,EAAI,EAGtBmQ,EAAIhS,EAAIoS,KACZJ,EAAIhS,GAAMgS,EAAIhS,EAAIoS,GAAMjK,MAAO,IAI1B6J,EAAIhS,KACTgS,EAAIhS,GAAM,IAIXgS,EAAIhS,GAAIiG,KAAMkM,EAAKtQ,EAAKgQ,EAAUC,GAGlC,IAAIxR,EAAIpB,KAAKC,IAAKgT,EAAItQ,GAClBtB,EAAID,EAAIN,EAGZ,KAAQO,EAAIqB,GAAKtB,EAAIqB,GAAKwO,EAAMnO,EAAGzB,GAAKvB,EAAGsB,KAC1CC,IACAD,IAEA0R,EAAIhS,GAAIiG,KAAM,SAGf,OAAO3F,CACR,CAEA,IACIN,EADAmB,EAAI,EAIR,EAAG,CAEF,IAAMnB,GAAKmB,EAAGnB,EAAIZ,EAAOY,IACxBiS,EAAIjS,GAAMkS,EAAOlS,GAIlB,IAAMA,EAAIZ,EAAQ+B,EAAGnB,EAAIZ,EAAOY,IAC/BiS,EAAIjS,GAAMkS,EAAOlS,GAKlBiS,EAAI7S,GAAU8S,EAAO9S,GAErB+B,GACD,OAAU8Q,EAAI7S,KAAYuC,GAI1B,OAAOqQ,EAAI5S,GAAQ+I,MAAO,EAC3B,CAIAvI,EAAKsQ,SAAWA,ECvHhB,QANA,WACC,OAAO,SAASmC,IACbA,EAAaC,QAAS,CACzB,CACD,ECPe,MAAMC,EAgDpB,WAAAnK,CAAamB,EAAgBlB,GAC5BhC,KAAKkD,OAASA,EACdlD,KAAKgC,KAAOA,EACZhC,KAAKP,KAAO,GAGZO,KAAKmM,KAAO,IACZnM,KAAKoM,IAAM,GACZ,EC1DD,MAAMC,EAAc,IAAI1L,MAAO,KAAMmK,KAAM,IACzC9N,KAAK,CAAEsP,EAAG1H,KAAa,IAAQA,EAAQlI,SAAU,KAAOoF,OAAQ,KAYnD,SAAS,IAWvB,MAAMyK,EAAqB,WAAhB1T,KAAK2T,WAA2B,EACrCC,EAAqB,WAAhB5T,KAAK2T,WAA2B,EACrCE,EAAqB,WAAhB7T,KAAK2T,WAA2B,EACrCG,EAAqB,WAAhB9T,KAAK2T,WAA2B,EAG3C,MAAO,IACNH,EAAaE,GAAM,EAAI,KACvBF,EAAaE,GAAM,EAAI,KACvBF,EAAaE,GAAM,GAAK,KACxBF,EAAaE,GAAM,GAAK,KACxBF,EAAaI,GAAM,EAAI,KACvBJ,EAAaI,GAAM,EAAI,KACvBJ,EAAaI,GAAM,GAAK,KACxBJ,EAAaI,GAAM,GAAK,KACxBJ,EAAaK,GAAM,EAAI,KACvBL,EAAaK,GAAM,EAAI,KACvBL,EAAaK,GAAM,GAAK,KACxBL,EAAaK,GAAM,GAAK,KACxBL,EAAaM,GAAM,EAAI,KACvBN,EAAaM,GAAM,EAAI,KACvBN,EAAaM,GAAM,GAAK,KACxBN,EAAaM,GAAM,GAAK,IAC1B,CCtBA,MAgBA,EAhBmC,CAClC,GAAAvE,CAAKwE,EAA2B,UAC/B,MAAwB,iBAAZA,EACJ5M,KAAM4M,IAAc5M,KAAK6M,OAEzBD,CAET,EAEAE,QAAS,IACTC,KAAM,IACNF,OAAQ,EACRG,KAAM,IACNC,QAAS,KCjBK,SAASC,EAAqDC,EAAmBC,GAC/F,MAAMR,EAAW,EAAWxE,IAAKgF,EAAeR,UAEhD,QAAUpS,EAAI,EAAGA,EAAI2S,EAAQ7U,OAAQkC,IACpC,GAAK,EAAW4N,IAAK+E,EAAS3S,GAAIoS,UAAaA,EAG9C,YAFAO,EAAQ3F,OAAQhN,EAAG,EAAG4S,GAMxBD,EAAQvN,KAAMwN,EACf,CChCO,MAAMC,EAAoB,sEA8BlB,MAAM,UAAsBhV,MAyB1C,WAAA0J,CAAauL,EAAmBC,EAAyBzK,GACxD0K,MA4GF,SAA0BF,EAAmBxK,GAC5C,MAAM2K,EAAmB,IAAIC,QACvBC,EAA6B,CAAE5W,EAAayB,KACjD,GAAsB,iBAAVA,GAAgC,OAAVA,EAAiB,CAClD,GAAKiV,EAAiBG,IAAKpV,GAC1B,MAAO,WAAYA,EAAMuJ,YAAYC,QAGtCyL,EAAiBI,IAAKrV,EACvB,CAEA,OAAOA,CAAK,EAGPsV,EAAkBhL,EAAO,IAAKF,KAAKC,UAAWC,EAAM6K,KAAkC,GACtFI,EAAoBC,EAA+BV,GAEzD,OAAOA,EAAYQ,EAAkBC,CACtC,CA9HSE,CAAiBX,EAAWxK,IAEnC9C,KAAKgC,KAAO,gBACZhC,KAAKuN,QAAUA,EACfvN,KAAK8C,KAAOA,CACb,CAKO,EAAAoL,CAAIjD,GACV,MAAgB,kBAATA,CACR,CAWA,6BAAckD,CAAwB3M,EAAY+L,GACjD,GAAO/L,EAAa0M,IAAQ1M,EAAa0M,GAAI,iBAC5C,MAAM1M,EAYP,MAAMoI,EAAQ,IAAI,EAAepI,EAAI4M,QAASb,GAM9C,MAFA3D,EAAMyE,MAAQ7M,EAAI6M,MAEZzE,CACP,EAwBM,SAAS,EAAY0D,EAAmBxK,GAC9CwL,QAAQC,QAASC,EAAwBlB,EAAWxK,GACrD,CAwBO,SAAS2L,EAAUnB,EAAmBxK,GAC5CwL,QAAQ1E,SAAU4E,EAAwBlB,EAAWxK,GACtD,CAKA,SAASkL,EAA+BV,GACvC,MAAO,gBAAiBD,WAA6BC,GACtD,CA4BA,SAASkB,EAAwBlB,EAAmBxK,GACnD,MAAM4L,EAAuBV,EAA+BV,GAE5D,OAAOxK,EAAO,CAAEwK,EAAWxK,EAAM4L,GAAyB,CAAEpB,EAAWoB,EACxE,CChMA,MAAMC,EAAU,SAKHC,EAAc,IAAIC,KAAM,KAAM,EAAG,IAGxCC,EAAmC,iBAAXvL,OAAsBA,OAAS,EAAA7K,EAQ7D,GAAKoW,EAAeC,iBAuInB,MAAM,IAAI,EACT,8BACA,MAGDD,EAAeC,iBAAmBJ,ECtJnC,MAAMK,EAAelO,OAAQ,eACvBmO,EAAanO,OAAQ,aACrBoO,EAAepO,OAAQ,eAEvBqO,EAAsBC,EAAcpY,QA+C3B,SAASoY,EAAc1K,GACrC,IAAMA,EACL,OAAOyK,EA2SR,OAxSA,cAA6BzK,EACrB,EAAA2K,CACNC,EACAC,EACAhL,GAEAvE,KAAKwP,SAAUxP,KAAMsP,EAAOC,EAAUhL,EACvC,CAEO,IAAAkL,CACNH,EACAC,EACAhL,GAEA,IAAImL,GAAW,EAiBf1P,KAAKwP,SAAUxP,KAAMsP,GAfiB,CAAEA,KAAUpT,KAG3CwT,IACLA,GAAW,EAGXJ,EAAMlD,MAGNmD,EAASlO,KAAMrB,KAAMsP,KAAUpT,GAChC,GAIyCqI,EAC3C,CAEO,GAAA6H,CAAKkD,EAAeC,GAC1BvP,KAAK2P,cAAe3P,KAAMsP,EAAOC,EAClC,CAEO,QAAAC,CACNI,EACAN,EACAC,EACAhL,EAA2B,CAAC,GAE5B,IAAIsL,EAAaC,EAgBX9P,KAAMgP,KACXhP,KAAMgP,GAAiB,CAAC,GAGzB,MAAMe,EAAW/P,KAAMgP,GAEjBgB,EAAeJ,IACpBK,EAAeL,GAGhB,MAAMM,EAAYF,EAAeJ,IAEzBC,EAAcE,EAAUG,MAC/BL,EAAcE,EAAUG,GAAc,CACrCN,UACAO,UAAW,CAAC,KAINL,EAAiBD,EAAYM,UAAWb,MAC/CQ,EAAiBD,EAAYM,UAAWb,GAAU,IAGnDQ,EAAelQ,KAAM2P,GAutBxB,SACCa,EACAR,EACAN,EACAC,EACAhL,GAEKqL,EAAQS,kBACZT,EAAQS,kBAAmBf,EAAOC,EAAUhL,GAI1C6L,EAASC,kBAA6BhP,KAAMuO,EAASN,EAAOC,EAAUhL,EAE1E,CAluBG+L,CAAkBtQ,KAAM4P,EAASN,EAAOC,EAAUhL,EACnD,CAEO,aAAAoL,CAAeC,EAAmBN,EAAgBC,GACxD,MAAMQ,EAAW/P,KAAMgP,GACvB,IAAIkB,EAAYN,GAAWI,EAAeJ,GAC1C,MAAMC,EAAgBE,GAAYG,EAAcH,EAAUG,QAAc,EAClEJ,EAAmBD,GAAeP,EAAUO,EAAYM,UAAWb,QAAU,EAGnF,MAAMS,GAAcH,IAAYC,GAAmBP,IAAUQ,GAK7D,GAAKP,EAAW,CACfgB,EAAqBvQ,KAAM4P,EAAUN,EAAQC,IAM7B,IAFFO,EAAgB9G,QAASuG,KAGN,IAA3BO,EAAgBxX,cACbuX,EAAaM,UAAWb,GAE/BiB,EAAqBvQ,KAAM4P,EAAUN,EAAQC,GAGhD,SAEUO,EAAiB,CAC1B,KAAUP,EAAWO,EAAe9Q,OACnCuR,EAAqBvQ,KAAM4P,EAAUN,EAAQC,UAGvCM,EAAaM,UAAWb,EAChC,SAEUO,EAAc,CACvB,IAAMP,KAASO,EAAYM,UAC1BnQ,KAAK2P,cAAeC,EAAUN,UAExBS,EAAUG,EAClB,KAEK,CACJ,IAAMA,KAAaH,EAClB/P,KAAK2P,cAAeI,EAAUG,GAAYN,gBAEpC5P,KAAMgP,EACd,CACD,CAEO,IAAAwB,CACNC,KACGvU,GAEH,IACC,MAAMwU,EAAYD,aAAuBvE,EAAYuE,EAAc,IAAIvE,EAAWlM,KAAMyQ,GAClFnB,EAAQoB,EAAU1O,KACxB,IAAImO,EAAYQ,EAAsB3Q,KAAMsP,GAM5C,GAHAoB,EAAUjR,KAAKG,KAAMI,MAGhBmQ,EAAY,CAEhB,MAAMS,EAAe,CAAEF,KAAcxU,GAOrCiU,EAAYxP,MAAMrB,KAAM6Q,GAExB,QAAU3V,EAAI,EAAGA,EAAI2V,EAAU7X,SAC9B6X,EAAW3V,GAAI+U,SAASsB,MAAO7Q,KAAM4Q,GAGhCF,EAAUtE,IAAIH,gBAEXyE,EAAUtE,IAAIH,OAErBjM,KAAK8Q,qBAAsBxB,EAAOa,EAAW3V,GAAI+U,YAI7CmB,EAAUvE,KAAKF,QAZkBzR,KAgBxC,CAGA,MAAMuW,EAAc/Q,KAAMkP,GAE1B,GAAK6B,EAAc,CAClB,MAAMC,EAAeD,EAAY3I,IAAKkH,GAChC2B,EAAsBF,EAAY3I,IAAK,KAExC4I,GACJE,EAAqBF,EAAcN,EAAWxU,GAG1C+U,GACJC,EAAqBD,EAAqBP,EAAWxU,EAEvD,CAEA,OAAOwU,EAAUS,MAClB,CAAE,MAAQ3P,GAGT,EAAc2M,uBAAwB3M,EAAcxB,KACrD,CACD,CAEO,QAAAoR,IAAaC,GACnB,MAAO,CACN9R,GAAI,CAAEqQ,EAAS0B,KACRtR,KAAMkP,KACXlP,KAAMkP,GAAiB,IAAIqC,KAK5BF,EAAOzT,SAAS4T,IACf,MAAMR,EAAehR,KAAMkP,GAAgB9G,IAAKoJ,GAE1CR,EAGLA,EAAaS,IAAK7B,EAAS0B,GAF3BtR,KAAMkP,GAAgBuC,IAAKD,EAAW,IAAID,IAAK,CAAE,CAAE3B,EAAS0B,KAG7D,GACE,EAGN,CAEO,cAAAI,CAAgBpC,EAAgBM,GACtC,GAAM5P,KAAMkP,GAIZ,GAAMI,EAEC,GAAMM,EAEN,CACN,MAAMoB,EAAehR,KAAMkP,GAAgB9G,IAAKkH,GAE3C0B,GACJA,EAAaW,OAAQ/B,EAEvB,MAPC5P,KAAMkP,GAAgByC,OAAQrC,QAF9BtP,KAAMkP,GAAgB0C,OAUxB,CAEO,iBAAAvB,CACNf,EACAC,EACAhL,IAuaH,SAA+BrB,EAAyBsO,GACvD,MAAMH,EAASQ,EAAW3O,GAG1B,GAAKmO,EAAQG,GAEZ,OASD,IAAIxP,EAAOwP,EAEPM,EAAiB,KAGrB,MAAMC,EAAgB,GAKtB,KAAiB,KAAT/P,IACFqP,EAAQrP,IAQbqP,EAAQrP,GA/CF,CACNmO,UAAW,GACX6B,YAAa,IA+CbD,EAAcnS,KAAMyR,EAAQrP,IAGvB8P,GACJT,EAAQrP,GAAOgQ,YAAYpS,KAAMkS,GAGlCA,EAAiB9P,EAEjBA,EAAOA,EAAKiQ,OAAQ,EAAGjQ,EAAKkQ,YAAa,MAG1C,GAAc,KAATlQ,EAAc,CAKlB,UAAY7C,KAAQ4S,EACnB5S,EAAKgR,UAAYkB,EAAQrP,GAAOmO,UAAUrO,QAI3CuP,EAAQrP,GAAOgQ,YAAYpS,KAAMkS,EAClC,CACD,CAjeGK,CAAsBnS,KAAMsP,GAE5B,MAAM8C,EAAQC,EAA+BrS,KAAMsP,GAG7CgD,EAAqB,CAC1B/C,WACA3C,SAJgB,EAAWxE,IAAK7D,EAAQqI,WAQzC,UAAYuD,KAAaiC,EAExBlF,EAAuBiD,EAAWmC,EAEpC,CAEO,oBAAAxB,CAAsBxB,EAAeC,GAC3C,MAAM6C,EAAQC,EAA+BrS,KAAMsP,GAEnD,UAAYa,KAAaiC,EACxB,QAAU5X,EAAI,EAAGA,EAAI2V,EAAU7X,OAAQkC,IACjC2V,EAAW3V,GAAI+U,UAAYA,IAE/BY,EAAU3I,OAAQhN,EAAG,GACrBA,IAIJ,EAiBF,CAkUO,SAASyV,EAAeL,EAAkB/P,GACxC+P,EAA8BX,KACnCW,EAA8BX,GAAepP,GAAM,IAEvD,CAQO,SAASmQ,EAAeJ,GAC9B,OAASA,EAA8BX,EACxC,CAYA,SAAS4C,EAAW3O,GAOnB,OANMA,EAAOqP,SACZvb,OAAOuB,eAAgB2K,EAAQ,UAAW,CACzC1K,MAAO,CAAC,IAIH0K,EAAOqP,OACf,CAsFA,SAASF,EAA+BnP,EAAyBsO,GAChE,MAAMgB,EAAYX,EAAW3O,GAAUsO,GAEvC,IAAMgB,EACL,MAAO,GAGR,IAAIC,EAAiB,CAAED,EAAUrC,WAEjC,QAAU3V,EAAI,EAAGA,EAAIgY,EAAUR,YAAY1Z,OAAQkC,IAAM,CACxD,MAAMkY,EAAsBL,EAA+BnP,EAAQsP,EAAUR,YAAaxX,IAE1FiY,EAAiBA,EAAetS,OAAQuS,EACzC,CAEA,OAAOD,CACR,CAOA,SAAS9B,EAAsBzN,EAAyBsO,GACvD,IAAIlC,EAEJ,OAAMpM,EAAOqP,UAAcjD,EAAQpM,EAAOqP,QAASf,KAAkBlC,EAAMa,UAAU7X,OAY9EgX,EAAMa,UATPqB,EAAUxI,QAAS,MAAS,EAEzB2H,EAAsBzN,EAAQsO,EAAUS,OAAQ,EAAGT,EAAUU,YAAa,OAG1E,IAKV,CASA,SAAShB,EACRF,EACAN,EACAiC,GAEA,QAAY/C,EAAS5N,KAAUgP,EAAe,CACvChP,EAEqB,mBAARA,IAClBA,EAAOA,EAAM0O,EAAU1O,OAFvBA,EAAO0O,EAAU1O,KAKlB,MAAM4Q,EAAgB,IAAI1G,EAAWwE,EAAUxN,OAAQlB,GAEvD4Q,EAAcnT,KAAO,IAAKiR,EAAUjR,MAEpCmQ,EAAQY,KAAMoC,KAAkBD,EACjC,CACD,CAwBA,SAASpC,EAAqBH,EAA2BR,EAA0BN,EAAeC,GAC5FK,EAAQkB,qBACZlB,EAAQkB,qBAAsBxB,EAAOC,GAIrCa,EAASU,qBAAsBzP,KAAMuO,EAASN,EAAOC,EAEvD,CAzhBE,CACD,KAAM,OAAQ,MAAO,WACrB,gBAAiB,OAAQ,WAAY,iBACrC,oBAAqB,wBAClB3R,SAAS7G,IACVqY,EAAuBrY,GAAUoY,EAAoBtN,UAAoB9K,EAAK,IC9VjF,QALA,SAAkByB,GAChB,IAAIyS,SAAczS,EAClB,OAAgB,MAATA,IAA0B,UAARyS,GAA4B,YAARA,EAC/C,ECXM4H,EAA6B/R,OAAQ,wBACrCgS,EAAyBhS,OAAQ,oBACjCiS,EAAwBjS,OAAQ,mBAEhCkS,EAAmBlS,OAAQ,oBAC3BmS,EAAoBnS,OAAQ,qBAE5BoS,EAAyBC,EAAiB/D,KAiDjC,SAAS+D,EAAiBzO,GACxC,IAAMA,EACL,OAAOwO,EAyPR,OAtPA,cAA6BxO,EACrB,GAAA+M,CAAKzP,EAA8CxJ,GAEzD,GAAK,EAAUwJ,GAKd,YAJAhL,OAAOC,KAAM+K,GAAOpE,SAASwV,IAC5BpT,KAAKyR,IAAK2B,EAAUpR,EAAMoR,GAAY,GACpCpT,MAKJqT,EAAgBrT,MAEhB,MAAMsT,EAAatT,KAAM6S,GAEzB,GAAO7Q,KAAQhC,OAAWsT,EAAY1F,IAAK5L,GAkB1C,MAAM,IAAI,EAAe,iCAAkChC,MAG5DhJ,OAAOuB,eAAgByH,KAAMgC,EAAM,CAClCmG,YAAY,EACZoL,cAAc,EAEdnL,IAAG,IACKkL,EAAYlL,IAAKpG,GAGzB,GAAAyP,CAAuBjZ,GACtB,MAAMgb,EAAWF,EAAYlL,IAAKpG,GAKlC,IAAIyR,EAAWzT,KAAKwQ,KAA0B,OAAQxO,IAASA,EAAMxJ,EAAOgb,QAE1D,IAAbC,IACJA,EAAWjb,GAKPgb,IAAaC,GAAaH,EAAY1F,IAAK5L,KAC/CsR,EAAY7B,IAAKzP,EAAMyR,GACvBzT,KAAKwQ,KAA6B,UAAWxO,IAASA,EAAMyR,EAAUD,GAExE,IAGCxT,KAAegC,GAASxJ,CAC3B,CAEO,IAAAwO,IAAS0M,GACf,IAAMA,EAAepb,SAAWqb,EAAeD,GAM9C,MAAM,IAAI,EAAe,mCAAoC1T,MAG9D,GAAO,IAAI4T,IAAKF,GAAmBG,OAASH,EAAepb,OAM1D,MAAM,IAAI,EAAe,uCAAwC0H,MAGlEqT,EAAgBrT,MAEhB,MAAM8T,EAAkB9T,KAAM+S,GAE9BW,EAAe9V,SAASmW,IACvB,GAAKD,EAAiBlG,IAAKmG,GAM1B,MAAM,IAAI,EAAe,yBAA0B/T,KACpD,IAGD,MAAMgU,EAAW,IAAIzC,IASrB,OAPAmC,EAAe9V,SAASjC,IACvB,MAAMsY,EAAU,CAAEb,SAAUzX,EAAG4D,GAAI,IAEnCuU,EAAiBrC,IAAK9V,EAAGsY,GACzBD,EAASvC,IAAK9V,EAAGsY,EAAS,IAGpB,CACN1U,GAAI2U,EACJC,OAAQC,EAERC,YAAarU,KACbsU,gBAAiBZ,EACjBa,IAAK,GACLC,UAAWR,EAEb,CAEO,MAAAS,IAAWC,GAEjB,IAAQ1U,KAAM6S,GACb,OAGD,MAAMiB,EAAkB9T,KAAM+S,GACxB4B,EAAmB3U,KAAM8S,GAE/B,GAAK4B,EAAiBpc,OAAS,CAC9B,IAAMqb,EAAee,GAMpB,MAAM,IAAI,EAAe,qCAAsC1U,MAGhE0U,EAAiB9W,SAASmW,IACzB,MAAME,EAAUH,EAAgB1L,IAAK2L,GAG/BE,IAINA,EAAQ1U,GAAG3B,SAAS,EAAIgX,EAAcC,MACrC,MAAMC,EAAeH,EAAiBvM,IAAKwM,GACrCG,EAAqBD,EAAcD,GAEzCE,EAAmBpD,OAAQsC,GAErBc,EAAmBlB,aACjBiB,EAAcD,GAGhB7d,OAAOC,KAAM6d,GAAexc,SACjCqc,EAAiBhD,OAAQiD,GACzB5U,KAAK2P,cAAeiF,EAAc,UACnC,IAGDd,EAAgBnC,OAAQoC,GAAc,GAExC,MACCY,EAAiB/W,SAAS,CAAEoW,EAAUgB,KACrChV,KAAK2P,cAAeqF,EAAiB,SAAU,IAGhDL,EAAiB/C,QACjBkC,EAAgBlC,OAElB,CAEO,QAAAqD,CAA+CC,GACrD7B,EAAgBrT,MAEhB,MAAMmV,EAAiBnV,KAAMkV,GAE7B,IAAMC,EAQL,MAAM,IAAI,EACT,4CACAnV,KACA,CAAEoV,OAAQpV,KAAMkV,eAIlBlV,KAAKqP,GAAI6F,GAAY,CAAEG,EAAKnZ,KAC3BmZ,EAAIlE,OAASgE,EAAetE,MAAO7Q,KAAM9D,EAAM,IAGhD8D,KAAMkV,GAAe,YAAahZ,GACjC,OAAO8D,KAAKwQ,KAAM0E,EAAYhZ,EAC/B,EAEA8D,KAAMkV,GAAcjC,GAAsBkC,EAEpCnV,KAAMgT,KACXhT,KAAMgT,GAAqB,IAG5BhT,KAAMgT,GAAoBpT,KAAMsV,EACjC,CAOgB,aAAAvF,CAEfC,EACAN,EACAC,GAGA,IAAMK,GAAW5P,KAAMgT,GAAqB,CAC3C,UAAYkC,KAAclV,KAAMgT,GAC/BhT,KAAMkV,GAAelV,KAAMkV,GAAcjC,UAGnCjT,KAAMgT,EACd,CAEAxF,MAAMmC,cAAeC,EAASN,EAAOC,EACtC,EAYF,CA0CA,SAAS8D,EAAgBiC,GAEnBA,EAAYzC,KAKjB7b,OAAOuB,eAAgB+c,EAAYzC,EAA4B,CAC9Dra,MAAO,IAAI+Y,MA8CZva,OAAOuB,eAAgB+c,EAAYxC,EAAwB,CAC1Dta,MAAO,IAAI+Y,MA6BZva,OAAOuB,eAAgB+c,EAAYvC,EAAuB,CACzDva,MAAO,IAAI+Y,MAEb,CAOA,SAAS2C,KAAoChY,GAC5C,MAAMqZ,EA2HP,YAA6BrZ,GAE5B,IAAMA,EAAK5D,OAMV,MAAM,IAAI,EAAe,iCAAkC,MAG5D,MAAMkd,EAAkE,CAAEjW,GAAI,IAC9E,IAAIkW,EAEmC,mBAA3BvZ,EAAMA,EAAK5D,OAAS,KAC/Bkd,EAAOjG,SAAWrT,EAAK8C,OAcxB,OAXA9C,EAAK0B,SAASjC,IACb,GAAiB,iBAALA,EACX8Z,EAAenC,WAAW1T,KAAMjE,OAC1B,IAAiB,iBAALA,EAIlB,MAAM,IAAI,EAAe,iCAAkC,MAH3D8Z,EAAiB,CAAEH,WAAY3Z,EAAG2X,WAAY,IAC9CkC,EAAOjW,GAAGK,KAAM6V,EAGjB,KAGMD,CACR,CAzJoBE,IAAoBxZ,GACjCyZ,EAAehV,MAAMrB,KAAMU,KAAKwU,UAAUvd,QAC1C2e,EAAmBD,EAAard,OAGtC,IAAMid,EAAWhG,UAAYgG,EAAWhW,GAAGjH,OAAS,EAMnD,MAAM,IAAI,EAAe,iCAAkC0H,MAI5D,GAAK4V,EAAmB,GAAKL,EAAWhG,SAMvC,MAAM,IAAI,EACT,oCACAvP,MAsQH,IAAgCsV,EAlQ/BC,EAAWhW,GAAG3B,SAAS2B,IAEtB,GAAKA,EAAG+T,WAAWhb,QAAUiH,EAAG+T,WAAWhb,SAAWsd,EAMrD,MAAM,IAAI,EAAe,uCAAwC5V,MAK5DT,EAAG+T,WAAWhb,SACnBiH,EAAG+T,WAAatT,KAAKsU,gBACtB,IAGDtU,KAAKuU,IAAMgB,EAAWhW,GAGjBgW,EAAWhG,WACfvP,KAAKwU,UAAUpM,IAAKuN,EAAc,IAAOpG,SAAWgG,EAAWhG,UA4OjC+F,EAzORtV,KAAKqU,YAAarU,KAAKuU,IA0OnC3W,SAAS2B,IACnB,MAAMoV,EAAmBW,EAAYxC,GACrC,IAAIkB,EAIEW,EAAiBvM,IAAK7I,EAAG+V,aAC9BA,EAAW9F,SAAiCjQ,EAAG+V,WAAY,UAAU,CAAED,EAAKtB,KAC3EC,EAAWW,EAAiBvM,IAAK7I,EAAG+V,YAAevB,GAI9CC,GACJA,EAASpW,SAASqW,IACjB4B,EAA+BP,EAAYrB,EAAQb,SAAU,GAE/D,GAEF,IAzEF,SAA4B0C,GAC3B,IAAIjB,EAEJiB,EAAMtB,UAAU5W,SAAS,CAAEqW,EAASF,KAInC+B,EAAMvB,IAAI3W,SAAS2B,IAClBsV,EAAatV,EAAG+T,WAAYW,EAAQ1E,SAAW,EAAIuG,EAAMxB,gBAAgBtL,QAAS+K,IAElFE,EAAQ1U,GAAGK,KAAM,CAAEL,EAAG+V,WAAYT,IA/ErC,SACCS,EACArB,EACAW,EACAmB,GAEA,MAAMpB,EAAmBW,EAAYxC,GAC/BkD,EAAuBrB,EAAiBvM,IAAKwM,GAC7CZ,EAAWgC,GAAwB,CAAC,EAEpChC,EAAU+B,KACf/B,EAAU+B,GAAmB,IAAInC,KAIlCI,EAAU+B,GAAiBlI,IAAKoG,GAE1B+B,GACLrB,EAAiBlD,IAAKmD,EAAcZ,EAEtC,CA4DGiC,CAAwBH,EAAMzB,YAAaJ,EAAS1U,EAAG+V,WAAYT,EAAY,GAC7E,GAEL,CA9LCqB,CAAmBlW,MAGnBA,KAAKsU,gBAAgB1W,SAASmW,IAC7B8B,EAA+B7V,KAAKqU,YAAaN,EAAc,GAEjE,CAKA,SAASK,EAAqC+B,EAAgCC,EAAmB7G,GAChG,GAAKvP,KAAKwU,UAAUX,KAAO,EAM1B,MAAM,IAAI,EAAe,0CAA2C7T,MAGrEA,KAAKT,MAYN,SAA4B4W,EAAgCC,GAC3D,MAAMC,EAA8BF,EAAYnZ,KAAKsY,GAAc,CAAEA,EAAYc,KAGjF,OAAOzV,MAAMkB,UAAU1B,OAAO0Q,MAAO,GAAIwF,EAC1C,CAfKC,CAAmBH,EAAaC,GAEnC7G,EAEF,CAgBA,SAASoE,EAAejT,GACvB,OAAOA,EAAI6V,OAAO5a,GAAiB,iBAALA,GAC/B,CA0JA,SAASka,EAA+BP,EAAgCvB,GACvE,MACME,EADkBqB,EAAYvC,GACJ3K,IAAK2L,GACrC,IAAIyC,EAOCvC,EAAQ1E,SACZiH,EAAgBvC,EAAQ1E,SAASsB,MAAOyE,EAAYrB,EAAQ1U,GAAGvC,KAAKuC,GAAQA,EAAI,GAAcA,EAAI,QAElGiX,EAAgBvC,EAAQ1U,GAAI,GAC5BiX,EAAkBA,EAAe,GAAcA,EAAe,KAG1Dxf,OAAO6K,UAAU2G,eAAenH,KAAMiU,EAAYvB,GACpDuB,EAAqBvB,GAAiByC,EAExClB,EAAW7D,IAAKsC,EAAqByC,EAEvC,CC/sBe,SAAS7R,EAAO8R,GAC9B,IAAI9R,EAAQ,EAEZ,UAAY2H,KAAKmK,EAChB9R,IAGD,OAAOA,CACR,CCAe,SAAS+R,EAAe/a,EAA2BhD,GACjE,MAAMgJ,EAAS9I,KAAKD,IAAK+C,EAAErD,OAAQK,EAAEL,QAErC,QAAUkC,EAAI,EAAGA,EAAImH,EAAQnH,IAC5B,GAAKmB,EAAGnB,IAAO7B,EAAG6B,GAEjB,OAAOA,EAKT,OAAKmB,EAAErD,QAAUK,EAAEL,OAEX,OACIqD,EAAErD,OAASK,EAAEL,OAEjB,SAGA,WAET,CCjCe,SAASqe,EAAYne,GACnC,SAAWA,IAASA,EAAOsI,OAAOC,UACnC,CHuTE,CACD,MAAO,OAAQ,SAAU,WACzB,KAAM,OAAQ,MAAO,WACrB,gBAAiB,OAAQ,WAAY,iBACrC,oBAAqB,wBAClBnD,SAAS7G,IACVoc,EAA0Bpc,GAAUmc,EAAuBrR,UAAoB9K,EAAK,II3UvF,QAFkC,iBAAV6f,QAAsBA,QAAUA,OAAO5f,SAAWA,QAAU4f,OCEpF,IAAIC,GAA0B,iBAARjgB,MAAoBA,MAAQA,KAAKI,SAAWA,QAAUJ,KAK5E,SAFW,GAAcigB,IAAYvO,SAAS,cAATA,GCDrC,SAFa,GAAKxH,OCAlB,IAAIgW,GAAc9f,OAAO6K,UAGrB,GAAiBiV,GAAYtO,eAO7BuO,GAAuBD,GAAYpa,SAGnCsa,GAAiB,GAAS,GAAOC,iBAAcpP,EA6BnD,SApBA,SAAmBrP,GACjB,IAAI0e,EAAQ,GAAe7V,KAAK7I,EAAOwe,IACnCG,EAAM3e,EAAMwe,IAEhB,IACExe,EAAMwe,SAAkBnP,EACxB,IAAIuP,GAAW,CACjB,CAAE,MAAOlT,GAAI,CAEb,IAAI/F,EAAS4Y,GAAqB1V,KAAK7I,GAQvC,OAPI4e,IACEF,EACF1e,EAAMwe,IAAkBG,SAEjB3e,EAAMwe,KAGV7Y,CACT,EC1CA,IAOI,GAPcnH,OAAO6K,UAOcnF,SAavC,SAJA,SAAwBlE,GACtB,OAAO,GAAqB6I,KAAK7I,EACnC,ECdA,IAII,GAAiB,GAAS,GAAOye,iBAAcpP,EAkBnD,SATA,SAAoBrP,GAClB,OAAa,MAATA,OACeqP,IAAVrP,EAdQ,qBADL,gBAiBJ,IAAkB,MAAkBxB,OAAOwB,GAC/C,GAAUA,GACV,GAAeA,EACrB,ECAA,SAFcmI,MAAMC,QCKpB,SAJA,SAAsBpI,GACpB,OAAgB,MAATA,GAAiC,iBAATA,CACjC,ECGA,SALA,SAAkBA,GAChB,MAAuB,iBAATA,IACV,GAAQA,IAAU,GAAaA,IArBrB,mBAqB+B,GAAWA,EAC1D,EC8De,SAAS,GACvB6e,EACArV,EACAuD,EAA2D,CAAC,EAC5D+R,EAA6B,IAE7B,MAAMC,EAAYhS,GAAcA,EAAWiS,MACrCC,EAAUF,EAAYF,EAAIK,gBAAiBH,EAAWvV,GAASqV,EAAI/R,cAAetD,GAExF,UAAYjL,KAAOwO,EAClBkS,EAAQhS,aAAc1O,EAAKwO,EAAYxO,KAGnC,GAAUugB,IAAeX,EAAYW,KACzCA,EAAW,CAAEA,IAGd,QAAUK,KAASL,EACb,GAAUK,KACdA,EAAQN,EAAIhR,eAAgBsR,IAG7BF,EAAQ9R,YAAagS,GAGtB,OAAOF,CACR,CCrGA,SANA,SAAiBG,EAAMC,GACrB,OAAO,SAASC,GACd,OAAOF,EAAKC,EAAUC,GACxB,CACF,ECPA,SAFmB,GAAQ9gB,OAAO+gB,eAAgB/gB,QCElD,IAGIghB,GAAY1P,SAASzG,UACrB,GAAc7K,OAAO6K,UAGrBoW,GAAeD,GAAUtb,SAGzB,GAAiB,GAAY8L,eAG7B0P,GAAmBD,GAAa5W,KAAKrK,QA2CzC,SAbA,SAAuBwB,GACrB,IAAK,GAAaA,IA5CJ,mBA4Cc,GAAWA,GACrC,OAAO,EAET,IAAI2f,EAAQ,GAAa3f,GACzB,GAAc,OAAV2f,EACF,OAAO,EAET,IAAIC,EAAO,GAAe/W,KAAK8W,EAAO,gBAAkBA,EAAMpW,YAC9D,MAAsB,mBAARqW,GAAsBA,aAAgBA,GAClDH,GAAa5W,KAAK+W,IAASF,EAC/B,EC/CA,SALA,WACElY,KAAKqY,SAAW,GAChBrY,KAAK6T,KAAO,CACd,EC0BA,SAJA,SAAYrb,EAAO8f,GACjB,OAAO9f,IAAU8f,GAAU9f,GAAUA,GAAS8f,GAAUA,CAC1D,ECdA,SAVA,SAAsBC,EAAOxhB,GAE3B,IADA,IAAIuB,EAASigB,EAAMjgB,OACZA,KACL,GAAI,GAAGigB,EAAMjgB,GAAQ,GAAIvB,GACvB,OAAOuB,EAGX,OAAQ,CACV,ECfA,IAGIkP,GAHa7G,MAAMkB,UAGC2F,OA4BxB,SAjBA,SAAyBzQ,GACvB,IAAI+L,EAAO9C,KAAKqY,SACZzT,EAAQ,GAAa9B,EAAM/L,GAE/B,QAAI6N,EAAQ,KAIRA,GADY9B,EAAKxK,OAAS,EAE5BwK,EAAK9D,MAELwI,GAAOnG,KAAKyB,EAAM8B,EAAO,KAEzB5E,KAAK6T,MACA,EACT,ECdA,SAPA,SAAsB9c,GACpB,IAAI+L,EAAO9C,KAAKqY,SACZzT,EAAQ,GAAa9B,EAAM/L,GAE/B,OAAO6N,EAAQ,OAAIiD,EAAY/E,EAAK8B,GAAO,EAC7C,ECDA,SAJA,SAAsB7N,GACpB,OAAO,GAAaiJ,KAAKqY,SAAUthB,IAAQ,CAC7C,ECYA,SAbA,SAAsBA,EAAKyB,GACzB,IAAIsK,EAAO9C,KAAKqY,SACZzT,EAAQ,GAAa9B,EAAM/L,GAQ/B,OANI6N,EAAQ,KACR5E,KAAK6T,KACP/Q,EAAKlD,KAAK,CAAC7I,EAAKyB,KAEhBsK,EAAK8B,GAAO,GAAKpM,EAEZwH,IACT,ECVA,SAASwY,GAAUC,GACjB,IAAI7T,GAAS,EACTtM,EAAoB,MAAXmgB,EAAkB,EAAIA,EAAQngB,OAG3C,IADA0H,KAAK4R,UACIhN,EAAQtM,GAAQ,CACvB,IAAIogB,EAAQD,EAAQ7T,GACpB5E,KAAKyR,IAAIiH,EAAM,GAAIA,EAAM,GAC3B,CACF,CAGAF,GAAU3W,UAAU+P,MAAQ,GAC5B4G,GAAU3W,UAAkB,OAAI,GAChC2W,GAAU3W,UAAUuG,IAAM,GAC1BoQ,GAAU3W,UAAU+L,IAAM,GAC1B4K,GAAU3W,UAAU4P,IAAM,GAE1B,YCjBA,SALA,WACEzR,KAAKqY,SAAW,IAAI,GACpBrY,KAAK6T,KAAO,CACd,ECKA,SARA,SAAqB9c,GACnB,IAAI+L,EAAO9C,KAAKqY,SACZla,EAAS2E,EAAa,OAAE/L,GAG5B,OADAiJ,KAAK6T,KAAO/Q,EAAK+Q,KACV1V,CACT,ECFA,SAJA,SAAkBpH,GAChB,OAAOiJ,KAAKqY,SAASjQ,IAAIrR,EAC3B,ECEA,SAJA,SAAkBA,GAChB,OAAOiJ,KAAKqY,SAASzK,IAAI7W,EAC3B,ECyBA,SAVA,SAAoByB,GAClB,IAAK,EAASA,GACZ,OAAO,EAIT,IAAI2e,EAAM,GAAW3e,GACrB,MA5BY,qBA4BL2e,GA3BI,8BA2BcA,GA7BZ,0BA6B6BA,GA1B7B,kBA0BgDA,CAC/D,EC7BA,SAFiB,GAAK,sBCAtB,IAAIwB,GAAc,WAChB,IAAIC,EAAM,SAASC,KAAK,IAAc,GAAW5hB,MAAQ,GAAWA,KAAK6hB,UAAY,IACrF,OAAOF,EAAO,iBAAmBA,EAAO,EAC1C,CAHiB,GAgBjB,SAJA,SAAkBhB,GAChB,QAASe,IAAeA,MAAcf,CACxC,EChBA,IAGI,GAHYtP,SAASzG,UAGInF,SAqB7B,SAZA,SAAkBkb,GAChB,GAAY,MAARA,EAAc,CAChB,IACE,OAAO,GAAavW,KAAKuW,EAC3B,CAAE,MAAO1T,GAAI,CACb,IACE,OAAQ0T,EAAO,EACjB,CAAE,MAAO1T,GAAI,CACf,CACA,MAAO,EACT,ECdA,IAGI6U,GAAe,8BAGf,GAAYzQ,SAASzG,UACrB,GAAc7K,OAAO6K,UAGrB,GAAe,GAAUnF,SAGzB,GAAiB,GAAY8L,eAG7BwQ,GAAarP,OAAO,IACtB,GAAatI,KAAK,IAAgB4X,QAjBjB,sBAiBuC,QACvDA,QAAQ,yDAA0D,SAAW,KAmBhF,SARA,SAAsBzgB,GACpB,SAAK,EAASA,IAAU,GAASA,MAGnB,GAAWA,GAASwgB,GAAaD,IAChC9W,KAAK,GAASzJ,GAC/B,EChCA,SAJA,SAAkB4c,EAAQre,GACxB,OAAiB,MAAVqe,OAAiBvN,EAAYuN,EAAOre,EAC7C,ECMA,SALA,SAAmBqe,EAAQre,GACzB,IAAIyB,EAAQ,GAAS4c,EAAQre,GAC7B,OAAO,GAAayB,GAASA,OAAQqP,CACvC,ECRA,SAFU,GAAU,GAAM,OCC1B,SAFmB,GAAU7Q,OAAQ,UCWrC,SALA,WACEgJ,KAAKqY,SAAW,GAAe,GAAa,MAAQ,CAAC,EACrDrY,KAAK6T,KAAO,CACd,ECIA,SANA,SAAoB9c,GAClB,IAAIoH,EAAS6B,KAAK4N,IAAI7W,WAAeiJ,KAAKqY,SAASthB,GAEnD,OADAiJ,KAAK6T,MAAQ1V,EAAS,EAAI,EACnBA,CACT,ECXA,IAMI,GAHcnH,OAAO6K,UAGQ2G,eAoBjC,SATA,SAAiBzR,GACf,IAAI+L,EAAO9C,KAAKqY,SAChB,GAAI,GAAc,CAChB,IAAIla,EAAS2E,EAAK/L,GAClB,MArBiB,8BAqBVoH,OAA4B0J,EAAY1J,CACjD,CACA,OAAO,GAAekD,KAAKyB,EAAM/L,GAAO+L,EAAK/L,QAAO8Q,CACtD,ECxBA,IAGI,GAHc7Q,OAAO6K,UAGQ2G,eAgBjC,SALA,SAAiBzR,GACf,IAAI+L,EAAO9C,KAAKqY,SAChB,OAAO,QAA8BxQ,IAAd/E,EAAK/L,GAAsB,GAAesK,KAAKyB,EAAM/L,EAC9E,ECEA,SAPA,SAAiBA,EAAKyB,GACpB,IAAIsK,EAAO9C,KAAKqY,SAGhB,OAFArY,KAAK6T,MAAQ7T,KAAK4N,IAAI7W,GAAO,EAAI,EACjC+L,EAAK/L,GAAQ,SAA0B8Q,IAAVrP,EAfV,4BAekDA,EAC9DwH,IACT,ECPA,SAASkZ,GAAKT,GACZ,IAAI7T,GAAS,EACTtM,EAAoB,MAAXmgB,EAAkB,EAAIA,EAAQngB,OAG3C,IADA0H,KAAK4R,UACIhN,EAAQtM,GAAQ,CACvB,IAAIogB,EAAQD,EAAQ7T,GACpB5E,KAAKyR,IAAIiH,EAAM,GAAIA,EAAM,GAC3B,CACF,CAGAQ,GAAKrX,UAAU+P,MAAQ,GACvBsH,GAAKrX,UAAkB,OAAI,GAC3BqX,GAAKrX,UAAUuG,IAAM,GACrB8Q,GAAKrX,UAAU+L,IAAM,GACrBsL,GAAKrX,UAAU4P,IAAM,GAErB,YCXA,SATA,WACEzR,KAAK6T,KAAO,EACZ7T,KAAKqY,SAAW,CACd,KAAQ,IAAI,GACZ,IAAO,IAAK,IAAO,IACnB,OAAU,IAAI,GAElB,ECJA,SAPA,SAAmB7f,GACjB,IAAIyS,SAAczS,EAClB,MAAgB,UAARyS,GAA4B,UAARA,GAA4B,UAARA,GAA4B,WAARA,EACrD,cAAVzS,EACU,OAAVA,CACP,ECKA,SAPA,SAAoBwE,EAAKjG,GACvB,IAAI+L,EAAO9F,EAAIqb,SACf,OAAO,GAAUthB,GACb+L,EAAmB,iBAAP/L,EAAkB,SAAW,QACzC+L,EAAK9F,GACX,ECEA,SANA,SAAwBjG,GACtB,IAAIoH,EAAS,GAAW6B,KAAMjJ,GAAa,OAAEA,GAE7C,OADAiJ,KAAK6T,MAAQ1V,EAAS,EAAI,EACnBA,CACT,ECAA,SAJA,SAAqBpH,GACnB,OAAO,GAAWiJ,KAAMjJ,GAAKqR,IAAIrR,EACnC,ECEA,SAJA,SAAqBA,GACnB,OAAO,GAAWiJ,KAAMjJ,GAAK6W,IAAI7W,EACnC,ECQA,SATA,SAAqBA,EAAKyB,GACxB,IAAIsK,EAAO,GAAW9C,KAAMjJ,GACxB8c,EAAO/Q,EAAK+Q,KAIhB,OAFA/Q,EAAK2O,IAAI1a,EAAKyB,GACdwH,KAAK6T,MAAQ/Q,EAAK+Q,MAAQA,EAAO,EAAI,EAC9B7T,IACT,ECNA,SAASmZ,GAASV,GAChB,IAAI7T,GAAS,EACTtM,EAAoB,MAAXmgB,EAAkB,EAAIA,EAAQngB,OAG3C,IADA0H,KAAK4R,UACIhN,EAAQtM,GAAQ,CACvB,IAAIogB,EAAQD,EAAQ7T,GACpB5E,KAAKyR,IAAIiH,EAAM,GAAIA,EAAM,GAC3B,CACF,CAGAS,GAAStX,UAAU+P,MAAQ,GAC3BuH,GAAStX,UAAkB,OAAI,GAC/BsX,GAAStX,UAAUuG,IAAM,GACzB+Q,GAAStX,UAAU+L,IAAM,GACzBuL,GAAStX,UAAU4P,IAAM,GAEzB,YCEA,SAhBA,SAAkB1a,EAAKyB,GACrB,IAAIsK,EAAO9C,KAAKqY,SAChB,GAAIvV,aAAgB,GAAW,CAC7B,IAAIsW,EAAQtW,EAAKuV,SACjB,IAAK,IAAQe,EAAM9gB,OAAS+gB,IAG1B,OAFAD,EAAMxZ,KAAK,CAAC7I,EAAKyB,IACjBwH,KAAK6T,OAAS/Q,EAAK+Q,KACZ7T,KAET8C,EAAO9C,KAAKqY,SAAW,IAAI,GAASe,EACtC,CAGA,OAFAtW,EAAK2O,IAAI1a,EAAKyB,GACdwH,KAAK6T,KAAO/Q,EAAK+Q,KACV7T,IACT,ECjBA,SAASsZ,GAAMb,GACb,IAAI3V,EAAO9C,KAAKqY,SAAW,IAAI,GAAUI,GACzCzY,KAAK6T,KAAO/Q,EAAK+Q,IACnB,CAGAyF,GAAMzX,UAAU+P,MAAQ,GACxB0H,GAAMzX,UAAkB,OAAI,GAC5ByX,GAAMzX,UAAUuG,IAAM,GACtBkR,GAAMzX,UAAU+L,IAAM,GACtB0L,GAAMzX,UAAU4P,IAAM,GAEtB,YCLA,SAZA,SAAmB8G,EAAOgB,GAIxB,IAHA,IAAI3U,GAAS,EACTtM,EAAkB,MAATigB,EAAgB,EAAIA,EAAMjgB,SAE9BsM,EAAQtM,IAC8B,IAAzCihB,EAAShB,EAAM3T,GAAQA,EAAO2T,KAIpC,OAAOA,CACT,ECTA,SARsB,WACpB,IACE,IAAIX,EAAO,GAAU5gB,OAAQ,kBAE7B,OADA4gB,EAAK,CAAC,EAAG,GAAI,CAAC,GACPA,CACT,CAAE,MAAO1T,GAAI,CACf,CANqB,GCsBrB,SAbA,SAAyBkR,EAAQre,EAAKyB,GACzB,aAAPzB,GAAsB,GACxB,GAAeqe,EAAQre,EAAK,CAC1B,cAAgB,EAChB,YAAc,EACd,MAASyB,EACT,UAAY,IAGd4c,EAAOre,GAAOyB,CAElB,EClBA,IAGI,GAHcxB,OAAO6K,UAGQ2G,eAoBjC,SARA,SAAqB4M,EAAQre,EAAKyB,GAChC,IAAIghB,EAAWpE,EAAOre,GAChB,GAAesK,KAAK+T,EAAQre,IAAQ,GAAGyiB,EAAUhhB,UACxCqP,IAAVrP,GAAyBzB,KAAOqe,IACnC,GAAgBA,EAAQre,EAAKyB,EAEjC,ECcA,SA1BA,SAAoB0K,EAAQuW,EAAOrE,EAAQsE,GACzC,IAAIC,GAASvE,EACbA,IAAWA,EAAS,CAAC,GAKrB,IAHA,IAAIxQ,GAAS,EACTtM,EAASmhB,EAAMnhB,SAEVsM,EAAQtM,GAAQ,CACvB,IAAIvB,EAAM0iB,EAAM7U,GAEZ6O,EAAWiG,EACXA,EAAWtE,EAAOre,GAAMmM,EAAOnM,GAAMA,EAAKqe,EAAQlS,QAClD2E,OAEaA,IAAb4L,IACFA,EAAWvQ,EAAOnM,IAEhB4iB,EACF,GAAgBvE,EAAQre,EAAK0c,GAE7B,GAAY2B,EAAQre,EAAK0c,EAE7B,CACA,OAAO2B,CACT,EClBA,SAVA,SAAmB9Z,EAAGie,GAIpB,IAHA,IAAI3U,GAAS,EACTzG,EAASwC,MAAMrF,KAEVsJ,EAAQtJ,GACf6C,EAAOyG,GAAS2U,EAAS3U,GAE3B,OAAOzG,CACT,ECAA,SAJA,SAAyB3F,GACvB,OAAO,GAAaA,IAVR,sBAUkB,GAAWA,EAC3C,ECXA,IAAI,GAAcxB,OAAO6K,UAGrB,GAAiB,GAAY2G,eAG7BoR,GAAuB,GAAYA,qBAyBvC,SALkB,GAAgB,WAAa,OAAOC,SAAW,CAA/B,IAAsC,GAAkB,SAASrhB,GACjG,OAAO,GAAaA,IAAU,GAAe6I,KAAK7I,EAAO,YACtDohB,GAAqBvY,KAAK7I,EAAO,SACtC,EChBA,SAJA,WACE,OAAO,CACT,ECXA,IAAIshB,GAAgC,iBAAXtjB,SAAuBA,UAAYA,QAAQujB,UAAYvjB,QAG5EwjB,GAAaF,IAAgC,iBAAVrjB,QAAsBA,SAAWA,OAAOsjB,UAAYtjB,OAMvFwjB,GAHgBD,IAAcA,GAAWxjB,UAAYsjB,GAG5B,GAAKG,YAASpS,EAwB3C,UArBqBoS,GAASA,GAAOC,cAAWrS,IAmBf,GClCjC,IAGIsS,GAAW,mBAoBf,SAVA,SAAiB3hB,EAAOF,GACtB,IAAI2S,SAAczS,EAGlB,SAFAF,EAAmB,MAAVA,EAfY,iBAewBA,KAGlC,UAAR2S,GACU,UAARA,GAAoBkP,GAASlY,KAAKzJ,KAChCA,GAAS,GAAKA,EAAQ,GAAK,GAAKA,EAAQF,CACjD,ECYA,SALA,SAAkBE,GAChB,MAAuB,iBAATA,GACZA,GAAS,GAAKA,EAAQ,GAAK,GAAKA,GA9Bb,gBA+BvB,EC3BA,IA2BI4hB,GAAiB,CAAC,EACtBA,GAZiB,yBAYYA,GAXZ,yBAYjBA,GAXc,sBAWYA,GAVX,uBAWfA,GAVe,uBAUYA,GATZ,uBAUfA,GATsB,8BASYA,GARlB,wBAShBA,GARgB,yBAQY,EAC5BA,GAjCc,sBAiCYA,GAhCX,kBAiCfA,GApBqB,wBAoBYA,GAhCnB,oBAiCdA,GApBkB,qBAoBYA,GAhChB,iBAiCdA,GAhCe,kBAgCYA,GA/Bb,qBAgCdA,GA/Ba,gBA+BYA,GA9BT,mBA+BhBA,GA9BgB,mBA8BYA,GA7BZ,mBA8BhBA,GA7Ba,gBA6BYA,GA5BT,mBA6BhBA,GA5BiB,qBA4BY,EAc7B,SALA,SAA0B5hB,GACxB,OAAO,GAAaA,IAClB,GAASA,EAAMF,WAAa8hB,GAAe,GAAW5hB,GAC1D,EC5CA,SANA,SAAmBof,GACjB,OAAO,SAASpf,GACd,OAAOof,EAAKpf,EACd,CACF,ECRA,IAAI,GAAgC,iBAAXhC,SAAuBA,UAAYA,QAAQujB,UAAYvjB,QAG5E,GAAa,IAAgC,iBAAVC,QAAsBA,SAAWA,OAAOsjB,UAAYtjB,OAMvF4jB,GAHgB,IAAc,GAAW7jB,UAAY,IAGtB,EAAW8jB,QAiB9C,SAdgB,WACd,IAEE,IAAIC,EAAQ,IAAc,GAAWC,SAAW,GAAWA,QAAQ,QAAQD,MAE3E,OAAIA,GAKGF,IAAeA,GAAYpG,SAAWoG,GAAYpG,QAAQ,OACnE,CAAE,MAAO/P,GAAI,CACf,CAZe,GCVf,IAAIuW,GAAmB,IAAY,GAASC,aAqB5C,SAFmBD,GAAmB,GAAUA,IAAoB,GChBpE,IAGI,GAHczjB,OAAO6K,UAGQ2G,eAqCjC,SA3BA,SAAuBhQ,EAAOmiB,GAC5B,IAAIC,EAAQ,GAAQpiB,GAChBqiB,GAASD,GAAS,GAAYpiB,GAC9BsiB,GAAUF,IAAUC,GAAS,GAASriB,GACtCuiB,GAAUH,IAAUC,IAAUC,GAAU,GAAatiB,GACrDwiB,EAAcJ,GAASC,GAASC,GAAUC,EAC1C5c,EAAS6c,EAAc,GAAUxiB,EAAMF,OAAQ2iB,QAAU,GACzD3iB,EAAS6F,EAAO7F,OAEpB,IAAK,IAAIvB,KAAOyB,GACTmiB,IAAa,GAAetZ,KAAK7I,EAAOzB,IACvCikB,IAEQ,UAAPjkB,GAEC+jB,IAAkB,UAAP/jB,GAA0B,UAAPA,IAE9BgkB,IAAkB,UAAPhkB,GAA0B,cAAPA,GAA8B,cAAPA,IAEtD,GAAQA,EAAKuB,KAElB6F,EAAOyB,KAAK7I,GAGhB,OAAOoH,CACT,EC7CA,IAAI,GAAcnH,OAAO6K,UAgBzB,SAPA,SAAqBrJ,GACnB,IAAI4f,EAAO5f,GAASA,EAAMuJ,YAG1B,OAAOvJ,KAFqB,mBAAR4f,GAAsBA,EAAKvW,WAAc,GAG/D,ECVA,SAFiB,GAAQ7K,OAAOC,KAAMD,QCCtC,IAGI,GAHcA,OAAO6K,UAGQ2G,eAsBjC,SAbA,SAAkB4M,GAChB,IAAK,GAAYA,GACf,OAAO,GAAWA,GAEpB,IAAIjX,EAAS,GACb,IAAK,IAAIpH,KAAOC,OAAOoe,GACjB,GAAe/T,KAAK+T,EAAQre,IAAe,eAAPA,GACtCoH,EAAOyB,KAAK7I,GAGhB,OAAOoH,CACT,ECKA,SAJA,SAAqB3F,GACnB,OAAgB,MAATA,GAAiB,GAASA,EAAMF,UAAY,GAAWE,EAChE,ECMA,SAJA,SAAc4c,GACZ,OAAO,GAAYA,GAAU,GAAcA,GAAU,GAASA,EAChE,EClBA,SAJA,SAAoBA,EAAQlS,GAC1B,OAAOkS,GAAU,GAAWlS,EAAQ,GAAKA,GAASkS,EACpD,ECKA,SAVA,SAAsBA,GACpB,IAAIjX,EAAS,GACb,GAAc,MAAViX,EACF,IAAK,IAAIre,KAAOC,OAAOoe,GACrBjX,EAAOyB,KAAK7I,GAGhB,OAAOoH,CACT,ECZA,IAGI,GAHcnH,OAAO6K,UAGQ2G,eAwBjC,SAfA,SAAoB4M,GAClB,IAAK,EAASA,GACZ,OAAO,GAAaA,GAEtB,IAAI8F,EAAU,GAAY9F,GACtBjX,EAAS,GAEb,IAAK,IAAIpH,KAAOqe,GACD,eAAPre,IAAyBmkB,GAAY,GAAe7Z,KAAK+T,EAAQre,KACrEoH,EAAOyB,KAAK7I,GAGhB,OAAOoH,CACT,ECCA,SAJA,SAAgBiX,GACd,OAAO,GAAYA,GAAU,GAAcA,GAAQ,GAAQ,GAAWA,EACxE,ECbA,SAJA,SAAsBA,EAAQlS,GAC5B,OAAOkS,GAAU,GAAWlS,EAAQ,GAAOA,GAASkS,EACtD,ECXA,IAAI,GAAgC,iBAAX5e,SAAuBA,UAAYA,QAAQujB,UAAYvjB,QAG5E,GAAa,IAAgC,iBAAVC,QAAsBA,SAAWA,OAAOsjB,UAAYtjB,OAMvF,GAHgB,IAAc,GAAWD,UAAY,GAG5B,GAAKyjB,YAASpS,EACvCsT,GAAc,GAAS,GAAOA,iBAActT,EAqBhD,SAXA,SAAqBuT,EAAQC,GAC3B,GAAIA,EACF,OAAOD,EAAOtZ,QAEhB,IAAIxJ,EAAS8iB,EAAO9iB,OAChB6F,EAASgd,GAAcA,GAAY7iB,GAAU,IAAI8iB,EAAOrZ,YAAYzJ,GAGxE,OADA8iB,EAAOE,KAAKnd,GACLA,CACT,ECbA,SAXA,SAAmB+E,EAAQqV,GACzB,IAAI3T,GAAS,EACTtM,EAAS4K,EAAO5K,OAGpB,IADAigB,IAAUA,EAAQ5X,MAAMrI,MACfsM,EAAQtM,GACfigB,EAAM3T,GAAS1B,EAAO0B,GAExB,OAAO2T,CACT,ECOA,SAfA,SAAqBA,EAAOgD,GAM1B,IALA,IAAI3W,GAAS,EACTtM,EAAkB,MAATigB,EAAgB,EAAIA,EAAMjgB,OACnCkjB,EAAW,EACXrd,EAAS,KAEJyG,EAAQtM,GAAQ,CACvB,IAAIE,EAAQ+f,EAAM3T,GACd2W,EAAU/iB,EAAOoM,EAAO2T,KAC1Bpa,EAAOqd,KAAchjB,EAEzB,CACA,OAAO2F,CACT,ECAA,SAJA,WACE,MAAO,EACT,EChBA,IAGI,GAHcnH,OAAO6K,UAGc+X,qBAGnC6B,GAAmBzkB,OAAO0kB,sBAmB9B,SAVkBD,GAA+B,SAASrG,GACxD,OAAc,MAAVA,EACK,IAETA,EAASpe,OAAOoe,GACT,GAAYqG,GAAiBrG,IAAS,SAASuG,GACpD,OAAO,GAAqBta,KAAK+T,EAAQuG,EAC3C,IACF,EARqC,GCJrC,SAJA,SAAqBzY,EAAQkS,GAC3B,OAAO,GAAWlS,EAAQ,GAAWA,GAASkS,EAChD,ECMA,SAXA,SAAmBmD,EAAOrN,GAKxB,IAJA,IAAItG,GAAS,EACTtM,EAAS4S,EAAO5S,OAChBsjB,EAASrD,EAAMjgB,SAEVsM,EAAQtM,GACfigB,EAAMqD,EAAShX,GAASsG,EAAOtG,GAEjC,OAAO2T,CACT,ECOA,SAlBuBvhB,OAAO0kB,sBASqB,SAAStG,GAE1D,IADA,IAAIjX,EAAS,GACNiX,GACL,GAAUjX,EAAQ,GAAWiX,IAC7BA,EAAS,GAAaA,GAExB,OAAOjX,CACT,EAPuC,GCAvC,SAJA,SAAuB+E,EAAQkS,GAC7B,OAAO,GAAWlS,EAAQ,GAAaA,GAASkS,EAClD,ECMA,SALA,SAAwBA,EAAQyG,EAAUC,GACxC,IAAI3d,EAAS0d,EAASzG,GACtB,OAAO,GAAQA,GAAUjX,EAAS,GAAUA,EAAQ2d,EAAY1G,GAClE,ECFA,SAJA,SAAoBA,GAClB,OAAO,GAAeA,EAAQ,GAAM,GACtC,ECGA,SAJA,SAAsBA,GACpB,OAAO,GAAeA,EAAQ,GAAQ,GACxC,ECRA,SAFe,GAAU,GAAM,YCE/B,SAFc,GAAU,GAAM,WCE9B,SAFU,GAAU,GAAM,OCE1B,SAFc,GAAU,GAAM,WCK9B,IAAI,GAAS,eAET2G,GAAa,mBACb,GAAS,eACT,GAAa,mBAEb,GAAc,oBAGdC,GAAqB,GAAS,IAC9BC,GAAgB,GAAS,IACzBC,GAAoB,GAAS,IAC7BC,GAAgB,GAAS,IACzBC,GAAoB,GAAS,IAS7BC,GAAS,IAGR,IAAYA,GAAO,IAAI,GAAS,IAAIC,YAAY,MAAQ,IACxD,IAAOD,GAAO,IAAI,KAAQ,IAC1B,IAAWA,GAAO,GAAQE,YAAcR,IACxC,IAAOM,GAAO,IAAI,KAAQ,IAC1B,IAAWA,GAAO,IAAI,KAAY,MACrCA,GAAS,SAAS7jB,GAChB,IAAI2F,EAAS,GAAW3F,GACpB4f,EA/BQ,mBA+BDja,EAAsB3F,EAAMuJ,iBAAc8F,EACjD2U,EAAapE,EAAO,GAASA,GAAQ,GAEzC,GAAIoE,EACF,OAAQA,GACN,KAAKR,GAAoB,OAAO,GAChC,KAAKC,GAAe,OAAO,GAC3B,KAAKC,GAAmB,OAAOH,GAC/B,KAAKI,GAAe,OAAO,GAC3B,KAAKC,GAAmB,OAAO,GAGnC,OAAOje,CACT,GAGF,YCxDA,IAGI,GAHcnH,OAAO6K,UAGQ2G,eAqBjC,SAZA,SAAwB+P,GACtB,IAAIjgB,EAASigB,EAAMjgB,OACf6F,EAAS,IAAIoa,EAAMxW,YAAYzJ,GAOnC,OAJIA,GAA6B,iBAAZigB,EAAM,IAAkB,GAAelX,KAAKkX,EAAO,WACtEpa,EAAOyG,MAAQ2T,EAAM3T,MACrBzG,EAAOse,MAAQlE,EAAMkE,OAEhBte,CACT,EClBA,SAFiB,GAAKue,WCYtB,SANA,SAA0BC,GACxB,IAAIxe,EAAS,IAAIwe,EAAY5a,YAAY4a,EAAYC,YAErD,OADA,IAAI,GAAWze,GAAQsT,IAAI,IAAI,GAAWkL,IACnCxe,CACT,ECEA,SALA,SAAuB0e,EAAUxB,GAC/B,IAAID,EAASC,EAAS,GAAiBwB,EAASzB,QAAUyB,EAASzB,OACnE,OAAO,IAAIyB,EAAS9a,YAAYqZ,EAAQyB,EAASC,WAAYD,EAASD,WACxE,ECZA,IAAIG,GAAU,OAed,SANA,SAAqBC,GACnB,IAAI7e,EAAS,IAAI6e,EAAOjb,YAAYib,EAAO9Z,OAAQ6Z,GAAQlE,KAAKmE,IAEhE,OADA7e,EAAOwM,UAAYqS,EAAOrS,UACnBxM,CACT,ECXA,IAAI8e,GAAc,GAAS,GAAOpb,eAAYgG,EAC1CqV,GAAgBD,GAAcA,GAAYE,aAAUtV,EAaxD,SAJA,SAAqB8T,GACnB,OAAOuB,GAAgBlmB,OAAOkmB,GAAc7b,KAAKsa,IAAW,CAAC,CAC/D,ECAA,SALA,SAAyByB,EAAY/B,GACnC,IAAID,EAASC,EAAS,GAAiB+B,EAAWhC,QAAUgC,EAAWhC,OACvE,OAAO,IAAIgC,EAAWrb,YAAYqZ,EAAQgC,EAAWN,WAAYM,EAAW9kB,OAC9E,EC+DA,SApCA,SAAwB8c,EAAQ+B,EAAKkE,GACnC,IAAIjD,EAAOhD,EAAOrT,YAClB,OAAQoV,GACN,IA3BiB,uBA4Bf,OAAO,GAAiB/B,GAE1B,IAvCU,mBAwCV,IAvCU,gBAwCR,OAAO,IAAIgD,GAAMhD,GAEnB,IAjCc,oBAkCZ,OAAO,GAAcA,EAAQiG,GAE/B,IAnCa,wBAmCI,IAlCJ,wBAmCb,IAlCU,qBAkCI,IAjCH,sBAiCkB,IAhClB,sBAiCX,IAhCW,sBAgCI,IA/BG,6BA+BmB,IA9BzB,uBA8ByC,IA7BzC,uBA8BV,OAAO,GAAgBjG,EAAQiG,GAEjC,IAjDS,eA2DT,IAxDS,eAyDP,OAAO,IAAIjD,EARb,IAnDY,kBAoDZ,IAjDY,kBAkDV,OAAO,IAAIA,EAAKhD,GAElB,IAtDY,kBAuDV,OAAO,GAAYA,GAKrB,IAzDY,kBA0DV,OAAO,GAAYA,GAEzB,ECvEA,IAAIiI,GAAermB,OAAOsmB,OA0B1B,SAhBkB,WAChB,SAASlI,IAAU,CACnB,OAAO,SAAS+C,GACd,IAAK,EAASA,GACZ,MAAO,CAAC,EAEV,GAAIkF,GACF,OAAOA,GAAalF,GAEtB/C,EAAOvT,UAAYsW,EACnB,IAAIha,EAAS,IAAIiX,EAEjB,OADAA,EAAOvT,eAAYgG,EACZ1J,CACT,CACF,CAdiB,GCIjB,SANA,SAAyBiX,GACvB,MAAqC,mBAAtBA,EAAOrT,aAA8B,GAAYqT,GAE5D,CAAC,EADD,GAAW,GAAaA,GAE9B,ECEA,SAJA,SAAmB5c,GACjB,OAAO,GAAaA,IAVT,gBAUmB,GAAOA,EACvC,ECVA,IAAI+kB,GAAY,IAAY,GAASC,MAqBrC,SAFYD,GAAY,GAAUA,IAAa,GCP/C,SAJA,SAAmB/kB,GACjB,OAAO,GAAaA,IAVT,gBAUmB,GAAOA,EACvC,ECVA,IAAIilB,GAAY,IAAY,GAASC,MAqBrC,SAFYD,GAAY,GAAUA,IAAa,GCA/C,IAKI,GAAU,qBAKV,GAAU,oBAIV,GAAY,kBAoBZE,GAAgB,CAAC,EACrBA,GAAc,IAAWA,GA7BV,kBA8BfA,GAfqB,wBAeWA,GAdd,qBAelBA,GA9Bc,oBA8BWA,GA7BX,iBA8BdA,GAfiB,yBAeWA,GAdX,yBAejBA,GAdc,sBAcWA,GAbV,uBAcfA,GAbe,uBAaWA,GA5Bb,gBA6BbA,GA5BgB,mBA4BWA,GAAc,IACzCA,GA3BgB,mBA2BWA,GA1Bd,gBA2BbA,GA1BgB,mBA0BWA,GAzBX,mBA0BhBA,GAhBe,uBAgBWA,GAfJ,8BAgBtBA,GAfgB,wBAeWA,GAdX,yBAcsC,EACtDA,GArCe,kBAqCWA,GAAc,IACxCA,GA5BiB,qBA4BW,EA8F5B,SA5EA,SAASC,EAAUplB,EAAOqlB,EAASnE,EAAY3iB,EAAKqe,EAAQ/G,GAC1D,IAAIlQ,EACAkd,EAnEgB,EAmEPwC,EACTC,EAnEgB,EAmEPD,EACTE,EAnEmB,EAmEVF,EAKb,GAHInE,IACFvb,EAASiX,EAASsE,EAAWlhB,EAAOzB,EAAKqe,EAAQ/G,GAASqL,EAAWlhB,SAExDqP,IAAX1J,EACF,OAAOA,EAET,IAAK,EAAS3F,GACZ,OAAOA,EAET,IAAIoiB,EAAQ,GAAQpiB,GACpB,GAAIoiB,GAEF,GADAzc,EAAS,GAAe3F,IACnB6iB,EACH,OAAO,GAAU7iB,EAAO2F,OAErB,CACL,IAAIgZ,EAAM,GAAO3e,GACbwlB,EAAS7G,GAAO,IA7EX,8BA6EsBA,EAE/B,GAAI,GAAS3e,GACX,OAAO,GAAYA,EAAO6iB,GAE5B,GAAIlE,GAAO,IAAaA,GAAO,IAAY6G,IAAW5I,GAEpD,GADAjX,EAAU2f,GAAUE,EAAU,CAAC,EAAI,GAAgBxlB,IAC9C6iB,EACH,OAAOyC,EACH,GAActlB,EAAO,GAAa2F,EAAQ3F,IAC1C,GAAYA,EAAO,GAAW2F,EAAQ3F,QAEvC,CACL,IAAKmlB,GAAcxG,GACjB,OAAO/B,EAAS5c,EAAQ,CAAC,EAE3B2F,EAAS,GAAe3F,EAAO2e,EAAKkE,EACtC,CACF,CAEAhN,IAAUA,EAAQ,IAAI,IACtB,IAAI4P,EAAU5P,EAAMjG,IAAI5P,GACxB,GAAIylB,EACF,OAAOA,EAET5P,EAAMoD,IAAIjZ,EAAO2F,GAEb,GAAM3F,GACRA,EAAMoF,SAAQ,SAASsgB,GACrB/f,EAAO0P,IAAI+P,EAAUM,EAAUL,EAASnE,EAAYwE,EAAU1lB,EAAO6V,GACvE,IACS,GAAM7V,IACfA,EAAMoF,SAAQ,SAASsgB,EAAUnnB,GAC/BoH,EAAOsT,IAAI1a,EAAK6mB,EAAUM,EAAUL,EAASnE,EAAY3iB,EAAKyB,EAAO6V,GACvE,IAGF,IAIIoL,EAAQmB,OAAQ/S,GAJLkW,EACVD,EAAS,GAAe,GACxBA,EAAS,GAAS,IAEkBtlB,GASzC,OARA,GAAUihB,GAASjhB,GAAO,SAAS0lB,EAAUnnB,GACvC0iB,IAEFyE,EAAW1lB,EADXzB,EAAMmnB,IAIR,GAAY/f,EAAQpH,EAAK6mB,EAAUM,EAAUL,EAASnE,EAAY3iB,EAAKyB,EAAO6V,GAChF,IACOlQ,CACT,EC5HA,SALA,SAAuB3F,EAAOkhB,GAE5B,OAAO,GAAUlhB,EAAO,EADxBkhB,EAAkC,mBAAdA,EAA2BA,OAAa7R,EAE9D,ECbA,SAJA,SAAmBrP,GACjB,OAAO,GAAaA,IAA6B,IAAnBA,EAAMuhB,WAAmB,GAAcvhB,EACvE,ECNe,MAAM2lB,GAYpB,WAAApc,CAAaqc,EAA+BC,GAC3Cre,KAAKse,QAAU,CAAC,EAGXD,GAGJre,KAAKtJ,OAAQ6nB,GAAaF,IAItBD,GACJpe,KAAKwe,mBAAoBxe,KAAKse,QAASF,EAEzC,CA+DO,GAAA3M,CAAKzP,EAAoCxJ,GAC/CwH,KAAKye,aAAcze,KAAKse,QAAStc,EAAMxJ,EACxC,CA4BO,MAAA9B,CAAQsL,EAAoCxJ,GAGlDwH,KAAKye,aAAcze,KAAKse,QAAStc,EAAMxJ,GAFtB,EAGlB,CAkBO,GAAA4P,CAAuBpG,GAC7B,OAAOhC,KAAK0e,eAAgB1e,KAAKse,QAAStc,EAC3C,CAKA,MAAQ2c,GACP,UAAY3c,KAAQhL,OAAOC,KAAM+I,KAAKse,eAC/Btc,CAER,CAWQ,YAAAyc,CAAc7a,EAAa5B,EAAWxJ,EAAYomB,GAAoB,GAE7E,GAAK,GAAe5c,GAGnB,YAFAhC,KAAKwe,mBAAoB5a,EAAQ5B,EAAM4c,GAMxC,MAAMC,EAAQ7c,EAAKjF,MAAO,KAG1BiF,EAAO6c,EAAM7f,MAGb,UAAY8f,KAAQD,EAEb,GAAejb,EAAQkb,MAC5Blb,EAAQkb,GAAS,CAAC,GAInBlb,EAASA,EAAQkb,GAIlB,GAAK,GAAetmB,GAWnB,OATM,GAAeoL,EAAQ5B,MAC5B4B,EAAQ5B,GAAS,CAAC,GAGnB4B,EAASA,EAAQ5B,QAGjBhC,KAAKwe,mBAAoB5a,EAAQpL,EAAOomB,GAMpCA,QAAqC,IAAlBhb,EAAQ5B,KAIhC4B,EAAQ5B,GAASxJ,EAClB,CASQ,cAAAkmB,CAAgBxb,EAAalB,GAEpC,MAAM6c,EAAQ7c,EAAKjF,MAAO,KAG1BiF,EAAO6c,EAAM7f,MAGb,UAAY8f,KAAQD,EAAQ,CAC3B,IAAM,GAAe3b,EAAQ4b,IAAW,CACvC5b,EAAS,KACT,KACD,CAGAA,EAASA,EAAQ4b,EAClB,CAGA,OAAO5b,EAASqb,GAAarb,EAAQlB,SAAW,CACjD,CASQ,kBAAAwc,CAAoB5a,EAAamb,EAAoBH,GAC5D5nB,OAAOC,KAAM8nB,GAAgBnhB,SAAS7G,IACrCiJ,KAAKye,aAAc7a,EAAQ7M,EAAKgoB,EAAehoB,GAAO6nB,EAAU,GAElE,EAMD,SAASL,GAAgBrb,GACxB,OAAO,GAAeA,EAAQ8b,GAC/B,CAMA,SAASA,GAAoBxmB,GAC5B,OAAO,GAAWA,GAAUA,OAAQ,CACrC,CC/Qe,SAASymB,GAAQpa,GAC/B,GAAKA,EAAM,CACV,GAAKA,EAAIqa,YACR,OAAOra,aAAeA,EAAIqa,YAAYC,SAChC,GAAKta,EAAIua,eAAiBva,EAAIua,cAAcF,YAClD,OAAOra,aAAeA,EAAIua,cAAcF,YAAYG,IAEtD,CAEA,OAAO,CACR,CCVe,SAASC,GAAUza,GACjC,MAAM0a,EAAoBvoB,OAAO6K,UAAUnF,SAASmU,MAAOhM,GAG3D,MAA0B,mBAArB0a,GAKqB,mBAArBA,CAKN,CCHA,MAAM,GAAsB,GAAiBnQ,KAuD9B,SAAS,GAAiB1K,GACxC,IAAMA,EACL,OAAO,GAyFR,OAtFA,cAA6BA,EAaZ,QAAA8K,CACfI,EACAN,EACAC,EACAhL,EAA8F,CAAC,GAG/F,GAAK0a,GAAQrP,IAAa0P,GAAU1P,GAAY,CAC/C,MAAM4P,EAAe,CACpBC,UAAWlb,EAAQmb,WACnBC,UAAWpb,EAAQqb,YAGdC,EAAe7f,KAAK8f,iBAAkBlQ,EAAS4P,IAAkB,IAAIO,GAAcnQ,EAAS4P,GAElGxf,KAAKwP,SAAUqQ,EAAcvQ,EAAOC,EAAUhL,EAC/C,MAECiJ,MAAMgC,SAAUI,EAASN,EAAOC,EAAUhL,EAE5C,CAEgB,aAAAoL,CACfC,EACAN,EACAC,GAGA,GAAK0P,GAAQrP,IAAa0P,GAAU1P,GAAY,CAC/C,MAAMoQ,EAAgBhgB,KAAKigB,qBAAsBrQ,GAEjD,UAAYsQ,KAASF,EACpBhgB,KAAK2P,cAAeuQ,EAAO5Q,EAAOC,EAEpC,MAEC/B,MAAMmC,cAAeC,EAASN,EAAOC,EAEvC,CAaQ,gBAAAuQ,CACP3gB,EACAoF,GAEA,OxH6gBI,SAAgC4b,EAA2BC,GACjE,MAAMC,EAAgBF,EAAuCnR,GAC7D,OAAKqR,GAAeA,EAAaD,GACzBC,EAAaD,GAAsBxQ,QAGpC,IACR,CwHphBU0Q,CAAuBtgB,KAAMugB,GAAmBphB,EAAMoF,GAC9D,CAOQ,oBAAA0b,CAAsB9gB,GAC7B,MAAO,CACN,CAAEsgB,SAAS,EAAOE,SAAS,GAC3B,CAAEF,SAAS,EAAOE,SAAS,GAC3B,CAAEF,SAAS,EAAME,SAAS,GAC1B,CAAEF,SAAS,EAAME,SAAS,IACzB3iB,KAAKuH,GAAWvE,KAAK8f,iBAAkB3gB,EAAMoF,KAAYwB,QAAQma,KAAWA,GAC/E,EAIF,CAGE,CACD,mBAAoB,uBACpB,KAAM,OAAQ,MAAO,WACrB,gBAAiB,OAAQ,WAAY,iBACrC,oBAAqB,wBAClBtiB,SAAS7G,IACV,GAA0BA,GAAU,GAAoB8K,UAAoB9K,EAAK,IA+BpF,MAAMgpB,WAAqB3Q,KAY1B,WAAArN,CACC5C,EACAoF,GAEAiJ,QAGAyC,EAAejQ,KAAMugB,GAAmBphB,EAAMoF,IAG9CvE,KAAKwgB,SAAWrhB,EAGhBa,KAAKygB,SAAWlc,CACjB,CAuBO,MAAAmc,CAAQpR,GAGd,GAAKtP,KAAK2gB,eAAiB3gB,KAAK2gB,cAAerR,GAC9C,OAGD,MAAMsR,EAAc5gB,KAAK6gB,mBAAoBvR,GAG7CtP,KAAKwgB,SAASlQ,iBAAkBhB,EAAOsR,EAAa5gB,KAAKygB,UAEnDzgB,KAAK2gB,gBACV3gB,KAAK2gB,cAAgB,CAAC,GAKvB3gB,KAAK2gB,cAAerR,GAAUsR,CAC/B,CAUO,MAAAE,CAAQxR,GACd,IAAI+B,GAMCrR,KAAK2gB,cAAgBrR,KAAgB+B,EAAWrR,KAAcuS,QAAUjD,KAAc+B,EAAOlB,UAAU7X,QAC3G0H,KAAK2gB,cAAgBrR,GAAQyR,gBAE/B,CAUO,iBAAA1Q,CACNf,EACAC,EACAhL,GAEAvE,KAAK0gB,OAAQpR,GACXF,IAAevN,UAAmBwO,kBAAkBhP,KAAMrB,KAAMsP,EAAOC,EAAUhL,EACpF,CASO,oBAAAuM,CAAsBxB,EAAeC,GACzCH,IAAevN,UAAmBiP,qBAAqBzP,KAAMrB,KAAMsP,EAAOC,GAC5EvP,KAAK8gB,OAAQxR,EACd,CAUQ,kBAAAuR,CAAoBvR,GAC3B,MAAMsR,EAAgBI,IACrBhhB,KAAKwQ,KAAMlB,EAAO0R,EAAQ,EAW3B,OALAJ,EAAYG,eAAiB,KAC5B/gB,KAAKwgB,SAASjQ,oBAAqBjB,EAAOsR,EAAa5gB,KAAKygB,iBACrDzgB,KAAK2gB,cAAgBrR,EAAO,EAG7BsR,CACR,EAeD,SAASL,GAAmBphB,EAAqBoF,GAChD,IAAI1E,EARL,SAAqBV,GACpB,OAAOA,EAAM,qBAAyBA,EAAM,mBAAsB,IACnE,CAMU8hB,CAAY9hB,GAErB,UAAY+hB,KAAUlqB,OAAOC,KAAMsN,GAAU4c,OACvC5c,EAAS2c,KACbrhB,GAAM,IAAMqhB,GAId,OAAOrhB,CACR,CCpVA,IAAIuhB,GAGJ,IACCA,GAAY,CAAE7d,OAAQC,SACvB,CAAE,MAAQU,GAQTkd,GAAY,CAAE7d,OAAQ,CAAC,EAAUC,SAAU,CAAC,EAC7C,CAEA,YCxCe,SAAS6d,GAAQxc,GAC/B,MAAgD,iBAAzC7N,OAAO6K,UAAUnF,SAAS2E,KAAMwD,EACxC,CCFe,SAASyc,GAASzc,GAChC,MAAiD,kBAA1C7N,OAAO6K,UAAUnF,SAASmU,MAAOhM,EACzC,CCEe,SAAS0c,GAAiB9J,GAExC,MAAMpS,EAAQoS,EAAQ2H,cAAcF,YAAasC,iBAAkB/J,GAEnE,MAAO,CACNgK,IAAKrkB,SAAUiI,EAAMqc,eAAgB,IACrCC,MAAOvkB,SAAUiI,EAAMuc,iBAAkB,IACzCC,OAAQzkB,SAAUiI,EAAMyc,kBAAmB,IAC3CC,KAAM3kB,SAAUiI,EAAM2c,gBAAiB,IAEzC,CCZA,MAAMC,GAAwC,CAAE,MAAO,QAAS,SAAU,OAAQ,QAAS,UAO5E,MAAM,GA+EpB,WAAAlgB,CAAamB,GACZ,MAAMgf,EAAgBZ,GAASpe,GAS/B,GAPAlM,OAAOuB,eAAgByH,KAAM,UAAW,CAEvCxH,MAAS0K,EAAgBif,SAAWjf,EACpCkf,UAAU,EACVja,YAAY,IAGRka,GAAcnf,IAAYgf,EAY9B,GAAKA,EAAgB,CACpB,MAAMI,EAAa,GAAKC,iBAAkBrf,GAC1Csf,GAAoBxiB,KAAM,GAAKyiB,gBAAiBH,GACjD,MACCE,GAAoBxiB,KAAMkD,EAAOwf,8BAE5B,GAAKpD,GAAUpc,GAAW,CAChC,MAAM,WAAEyf,EAAU,YAAEC,GAAgB1f,EAEpCsf,GAAoBxiB,KAAM,CACzByhB,IAAK,EACLE,MAAOgB,EACPd,OAAQe,EACRb,KAAM,EACNc,MAAOF,EACPG,OAAQF,GAEV,MACCJ,GAAoBxiB,KAAMkD,EAE5B,CAOO,KAAA6f,GACN,OAAO,IAAI,GAAM/iB,KAClB,CASO,MAAAgjB,CAAQ9oB,EAAWD,GAMzB,OALA+F,KAAKyhB,IAAMxnB,EACX+F,KAAK2hB,MAAQznB,EAAI8F,KAAK6iB,MACtB7iB,KAAK6hB,OAAS5nB,EAAI+F,KAAK8iB,OACvB9iB,KAAK+hB,KAAO7nB,EAEL8F,IACR,CASO,MAAAijB,CAAQ/oB,EAAWD,GAMzB,OALA+F,KAAKyhB,KAAOxnB,EACZ+F,KAAK2hB,OAASznB,EACd8F,KAAK+hB,MAAQ7nB,EACb8F,KAAK6hB,QAAU5nB,EAER+F,IACR,CAKO,eAAAkjB,CAAiBC,GACvB,MAAMC,EAAO,CACZ3B,IAAK5oB,KAAKC,IAAKkH,KAAKyhB,IAAK0B,EAAY1B,KACrCE,MAAO9oB,KAAKD,IAAKoH,KAAK2hB,MAAOwB,EAAYxB,OACzCE,OAAQhpB,KAAKD,IAAKoH,KAAK6hB,OAAQsB,EAAYtB,QAC3CE,KAAMlpB,KAAKC,IAAKkH,KAAK+hB,KAAMoB,EAAYpB,MACvCc,MAAO,EACPC,OAAQ,GAMT,OAHAM,EAAKP,MAAQO,EAAKzB,MAAQyB,EAAKrB,KAC/BqB,EAAKN,OAASM,EAAKvB,OAASuB,EAAK3B,IAE5B2B,EAAKP,MAAQ,GAAKO,EAAKN,OAAS,EAC7B,KAEA,IAAI,GAAMM,EAEnB,CAOO,mBAAAC,CAAqBF,GAC3B,MAAMC,EAAOpjB,KAAKkjB,gBAAiBC,GAEnC,OAAKC,EACGA,EAAKE,UAEL,CAET,CAKO,OAAAA,GACN,OAAOtjB,KAAK6iB,MAAQ7iB,KAAK8iB,MAC1B,CAmBO,UAAAS,GACN,MAAMrgB,EAA2FlD,KAAKmiB,QAEtG,IAAIqB,EAAcxjB,KAAK+iB,QAGvB,GAAKU,GAAQvgB,GACZ,OAAOsgB,EAGR,IAEIE,EAFA/L,EAAazU,EACbtE,EAASsE,EAAO+D,YAAc/D,EAAOygB,wBAIzC,KAAQ/kB,IAAW6kB,GAAQ7kB,IAAW,CAQrC,GAPK+Y,aAAiBiM,aAA+C,aAAhCC,GAAoBlM,KACxD+L,EAAmC/L,GAOnC+L,IACkD,aAAhDG,GAAoBjlB,IAA0F,aAwOvF6Y,EAxO2D7Y,GAyOxEwgB,cAAcF,YAAasC,iBAAkB/J,GAAUqM,UAxOlE,CACDnM,EAAQ/Y,EACRA,EAASA,EAAOqI,WAChB,QACD,CAEA,MAAM8c,EAAa,IAAI,GAAMnlB,GACvBolB,EAAmBR,EAAYN,gBAAiBa,GAEtD,IAAKC,EAOJ,OAAO,KANFA,EAAiBV,UAAYE,EAAYF,YAE7CE,EAAcQ,GAOhBrM,EAAQ/Y,EACRA,EAASA,EAAOqI,UACjB,CAkNF,IAA6BwQ,EAhN3B,OAAO+L,CACR,CAUO,OAAAS,CAASd,GACf,UAAY5a,KAAQ0Z,GACnB,GAAKjiB,KAAMuI,KAAW4a,EAAa5a,GAClC,OAAO,EAIT,OAAO,CACR,CAQO,QAAA2b,CAAUf,GAChB,MAAMgB,EAAgBnkB,KAAKkjB,gBAAiBC,GAE5C,SAAWgB,IAAiBA,EAAcF,QAASd,GACpD,CAUO,2BAAAiB,GACN,MAAMlhB,EAASlD,KAAKmiB,QACpB,IAAIkC,EAAgBC,EAAiBC,EAErC,GAAKjF,GAAUpc,GACdmhB,EAAiBnhB,EAAOyf,WAAazf,EAAOM,SAASghB,gBAAgBC,YACrEH,EAAkBphB,EAAO0f,YAAc1f,EAAOM,SAASghB,gBAAgBE,aACvEH,EAAYrhB,EAAOse,iBAAkBte,EAAOM,SAASghB,iBAAkBD,cACjE,CACN,MAAMI,EAAepD,GAAiBre,GAEtCmhB,EAAiBnhB,EAAO0hB,YAAc1hB,EAAOuhB,YAAcE,EAAa5C,KAAO4C,EAAahD,MAC5F2C,EAAkBphB,EAAO2hB,aAAe3hB,EAAOwhB,aAAeC,EAAalD,IAAMkD,EAAa9C,OAC9F0C,EAAYrhB,EAAOkc,cAAcF,YAAasC,iBAAkBte,GAASqhB,UAEzEvkB,KAAK+hB,MAAQ4C,EAAa5C,KAC1B/hB,KAAKyhB,KAAOkD,EAAalD,IACzBzhB,KAAK2hB,OAASgD,EAAahD,MAC3B3hB,KAAK6hB,QAAU8C,EAAa9C,OAC5B7hB,KAAK6iB,MAAQ7iB,KAAK2hB,MAAQ3hB,KAAK+hB,KAC/B/hB,KAAK8iB,OAAS9iB,KAAK6hB,OAAS7hB,KAAKyhB,GAClC,CAaA,OAXAzhB,KAAK6iB,OAASwB,EAEK,QAAdE,EACJvkB,KAAK2hB,OAAS0C,EAEdrkB,KAAK+hB,MAAQsC,EAGdrkB,KAAK8iB,QAAUwB,EACftkB,KAAK6hB,QAAUyC,EAERtkB,IACR,CAQA,uBAAcuiB,CAAkBuC,GAC/B,MAAMC,EAAqB,GAErBC,EAAcrkB,MAAMrB,KAAMwlB,EAAMG,kBAEtC,GAAKD,EAAY1sB,OAChB,UAAY8qB,KAAQ4B,EACnBD,EAAMnlB,KAAM,IAAI,GAAMwjB,QAOnB,CACJ,IAAI8B,EAAiBJ,EAAMI,eAEtB7D,GAAQ6D,KACZA,EAAiBA,EAAeje,YAGjC,MAAMmc,EAAO,IAAI,GAAQ8B,EAA4BxC,yBACrDU,EAAKzB,MAAQyB,EAAKrB,KAClBqB,EAAKP,MAAQ,EAEbkC,EAAMnlB,KAAMwjB,EACb,CAEA,OAAO2B,CACR,CAQA,sBAActC,CAAiBsC,GAC9B,MAAMI,EAAmB,CACxBpD,KAAMqD,OAAOC,kBACb5D,IAAK2D,OAAOC,kBACZ1D,MAAOyD,OAAOE,kBACdzD,OAAQuD,OAAOE,kBACfzC,MAAO,EACPC,OAAQ,GAET,IAAIyC,EAAiB,EAErB,UAAYnC,KAAQ2B,EACnBQ,IAEAJ,EAAiBpD,KAAOlpB,KAAKD,IAAKusB,EAAiBpD,KAAMqB,EAAKrB,MAC9DoD,EAAiB1D,IAAM5oB,KAAKD,IAAKusB,EAAiB1D,IAAK2B,EAAK3B,KAC5D0D,EAAiBxD,MAAQ9oB,KAAKC,IAAKqsB,EAAiBxD,MAAOyB,EAAKzB,OAChEwD,EAAiBtD,OAAShpB,KAAKC,IAAKqsB,EAAiBtD,OAAQuB,EAAKvB,QAGnE,OAAuB,GAAlB0D,EACG,MAGRJ,EAAiBtC,MAAQsC,EAAiBxD,MAAQwD,EAAiBpD,KACnEoD,EAAiBrC,OAASqC,EAAiBtD,OAASsD,EAAiB1D,IAE9D,IAAI,GAAM0D,GAClB,EAuBD,SAAS3C,GAAoBY,EAAYlgB,GACxC,UAAYpI,KAAKmnB,GAChBmB,EAAMtoB,GAAMoI,EAAQpI,EAEtB,CAKA,SAAS2oB,GAAQjrB,GAChB,QAAM6pB,GAAc7pB,IAIbA,IAAUA,EAAM4mB,cAAcoG,IACtC,CAKA,SAASnD,GAAc7pB,GAGtB,OAAiB,OAAVA,GAAmC,iBAAVA,GAAyC,IAAnBA,EAAMuhB,UAAyD,mBAAhCvhB,EAAMkqB,qBAC5F,CAKA,SAASmB,GAAoBpM,GAC5B,OAAOA,EAAQ2H,cAAcF,YAAasC,iBAAkB/J,GAAUgO,QACvE,CCxdA,MAAqB,GAArB,MA+BC,WAAA1jB,CAAa0V,EAAkBlI,GAGxB,GAAemW,mBACpB,GAAeC,kBAGhB3lB,KAAK4lB,SAAWnO,EAChBzX,KAAK6lB,UAAYtW,EAEjB,GAAeuW,oBAAqBrO,EAASlI,GAC7C,GAAemW,kBAAmBK,QAAStO,EAC5C,CAKA,WAAWA,GACV,OAAOzX,KAAK4lB,QACb,CAKO,OAAAI,GACN,GAAeC,uBAAwBjmB,KAAK4lB,SAAU5lB,KAAK6lB,UAC5D,CAKA,0BAAeC,CAAqBrO,EAAkBlI,GAC/C,GAAe2W,oBACpB,GAAeA,kBAAoB,IAAI3U,KAGxC,IAAIpB,EAAY,GAAe+V,kBAAkB9d,IAAKqP,GAEhDtH,IACLA,EAAY,IAAIyD,IAChB,GAAesS,kBAAkBzU,IAAKgG,EAAStH,IAGhDA,EAAUtC,IAAK0B,EAChB,CAMA,6BAAe0W,CAAwBxO,EAAkBlI,GACxD,MAAMY,EAAY,GAAegW,qBAAsB1O,GAIlDtH,IACJA,EAAUwB,OAAQpC,GAGZY,EAAU0D,OACf,GAAeqS,kBAAmBvU,OAAQ8F,GAC1C,GAAeiO,kBAAmBU,UAAW3O,KAI1C,GAAeyO,oBAAsB,GAAeA,kBAAkBrS,OAC1E,GAAe6R,kBAAoB,KACnC,GAAeQ,kBAAoB,KAErC,CAKA,2BAAeC,CAAsB1O,GACpC,OAAM,GAAeyO,kBAId,GAAeA,kBAAkB9d,IAAKqP,GAHrC,IAIT,CAKA,sBAAekO,GACd,GAAeD,kBAAoB,IAAI,GAAOniB,OAAO8iB,gBAAgB5N,IACpE,UAAYC,KAASD,EAAU,CAC9B,MAAMtI,EAAY,GAAegW,qBAAsBzN,EAAM9U,QAE7D,GAAKuM,EACJ,UAAYZ,KAAYY,EACvBZ,EAAUmJ,EAGb,IAEF,GAhID,IAAqB,GAArB,GCVe,SAAS4N,GAAkBC,EAAiBzjB,GACrDyjB,aAAcC,sBAClBD,EAAG/tB,MAAQsK,GAGZyjB,EAAGE,UAAY3jB,CAChB,CCRe,SAAS4jB,GAAQC,GAC/B,OAAOnuB,GAASA,EAAQmuB,CACzB,CCFe,SAAS3d,GAAS7J,GAChC,IAAIyF,EAAQ,EAEZ,KAAQzF,EAAKynB,iBACZznB,EAAOA,EAAKynB,gBACZhiB,IAGD,OAAOA,CACR,CCRe,SAASiiB,GACvBC,EACAliB,EACAmiB,GAEAD,EAActgB,aAAcugB,EAAcD,EAAcxgB,WAAY1B,IAAW,KAChF,CCRe,SAASoiB,GAAWniB,GAClC,OAAOA,GAAOA,EAAIkV,WAAasF,KAAK4H,YACrC,CCGe,SAASC,GAAWzP,GAClC,SAAWA,GAAWA,EAAQwN,gBAAkBxN,EAAQwN,iBAAiB3sB,OAC1E,CNMqB,GAcLotB,kBAA8E,KAdzE,GAoBLQ,kBAAsF,K,gBOmC/F,SAASiB,IAAoB,QAAE1P,EAAO,OAAE7T,EAAM,UAAEwjB,EAAS,QAAEC,EAAO,cAAEC,EAAa,qBAAEC,IAGpF,GAAY3jB,KAChBA,EAASA,KAKL,GAAYyjB,KAChBA,EAAUA,KAGX,MAAMG,EC/EQ,SAAgC/P,GAC9C,OAAMA,GAAYA,EAAQxQ,WAIrBwQ,EAAQgQ,eAAiB,GAAOjkB,SAASgiB,KACtC,KAGD/N,EAAQgQ,aAPP,IAQT,CDqEmCC,CAAuBjQ,GACnDkQ,EAAc,IAAI,GAAMlQ,GACxBmQ,EAAa,IAAI,GAAMhkB,GAE7B,IAAIikB,EAKJ,MAAMC,EAAeR,GA8BtB,SAAqCC,GACpCA,EAAuBvwB,OAAO+wB,OAAQ,CAAEtG,IAAK,EAAGI,OAAQ,EAAGE,KAAM,EAAGJ,MAAO,GAAK4F,GAEhF,MAAMO,EAAe,IAAI,GAAM,GAAOvkB,QAOtC,OALAukB,EAAarG,KAAO8F,EAAqB9F,IACzCqG,EAAahF,QAAUyE,EAAqB9F,IAC5CqG,EAAajG,QAAU0F,EAAqB1F,OAC5CiG,EAAahF,QAAUyE,EAAqB1F,OAErCiG,CACR,CAzCuCE,CAA4BT,IAA0B,KACtFU,EAAkB,CAAEL,aAAYD,cAAaH,4BAA2BM,gBAG9E,GAAMT,GAAYC,EAEX,CACN,MAAMY,EAAcb,GAAW,IAAI,GAAMA,GAAU9D,aAUnDvsB,OAAO+wB,OAAQE,EAAiB,CAAEC,cAAaJ,iBAI/CD,EA0BF,SACCT,EACA7iB,GAEA,MAAM,YAAEojB,GAAgBpjB,EAGlB4jB,EAAkBR,EAAYrE,UAE9B8E,EAAoBhB,EACxBpqB,KAAKqrB,GAAuB,IAAIC,GAAgBD,EAAqB9jB,KAErEwB,QAAQ0f,KAAcA,EAASzjB,OAEjC,IAAIumB,EAAe,EACfV,EAAe,KAEnB,UAAYpC,KAAY2C,EAAoB,CAC3C,MAAM,wBAAEI,EAAuB,yBAAEC,GAA6BhD,EAI9D,GAAK+C,IAA4BL,EAChC,OAAO1C,EAKR,MAAMiD,EAAY,GAAAD,EAA4B,GAAI,GAAAD,EAA2B,GAExEE,EAAYH,IAChBA,EAAeG,EACfb,EAAepC,EAEjB,CAEA,OAAOoC,CACR,CA/DiBc,CAAiBvB,EAAWa,IAAqB,IAAIK,GAAgBlB,EAAW,GAAKa,EACrG,MAjBCJ,EAAe,IAAIS,GAAgBlB,EAAW,GAAKa,GAmBpD,OAAOJ,CACR,CA0GA,SAASe,GAA+BxF,GACvC,MAAM,QAAEyF,EAAO,QAAEC,GAAY,GAAOvlB,OAEpC,OAAO6f,EAAKL,QAAQE,OAAQ4F,EAASC,EACtC,CA4CA,MAAMR,GAqBL,WAAAvmB,CACCsmB,EACA9jB,GAQA,MAAMwkB,EAA4BV,EAAqB9jB,EAAQqjB,WAAYrjB,EAAQojB,YAAapjB,EAAQujB,cAGxG,IAAMiB,EACL,OAGD,MAAM,KAAEhH,EAAI,IAAEN,EAAG,KAAEzf,EAAI,OAAEgnB,GAAWD,EAEpC/oB,KAAKgC,KAAOA,EACZhC,KAAKgpB,OAASA,EAEdhpB,KAAKipB,gCAAkC,CAAElH,OAAMN,OAC/CzhB,KAAKygB,SAAWlc,CACjB,CAMA,QAAWwd,GACV,OAAO/hB,KAAKkpB,cAAcnH,IAC3B,CAMA,OAAWN,GACV,OAAOzhB,KAAKkpB,cAAczH,GAC3B,CAKA,2BAAW+G,GACV,MAAMN,EAAcloB,KAAKygB,SAASyH,YAElC,GAAKA,EAAc,CAClB,MAAMJ,EAAe9nB,KAAKygB,SAASqH,aAEnC,IAAKA,EAUJ,OAAOI,EAAY7E,oBAAqBrjB,KAAKmpB,OAV1B,CAEnB,MAAMC,EAA+BlB,EAAYhF,gBAAiB4E,GAElE,GAAKsB,EAGJ,OAAOA,EAA6B/F,oBAAqBrjB,KAAKmpB,MAEhE,CAGD,CAEA,OAAO,CACR,CAKA,4BAAWV,GACV,MAAMX,EAAe9nB,KAAKygB,SAASqH,aAEnC,OAAKA,EACGA,EAAazE,oBAAqBrjB,KAAKmpB,OAGxC,CACR,CAMA,SAAYA,GACX,OAAKnpB,KAAKqpB,cAIVrpB,KAAKqpB,YAAcrpB,KAAKygB,SAASkH,YAAY5E,QAAQC,OACpDhjB,KAAKipB,gCAAgClH,KACrC/hB,KAAKipB,gCAAgCxH,MAL9BzhB,KAAKqpB,WASd,CAKA,iBAAYH,GACX,OAAKlpB,KAAKspB,sBAIVtpB,KAAKspB,oBAAsBV,GAA+B5oB,KAAKmpB,OAE1DnpB,KAAKygB,SAAS+G,2BAtNrB,SAAkDpE,EAAYoE,GAC7D,MAAM+B,EAAmBX,GAA+B,IAAI,GAAMpB,IAC5DgC,EAAuBjI,GAAiBiG,GAE9C,IAAIiC,EAAQ,EACRC,EAAQ,EAMZD,GAASF,EAAiBxH,KAC1B2H,GAASH,EAAiB9H,IAO1BgI,GAASjC,EAA0BmC,WACnCD,GAASlC,EAA0BoC,UAOnCH,GAASD,EAAqBzH,KAC9B2H,GAASF,EAAqB/H,IAE9B2B,EAAKH,OAAQwG,EAAOC,EACrB,CAwLGG,CAAyC7pB,KAAKspB,oBAAqBtpB,KAAKygB,SAAS+G,4BAN1ExnB,KAAKspB,mBAUd,EEnZc,SAASrjB,GAAQ9G,GAC/B,MAAMP,EAASO,EAAK8H,WAEfrI,GACJA,EAAO2H,YAAapH,EAEtB,CC6MA,SAAS2qB,IACR,OACCvmB,EAAM,KACN6f,EAAI,WACJ2G,EAAU,YACVC,EAAW,eACXC,IASD,MAAMC,EAAwB9G,EAAKL,QAAQE,OAAQ,EAAGgH,EAAepI,QAC/DsI,EAAsB/G,EAAKL,QAAQE,OAAQ,GAAIgH,EAAexI,KAC9DqG,EAAe,IAAI,GAAMvkB,GAAS6gB,8BAGlCgG,EAAmBL,GAAcC,EACjCK,EAFQ,CAAEF,EAAqBD,GAED3T,OAAO6M,GAAQ0E,EAAa5D,SAAUd,KAE1E,IAAI,QAAEyF,EAAO,QAAEC,GAAYvlB,EAC3B,MAAM+mB,EAAiBzB,EACjB0B,EAAiBzB,EAElBsB,EACJtB,GAAahB,EAAarG,IAAM2B,EAAK3B,IAAQwI,EAAexI,IAChD4I,IACPG,GAASL,EAAqBrC,GAClCgB,GAAWhB,EAAarG,IAAM2B,EAAK3B,IAAMwI,EAAexI,IAC7CgJ,GAASP,EAAuBpC,KAE1CgB,GADIiB,EACO3G,EAAK3B,IAAMqG,EAAarG,IAAMwI,EAAexI,IAE7C2B,EAAKvB,OAASiG,EAAajG,OAASoI,EAAepI,SAK3DwI,IAGAK,GAAUtH,EAAM0E,GACpBe,GAAWf,EAAa/F,KAAOqB,EAAKrB,KAAOkI,EAAelI,KAC/C4I,GAAWvH,EAAM0E,KAC5Be,GAAWzF,EAAKzB,MAAQmG,EAAanG,MAAQsI,EAAetI,QAIzDkH,GAAWyB,GAAkBxB,IAAYyB,GAC7ChnB,EAAOqnB,SAAU/B,EAASC,EAE5B,CAmBA,SAAS+B,IACR,OACCjsB,EAAM,QACNksB,EAAO,WACPf,EAAU,YACVC,EAAW,eACXe,EAAiB,EAAC,eAClBC,IAUD,MAAMC,EAAeC,GAAWtsB,GAC1BwrB,EAAmBL,GAAcC,EACvC,IAAIjG,EAAkB6D,EAAkBuD,EAExC,MAAM9D,EAAU2D,GAAkBC,EAAaznB,SAASgiB,KAExD,KAAQ5mB,GAAUyoB,GACjBO,EAAakD,IACb/G,EAAa,IAAI,GAAMnlB,GAASwlB,8BAChC+G,EAAqBpH,EAAWG,SAAU0D,GAErCwC,EACJxrB,EAAOgrB,WAAe7F,EAAWtC,IAAMmG,EAAWnG,IAAQsJ,EAC9CI,IACPX,GAAS5C,EAAY7D,GACzBnlB,EAAOgrB,WAAa7F,EAAWtC,IAAMmG,EAAWnG,IAAMsJ,EAC3CN,GAAS7C,EAAY7D,KAE/BnlB,EAAOgrB,WADHG,EACgBnC,EAAWnG,IAAMsC,EAAWtC,IAAMsJ,EAElCnD,EAAW/F,OAASkC,EAAWlC,OAASkJ,IAKzDI,IACAT,GAAU9C,EAAY7D,GAC1BnlB,EAAO+qB,YAAc5F,EAAWhC,KAAO6F,EAAW7F,KAAOgJ,EAC9CJ,GAAW/C,EAAY7D,KAClCnlB,EAAO+qB,YAAc/B,EAAWjG,MAAQoC,EAAWpC,MAAQoJ,IAI7DnsB,EAASA,EAAOqI,UAElB,CAKA,SAASwjB,GAASW,EAAiBC,GAClC,OAAOD,EAAUvJ,OAASwJ,EAAWxJ,MACtC,CAKA,SAAS2I,GAASY,EAAiBC,GAClC,OAAOD,EAAU3J,IAAM4J,EAAW5J,GACnC,CAKA,SAASiJ,GAAUU,EAAiBC,GACnC,OAAOD,EAAUrJ,KAAOsJ,EAAWtJ,IACpC,CAKA,SAAS4I,GAAWS,EAAiBC,GACpC,OAAOD,EAAUzJ,MAAQ0J,EAAW1J,KACrC,CAKA,SAASuJ,GAAWI,GACnB,OAAKhK,GAASgK,GACNA,EAAepG,eAAe9F,cAAeF,YAE7CoM,EAAelM,cAAcF,WAEtC,CAKA,SAASqM,GAAkBD,GAC1B,GAAKhK,GAASgK,GAAmB,CAChC,IAAI1sB,EAAS0sB,EAAe3H,wBAO5B,OAJKtC,GAAQziB,KACZA,EAASA,EAAOqI,YAGVrI,CACR,CACC,OAAO0sB,EAAerkB,UAExB,CASA,SAASukB,GAAyB5nB,EAA6B6nB,GAC9D,MAAMC,EAAeR,GAAWtnB,GAC1Bwf,EAAO,IAAI,GAAMxf,GAEvB,GAAK8nB,IAAiBD,EACrB,OAAOrI,EACD,CACN,IAAIuI,EAAgBD,EAEpB,KAAQC,GAAiBF,GAAiB,CACzC,MAAMG,EAAQD,EAAcE,aACtBC,EAAY,IAAI,GAAMF,GAAQxH,8BAEpChB,EAAKH,OAAQ6I,EAAU/J,KAAM+J,EAAUrK,KAEvCkK,EAAgBA,EAAc/sB,MAC/B,CACD,CAEA,OAAOwkB,CACR,CCraA,MAAM2I,GAAuB,CAC5BC,KAAM,IACNC,IAAK,IACLC,IAAK,IACLC,MAAO,KAGFC,GAA0B,CAC/BJ,KAAM,QACNE,IAAK,OACLC,MAAO,UAgBKE,GA0Lb,WACC,MAAMA,EAA0C,CAC/CC,UAAW,GACXC,QAAS,GACTC,WAAY,GACZC,UAAW,GACXC,UAAW,EACX/a,OAAQ,GACRgb,MAAO,GACPC,MAAO,GACPC,IAAK,GACLC,IAAK,EAILd,KAAM,QACNG,MAAO,QACPD,IAAK,QACLD,IAAK,SAIN,QAAUc,EAAO,GAAIA,GAAQ,GAAIA,IAAS,CAGzCV,EAFepR,OAAO+R,aAAcD,GAEnBnkB,eAAkBmkB,CACpC,CAGA,QAAUA,EAAO,GAAIA,GAAQ,GAAIA,IAChCV,EAAUU,EAAO,IAAOA,EAIzB,QAAUA,EAAO,IAAKA,GAAQ,IAAKA,IAClCV,EAAU,KAAQU,EAAO,MAAUA,EAIpC,UAAY9vB,IAAQ,eACnBovB,EAAUpvB,GAASA,EAAKgwB,WAAY,GAGrC,OAAOZ,CACR,CAtOwBa,GAElBC,GAAen2B,OAAOo2B,YAC3Bp2B,OAAOyhB,QAAS4T,IAAWrvB,KAAK,EAAIgF,EAAM+qB,KAAY,CAAEA,EAAM/qB,EAAKqrB,OAAQ,GAAI1wB,cAAgBqF,EAAKF,MAAO,OAWrG,SAAS,GAAS/K,GACxB,IAAIu2B,EAEJ,GAAmB,iBAAPv2B,GAGX,GAFAu2B,EAAUjB,GAAUt1B,EAAI6R,gBAElB0kB,EAOL,MAAM,IAAI,EAAe,uBAAwB,KAAM,CAAEv2B,aAG1Du2B,EAAUv2B,EAAIu2B,SACXv2B,EAAIw2B,OAASlB,GAASH,IAAM,IAC5Bn1B,EAAIy2B,QAAUnB,GAASL,KAAO,IAC9Bj1B,EAAI02B,SAAWpB,GAASF,MAAQ,IAChCp1B,EAAI22B,QAAUrB,GAASJ,IAAM,GAGjC,OAAOqB,CACR,CAwBO,SAASK,GAAgBC,GAK/B,MAJyB,iBAAbA,IACXA,EAwKF,SAA6BA,GAC5B,OAAOA,EAAU7wB,MAAO,KAAMC,KAAKjG,GAAOA,EAAI82B,QAC/C,CA1KcC,CAAoBF,IAG1BA,EACL5wB,KAAKjG,GAAuB,iBAAPA,EAuFxB,SAAwBA,GAEvB,GAAKA,EAAIg3B,SAAU,KAClB,OAAO,GAASh3B,EAAI+K,MAAO,GAAI,IAGhC,MAAMirB,EAAO,GAASh2B,GAEtB,OAAO,SAAag2B,GAAQV,GAASL,KAAOK,GAASJ,IAAMc,CAC5D,CAhG4CiB,CAAej3B,GAAQA,IAChEk3B,QAAQ,CAAEl3B,EAAKm3B,IAASA,EAAMn3B,GAAK,EACtC,CASO,SAASo3B,GAAqBP,GACpC,IAAIQ,EAAgBT,GAAgBC,GAcpC,OAZ0B52B,OAAOyhB,QAAS,QAAYsT,GAAuBK,IAEzC6B,QAAQ,CAAEI,GAAarsB,EAAMssB,MAEnB,IAAtCF,EAAgB/B,GAAUrqB,MAChCosB,IAAkB/B,GAAUrqB,GAC5BqsB,GAAaC,GAGPD,IACL,KAEkBD,EAAgBjB,GAAciB,GAAkB,GACtE,CA+BO,SAAS,GACfd,EACAiB,GAEA,MAAMC,EAA4C,QAA7BD,EAErB,OAASjB,GACR,KAAKjB,GAASC,UACb,OAAOkC,EAAe,OAAS,QAEhC,KAAKnC,GAASG,WACb,OAAOgC,EAAe,QAAU,OAEjC,KAAKnC,GAASE,QACb,MAAO,KAER,KAAKF,GAASI,UACb,MAAO,OAEV,CC/Je,SAAS,GAAY3pB,GACnC,OAAOnC,MAAMC,QAASkC,GAASA,EAAO,CAAEA,EACzC,CC0IO,SAAS2rB,GAAYC,EAAkBtgB,EAAkBugB,EAAmB,GAClF,GAAyB,iBAAbA,EAQX,MAAM,IAAI,EAAe,4CAA6C,KAAM,CAAEA,aAG/E,MAAMC,EAqDC53B,OAAOC,KAAM,GAAOsM,OAAOsrB,uBAAwBv2B,OAnD/B,IAAtBs2B,IAGJF,EAAW13B,OAAOC,KAAM,GAAOsM,OAAOsrB,uBAAyB,IAGhE,MAAMC,EAAY1gB,EAAQvO,IAAMuO,EAAQ3R,OAExC,GAA2B,IAAtBmyB,IAmCN,SAAyBF,EAAkBI,GAC1C,QACG,GAAOvrB,OAAOsrB,sBAAuBH,MACrC,GAAOnrB,OAAOsrB,sBAAuBH,GAAWK,WAAYD,EAEhE,CAxCkCE,CAAgBN,EAAUI,GAC1D,OAAkB,IAAbH,EAEGvgB,EAAQ6gB,OAGT7gB,EAAQ3R,OAGhB,MAAMsyB,EAAa,GAAOxrB,OAAOsrB,sBAAuBH,GAAWK,WAC7DG,EAAgB,GAAO3rB,OAAOsrB,sBAAuBH,GAAWQ,eAAiB,CAAE5zB,GAAW,IAANA,EAAU,EAAI,GACtG6zB,EAAcJ,EAAYD,GAEhC,GAA4B,iBAAhBK,EACX,OAAOA,EAMR,OAAOA,EAHiB/J,OAAQ8J,EAAeP,IAIhD,CA1LM,GAAOprB,OAAOsrB,wBACnB,GAAOtrB,OAAOsrB,sBAAwB,CAAC,GCXxC,MAAMO,GAAqB,CAC1B,KAAM,MACN,KAAM,MAAO,MACb,KAAM,MACN,KAAM,MACN,KAAM,OAQA,SAASC,GAAsBC,GACrC,OAAOF,GAAmBG,SAAUD,GAAiB,MAAQ,KAC9D,CCXe,MAAME,GA4FpB,WAAAztB,EAAa,WAAE0tB,EAAa,qBAAMC,GAAyF,CAAC,GAC3H1vB,KAAKyvB,WAAaA,EAClBzvB,KAAK0vB,gBAAkBA,GAAmB1vB,KAAKyvB,WAC/CzvB,KAAK2vB,oBAAsBN,GAAsBrvB,KAAKyvB,YACtDzvB,KAAKuuB,yBAA2Bc,GAAsBrvB,KAAK0vB,iBAE3D1vB,KAAKhF,EAAI,CAAEoT,EAASlD,IAAYlL,KAAK4vB,GAAIxhB,EAASlD,EACnD,CAUA,YAAWwjB,GAaV,OALApgB,QAAQC,KACP,iMAIMvO,KAAKyvB,UACb,CAKQ,EAAAG,CAAIxhB,EAA2BlD,EAA2D,IACjGA,EAAS,GAASA,GAEM,iBAAZkD,IACXA,EAAU,CAAE3R,OAAQ2R,IAGrB,MACMugB,IADkBvgB,EAAQ6gB,OACC/jB,EAAQ,GAAgB,EAIzD,OAcF,SAA4BzO,EAAgByO,GAC3C,OAAOzO,EAAOwc,QAAS,WAAW,CAAEpc,EAAO+H,IACjCA,EAAQsG,EAAO5S,OAAW4S,EAAQtG,GAAU/H,GAEvD,CAlBSgzB,CAFkBpB,GAAYzuB,KAAKyvB,WAAYrhB,EAASugB,GAEnBzjB,EAC7C,ECrIc,MAAM,WAAkDkE,KAoGtE,WAAArN,CACC+tB,EAAwE,CAAC,EACzEvrB,EAA4C,CAAC,GAE7CiJ,QAEA,MAAMuiB,EAAkBpZ,EAAYmZ,GAcpC,GAZMC,IACLxrB,EAAUurB,GAGX9vB,KAAKgwB,OAAS,GACdhwB,KAAKiwB,SAAW,IAAI1e,IACpBvR,KAAKkwB,YAAc3rB,EAAQ4rB,YAAc,KACzCnwB,KAAKowB,6BAA+B,IAAIC,QACxCrwB,KAAKswB,6BAA+B,IAAID,QACxCrwB,KAAKuwB,4BAA8B,GAG9BR,EACJ,UAAY9vB,KAAQ6vB,EACnB9vB,KAAKgwB,OAAOpwB,KAAMK,GAClBD,KAAKiwB,SAASxe,IAAKzR,KAAKwwB,uBAAwBvwB,GAAQA,EAG3D,CAKA,UAAW3H,GACV,OAAO0H,KAAKgwB,OAAO13B,MACpB,CAKA,SAAWm4B,GACV,OAAOzwB,KAAKgwB,OAAQ,IAAO,IAC5B,CAKA,QAAWU,GACV,OAAO1wB,KAAKgwB,OAAQhwB,KAAK1H,OAAS,IAAO,IAC1C,CAaO,GAAAuV,CAAK5N,EAAS2E,GACpB,OAAO5E,KAAK2wB,QAAS,CAAE1wB,GAAQ2E,EAChC,CAYO,OAAA+rB,CAASC,EAAoBhsB,GACnC,QAAe,IAAVA,EACJA,EAAQ5E,KAAKgwB,OAAO13B,YACd,GAAKsM,EAAQ5E,KAAKgwB,OAAO13B,QAAUsM,EAAQ,EAOjD,MAAM,IAAI,EAAe,oCAAqC5E,MAG/D,IAAI4b,EAAS,EAEb,UAAY3b,KAAQ2wB,EAAQ,CAC3B,MAAMC,EAAS7wB,KAAKwwB,uBAAwBvwB,GACtC6wB,EAAmBlsB,EAAQgX,EAEjC5b,KAAKgwB,OAAOxoB,OAAQspB,EAAkB,EAAG7wB,GACzCD,KAAKiwB,SAASxe,IAAKof,EAAQ5wB,GAE3BD,KAAKwQ,KAA6B,MAAOvQ,EAAM6wB,GAE/ClV,GACD,CAQA,OANA5b,KAAKwQ,KAAgC,SAAU,CAC9CugB,MAAOH,EACPI,QAAS,GACTpsB,UAGM5E,IACR,CAQO,GAAAoI,CAAK6oB,GACX,IAAIhxB,EAEJ,GAAyB,iBAAbgxB,EACXhxB,EAAOD,KAAKiwB,SAAS7nB,IAAK6oB,OACpB,IAAyB,iBAAbA,EAQlB,MAAM,IAAI,EAAe,6BAA8BjxB,MAPvDC,EAAOD,KAAKgwB,OAAQiB,EAQrB,CAEA,OAAOhxB,GAAQ,IAChB,CAQO,GAAA2N,CAAKsjB,GACX,GAAwB,iBAAZA,EACX,OAAOlxB,KAAKiwB,SAASriB,IAAKsjB,GACpB,CACN,MACMrxB,EAAKqxB,EADQlxB,KAAKkwB,aAGxB,OAAOrwB,GAAMG,KAAKiwB,SAASriB,IAAK/N,EACjC,CACD,CASO,QAAAsxB,CAAUD,GAChB,IAAIjxB,EAQJ,OALCA,EADuB,iBAAZixB,EACJlxB,KAAKiwB,SAAS7nB,IAAK8oB,GAEnBA,EAGDjxB,EAAOD,KAAKgwB,OAAOhnB,QAAS/I,IAAU,CAC9C,CAUO,MAAAgG,CAAQmrB,GACd,MAAQnxB,EAAM2E,GAAU5E,KAAKqxB,QAASD,GAQtC,OANApxB,KAAKwQ,KAAgC,SAAU,CAC9CugB,MAAO,GACPC,QAAS,CAAE/wB,GACX2E,UAGM3E,CACR,CAUO,GAAAjD,CACNuS,EACA+hB,GAEA,OAAOtxB,KAAKgwB,OAAOhzB,IAAKuS,EAAU+hB,EACnC,CASO,IAAAC,CACNhiB,EACA+hB,GAEA,OAAOtxB,KAAKgwB,OAAOuB,KAAMhiB,EAAU+hB,EACpC,CASO,MAAAvrB,CACNwJ,EACA+hB,GAEA,OAAOtxB,KAAKgwB,OAAOjqB,OAAQwJ,EAAU+hB,EACtC,CASO,KAAA1f,GACD5R,KAAKwxB,oBACTxxB,KAAK2P,cAAe3P,KAAKwxB,mBACzBxxB,KAAKwxB,kBAAoB,MAG1B,MAAMC,EAAe9wB,MAAMrB,KAAMU,KAAKgwB,QAEtC,KAAQhwB,KAAK1H,QACZ0H,KAAKqxB,QAAS,GAGfrxB,KAAKwQ,KAAgC,SAAU,CAC9CugB,MAAO,GACPC,QAASS,EACT7sB,MAAO,GAET,CAiHO,MAAAsP,CACNwd,GAEA,GAAK1xB,KAAKwxB,kBAMT,MAAM,IAAI,EAAe,4BAA6BxxB,MAKvD,OAFAA,KAAKwxB,kBAAoBE,EAElB,CACNC,GAAIC,IACH5xB,KAAK6xB,qBAAwB5xB,GAAQ,IAAI2xB,EAAO3xB,IAAQ,EAGzD6xB,MAAOC,IAC4B,mBAAtBA,EACX/xB,KAAK6xB,oBAAwBE,GAE7B/xB,KAAK6xB,qBAAwB5xB,GAAQA,EAAM8xB,IAC5C,EAGH,CAOQ,mBAAAF,CAAwBt7B,GAC/B,MAAMm7B,EAAqB1xB,KAAKwxB,kBAG1BQ,EAAU,CAAE3c,EAAc4c,EAAiBrtB,KAChD,MAAMstB,EAAwBR,EAAmBF,mBAAqBxxB,KAChEmyB,EAAoBT,EAAmBpB,6BAA6BloB,IAAK6pB,GAM/E,GAAKC,GAAyBC,EAC7BnyB,KAAKowB,6BAA6B3e,IAAKwgB,EAAcE,GACrDnyB,KAAKswB,6BAA6B7e,IAAK0gB,EAAmBF,OACpD,CACN,MAAMhyB,EAAO1J,EAAS07B,GAGtB,IAAMhyB,EAGL,YAFAD,KAAKuwB,4BAA4B3wB,KAAMgF,GAOxC,IAAIwtB,EAAaxtB,EAmBjB,UAAYytB,KAAWryB,KAAKuwB,4BACtB3rB,EAAQytB,GACZD,IAiBF,UAAYC,KAAWX,EAAmBnB,4BACpC6B,GAAcC,GAClBD,IAIFpyB,KAAKowB,6BAA6B3e,IAAKwgB,EAAchyB,GACrDD,KAAKswB,6BAA6B7e,IAAKxR,EAAMgyB,GAC7CjyB,KAAK6N,IAAK5N,EAAMmyB,GAIhB,QAAU53B,EAAI,EAAGA,EAAIk3B,EAAmBnB,4BAA4Bj4B,OAAQkC,IACtE43B,GAAcV,EAAmBnB,4BAA6B/1B,IAClEk3B,EAAmBnB,4BAA6B/1B,IAGnD,GAID,UAAYy3B,KAAgBP,EAC3BM,EAAS,EAAMC,EAAcP,EAAmBP,SAAUc,IAI3DjyB,KAAKwP,SAAiCkiB,EAAoB,MAAOM,GAGjEhyB,KAAKwP,SAAoCkiB,EAAoB,UAAU,CAAErc,EAAK4c,EAAcrtB,KAC3F,MAAM3E,EAAOD,KAAKowB,6BAA6BhoB,IAAK6pB,GAE/ChyB,GACJD,KAAKiG,OAAQhG,GAKdD,KAAKuwB,4BAA8BvwB,KAAKuwB,4BAA4BtC,QAAQ,CAAE9vB,EAAQk0B,KAChFztB,EAAQytB,GACZl0B,EAAOyB,KAAMyyB,EAAU,GAGnBztB,EAAQytB,GACZl0B,EAAOyB,KAAMyyB,GAGPl0B,IACL,GAAqB,GAE1B,CASQ,sBAAAqyB,CAAwBvwB,GAC/B,MAAMkwB,EAAanwB,KAAKkwB,YACxB,IAAIW,EAEJ,GAAOV,KAAclwB,EAAS,CAG7B,GAFA4wB,EAAS5wB,EAAMkwB,GAEO,iBAAVU,EAMX,MAAM,IAAI,EAAe,4BAA6B7wB,MAGvD,GAAKA,KAAKoI,IAAKyoB,GAMd,MAAM,IAAI,EAAe,qCAAsC7wB,KAEjE,MACCC,EAAMkwB,GAAeU,EAAS,IAG/B,OAAOA,CACR,CAWQ,OAAAQ,CAASD,GAChB,IAAIxsB,EAAe/E,EAAYI,EAC3BqyB,GAAmB,EACvB,MAAMnC,EAAanwB,KAAKkwB,YAyBxB,GAvBuB,iBAAXkB,GACXvxB,EAAKuxB,EACLnxB,EAAOD,KAAKiwB,SAAS7nB,IAAKvI,GAC1ByyB,GAAoBryB,EAEfA,IACJ2E,EAAQ5E,KAAKgwB,OAAOhnB,QAAS/I,KAED,iBAAXmxB,GAClBxsB,EAAQwsB,EACRnxB,EAAOD,KAAKgwB,OAAQprB,GACpB0tB,GAAoBryB,EAEfA,IACJJ,EAAKI,EAAMkwB,MAGZlwB,EAAOmxB,EACPvxB,EAAKI,EAAMkwB,GACXvrB,EAAQ5E,KAAKgwB,OAAOhnB,QAAS/I,GAC7BqyB,GAA+B,GAAV1tB,IAAgB5E,KAAKiwB,SAAS7nB,IAAKvI,IAGpDyyB,EAMJ,MAAM,IAAI,EAAe,wBAAyBtyB,MAGnDA,KAAKgwB,OAAOxoB,OAAQ5C,EAAQ,GAC5B5E,KAAKiwB,SAASte,OAAQ9R,GAEtB,MAAMoyB,EAAejyB,KAAKswB,6BAA6BloB,IAAKnI,GAM5D,OALAD,KAAKswB,6BAA6B3e,OAAQ1R,GAC1CD,KAAKowB,6BAA6Bze,OAAQsgB,GAE1CjyB,KAAKwQ,KAAgC,SAAUvQ,EAAM2E,GAE9C,CAAE3E,EAAM2E,EAChB,CAKA,CAAS9D,OAAOC,YACf,OAAOf,KAAKgwB,OAAQlvB,OAAOC,WAC5B,ECjuBc,SAAS,GAAUA,GACjC,MAAMwxB,EAAexxB,EAASO,OAE9B,OAAKixB,EAAahxB,KACV,KAGDgxB,EAAa/5B,KACrB,CCOe,MAAM,WAAqB,GAAiB2a,MAiC1D,WAAApR,GACCyL,QARD,KAAOglB,UAA0B,IAAI5e,IAKrC,KAAQ6e,sBAA8D,KAKrEzyB,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,iBAAkB,KAC7B,CAKO,GAAA5D,CAAK4J,GACX,GAAKzX,KAAKwyB,UAAU5kB,IAAK6J,GAMxB,MAAM,IAAI,EAAe,yCAA0CzX,MAGpEA,KAAKwP,SAAUiI,EAAS,SAAS,IAAMzX,KAAK0yB,OAAQjb,IAAW,CAAEiI,YAAY,IAC7E1f,KAAKwP,SAAUiI,EAAS,QAAQ,IAAMzX,KAAK2yB,SAAS,CAAEjT,YAAY,IAClE1f,KAAKwyB,UAAU3kB,IAAK4J,EACrB,CAKO,MAAAxR,CAAQwR,GACTA,IAAYzX,KAAK4yB,gBACrB5yB,KAAK2yB,QAGD3yB,KAAKwyB,UAAU5kB,IAAK6J,KACxBzX,KAAK2P,cAAe8H,GACpBzX,KAAKwyB,UAAU7gB,OAAQ8F,GAEzB,CAOO,OAAAuO,GACNhmB,KAAK2P,eACN,CAKQ,MAAA+iB,CAAQjb,GACfob,aAAc7yB,KAAKyyB,uBAEnBzyB,KAAK4yB,eAAiBnb,EACtBzX,KAAK8yB,WAAY,CAClB,CAMQ,KAAAH,GACPE,aAAc7yB,KAAKyyB,uBAEnBzyB,KAAKyyB,sBAAwBM,YAAY,KACxC/yB,KAAK4yB,eAAiB,KACtB5yB,KAAK8yB,WAAY,CAAK,GACpB,EACJ,EChFc,MAAM,GASpB,WAAA/wB,GACC/B,KAAKgzB,UAAY,IAAM,KACxB,CAKO,QAAAxjB,CAAUI,GAUhB5P,KAAKgzB,UAAUxjB,SAAUI,EAAiC,WAAW,CAAEyF,EAAK4d,KAC3EjzB,KAAKgzB,UAAUxiB,KAAM,YAAc,GAASyiB,GAAcA,EAAY,GAExE,CAeO,GAAAxhB,CACNmc,EACAre,EACAhL,EAAkD,CAAC,GAEnD,MAAM+oB,EAAUK,GAAgBC,GAC1BhhB,EAAWrI,EAAQqI,SAIzB5M,KAAKgzB,UAAUxjB,SAAUxP,KAAKgzB,UAAW,YAAc1F,GAAS,CAAEjY,EAAK4d,KACtE1jB,EAAU0jB,GAAY,KAGrBA,EAAWC,iBACXD,EAAWE,kBAIX9d,EAAIlJ,MAAM,IAIXkJ,EAAIlE,QAAS,CAAI,GACf,CAAEvE,YACN,CAQO,KAAAwmB,CAAOH,GACb,QAASjzB,KAAKgzB,UAAUxiB,KAAM,YAAc,GAASyiB,GAAcA,EACpE,CAKO,aAAAtjB,CAAeC,GACrB5P,KAAKgzB,UAAUrjB,cAAeC,EAC/B,CAKO,OAAAoW,GACNhmB,KAAK2P,eACN,ECtHc,SAAS,GAAU7M,GAGjC,OAAK6T,EAAY7T,GACT,IAAIyO,IAAKzO,GCNH,SAAyB+B,GACvC,MAAM7H,EAAM,IAAIuU,IAEhB,UAAYxa,KAAO8N,EAClB7H,EAAIyU,IAAK1a,EAAK8N,EAAK9N,IAGpB,OAAOiG,CACR,CDASq2B,CAAavwB,EAEtB,CEde,SAAS,GAAiD8U,EAAS0b,GACjF,IAAIC,EAEJ,SAASC,KAAYt3B,GACpBs3B,EAAQC,SACRF,EAAQR,YAAY,IAAMnb,KAAS1b,IAAQo3B,EAC5C,CAMA,OAJAE,EAAQC,OAAS,KAChBZ,aAAcU,EAAO,EAGfC,CACR,CCoBO,SAASE,GAAuBj3B,EAAgBmf,GACtD,SAvBoC+X,EAuBRl3B,EAAO4wB,OAAQzR,EAAS,KAtBV,GAApB+X,EAAUr7B,QAAe,kBAAkB2J,KAAM0xB,IAWjE,SAA6BA,GACnC,QAASA,GAAiC,GAApBA,EAAUr7B,QAAe,kBAAkB2J,KAAM0xB,EACxE,CAS8DC,CAAoBn3B,EAAO4wB,OAAQzR,IAvB1F,IAA8B+X,CAwBrC,CAQO,SAASE,GAAwBp3B,EAAgBmf,GACvD,SA9CgC+X,EA8CRl3B,EAAO4wB,OAAQzR,KA5CG,GAApB+X,EAAUr7B,QAAe,sEAAsE2J,KAAM0xB,GAFrH,IAA0BA,CA+CjC,CAEA,MAAMG,GAcN,WACC,MAAMjV,EAAQ,CAEb,WAAC,+CAAyC,KAG1C,WAAC,gCAA2B,KAG5B,WAAC,sBAAkB,KAGnB,WAAC,uEAAkE,MAG9DkV,EAAe,WAAC,6BAA0B,KAAE7wB,OAC5C8wB,EAAQ,MAAQnV,EAAM7hB,KAAK8hB,GAAQA,EAAK5b,SAAShG,KAAM,KAAQ,IAGrE,OAAO,IAAIyM,OAFM,GAAIoqB,KAAkBC,QAAqBA,MAE/B,KAC9B,CAlCsBC,GAQf,SAASC,GAAuBz3B,EAAgBmf,GACtD,MAAMuY,EAAUlZ,OAAQxe,GAAS23B,SAAUN,IAE3C,OAAOnzB,MAAMrB,KAAM60B,GAAUE,MAAMx3B,GAASA,EAAM+H,MAASgX,GAAUA,EAAS/e,EAAM+H,MAAS/H,EAAO,GAAIvE,QACzG,CC3De,MAAM,WAAe6a,KAwCnC,WAAApR,CAAauyB,GACZ9mB,QAND,KAAQ+mB,cAAgB,IAAI3gB,IAQ3B5T,KAAKs0B,OAASA,EAEdt0B,KAAKyR,IAAK,aAAa,EACxB,CA4CO,aAAA+iB,CAAe30B,GACrBG,KAAKu0B,cAAc1mB,IAAKhO,GAEQ,GAA3BG,KAAKu0B,cAAc1gB,OACvB7T,KAAKqP,GAAiC,gBAAiBolB,GAAc,CAAE7nB,SAAU,YACjF5M,KAAK00B,WAAY,EAEnB,CAOO,kBAAAC,CAAoB90B,GAC1BG,KAAKu0B,cAAc5iB,OAAQ9R,GAEK,GAA3BG,KAAKu0B,cAAc1gB,OACvB7T,KAAKoM,IAAK,gBAAiBqoB,IAC3Bz0B,KAAK00B,WAAY,EAEnB,CAKO,OAAA1O,GACNhmB,KAAK2P,eACN,CAKA,0BAAkBilB,GACjB,OAAO,CACR,EAyLD,SAASH,GAAcpf,GACtBA,EAAIlE,QAAS,EACbkE,EAAIlJ,MACL,CC3Se,MAAM,WAAgBgH,KA+DpC,WAAApR,CAAauyB,GACZ9mB,QAEAxN,KAAKs0B,OAASA,EACdt0B,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,aAAa,GAEvBzR,KAAK60B,cAAe,EACpB70B,KAAK80B,4BAA6B,EAClC90B,KAAKu0B,cAAgB,IAAI3gB,IAEzB5T,KAAKiV,SAAU,WAGfjV,KAAKwP,SAAUxP,KAAKs0B,OAAOl8B,MAAMoL,SAAU,UAAU,KACpDxD,KAAK+0B,SAAS,IAGf/0B,KAAKwP,SAA0C8kB,EAAQ,qBAAqB,KAC3Et0B,KAAK+0B,SAAS,IAIf/0B,KAAKqP,GAAiC,iBAAiBgG,IACtD,IAAMrV,KAAKg1B,YACV,OAGD,MAAMC,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAElCC,IADsE,cAA/CD,EAAUE,mBAAoB7+B,KAAK8+B,WACZd,EAAOl8B,MAAMi9B,UAAWJ,IAMvEX,EAAOgB,YAAct1B,KAAK80B,6BAA+BI,KAC7D7f,EAAIlE,QAAS,EACbkE,EAAIlJ,OACL,GACE,CAAES,SAAU,YAEf5M,KAAKqP,GAAyB,WAAWgG,IAClCrV,KAAK00B,WACVrf,EAAIlJ,MACL,GACE,CAAES,SAAU,QAChB,CAaA,eAAWooB,GACV,OAAOh1B,KAAK60B,YACb,CAEA,eAAcG,CAAaA,GAC1Bh1B,KAAK60B,aAAeG,CACrB,CASO,OAAAD,GACN/0B,KAAK00B,WAAY,CAClB,CA4CO,aAAAF,CAAe30B,GACrBG,KAAKu0B,cAAc1mB,IAAKhO,GAEQ,GAA3BG,KAAKu0B,cAAc1gB,OACvB7T,KAAKqP,GAAiC,gBAAiB,GAAc,CAAEzC,SAAU,YACjF5M,KAAK00B,WAAY,EAEnB,CAOO,kBAAAC,CAAoB90B,GAC1BG,KAAKu0B,cAAc5iB,OAAQ9R,GAEK,GAA3BG,KAAKu0B,cAAc1gB,OACvB7T,KAAKoM,IAAK,gBAAiB,IAC3BpM,KAAK+0B,UAEP,CAkBO,OAAAQ,IAAYr5B,GAAoD,CAKhE,OAAA8pB,GACNhmB,KAAK2P,eACN,EAMD,SAAS,GAAc0F,GACtBA,EAAIlE,QAAS,EACbkE,EAAIlJ,MACL,CCrOe,MAAMqpB,WAAqB,GAA1C,kCAIC,KAAQC,0BAAmF,EAAC,CAK5E,OAAAV,GAEhB,CAOgB,OAAAQ,IAAYr5B,GAC3B,MAAMw5B,EAAU11B,KAAK21B,0BAErB,QAASD,GAAWA,EAAQH,QAASr5B,EACtC,CAQO,oBAAA05B,CACNF,EACAnxB,EAAyC,CAAC,GAE1C2I,EAAuBlN,KAAKy1B,0BAA2B,CAAEC,UAAS9oB,SAAUrI,EAAQqI,UAAY,WAGhG8oB,EAAQrmB,GAAoC,oBAAoB,IAAMrP,KAAK61B,kBAE3E71B,KAAK61B,eACN,CAKQ,aAAAA,GACP71B,KAAK00B,YAAc10B,KAAK21B,yBACzB,CAKQ,uBAAAA,GACP,MAAMG,EAAoB91B,KAAKy1B,0BAA0BlE,MAAM,EAAImE,aAAeA,EAAQhB,YAE1F,OAAOoB,GAAqBA,EAAkBJ,OAC/C,ECnFc,MAAMK,WAAkD3mB,KA0BtE,WAAArN,CACCwL,EACAyoB,EAA0D,GAC1DC,EAAkD,IAElDzoB,QA5BD,KAAQ0oB,SAAW,IAAI3kB,IA8BtBvR,KAAKm2B,SAAW5oB,EAChBvN,KAAKo2B,kBAAoB,IAAI7kB,IAE7B,UAAY8kB,KAAqBL,EAC3BK,EAAkBC,YACtBt2B,KAAKo2B,kBAAkB3kB,IAAK4kB,EAAkBC,WAAYD,GAI5Dr2B,KAAKu2B,gBAAkB,IAAIhlB,IAE3B,UAAc8kB,EAAmBG,KAAoBP,EACpDj2B,KAAKu2B,gBAAgB9kB,IAAK4kB,EAAmBG,GAC7Cx2B,KAAKu2B,gBAAgB9kB,IAAK+kB,EAAgBH,GAGrCA,EAAkBC,YACtBt2B,KAAKo2B,kBAAkB3kB,IAAK4kB,EAAkBC,WAAYD,EAG7D,CAOA,EAAUv1B,OAAOC,YAChB,UAAY2X,KAAS1Y,KAAKk2B,SACC,mBAAdxd,EAAO,WACZA,EAGT,CAyBO,GAAAtQ,CAAKrR,GACX,MAAM0/B,EAASz2B,KAAKk2B,SAAS9tB,IAAKrR,GAElC,IAAM0/B,EAAS,CACd,IAAIH,EAAav/B,EAoBjB,KAlBmB,mBAAPA,IACXu/B,EAAav/B,EAAIu/B,YAAcv/B,EAAIiL,MAiB9B,IAAI,EAAe,qCAAsChC,KAAKm2B,SAAU,CAAEM,OAAQH,GACzF,CAEA,OAAOG,CACR,CAiBO,GAAA7oB,CAAK7W,GACX,OAAOiJ,KAAKk2B,SAAStoB,IAAK7W,EAC3B,CAkBO,IAAA2/B,CACNC,EACAC,EAAuE,GACvEC,EAAmE,IAiBnE,MAAMC,EAAO92B,KACPuN,EAAUvN,KAAKm2B,UAuDrB,SAASY,EACRJ,EACAK,EAAY,IAAIpjB,KAEhB+iB,EAAQ/4B,SAAS64B,IACVQ,EAAqBR,KAItBO,EAAUppB,IAAK6oB,KAIpBO,EAAUnpB,IAAK4oB,GAEVA,EAAOH,aAAeQ,EAAKV,kBAAkBxoB,IAAK6oB,EAAOH,aAC7DQ,EAAKV,kBAAkB3kB,IAAKglB,EAAOH,WAAYG,GAG3CA,EAAOS,UACXH,EAAiCN,EAAOS,SAAUF,IACnD,GAEF,CA5EAD,CAAiCJ,GAEjCQ,EAAiBR,GAEjB,MAEMS,EAAqB,IAwE3B,SAASC,EACRV,EACAK,EAAY,IAAIpjB,KAEhB,OAAO+iB,EACL35B,KAAKy5B,GACEQ,EAAqBR,GAC3BA,EACAK,EAAKV,kBAAkBhuB,IAAKquB,KAE7BxI,QAAQ,CAAE9vB,EAAQs4B,IACbO,EAAUppB,IAAK6oB,GACZt4B,GAGR64B,EAAUnpB,IAAK4oB,GAEVA,EAAOS,WACXC,EAAiBV,EAAOS,SAAUT,GAElCY,EAAuBZ,EAAOS,SAAUF,GAAYp5B,SAAS64B,GAAUt4B,EAAO0P,IAAK4oB,MAG7Et4B,EAAO0P,IAAK4oB,KACjB,IAAI7iB,IACT,CAjGgCyjB,CAFVV,EAAQ5wB,QAAQ0wB,IAAWa,EAAiBb,EAAQG,QAgR1E,SACCQ,EACAP,GAEA,UAAYU,KAAcV,EAAuB,CAChD,GAA0B,mBAAdU,EAMX,MAAM,IAAI,EAAe,+CAAgD,KAAM,CAAEA,eAGlF,MAAMjB,EAAaiB,EAAWjB,WAE9B,IAAMA,EAML,MAAM,IAAI,EAAe,+CAAgD,KAAM,CAAEiB,eAGlF,GAAKA,EAAWL,UAAYK,EAAWL,SAAS5+B,OAM/C,MAAM,IAAI,EAAe,iEAAkE,KAAM,CAAEg+B,eAGpG,MAAMkB,EAAkBV,EAAKV,kBAAkBhuB,IAAKkuB,GAEpD,IAAMkB,EAOL,MAAM,IAAI,EAAe,kDAAmD,KAAM,CAAElB,eAGrF,MAAMmB,EAA4BL,EAAmBpuB,QAASwuB,GAE9D,IAAoC,IAA/BC,EAAmC,CAIvC,GAAKX,EAAKP,gBAAgB3oB,IAAK4pB,GAC9B,OAQD,MAAM,IAAI,EAAe,mDAAoD,KAAM,CAAElB,cACtF,CAEA,GAAKkB,EAAgBN,UAAYM,EAAgBN,SAAS5+B,OAMzD,MAAM,IAAI,EAAe,4DAA6D,KAAM,CAAEg+B,eAG/Fc,EAAmB5vB,OAAQiwB,EAA2B,EAAGF,GACzDT,EAAKV,kBAAkB3kB,IAAK6kB,EAAYiB,EACzC,CACD,CAxVAG,CAAmBN,EAAoBP,GAEvC,MAAMc,EAA+BP,EA8OVp6B,KAAKq5B,IAC9B,IAAIG,EAAiBM,EAAKP,gBAAgBnuB,IAAKiuB,GAM/C,OAJAG,EAAiBA,GAAkB,IAAMH,EAAyD9oB,GAElGupB,EAAKc,KAAMvB,EAAmBG,GAEvBA,CAAc,IAnPvB,OAAOqB,EAAaF,EAAiB,QACnCG,MAAM,IAAMD,EAAaF,EAAiB,eAC1CG,MAAM,IAAMH,IAEd,SAASV,EAAqBR,GAC7B,MAAyB,mBAAXA,CACf,CAEA,SAAS7B,EACR6B,GAEA,OAAOQ,EAAqBR,MAAcA,EAAO7B,eAClD,CAEA,SAAS0C,EACRb,EACAG,GAEA,OAAOA,EAAgBvC,MAAM0D,GACvBA,IAAkBtB,IAIlBuB,EAAevB,KAAasB,GAI5BC,EAAeD,KAAoBtB,IAM1C,CAEA,SAASuB,EAAevB,GACvB,OAAOQ,EAAqBR,GAC3BA,EAAOH,YAAcG,EAAOz0B,KAC5By0B,CACF,CAsDA,SAASU,EACRR,EACAsB,EAA8D,MAE9DtB,EACE35B,KAAKy5B,GACEQ,EAAqBR,GAC3BA,EACAK,EAAKV,kBAAkBhuB,IAAKquB,IAAYA,IAEzC74B,SAAS64B,KAOZ,SACCA,EACAwB,GAEA,GAAKhB,EAAqBR,GACzB,OAGD,GAAKwB,EAwBJ,MAAM,IAAI,EACT,iCACA1qB,EACA,CAAE2qB,cAAezB,EAAQ0B,WAAYH,EAAeC,KA0BtD,MAAM,IAAI,EACT,oCACA1qB,EACA,CAAEkpB,UAEJ,CAxEG2B,CAAoB3B,EAAQwB,GA0E/B,SACCxB,EACAwB,GAEA,IAAMrD,EAAiBqD,GACtB,OAGD,GAAKrD,EAAiB6B,GACrB,OAeD,MAAM,IAAI,EACT,oCACAlpB,EACA,CAAEkpB,OAAQuB,EAAevB,GAAU0B,WAAYH,EAAeC,IAEhE,CAtGGI,CAAoB5B,EAAQwB,GAwG/B,SACCxB,EACAwB,GAEA,IAAMA,EACL,OAGD,IAAMX,EAAiBb,EAAQG,GAC9B,OAUD,MAAM,IAAI,EACT,4BACArpB,EACA,CAAEkpB,OAAQuB,EAAevB,GAAU0B,WAAYH,EAAeC,IAEhE,CA/HGK,CAAoB7B,EAAQwB,EAAyB,GAExD,CA2IA,SAASJ,EAAaF,EAAiDY,GACtE,OAAOZ,EAAgB1J,QAA0B,CAAEuK,EAAS/B,IACrDA,EAAQ8B,GAITzB,EAAKP,gBAAgB3oB,IAAK6oB,GACvB+B,EAGDA,EAAQV,KAAMrB,EAAQ8B,GAAUvxB,KAAMyvB,IAPrC+B,GAQNC,QAAQlc,UACZ,CAkFD,CAKO,OAAAyJ,GACN,MAAM0S,EAA2B,GAEjC,UAAc,CAAElC,KAAoBx2B,KACG,mBAA1Bw2B,EAAexQ,SAA0BhmB,KAAKu2B,gBAAgB3oB,IAAK4oB,IAC9EkC,EAAS94B,KAAM42B,EAAexQ,WAIhC,OAAOyS,QAAQh1B,IAAKi1B,EACrB,CAQQ,IAAAd,CAAMvB,EAAgDI,GAC7Dz2B,KAAKk2B,SAASzkB,IAAK4kB,EAAmBI,GAEtC,MAAMH,EAAaD,EAAkBC,WAErC,GAAMA,EAAN,CAIA,GAAKt2B,KAAKk2B,SAAStoB,IAAK0oB,GA+BvB,MAAM,IAAI,EACT,wCACA,KACA,CAAEA,aAAYqC,QAAS34B,KAAKk2B,SAAS9tB,IAAKkuB,GAAcv0B,YAAa62B,QAASvC,IAIhFr2B,KAAKk2B,SAASzkB,IAAK6kB,EAAYG,EAxC/B,CAyCD,ECzjBc,MAAMoC,GAkGpB,WAAA92B,CAAainB,GATb,KAAQ8P,cAA+B,KAUtC94B,KAAKgpB,OAAS,IAAI7K,GAAuB6K,EAAUhpB,KAAK+B,YAAgCg3B,eAExF,MAAM/C,EAAqBh2B,KAAK+B,YAAgCi3B,eAEhEh5B,KAAKgpB,OAAOtyB,OAAQ,UAAWs/B,GAE/Bh2B,KAAK22B,QAAU,IAAIZ,GAAoC/1B,KAAMg2B,GAE7D,MAAMiD,EAAiBj5B,KAAKgpB,OAAO5gB,IAAK,aAAgB,CAAC,EAEzDpI,KAAKk5B,OAAS,IAAI1J,GAAQ,CACzBC,WAAsC,iBAAnBwJ,EAA8BA,EAAiBA,EAAeE,GACjFzJ,gBAAiB1vB,KAAKgpB,OAAO5gB,IAAK,sBAGnCpI,KAAKhF,EAAIgF,KAAKk5B,OAAOl+B,EAErBgF,KAAKo5B,QAAU,IAAI,EACpB,CAOO,WAAAvB,GACN,MAAMlB,EAAU32B,KAAKgpB,OAAO5gB,IAAK,YAAe,GAC1CsvB,EAAoB13B,KAAKgpB,OAAO5gB,IAAK,sBAAyB,GAGpE,UAAYixB,KAAU1C,EAAQx2B,OAAQu3B,GAAsB,CAC3D,GAAsB,mBAAV2B,EAMX,MAAM,IAAI,EACT,uCACA,KACA,CAAEA,WAIJ,IAAgC,IAA3BA,EAAOzE,gBAOX,MAAM,IAAI,EACT,qCACA,KACA,CAAEyE,UAGL,CAEA,OAAOr5B,KAAK22B,QAAQD,KAAMC,EAAS,GAAIe,EACxC,CAQO,OAAA1R,GACN,OAAOyS,QAAQh1B,IAAK9C,MAAMrB,KAAMU,KAAKo5B,SAAS9E,GAAUA,EAAOtO,aAC7D8R,MAAM,IAAM93B,KAAK22B,QAAQ3Q,WAC5B,CAaO,UAAAsT,CAAYhF,EAAgBiF,GAClC,GAAKv5B,KAAK84B,cAMT,MAAM,IAAI,EAAe,qCAG1B94B,KAAKo5B,QAAQvrB,IAAKymB,GAEbiF,IACJv5B,KAAK84B,cAAgBxE,EAEvB,CAWO,aAAAkF,CAAelF,GAKrB,OAJKt0B,KAAKo5B,QAAQxrB,IAAK0mB,IACtBt0B,KAAKo5B,QAAQnzB,OAAQquB,GAGjBt0B,KAAK84B,gBAAkBxE,EACpBt0B,KAAKgmB,UAGNyS,QAAQlc,SAChB,CAaO,gBAAAkd,GACN,MAAMt7B,EAAkC,CAAC,EAEzC,UAAY6D,KAAQhC,KAAKgpB,OAAOrK,QACzB,CAAE,UAAW,gBAAiB,gBAAiB4Q,SAAUvtB,KAC9D7D,EAAQ6D,GAAShC,KAAKgpB,OAAO5gB,IAAKpG,IAIpC,OAAO7D,CACR,CAqDA,aAAcmf,CAAQ0L,GACrB,OAAO,IAAIyP,SAASlc,IACnB,MAAMhP,EAAU,IAAIvN,KAAMgpB,GAE1BzM,EAAShP,EAAQsqB,cAAcC,MAAM,IAAMvqB,IAAW,GAExD,ECzTc,MAAMmsB,WAAsBvmB,KAS1C,WAAApR,CAAawL,GACZC,QAEAxN,KAAKuN,QAAUA,CAChB,CAKO,OAAAyY,GACNhmB,KAAK2P,eACN,CAKA,0BAAkBilB,GACjB,OAAO,CACR,ECtCc,MAAM+E,WAAgC,GASpD,WAAA53B,CAAauyB,GACZ9mB,QACAxN,KAAKs0B,OAASA,CACf,CAmBgB,GAAA7iB,CACfmc,EACAre,EACAhL,EAAkD,CAAC,GAEnD,GAAwB,iBAAZgL,EAAuB,CAClC,MAAMqqB,EAAcrqB,EAEpBA,EAAW,CAAEsqB,EAASpG,KACrBzzB,KAAKs0B,OAAOiB,QAASqE,GACrBnG,GAAQ,CAEV,CAEAjmB,MAAMiE,IAAKmc,EAAWre,EAAUhL,EACjC,E,qCCrEGA,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzEA,OAAiB,OACjBA,WAAoB,GAEP,KAAI,KAASA,IAIX,KAAQu1B,OAAvB,MCQMC,GAAuB,IAAI1J,QAEjC,IAAI2J,IAA4C,EAqBzC,SAAS,IAAmB,KAAEC,EAAI,QAAExiB,EAAO,KAAEyiB,EAAI,aAAEC,GAAe,EAAI,YAAEC,GAAc,IAO5F,MAAM/iB,EAAM4iB,EAAKz2B,SAgCjB,SAAS62B,EAAgBH,GAExBH,GAAqB3xB,IAAKiP,GAAO5F,IAAKgG,EAAS,CAC9CyiB,KAAAA,EACAC,eACAC,cACAE,YAAaH,EAAe1iB,EAAU,OAIvCwiB,EAAKM,QAAQC,GAAUC,GAA4BpjB,EAAKmjB,IACzD,CAxCMT,GAAqBnsB,IAAKyJ,KAC/B0iB,GAAqBtoB,IAAK4F,EAAK,IAAI9F,KAInC8F,EAAIqjB,mBAAmBF,GAAUC,GAA4BpjB,EAAKmjB,KAGlEnjB,EAAIhI,GAA2B,sBAAsB,KACpD4qB,EAAKM,QAAQC,GAAUC,GAA4BpjB,EAAKmjB,IAAU,GAChE,CAAE5tB,SAAU,UAGX6K,EAAQvJ,GAAI,oBAChBuJ,EAAQpI,GAAI,sBAAsB,CAAEsrB,EAAStlB,EAAK6kB,KACjDG,EAAgBH,EAAM,IAInBziB,EAAQmjB,YACZP,EAAgB5iB,EAAQmjB,aACbV,GACXG,EAAgBH,GAGZA,GA6ON,WACOF,IAUL,EAAY,4CAGbA,IAA4C,CAC7C,CA3PEa,EAeF,CA8DO,SAAS,GAAiBL,EAAwB/iB,GACxD,QAAKA,EAAQqjB,SAAU,oBACtBN,EAAOO,YAAa,iBAAkBtjB,IAE/B,EAIT,CAuDA,SAASgjB,GAA4BpjB,EAAemjB,GACnD,MAAMQ,EAAejB,GAAqB3xB,IAAKiP,GACzC4jB,EAAqC,GAC3C,IAAIC,GAAkB,EAGtB,UAAczjB,EAASuR,KAAYgS,EAC7BhS,EAAOmR,eACXc,EAAmBr7B,KAAM6X,GAEpB0jB,GAAmBX,EAAQ/iB,EAASuR,KACxCkS,GAAkB,IAMrB,UAAczjB,EAASuR,KAAYgS,EAAe,CACjD,GAAKhS,EAAOmR,aACX,SAGD,MAAMG,EAAcc,GAAmC3jB,GAIjD6iB,IAKDW,EAAmB1L,SAAU+K,KAKlCtR,EAAOsR,YAAcA,EAEhBa,GAAmBX,EAAQ/iB,EAASuR,KACxCkS,GAAkB,IAEpB,CAEA,OAAOA,CACR,CAOA,SAASC,GAAmBX,EAAwB/iB,EAAkBuR,GACrE,MAAM,KAAEkR,EAAI,aAAEC,EAAY,YAAEG,GAAgBtR,EAE5C,IAAIkS,GAAkB,EAGjBZ,EAAae,aAAc,sBAAyBnB,IACxDM,EAAO/0B,aAAc,mBAAoBy0B,EAAMI,GAC/CY,GAAkB,GAcnB,OAVoBf,GAAsC,GAAtB1iB,EAAQ6jB,aAvGtC,SAA2B7jB,EAAkB2iB,GACnD,IAAM3iB,EAAQ8jB,aACb,OAAO,EAOR,GAHmB56B,MAAMrB,KAAMmY,EAAQ+jB,eACrCnH,MAAM5c,IAAYA,EAAQvJ,GAAI,eAG/B,OAAO,EAGR,MAAMmJ,EAAMI,EAAQjU,SAEdi4B,EADgBpkB,EAAI4d,UACYyG,OAEtC,QAAKrkB,EAAIskB,aAAeF,GAAmBA,EAAgB78B,SAAW6Y,IAKjE2iB,GAKC/iB,EAAIyb,aAKD2I,GAAmBA,EAAgB78B,SAAW6Y,GACxD,CAwEqB,CAAkB6iB,EAActR,EAAOoR,aArJrD,SAA0BI,EAAwB/iB,GACxD,OAAMA,EAAQqjB,SAAU,oBACvBN,EAAOoB,SAAU,iBAAkBnkB,IAE5B,EAIT,CA8IO,CAAiB+iB,EAAQF,KAC7BY,GAAkB,GAER,GAAiBV,EAAQF,KACpCY,GAAkB,GAGZA,CACR,CAOA,SAASE,GAAmCx8B,GAC3C,GAAKA,EAAO08B,WAAa,CACxB,MAAM30B,EAAa/H,EAAOi9B,SAAU,GAEpC,GAAKl1B,EAAWuH,GAAI,aAAgBvH,EAAWuH,GAAI,eAAkBvH,EAAWuH,GAAI,oBACnF,OAAOvH,CAET,CAEA,OAAO,IACR,CC5Re,MAAem1B,GAqdtB,EAAA5tB,GAON,MAAM,IAAI7V,MAAO,0BAClB,ECpdD,SAJA,SAAeG,GACb,OAAO,GAAUA,EA7BM,EA8BzB,ECAe,MAAe,WAAa4W,EAAc0sB,KAgB9C,WAAA/5B,CAAayB,GACtBgK,QAEAxN,KAAKwD,SAAWA,EAChBxD,KAAKpB,OAAS,IACf,CAQA,SAAWgG,GACV,IAAIm3B,EAEJ,IAAM/7B,KAAKpB,OACV,OAAO,KAIR,IAAqD,IAA9Cm9B,EAAM/7B,KAAKpB,OAAOo9B,cAAeh8B,OAMvC,MAAM,IAAI,EAAe,gCAAiCA,MAG3D,OAAO+7B,CACR,CAKA,eAAWE,GACV,MAAMr3B,EAAQ5E,KAAK4E,MAEnB,OAAmB,OAAVA,GAAkB5E,KAAKpB,OAAQi9B,SAAUj3B,EAAQ,IAAS,IACpE,CAKA,mBAAWgiB,GACV,MAAMhiB,EAAQ5E,KAAK4E,MAEnB,OAAmB,OAAVA,GAAkB5E,KAAKpB,OAAQi9B,SAAUj3B,EAAQ,IAAS,IACpE,CAKA,QAAWtO,GAEV,IAAIA,EAAgC0J,KAEpC,KAAQ1J,EAAKsI,QACZtI,EAAOA,EAAKsI,OAGb,OAAOtI,CACR,CAKO,UAAAilC,GACN,OAAOv7B,KAAK1J,KAAK4X,GAAI,cACtB,CAmBO,OAAAguB,GACN,MAAMz8B,EAAO,GAEb,IAAIN,EAAgCa,KAEpC,KAAQb,EAAKP,QACZa,EAAKL,QAASD,EAAKyF,OACnBzF,EAAOA,EAAKP,OAGb,OAAOa,CACR,CAWO,YAAA08B,CAAc53B,EAA4D,CAAC,GACjF,MAAM63B,EAA4C,GAClD,IAAIx9B,EAAS2F,EAAQ83B,YAAcr8B,KAAOA,KAAKpB,OAE/C,KAAQA,GACPw9B,EAAW73B,EAAQ+3B,YAAc,OAAS,WAAa19B,GACvDA,EAASA,EAAOA,OAGjB,OAAOw9B,CACR,CAWO,iBAAAG,CAAmBp9B,EAAYoF,EAAqC,CAAC,GAC3E,MAAMi4B,EAAax8B,KAAKm8B,aAAc53B,GAChCk4B,EAAat9B,EAAKg9B,aAAc53B,GAEtC,IAAI/J,EAAI,EAER,KAAQgiC,EAAYhiC,IAAOiiC,EAAYjiC,IAAOgiC,EAAYhiC,IACzDA,IAGD,OAAa,IAANA,EAAU,KAAOgiC,EAAYhiC,EAAI,EACzC,CAQO,QAAAkiC,CAAUv9B,GAEhB,GAAKa,MAAQb,EACZ,OAAO,EAIR,GAAKa,KAAK1J,OAAS6I,EAAK7I,KACvB,OAAO,EAGR,MAAMqmC,EAAW38B,KAAKk8B,UAChBU,EAAWz9B,EAAK+8B,UAEhB/9B,EAASuY,EAAeimB,EAAUC,GAExC,OAASz+B,GACR,IAAK,SACJ,OAAO,EAER,IAAK,YACJ,OAAO,EAER,QACC,OAAOw+B,EAAUx+B,GAAqBy+B,EAAUz+B,GAEnD,CAQO,OAAA0+B,CAAS19B,GAEf,OAAKa,MAAQb,IAKRa,KAAK1J,OAAS6I,EAAK7I,OAKhB0J,KAAK08B,SAAUv9B,GACxB,CAOO,OAAAkyB,GACNrxB,KAAKpB,OAAQk+B,gBAAiB98B,KAAK4E,MACpC,CAQO,WAAAm4B,CAAa9xB,EAAkB9L,GACrCa,KAAKwQ,KAA2B,UAAWvF,IAAS9L,GAE/Ca,KAAKpB,QACToB,KAAKpB,OAAOm+B,YAAa9xB,EAAM9L,EAEjC,CAOO,MAAA69B,GACN,MAAMC,EAAY,GAAOj9B,MAKzB,cAFOi9B,EAAKr+B,OAELq+B,CACR,EAoBD,GAAKp7B,UAAUqM,GAAK,SAAUjD,GAC7B,MAAgB,SAATA,GAA4B,cAATA,CAC3B,EC5Re,MAAMiyB,WAAa,GAgBjC,WAAAn7B,CAAayB,EAAoBV,GAChC0K,MAAOhK,GAEPxD,KAAKm9B,UAAYr6B,CAClB,CAKA,QAAWA,GACV,OAAO9C,KAAKm9B,SACb,CAqBA,SAAWC,GACV,OAAOp9B,KAAK8C,IACb,CAEA,SAAWs6B,CAAOt6B,GACjB9C,KAAK+8B,YAAa,OAAQ/8B,MAE1BA,KAAKm9B,UAAYr6B,CAClB,CAQO,SAAAu6B,CAAWC,GACjB,OAAQA,aAAqBJ,KAItBl9B,OAASs9B,GAAat9B,KAAK8C,OAASw6B,EAAUx6B,KACtD,CAQO,MAAAy6B,GACN,OAAO,IAAIL,GAAMl9B,KAAKwD,SAAUxD,KAAK8C,KACtC,EAiBDo6B,GAAKr7B,UAAUqM,GAAK,SAAUjD,GAC7B,MAAgB,UAATA,GAA6B,eAATA,GAEjB,SAATA,GAA4B,cAATA,GAEV,SAATA,GAA4B,cAATA,CACrB,ECvFe,MAAMuyB,WAAkB1B,GA0BtC,WAAA/5B,CAAa07B,EAAgBC,EAAsBplC,GAKlD,GAJAkV,QAEAxN,KAAKy9B,SAAWA,EAEXC,EAAe,GAAKA,EAAeD,EAAS36B,KAAKxK,OAMrD,MAAM,IAAI,EAAe,oCAAqC0H,MAG/D,GAAK1H,EAAS,GAAKolC,EAAeplC,EAASmlC,EAAS36B,KAAKxK,OAMxD,MAAM,IAAI,EAAe,8BAA+B0H,MAGzDA,KAAK8C,KAAO26B,EAAS36B,KAAKlG,UAAW8gC,EAAcA,EAAeplC,GAClE0H,KAAK09B,aAAeA,CACrB,CAKA,cAAWC,GACV,OAAO39B,KAAK8C,KAAKxK,MAClB,CAUA,aAAWslC,GACV,OAAO59B,KAAK8C,KAAKxK,SAAW0H,KAAKy9B,SAAS36B,KAAKxK,MAChD,CAKA,UAAWsG,GACV,OAAOoB,KAAKy9B,SAAS7+B,MACtB,CAKA,QAAWtI,GACV,OAAO0J,KAAKy9B,SAASnnC,IACtB,CAMA,YAAWkN,GACV,OAAOxD,KAAKy9B,SAASj6B,QACtB,CAWO,YAAA24B,CAAc53B,EAGjB,CAAC,GACJ,MAAM63B,EAAsD,GAC5D,IAAIx9B,EAAmD2F,EAAQ83B,YAAcr8B,KAAKy9B,SAAWz9B,KAAKpB,OAElG,KAAmB,OAAXA,GACPw9B,EAAW73B,EAAQ+3B,YAAc,OAAS,WAAa19B,GACvDA,EAASA,EAAOA,OAGjB,OAAOw9B,CACR,EAiBDoB,GAAU37B,UAAUqM,GAAK,SAAUjD,GAClC,MAAgB,eAATA,GAAkC,oBAATA,GAEtB,cAATA,GAAiC,mBAATA,CAC1B,EC5Je,MAAM,GAQpB,WAAAlJ,IAAgB87B,GAPhB,KAAiBC,UAAkE,GAQlF99B,KAAK6N,OAAQgwB,EACd,CAgCO,GAAAhwB,IAAQgwB,GACd,QAAU59B,KAAQ49B,GAEG,iBAAR59B,GAAoBA,aAAgB0J,UAC/C1J,EAAO,CAAE+B,KAAM/B,IAGhBD,KAAK89B,UAAUl+B,KAAMK,EAEvB,CAyBO,KAAApD,IAAU4a,GAChB,UAAYsmB,KAAiBtmB,EAC5B,UAAYomB,KAAW79B,KAAK89B,UAAY,CACvC,MAAMjhC,EAAQmhC,GAAmBD,EAAeF,GAEhD,GAAKhhC,EACJ,MAAO,CACN4a,QAASsmB,EACTF,UACAhhC,QAGH,CAGD,OAAO,IACR,CAYO,QAAAu3B,IAAa3c,GACnB,MAAMwmB,EAA8B,GAEpC,UAAYF,KAAiBtmB,EAC5B,UAAYomB,KAAW79B,KAAK89B,UAAY,CACvC,MAAMjhC,EAAQmhC,GAAmBD,EAAeF,GAE3ChhC,GACJohC,EAAQr+B,KAAM,CACb6X,QAASsmB,EACTF,UACAhhC,SAGH,CAGD,OAAOohC,EAAQ3lC,OAAS,EAAI2lC,EAAU,IACvC,CAQO,cAAAC,GACN,GAA+B,IAA1Bl+B,KAAK89B,UAAUxlC,OACnB,OAAO,KAGR,MAAMulC,EAAU79B,KAAK89B,UAAW,GAC1B97B,EAAO67B,EAAQ77B,KAErB,MAA2B,mBAAX67B,IAAyB77B,GAAWA,aAAgB2H,OAAoB,KAAP3H,CAClF,EASD,SAASg8B,GAAmBvmB,EAAkBomB,GAE7C,GAAuB,mBAAXA,EACX,OAAOA,EAASpmB,GAGjB,MAAM5a,EAAe,CAAC,EAGtB,OAAKghC,EAAQ77B,OACZnF,EAAMmF,KA0CR,SAAoB67B,EAA0B77B,GAE7C,GAAK67B,aAAmBl0B,OACvB,QAAS3H,EAAKnF,MAAOghC,GAGtB,OAAOA,IAAY77B,CACpB,CAjDem8B,CAAWN,EAAQ77B,KAAMyV,EAAQzV,OAExCnF,EAAMmF,OAMR67B,EAAQt4B,aACZ1I,EAAM0I,WA6QR,SACC64B,EACA3mB,GAEA,MAAM4mB,EAAgB,IAAIzqB,IAAK6D,EAAQ6mB,oBAIlC,GAAeF,SACgB,IAA5BA,EAAkB/4B,OAExB,EAAY,kDAAmD+4B,QAE7B,IAA5BA,EAAkBG,OAExB,EAAY,kDAAmDH,KAGhEC,EAAc1sB,OAAQ,SACtB0sB,EAAc1sB,OAAQ,UAGvB,OAAO6sB,GAAeJ,EAAUC,GAAetnC,GAAO0gB,EAAQ4jB,aAActkC,IAC7E,CApSqB0nC,CAAiBZ,EAAQt4B,WAAYkS,IAElD5a,EAAM0I,aAMRs4B,EAAQa,UACZ7hC,EAAM6hC,QAoSR,SAAuBN,EAAyB3mB,GAE/C,OAAO+mB,GAAeJ,EAAU3mB,EAAQknB,iBAAyD,QAClG,CAvSkBC,CAAcf,EAAQa,QAASjnB,IAEzC5a,EAAM6hC,UAMRb,EAAQgB,SACZhiC,EAAMgiC,OAwSR,SAAsBT,EAAyB3mB,GAC9C,OAAO+mB,GAAeJ,EAAU3mB,EAAQqnB,eAAe,IAAQ/nC,GAAO0gB,EAAQsnB,SAAUhoC,IACzF,CA1SiBioC,CAAanB,EAAQgB,OAAQpnB,IAEtC5a,EAAMgiC,QA1BJ,KA+BFhiC,CACR,CAkFA,SAAS2hC,GACRJ,EACAnnC,EACAgoC,GAEA,MAAMC,EA0FP,SAA4Bd,GAC3B,GAAKz9B,MAAMC,QAASw9B,GACnB,OAAOA,EAASphC,KAAO6gC,GACjB,GAAeA,SACE,IAAhBA,EAAQ9mC,UAAuC,IAAlB8mC,EAAQrlC,OAEzC,EAAY,uCAAwCqlC,GAG9C,CAAEA,EAAQ9mC,IAAK8mC,EAAQrlC,QAIxB,CAAEqlC,GAAS,KAIpB,GAAK,GAAeO,GACnB,OAAOpnC,OAAOyhB,QAAS2lB,GAIxB,MAAO,CAAE,CAAEA,GAAiB,GAC7B,CAjH4Be,CAAmBf,GACxCgB,EAAkBz+B,MAAMrB,KAAMrI,GAC9B4F,EAAuB,GAe7B,GAbAqiC,EAAmBthC,SAAS,EAAIyhC,EAAYC,MAC3CF,EAAgBxhC,SAAS2hC,KAkH3B,SAAuBF,EAAoCE,GAC1D,OAAsB,IAAfF,GACNA,IAAeE,GACfF,aAAsB11B,QAAU41B,EAAQ1iC,MAAOwiC,EACjD,EApHIG,CAAcH,EAAYE,IA2H9B,SACCD,EACAC,EACAN,GAEA,IAAsB,IAAjBK,EACJ,OAAO,EAGR,MAAMG,EAAYR,EAAaM,GAK/B,OAAOD,IAAiBG,GACvBH,aAAwB31B,UAAYsR,OAAQwkB,GAAY5iC,MAAOyiC,EACjE,CA1III,CAAgBJ,EAAcC,EAASN,IAEvCpiC,EAAM+C,KAAM2/B,EACb,GACE,IAKEL,EAAmB5mC,UAAUuE,EAAMvE,OAAS4mC,EAAmB5mC,QAIrE,OAAOuE,CACR,CCrSA,SALA,SAAkBrE,GAChB,MAAuB,iBAATA,GACX,GAAaA,IArBF,mBAqBY,GAAWA,EACvC,ECtBA,IAAImnC,GAAe,mDACfC,GAAgB,QAuBpB,SAbA,SAAepnC,EAAO4c,GACpB,GAAI,GAAQ5c,GACV,OAAO,EAET,IAAIyS,SAAczS,EAClB,QAAY,UAARyS,GAA4B,UAARA,GAA4B,WAARA,GAC/B,MAATzS,IAAiB,GAASA,MAGvBonC,GAAc39B,KAAKzJ,KAAWmnC,GAAa19B,KAAKzJ,IAC1C,MAAV4c,GAAkB5c,KAASxB,OAAOoe,GACvC,ECuBA,SAASyqB,GAAQjoB,EAAMkoB,GACrB,GAAmB,mBAARloB,GAAmC,MAAZkoB,GAAuC,mBAAZA,EAC3D,MAAM,IAAI39B,UAhDQ,uBAkDpB,IAAI49B,EAAW,WACb,IAAI7jC,EAAO2d,UACP9iB,EAAM+oC,EAAWA,EAASjvB,MAAM7Q,KAAM9D,GAAQA,EAAK,GACnD8jC,EAAQD,EAASC,MAErB,GAAIA,EAAMpyB,IAAI7W,GACZ,OAAOipC,EAAM53B,IAAIrR,GAEnB,IAAIoH,EAASyZ,EAAK/G,MAAM7Q,KAAM9D,GAE9B,OADA6jC,EAASC,MAAQA,EAAMvuB,IAAI1a,EAAKoH,IAAW6hC,EACpC7hC,CACT,EAEA,OADA4hC,EAASC,MAAQ,IAAKH,GAAQI,OAAS,IAChCF,CACT,CAGAF,GAAQI,MAAQ,GAEhB,YC/CA,SAZA,SAAuBroB,GACrB,IAAIzZ,EAAS,GAAQyZ,GAAM,SAAS7gB,GAIlC,OAfmB,MAYfipC,EAAMnsB,MACRmsB,EAAMpuB,QAED7a,CACT,IAEIipC,EAAQ7hC,EAAO6hC,MACnB,OAAO7hC,CACT,ECpBA,IAAI+hC,GAAa,mGAGbC,GAAe,WASfC,GAAe,IAAc,SAAS3jC,GACxC,IAAI0B,EAAS,GAOb,OAN6B,KAAzB1B,EAAOwwB,WAAW,IACpB9uB,EAAOyB,KAAK,IAEdnD,EAAOwc,QAAQinB,IAAY,SAASrjC,EAAOwjC,EAAQC,EAAOC,GACxDpiC,EAAOyB,KAAK0gC,EAAQC,EAAUtnB,QAAQknB,GAAc,MAASE,GAAUxjC,EACzE,IACOsB,CACT,IAEA,YCNA,SAXA,SAAkBoa,EAAOgB,GAKvB,IAJA,IAAI3U,GAAS,EACTtM,EAAkB,MAATigB,EAAgB,EAAIA,EAAMjgB,OACnC6F,EAASwC,MAAMrI,KAEVsM,EAAQtM,GACf6F,EAAOyG,GAAS2U,EAAShB,EAAM3T,GAAQA,EAAO2T,GAEhD,OAAOpa,CACT,ECZA,IAGI,GAAc,GAAS,GAAO0D,eAAYgG,EAC1C24B,GAAiB,GAAc,GAAY9jC,cAAWmL,EA0B1D,SAhBA,SAAS44B,EAAajoC,GAEpB,GAAoB,iBAATA,EACT,OAAOA,EAET,GAAI,GAAQA,GAEV,OAAO,GAASA,EAAOioC,GAAgB,GAEzC,GAAI,GAASjoC,GACX,OAAOgoC,GAAiBA,GAAen/B,KAAK7I,GAAS,GAEvD,IAAI2F,EAAU3F,EAAQ,GACtB,MAAkB,KAAV2F,GAAkB,EAAI3F,IA3BjB,SA2BwC,KAAO2F,CAC9D,ECPA,SAJA,SAAkB3F,GAChB,OAAgB,MAATA,EAAgB,GAAK,GAAaA,EAC3C,ECLA,SAPA,SAAkBA,EAAO4c,GACvB,OAAI,GAAQ5c,GACHA,EAEF,GAAMA,EAAO4c,GAAU,CAAC5c,GAAS,GAAa,GAASA,GAChE,ECCA,SALA,SAAc+f,GACZ,IAAIjgB,EAAkB,MAATigB,EAAgB,EAAIA,EAAMjgB,OACvC,OAAOA,EAASigB,EAAMjgB,EAAS,QAAKuP,CACtC,ECGA,SARA,SAAerP,GACb,GAAoB,iBAATA,GAAqB,GAASA,GACvC,OAAOA,EAET,IAAI2F,EAAU3F,EAAQ,GACtB,MAAkB,KAAV2F,GAAkB,EAAI3F,IAdjB,SAcwC,KAAO2F,CAC9D,ECKA,SAZA,SAAiBiX,EAAQ3V,GAMvB,IAHA,IAAImF,EAAQ,EACRtM,GAHJmH,EAAO,GAASA,EAAM2V,IAGJ9c,OAED,MAAV8c,GAAkBxQ,EAAQtM,GAC/B8c,EAASA,EAAO,GAAM3V,EAAKmF,OAE7B,OAAQA,GAASA,GAAStM,EAAU8c,OAASvN,CAC/C,ECSA,SArBA,SAAmB0Q,EAAOmoB,EAAOC,GAC/B,IAAI/7B,GAAS,EACTtM,EAASigB,EAAMjgB,OAEfooC,EAAQ,IACVA,GAASA,EAAQpoC,EAAS,EAAKA,EAASooC,IAE1CC,EAAMA,EAAMroC,EAASA,EAASqoC,GACpB,IACRA,GAAOroC,GAETA,EAASooC,EAAQC,EAAM,EAAMA,EAAMD,IAAW,EAC9CA,KAAW,EAGX,IADA,IAAIviC,EAASwC,MAAMrI,KACVsM,EAAQtM,GACf6F,EAAOyG,GAAS2T,EAAM3T,EAAQ87B,GAEhC,OAAOviC,CACT,ECbA,SAJA,SAAgBiX,EAAQ3V,GACtB,OAAOA,EAAKnH,OAAS,EAAI8c,EAAS,GAAQA,EAAQ,GAAU3V,EAAM,GAAI,GACxE,ECMA,SANA,SAAmB2V,EAAQ3V,GAGzB,OAFAA,EAAO,GAASA,EAAM2V,GAEL,OADjBA,EAAS,GAAOA,EAAQ3V,YACQ2V,EAAO,GAAM,GAAK3V,IACpD,ECgBA,SAJA,SAAe2V,EAAQ3V,GACrB,OAAiB,MAAV2V,GAAwB,GAAUA,EAAQ3V,EACnD,ECCA,SALA,SAAa2V,EAAQ3V,EAAMmhC,GACzB,IAAIziC,EAAmB,MAAViX,OAAiBvN,EAAY,GAAQuN,EAAQ3V,GAC1D,YAAkBoI,IAAX1J,EAAuByiC,EAAeziC,CAC/C,ECXA,SAPA,SAA0BiX,EAAQre,EAAKyB,SACtBqP,IAAVrP,IAAwB,GAAG4c,EAAOre,GAAMyB,SAC9BqP,IAAVrP,KAAyBzB,KAAOqe,KACnC,GAAgBA,EAAQre,EAAKyB,EAEjC,ECFA,SCRA,SAAuBqoC,GACrB,OAAO,SAASzrB,EAAQmE,EAAUsC,GAMhC,IALA,IAAIjX,GAAS,EACT6R,EAAWzf,OAAOoe,GAClBqE,EAAQoC,EAASzG,GACjB9c,EAASmhB,EAAMnhB,OAEZA,KAAU,CACf,IAAIvB,EAAM0iB,EAAMonB,EAAYvoC,IAAWsM,GACvC,IAA+C,IAA3C2U,EAAS9C,EAAS1f,GAAMA,EAAK0f,GAC/B,KAEJ,CACA,OAAOrB,CACT,CACF,CDTc,GEmBd,SAJA,SAA2B5c,GACzB,OAAO,GAAaA,IAAU,GAAYA,EAC5C,ECVA,SAZA,SAAiB4c,EAAQre,GACvB,IAAY,gBAARA,GAAgD,mBAAhBqe,EAAOre,KAIhC,aAAPA,EAIJ,OAAOqe,EAAOre,EAChB,ECaA,SAJA,SAAuByB,GACrB,OAAO,GAAWA,EAAO,GAAOA,GAClC,ECgEA,SA9DA,SAAuB4c,EAAQlS,EAAQnM,EAAK+pC,EAAUC,EAAWrnB,EAAYrL,GAC3E,IAAImL,EAAW,GAAQpE,EAAQre,GAC3BiqC,EAAW,GAAQ99B,EAAQnM,GAC3BknB,EAAU5P,EAAMjG,IAAI44B,GAExB,GAAI/iB,EACF,GAAiB7I,EAAQre,EAAKknB,OADhC,CAIA,IAAIxK,EAAWiG,EACXA,EAAWF,EAAUwnB,EAAWjqC,EAAM,GAAKqe,EAAQlS,EAAQmL,QAC3DxG,EAEAo5B,OAAwBp5B,IAAb4L,EAEf,GAAIwtB,EAAU,CACZ,IAAIrmB,EAAQ,GAAQomB,GAChBlmB,GAAUF,GAAS,GAASomB,GAC5BE,GAAWtmB,IAAUE,GAAU,GAAakmB,GAEhDvtB,EAAWutB,EACPpmB,GAASE,GAAUomB,EACjB,GAAQ1nB,GACV/F,EAAW+F,EAEJ,GAAkBA,GACzB/F,EAAW,GAAU+F,GAEdsB,GACPmmB,GAAW,EACXxtB,EAAW,GAAYutB,GAAU,IAE1BE,GACPD,GAAW,EACXxtB,EAAW,GAAgButB,GAAU,IAGrCvtB,EAAW,GAGN,GAAcutB,IAAa,GAAYA,IAC9CvtB,EAAW+F,EACP,GAAYA,GACd/F,EAAW,GAAc+F,GAEjB,EAASA,KAAa,GAAWA,KACzC/F,EAAW,GAAgButB,KAI7BC,GAAW,CAEf,CACIA,IAEF5yB,EAAMoD,IAAIuvB,EAAUvtB,GACpBstB,EAAUttB,EAAUutB,EAAUF,EAAUpnB,EAAYrL,GACpDA,EAAc,OAAE2yB,IAElB,GAAiB5rB,EAAQre,EAAK0c,EAnD9B,CAoDF,EClDA,SAtBA,SAAS0tB,EAAU/rB,EAAQlS,EAAQ49B,EAAUpnB,EAAYrL,GACnD+G,IAAWlS,GAGf,GAAQA,GAAQ,SAAS89B,EAAUjqC,GAEjC,GADAsX,IAAUA,EAAQ,IAAI,IAClB,EAAS2yB,GACX,GAAc5rB,EAAQlS,EAAQnM,EAAK+pC,EAAUK,EAAWznB,EAAYrL,OAEjE,CACH,IAAIoF,EAAWiG,EACXA,EAAW,GAAQtE,EAAQre,GAAMiqC,EAAWjqC,EAAM,GAAKqe,EAAQlS,EAAQmL,QACvExG,OAEaA,IAAb4L,IACFA,EAAWutB,GAEb,GAAiB5rB,EAAQre,EAAK0c,EAChC,CACF,GAAG,GACL,ECnBA,SAJA,SAAkBjb,GAChB,OAAOA,CACT,ECEA,SAVA,SAAeof,EAAMwpB,EAASllC,GAC5B,OAAQA,EAAK5D,QACX,KAAK,EAAG,OAAOsf,EAAKvW,KAAK+/B,GACzB,KAAK,EAAG,OAAOxpB,EAAKvW,KAAK+/B,EAASllC,EAAK,IACvC,KAAK,EAAG,OAAO0b,EAAKvW,KAAK+/B,EAASllC,EAAK,GAAIA,EAAK,IAChD,KAAK,EAAG,OAAO0b,EAAKvW,KAAK+/B,EAASllC,EAAK,GAAIA,EAAK,GAAIA,EAAK,IAE3D,OAAO0b,EAAK/G,MAAMuwB,EAASllC,EAC7B,ECfA,IAAImlC,GAAYxoC,KAAKC,IAgCrB,SArBA,SAAkB8e,EAAM8oB,EAAO7oB,GAE7B,OADA6oB,EAAQW,QAAoBx5B,IAAV64B,EAAuB9oB,EAAKtf,OAAS,EAAKooC,EAAO,GAC5D,WAML,IALA,IAAIxkC,EAAO2d,UACPjV,GAAS,EACTtM,EAAS+oC,GAAUnlC,EAAK5D,OAASooC,EAAO,GACxCnoB,EAAQ5X,MAAMrI,KAETsM,EAAQtM,GACfigB,EAAM3T,GAAS1I,EAAKwkC,EAAQ97B,GAE9BA,GAAS,EAET,IADA,IAAI08B,EAAY3gC,MAAM+/B,EAAQ,KACrB97B,EAAQ87B,GACfY,EAAU18B,GAAS1I,EAAK0I,GAG1B,OADA08B,EAAUZ,GAAS7oB,EAAUU,GACtB,GAAMX,EAAM5X,KAAMshC,EAC3B,CACF,ECRA,SANA,SAAkB9oC,GAChB,OAAO,WACL,OAAOA,CACT,CACF,ECFA,SATuB,GAA4B,SAASof,EAAMnb,GAChE,OAAO,GAAemb,EAAM,WAAY,CACtC,cAAgB,EAChB,YAAc,EACd,MAAS,GAASnb,GAClB,UAAY,GAEhB,EAPwC,GCXxC,IAII8kC,GAAY1yB,KAAK2yB,IA+BrB,SApBA,SAAkB5pB,GAChB,IAAIjT,EAAQ,EACR88B,EAAa,EAEjB,OAAO,WACL,IAAIC,EAAQH,KACRI,EApBO,IAoBiBD,EAAQD,GAGpC,GADAA,EAAaC,EACTC,EAAY,GACd,KAAMh9B,GAzBI,IA0BR,OAAOkV,UAAU,QAGnBlV,EAAQ,EAEV,OAAOiT,EAAK/G,WAAMhJ,EAAWgS,UAC/B,CACF,ECrBA,SAFkB,GAAS,ICK3B,SAJA,SAAkBjC,EAAM8oB,GACtB,OAAO,GAAY,GAAS9oB,EAAM8oB,EAAO,IAAW9oB,EAAO,GAC7D,ECeA,SAdA,SAAwBpf,EAAOoM,EAAOwQ,GACpC,IAAK,EAASA,GACZ,OAAO,EAET,IAAInK,SAAcrG,EAClB,SAAY,UAARqG,EACK,GAAYmK,IAAW,GAAQxQ,EAAOwQ,EAAO9c,QACrC,UAAR2S,GAAoBrG,KAASwQ,IAE7B,GAAGA,EAAOxQ,GAAQpM,EAG7B,ECSA,SA1BA,SAAwBopC,GACtB,OAAO,IAAS,SAASxsB,EAAQnS,GAC/B,IAAI2B,GAAS,EACTtM,EAAS2K,EAAQ3K,OACjBohB,EAAaphB,EAAS,EAAI2K,EAAQ3K,EAAS,QAAKuP,EAChDg6B,EAAQvpC,EAAS,EAAI2K,EAAQ,QAAK4E,EAWtC,IATA6R,EAAckoB,EAAStpC,OAAS,GAA0B,mBAAdohB,GACvCphB,IAAUohB,QACX7R,EAEAg6B,GAAS,GAAe5+B,EAAQ,GAAIA,EAAQ,GAAI4+B,KAClDnoB,EAAaphB,EAAS,OAAIuP,EAAY6R,EACtCphB,EAAS,GAEX8c,EAASpe,OAAOoe,KACPxQ,EAAQtM,GAAQ,CACvB,IAAI4K,EAASD,EAAQ2B,GACjB1B,GACF0+B,EAASxsB,EAAQlS,EAAQ0B,EAAO8U,EAEpC,CACA,OAAOtE,CACT,GACF,ECIA,SAJY,IAAe,SAASA,EAAQlS,EAAQ49B,GAClD,GAAU1rB,EAAQlS,EAAQ49B,EAC5B,ICcA,SAlCA,SAAiB1rB,EAAQ3V,EAAMjH,EAAOkhB,GACpC,IAAK,EAAStE,GACZ,OAAOA,EAST,IALA,IAAIxQ,GAAS,EACTtM,GAHJmH,EAAO,GAASA,EAAM2V,IAGJ9c,OACdqS,EAAYrS,EAAS,EACrBwpC,EAAS1sB,EAEI,MAAV0sB,KAAoBl9B,EAAQtM,GAAQ,CACzC,IAAIvB,EAAM,GAAM0I,EAAKmF,IACjB6O,EAAWjb,EAEf,GAAY,cAARzB,GAA+B,gBAARA,GAAiC,cAARA,EAClD,OAAOqe,EAGT,GAAIxQ,GAAS+F,EAAW,CACtB,IAAI6O,EAAWsoB,EAAO/qC,QAEL8Q,KADjB4L,EAAWiG,EAAaA,EAAWF,EAAUziB,EAAK+qC,QAAUj6B,KAE1D4L,EAAW,EAAS+F,GAChBA,EACC,GAAQ/Z,EAAKmF,EAAQ,IAAM,GAAK,CAAC,EAE1C,CACA,GAAYk9B,EAAQ/qC,EAAK0c,GACzBquB,EAASA,EAAO/qC,EAClB,CACA,OAAOqe,CACT,ECdA,SAJA,SAAaA,EAAQ3V,EAAMjH,GACzB,OAAiB,MAAV4c,EAAiBA,EAAS,GAAQA,EAAQ3V,EAAMjH,EACzD,EChBe,MAAMupC,GAiBpB,WAAAhgC,CAAaigC,GACZhiC,KAAKiiC,QAAU,CAAC,EAChBjiC,KAAKkiC,gBAAkBF,CACxB,CAKA,WAAWG,GACV,MAAM1pB,EAAUzhB,OAAOyhB,QAASzY,KAAKiiC,SAGrC,OAFathC,MAAMrB,KAAMmZ,GAEZngB,MACd,CAKA,QAAWub,GACV,OAAK7T,KAAKmiC,QACF,EAGDniC,KAAK8+B,gBAAgBxmC,MAC7B,CASO,KAAA8pC,CAAOC,GACbriC,KAAK4R,QAEL,MAAM0wB,EAAe3hC,MAAMrB,KAwxB7B,SAA4BijC,GAE3B,IAAIC,EAAY,KACZC,EAAoB,EACpBC,EAAqB,EACrB3uB,EAAe,KAEnB,MAAM4uB,EAAY,IAAIpxB,IAGtB,GAAsB,KAAjBgxB,EACJ,OAAOI,EAI+C,KAAlDJ,EAAalV,OAAQkV,EAAajqC,OAAS,KAC/CiqC,GAA8B,KAI/B,QAAU/nC,EAAI,EAAGA,EAAI+nC,EAAajqC,OAAQkC,IAAM,CAC/C,MAAMyC,EAAOslC,EAAalV,OAAQ7yB,GAElC,GAAmB,OAAdgoC,EAEJ,OAASvlC,GACR,IAAK,IAGE8W,IAGLA,EAAewuB,EAAatwB,OAAQwwB,EAAmBjoC,EAAIioC,GAE3DC,EAAqBloC,EAAI,GAG1B,MAED,IAAK,IACL,IAAK,IAEJgoC,EAAYvlC,EAEZ,MAED,IAAK,IAAK,CAGT,MAAMuZ,EAAgB+rB,EAAatwB,OAAQywB,EAAoBloC,EAAIkoC,GAE9D3uB,GAEJ4uB,EAAUlxB,IAAKsC,EAAa8Z,OAAQrX,EAAcqX,QAGnD9Z,EAAe,KAGf0uB,EAAoBjoC,EAAI,EAExB,KACD,OAEUyC,IAASulC,IAEpBA,EAAY,KAEd,CAEA,OAAOG,CACR,CA/1BmCC,CAAmBP,GAAc5pB,WAElE,UAAc1hB,EAAKyB,KAAW8pC,EAC7BtiC,KAAKkiC,gBAAgBW,iBAAkB9rC,EAAKyB,EAAOwH,KAAKiiC,QAE1D,CAiCO,GAAAr0B,CAAK5L,GACX,GAAKhC,KAAKmiC,QACT,OAAO,EAGR,MAEMW,EAFS9iC,KAAKkiC,gBAAgBa,eAAgB/gC,EAAMhC,KAAKiiC,SAE7B1Q,MAAM,EAAIne,KAAgBA,IAAapR,IAGzE,OAAOrB,MAAMC,QAASkiC,EACvB,CAkFO,GAAArxB,CAAKuxB,EAA+BC,GAC1C,GAAK,EAAUD,GACd,UAAcjsC,EAAKyB,KAAWxB,OAAOyhB,QAASuqB,GAC7ChjC,KAAKkiC,gBAAgBW,iBAAkB9rC,EAAKyB,EAAOwH,KAAKiiC,cAGzDjiC,KAAKkiC,gBAAgBW,iBAAkBG,EAAcC,EAAgBjjC,KAAKiiC,QAE5E,CA8BO,MAAAh8B,CAAQjE,GACd,MAAMvC,EAAOyjC,GAAQlhC,GAErB,GAAOhC,KAAKiiC,QAASxiC,UACdO,KAAKiiC,QAASjgC,GAErBhC,KAAKmjC,yBAA0B1jC,EAChC,CA4BO,aAAA2jC,CAAephC,GACrB,OAAOhC,KAAKkiC,gBAAgBkB,cAAephC,EAAMhC,KAAKiiC,QACvD,CA0BO,QAAAvlC,GACN,OAAKsD,KAAKmiC,QACF,GAGDniC,KAAKqjC,oBACVrmC,KAAK0D,GAAOA,EAAIxD,KAAM,OACtBikB,OACAjkB,KAAM,KAAQ,GACjB,CA0DO,WAAAomC,CAAavvB,GACnB,GAAK/T,KAAKmiC,QACT,OAGD,GAAKniC,KAAKiiC,QAASluB,KAAmB,EAAU/T,KAAKiiC,QAASluB,IAE7D,OAAO/T,KAAKiiC,QAASluB,GAGtB,MAEM+uB,EAFS9iC,KAAKkiC,gBAAgBa,eAAgBhvB,EAAc/T,KAAKiiC,SAErC1Q,MAAM,EAAIne,KAAgBA,IAAaW,IAGzE,OAAKpT,MAAMC,QAASkiC,GACZA,EAAoB,QAD5B,CAGD,CAmBO,aAAAhE,CAAeyE,GAAS,GAC9B,GAAKvjC,KAAKmiC,QACT,MAAO,GAGR,GAAKoB,EACJ,OAAOvjC,KAAKkiC,gBAAgBpD,cAAe9+B,KAAKiiC,SAKjD,OAFgBjiC,KAAKqjC,oBAENrmC,KAAK,EAAIjG,KAAWA,GACpC,CAKO,KAAA6a,GACN5R,KAAKiiC,QAAU,CAAC,CACjB,CAKQ,iBAAAoB,GACP,MAAM7tB,EAAoC,GAEpCve,EAAOD,OAAOC,KAAM+I,KAAKiiC,SAE/B,UAAYlrC,KAAOE,EAClBue,EAAO5V,QAASI,KAAKkiC,gBAAgBa,eAAgBhsC,EAAKiJ,KAAKiiC,UAGhE,OAAOzsB,CACR,CAKQ,wBAAA2tB,CAA0B1jC,GACjC,MAAM+jC,EAAY/jC,EAAK1C,MAAO,KAG9B,KAFoBymC,EAAUlrC,OAAS,GAGtC,OAGD,MAAMmrC,EAAaD,EAAUh8B,OAAQ,EAAGg8B,EAAUlrC,OAAS,GAAI4E,KAAM,KAE/DwmC,EAAe,GAAK1jC,KAAKiiC,QAASwB,GAExC,IAAMC,EACL,QAGsB/iC,MAAMrB,KAAMtI,OAAOC,KAAMysC,IAAiBprC,QAGhE0H,KAAKiG,OAAQw9B,EAEf,EAMM,MAAME,GAWZ,WAAA5hC,GACC/B,KAAK4jC,aAAe,IAAIryB,IACxBvR,KAAK6jC,YAAc,IAAItyB,IACvBvR,KAAK8jC,UAAY,IAAIvyB,IACrBvR,KAAK+jC,aAAe,IAAIxyB,GACzB,CAmBO,gBAAAsxB,CAAkB7gC,EAAcwU,EAA2BqoB,GACjE,GAAK,EAAUroB,GACdwtB,GAAkBnF,EAAQqE,GAAQlhC,GAAQwU,QAK3C,GAAKxW,KAAK4jC,aAAah2B,IAAK5L,GAAS,CACpC,MAAMiiC,EAAajkC,KAAK4jC,aAAax7B,IAAKpG,IAEpC,KAAEvC,EAAI,MAAEjH,GAAUyrC,EAAYztB,GAEpCwtB,GAAkBnF,EAAQp/B,EAAMjH,EACjC,MACCwrC,GAAkBnF,EAAQ78B,EAAMwU,EAElC,CAuBO,aAAA4sB,CAAephC,EAA0B68B,GAC/C,IAAM78B,EACL,OAAO,GAAO,CAAC,EAAG68B,GAInB,QAAwB,IAAnBA,EAAQ78B,GACZ,OAAO68B,EAAQ78B,GAGhB,GAAKhC,KAAK6jC,YAAYj2B,IAAK5L,GAAS,CACnC,MAAMkiC,EAAYlkC,KAAK6jC,YAAYz7B,IAAKpG,GAExC,GAA0B,iBAAdkiC,EACX,OAAO,GAAKrF,EAAQqF,GAGrB,MAAM1rC,EAAQ0rC,EAAWliC,EAAM68B,GAE/B,GAAKrmC,EACJ,OAAOA,CAET,CAEA,OAAO,GAAKqmC,EAAQqE,GAAQlhC,GAC7B,CAoCO,cAAA+gC,CAAgB/gC,EAAc68B,GACpC,MAAMsF,EAAkBnkC,KAAKojC,cAAephC,EAAM68B,GAGlD,QAAyB,IAApBsF,EACJ,MAAO,GAGR,GAAKnkC,KAAK8jC,UAAUl2B,IAAK5L,GAAS,CAGjC,OAFgBhC,KAAK8jC,UAAU17B,IAAKpG,EAE7BoiC,CAASD,EACjB,CAEA,MAAO,CAAE,CAAEniC,EAAMmiC,GAClB,CAOO,aAAArF,CAAeD,GAErB,MAAMwF,EAAqB1jC,MAAMrB,KAAMU,KAAK+jC,aAAa9sC,QAAS8O,QAAQ/D,IACzE,MAAMqD,EAAQrF,KAAKojC,cAAephC,EAAM68B,GAExC,OAAKx5B,GAAyB,iBAATA,EACbrO,OAAOC,KAAMoO,GAAQ/M,OAGtB+M,CAAK,IAKPi/B,EAAoB,IAAI1wB,IAAK,IAC/BywB,KACArtC,OAAOC,KAAM4nC,KAGjB,OAAOl+B,MAAMrB,KAAMglC,EAAkBp5B,SACtC,CAgBO,gBAAAq5B,CAAkBviC,GACxB,OAAOhC,KAAK+jC,aAAa37B,IAAKpG,IAAU,EACzC,CAwDO,aAAAwiC,CAAexiC,EAAcuN,GACnCvP,KAAK4jC,aAAanyB,IAAKzP,EAAMuN,EAC9B,CA8CO,YAAAk1B,CAAcziC,EAAc0iC,GAClC1kC,KAAK6jC,YAAYpyB,IAAKzP,EAAM0iC,EAC7B,CAsCO,UAAAC,CAAY3iC,EAAcuN,GAChCvP,KAAK8jC,UAAUryB,IAAKzP,EAAMuN,EAC3B,CAuBO,gBAAAq1B,CAAkBC,EAAuBC,GAC/C9kC,KAAK+kC,eAAgBF,EAAeC,GAEpC,UAAYE,KAAYF,EACvB9kC,KAAK+kC,eAAgBC,EAAU,CAAEH,GAEnC,CAKQ,cAAAE,CAAgB/iC,EAAc8iC,GAC/B9kC,KAAK+jC,aAAan2B,IAAK5L,IAC5BhC,KAAK+jC,aAAatyB,IAAKzP,EAAM,IAG9BhC,KAAK+jC,aAAa37B,IAAKpG,GAAQpC,QAASklC,EACzC,EAqFD,SAAS5B,GAAQlhC,GAChB,OAAOA,EAAKiX,QAAS,IAAK,IAC3B,CAKA,SAAS+qB,GAAkBiB,EAAsBC,EAAoBjC,GACpE,IAAIkC,EAAalC,EAEZ,EAAUA,KACdkC,EAAa,GAAO,CAAC,EAAG,GAAKF,EAAcC,GAAcjC,IAG1D,GAAKgC,EAAcC,EAAYC,EAChC,CC54Be,MAAM,WAAgB,GA8DpC,WAAApjC,CACCyB,EACAxB,EACAojC,EACA9tB,GAeA,GAbA9J,MAAOhK,GAnDR,KAAgB6hC,0BAA2C,GA0B3D,KAAiBC,kBAAoB,IAAI/zB,IA2BxCvR,KAAKgC,KAAOA,EAEZhC,KAAKulC,OAiwBP,SAA0BH,GACzB,MAAMI,EAAW,GAAOJ,GAExB,UAAcruC,EAAKyB,KAAWgtC,EACd,OAAVhtC,EACJgtC,EAAS7zB,OAAQ5a,GACU,iBAATyB,GAClBgtC,EAAS/zB,IAAK1a,EAAKkkB,OAAQziB,IAI7B,OAAOgtC,CACR,CA7wBgBC,CAAiBL,GAC/BplC,KAAK0lC,UAAY,GAEZpuB,GACJtX,KAAK2lC,aAAc,EAAGruB,GAGvBtX,KAAK4lC,SAAW,IAAIhyB,IAEf5T,KAAKulC,OAAO33B,IAAK,SAAY,CAEjC,MAAMi4B,EAAc7lC,KAAKulC,OAAOn9B,IAAK,SACrC09B,GAAc9lC,KAAK4lC,SAAUC,GAC7B7lC,KAAKulC,OAAO5zB,OAAQ,QACrB,CAEA3R,KAAKiiC,QAAU,IAAIF,GAAW/hC,KAAKwD,SAASuiC,iBAEvC/lC,KAAKulC,OAAO33B,IAAK,WAErB5N,KAAKiiC,QAAQG,MAAOpiC,KAAKulC,OAAOn9B,IAAK,UAErCpI,KAAKulC,OAAO5zB,OAAQ,SAEtB,CAKA,cAAW2pB,GACV,OAAOt7B,KAAK0lC,UAAUptC,MACvB,CAKA,WAAW6pC,GACV,OAAiC,IAA1BniC,KAAK0lC,UAAUptC,MACvB,CAQO,QAAAujC,CAAUj3B,GAChB,OAAO5E,KAAK0lC,UAAW9gC,EACxB,CAQO,aAAAo3B,CAAe78B,GACrB,OAAOa,KAAK0lC,UAAU18B,QAAS7J,EAChC,CAOO,WAAAq8B,GACN,OAAOx7B,KAAK0lC,UAAW5kC,OAAOC,WAC/B,CAOA,iBAAQu9B,GACFt+B,KAAK4lC,SAAS/xB,KAAO,SACnB,SAGD7T,KAAKiiC,QAAQE,eACZ,eAGAniC,KAAKulC,OAAOtuC,MACpB,CAQA,cAAQ+uC,SACAhmC,KAAKulC,OAAO9sB,UAEdzY,KAAK4lC,SAAS/xB,KAAO,SACnB,CAAE,QAAS7T,KAAKq7B,aAAc,WAG/Br7B,KAAKiiC,QAAQE,eACZ,CAAE,QAASniC,KAAKq7B,aAAc,UAEtC,CAQO,YAAAA,CAActkC,GACpB,GAAY,SAAPA,EACJ,OAAKiJ,KAAK4lC,SAAS/xB,KAAO,EAClB,IAAK7T,KAAK4lC,UAAW1oC,KAAM,UAGnC,EAGD,GAAY,SAAPnG,EAAiB,CACrB,MAAMsrC,EAAcriC,KAAKiiC,QAAQvlC,WAEjC,MAAsB,IAAf2lC,OAAoB,EAAYA,CACxC,CAEA,OAAOriC,KAAKulC,OAAOn9B,IAAKrR,EACzB,CAQO,YAAAkvC,CAAclvC,GACpB,MAAY,SAAPA,EACGiJ,KAAK4lC,SAAS/xB,KAAO,EAGjB,SAAP9c,GACIiJ,KAAKiiC,QAAQE,QAGfniC,KAAKulC,OAAO33B,IAAK7W,EACzB,CAOO,SAAAsmC,CAAW6I,GACjB,KAAQA,aAAwB,IAC/B,OAAO,EAIR,GAAKlmC,OAASkmC,EACb,OAAO,EAIR,GAAKlmC,KAAKgC,MAAQkkC,EAAalkC,KAC9B,OAAO,EAIR,GAAKhC,KAAKulC,OAAO1xB,OAASqyB,EAAaX,OAAO1xB,MAAQ7T,KAAK4lC,SAAS/xB,OAASqyB,EAAaN,SAAS/xB,MAClG7T,KAAKiiC,QAAQpuB,OAASqyB,EAAajE,QAAQpuB,KAC3C,OAAO,EAIR,UAAc9c,EAAKyB,KAAWwH,KAAKulC,OAClC,IAAMW,EAAaX,OAAO33B,IAAK7W,IAASmvC,EAAaX,OAAOn9B,IAAKrR,KAAUyB,EAC1E,OAAO,EAKT,UAAY2tC,KAAanmC,KAAK4lC,SAC7B,IAAMM,EAAaN,SAASh4B,IAAKu4B,GAChC,OAAO,EAKT,UAAY/yB,KAAYpT,KAAKiiC,QAAQnD,gBACpC,IACEoH,EAAajE,QAAQr0B,IAAKwF,IAC3B8yB,EAAajE,QAAQqB,YAAalwB,KAAepT,KAAKiiC,QAAQqB,YAAalwB,GAE3E,OAAO,EAIT,OAAO,CACR,CAWO,QAAA0nB,IAAaqL,GACnB,UAAYnkC,KAAQmkC,EACnB,IAAMnmC,KAAK4lC,SAASh4B,IAAK5L,GACxB,OAAO,EAIT,OAAO,CACR,CAKO,aAAA28B,GACN,OAAO3+B,KAAK4lC,SAAS3uC,MACtB,CA2BO,QAAA8nC,CAAU3rB,GAChB,OAAOpT,KAAKiiC,QAAQqB,YAAalwB,EAClC,CAoCO,kBAAAgzB,CAAoBhzB,GAC1B,OAAOpT,KAAKiiC,QAAQmB,cAAehwB,EACpC,CAOO,aAAA0rB,CAAeyE,GACrB,OAAOvjC,KAAKiiC,QAAQnD,cAAeyE,EACpC,CAWO,QAAA8C,IAAajzB,GACnB,UAAYpR,KAAQoR,EACnB,IAAMpT,KAAKiiC,QAAQr0B,IAAK5L,GACvB,OAAO,EAIT,OAAO,CACR,CAUO,YAAAskC,IAAiBlI,GACvB,MAAMmI,EAAU,IAAI,MAAYnI,GAChC,IAAIx/B,EAASoB,KAAKpB,OAElB,KAAQA,IAAWA,EAAOsP,GAAI,qBAAuB,CACpD,GAAKq4B,EAAQ1pC,MAAO+B,GACnB,OAAOA,EAGRA,EAASA,EAAOA,MACjB,CAEA,OAAO,IACR,CAKO,iBAAA4nC,CAAmBzvC,GACzB,OAAOiJ,KAAKslC,kBAAkBl9B,IAAKrR,EACpC,CAMA,oBAAQ0vC,SACAzmC,KAAKslC,kBAAkB7sB,SAC/B,CA2BO,WAAAiuB,GACN,MAAMhI,EAAU/9B,MAAMrB,KAAMU,KAAK4lC,UAAWzkB,OAAOjkB,KAAM,KACnD2hC,EAAS7+B,KAAKiiC,QAAQvlC,WACtB6I,EAAa5E,MAAMrB,KAAMU,KAAKulC,QAASvoC,KAAKxC,GAAK,GAAIA,EAAG,OAAUA,EAAG,QAAU2mB,OAAOjkB,KAAM,KAElG,OAAO8C,KAAKgC,MACE,IAAX08B,EAAgB,GAAK,WAAYA,OAChCG,EAAc,WAAYA,KAAjB,KACI,IAAdt5B,EAAmB,GAAK,IAAKA,IACjC,CAUO,2BAAAohC,CAA6BC,GACnC,OAAO5mC,KAAKqlC,0BAA0B9V,SAAUqX,EACjD,CAUO,MAAArJ,CAAQsJ,GAAO,GACrB,MAAMC,EAA6B,GAEnC,GAAKD,EACJ,UAAYlvB,KAAS3X,KAAKw7B,cACzBsL,EAAclnC,KAAM+X,EAAM4lB,OAAQsJ,IAKpC,MAAME,EAAS,IAAM/mC,KAAK+B,YAAsB/B,KAAKwD,SAAUxD,KAAKgC,KAAMhC,KAAKulC,OAAQuB,GAkBvF,OAdAC,EAAOnB,SAAW,IAAIhyB,IAAK5T,KAAK4lC,UAChCmB,EAAO9E,QAAQxwB,IAAKzR,KAAKiiC,QAAQmB,iBAGjC2D,EAAOzB,kBAAoB,IAAI/zB,IAAKvR,KAAKslC,mBAKzCyB,EAAOC,gBAAkBhnC,KAAKgnC,gBAG9BD,EAAO1B,0BAA4BrlC,KAAKqlC,0BAEjC0B,CACR,CAYO,YAAAE,CAAcrW,GACpB,OAAO5wB,KAAK2lC,aAAc3lC,KAAKs7B,WAAY1K,EAC5C,CAaO,YAAA+U,CAAc/gC,EAAegsB,GACnC5wB,KAAK+8B,YAAa,WAAY/8B,MAC9B,IAAI2E,EAAQ,EAEZ,MAAMuiC,EAqUR,SAAoB1jC,EAAoB0jC,GAEvC,GAAqB,iBAATA,EACX,MAAO,CAAE,IAAIhK,GAAM15B,EAAU0jC,IAGxBvwB,EAAYuwB,KACjBA,EAAQ,CAAEA,IAIX,OAAOvmC,MAAMrB,KAAM4nC,GACjBlqC,KAAKmC,GACe,iBAARA,EACJ,IAAI+9B,GAAM15B,EAAUrE,GAGvBA,aAAgBq+B,GACb,IAAIN,GAAM15B,EAAUrE,EAAK2D,MAG1B3D,GAEV,CA5VgBgoC,CAAWnnC,KAAKwD,SAAUotB,GAExC,UAAYzxB,KAAQ+nC,EAEE,OAAhB/nC,EAAKP,QACTO,EAAKkyB,UAGJlyB,EAAcP,OAASoB,KACvBb,EAAcqE,SAAWxD,KAAKwD,SAEhCxD,KAAK0lC,UAAUl+B,OAAQ5C,EAAO,EAAGzF,GACjCyF,IACAD,IAGD,OAAOA,CACR,CAYO,eAAAm4B,CAAiBl4B,EAAeuG,EAAkB,GACxDnL,KAAK+8B,YAAa,WAAY/8B,MAE9B,QAAUxF,EAAIoK,EAAOpK,EAAIoK,EAAQuG,EAAS3Q,IACvCwF,KAAK0lC,UAAWlrC,GAAaoE,OAAS,KAGzC,OAAOoB,KAAK0lC,UAAUl+B,OAAQ5C,EAAOuG,EACtC,CAWO,aAAAi8B,CAAerwC,EAAayB,GAClC,MAAM6uC,EAAcpsB,OAAQziB,GAE5BwH,KAAK+8B,YAAa,aAAc/8B,MAEpB,SAAPjJ,EACJ+uC,GAAc9lC,KAAK4lC,SAAUyB,GACX,SAAPtwC,EACXiJ,KAAKiiC,QAAQG,MAAOiF,GAEpBrnC,KAAKulC,OAAO9zB,IAAK1a,EAAKswC,EAExB,CAWO,gBAAAC,CAAkBvwC,GAIxB,OAHAiJ,KAAK+8B,YAAa,aAAc/8B,MAGpB,SAAPjJ,EACCiJ,KAAK4lC,SAAS/xB,KAAO,IACzB7T,KAAK4lC,SAASh0B,SAEP,GAOG,SAAP7a,GACEiJ,KAAKiiC,QAAQE,UAClBniC,KAAKiiC,QAAQrwB,SAEN,GAOF5R,KAAKulC,OAAO5zB,OAAQ5a,EAC5B,CAcO,SAAAwwC,CAAWpB,GACjBnmC,KAAK+8B,YAAa,aAAc/8B,MAEhC,UAAYgC,KAAQ,GAASmkC,GAC5BnmC,KAAK4lC,SAAS/3B,IAAK7L,EAErB,CAcO,YAAAwlC,CAAcrB,GACpBnmC,KAAK+8B,YAAa,aAAc/8B,MAEhC,UAAYgC,KAAQ,GAASmkC,GAC5BnmC,KAAK4lC,SAASj0B,OAAQ3P,EAExB,CA4CO,SAAAylC,CAAWr0B,EAA2C5a,GAC5DwH,KAAK+8B,YAAa,aAAc/8B,MAER,iBAAZoT,EACXpT,KAAKiiC,QAAQxwB,IAAK2B,GAElBpT,KAAKiiC,QAAQxwB,IAAK2B,EAAU5a,EAE9B,CAkBO,YAAAkvC,CAAct0B,GACpBpT,KAAK+8B,YAAa,aAAc/8B,MAEhC,UAAYgC,KAAQ,GAASoR,GAC5BpT,KAAKiiC,QAAQh8B,OAAQjE,EAEvB,CASO,kBAAA2lC,CAAoB5wC,EAAsByB,GAChDwH,KAAKslC,kBAAkB7zB,IAAK1a,EAAKyB,EAClC,CASO,qBAAAovC,CAAuB7wC,GAC7B,OAAOiJ,KAAKslC,kBAAkB3zB,OAAQ5a,EACvC,EAgFD,SAAS+uC,GAAc+B,EAAyBC,GAC/C,MAAMC,EAAaD,EAAc/qC,MAAO,OACxC8qC,EAAWj2B,QACXm2B,EAAWnqC,SAASoE,GAAQ6lC,EAAWh6B,IAAK7L,IAC7C,CAhDA,GAAQH,UAAUqM,GAAK,SAAUjD,EAAcjJ,GAC9C,OAAMA,EAKEA,IAAShC,KAAKgC,OAAmB,YAATiJ,GAA+B,iBAATA,GAJrC,YAATA,GAA+B,iBAATA,GAEnB,SAATA,GAA4B,cAATA,CAItB,ECt0Be,MAAM+8B,WAAyB,GAY7C,WAAAjmC,CACCyB,EACAxB,EACAojC,EACA9tB,GAEA9J,MAAOhK,EAAUxB,EAAMojC,EAAO9tB,GAE9BtX,KAAKgnC,gBAAkBA,EACxB,EAyBM,SAASA,KACf,MAAM1vB,EAAW,IAAKtX,KAAKw7B,eACrByM,EAAY3wB,EAAUtX,KAAKs7B,WAAa,GAG9C,GAAK2M,GAAaA,EAAU/5B,GAAI,UAAW,MAC1C,OAAOlO,KAAKs7B,WAGb,UAAY3jB,KAASL,EAEpB,IAAMK,EAAMzJ,GAAI,aACf,OAAO,KAKT,OAAOlO,KAAKs7B,UACb,CAtCA0M,GAAiBnmC,UAAUqM,GAAK,SAAUjD,EAAcjJ,GACvD,OAAMA,EAMEA,IAAShC,KAAKgC,OACX,qBAATiJ,GAAwC,0BAATA,GAEtB,YAATA,GAA+B,iBAATA,GARP,qBAATA,GAAwC,0BAATA,GAE5B,YAATA,GAA+B,iBAATA,GACb,SAATA,GAA4B,cAATA,CAQtB,EC9Ce,MAAMi9B,WAAwB/0B,EAAiB60B,KAwC7D,WAAAjmC,CACCyB,EACAxB,EACAuD,EACA+R,GAEA9J,MAAOhK,EAAUxB,EAAMuD,EAAY+R,GAEnCtX,KAAKyR,IAAK,cAAc,GACxBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,mBAAe,GAEzBzR,KAAKgH,KAAM,cAAezH,GAAIiE,GAE9BxD,KAAKgH,KAAM,aAAczH,GACxBiE,EACA,aACAsvB,GAAaA,GAAatvB,EAASyxB,UAAUkT,iBAAmBnoC,OAIjEA,KAAKwP,SAAoChM,EAASyxB,UAAW,UAAU,KACtEj1B,KAAK8yB,UAAYtvB,EAASsvB,WAAatvB,EAASyxB,UAAUkT,iBAAmBnoC,IAAI,GAEnF,CAEO,OAAAgmB,GACNhmB,KAAK2P,eACN,EAKDu4B,GAAgBrmC,UAAUqM,GAAK,SAAUjD,EAAcjJ,GACtD,OAAMA,EAOEA,IAAShC,KAAKgC,OACX,oBAATiJ,GAAuC,yBAATA,GAErB,qBAATA,GAAwC,0BAATA,GACtB,YAATA,GAA+B,iBAATA,GAVP,oBAATA,GAAuC,yBAATA,GAE3B,qBAATA,GAAwC,0BAATA,GACtB,YAATA,GAA+B,iBAATA,GACb,SAATA,GAA4B,cAATA,CAStB,ECpGA,MAAMm9B,GAAiBtnC,OAAQ,YAOhB,MAAMunC,WAA4BH,GAOhD,WAAAnmC,CACCyB,EACAxB,GAEAwL,MAAOhK,EAAUxB,GAEjBhC,KAAKo1B,SAAW,MACjB,CAQA,YAAWA,GACV,OAAOp1B,KAAKwmC,kBAAmB4B,GAChC,CAEA,YAAWhT,CAAUA,GACpBp1B,KAAK2nC,mBAAoBS,GAAgBhT,EAC1C,CAWA,SAAWkT,CAAOtmC,GACfhC,KAAcgC,KAAOA,CACxB,EAKDqmC,GAAoBxmC,UAAUqM,GAAK,SAAUjD,EAAcjJ,GAC1D,OAAMA,EAQEA,IAAShC,KAAKgC,OACX,gBAATiJ,GAAmC,qBAATA,GAEjB,oBAATA,GAAuC,yBAATA,GACrB,qBAATA,GAAwC,0BAATA,GACtB,YAATA,GAA+B,iBAATA,GAZP,gBAATA,GAAmC,qBAATA,GAEvB,oBAATA,GAAuC,yBAATA,GACrB,qBAATA,GAAwC,0BAATA,GACtB,YAATA,GAA+B,iBAATA,GACb,SAATA,GAA4B,cAATA,CAUtB,EC5De,MAAMs9B,GAwDpB,WAAAxmC,CAAawC,EAA6B,CAAC,GAC1C,IAAMA,EAAQikC,aAAejkC,EAAQkkC,cAMpC,MAAM,IAAI,EACT,qCACA,MAIF,GAAKlkC,EAAQggB,WAAkC,WAArBhgB,EAAQggB,WAA+C,YAArBhgB,EAAQggB,UAMnE,MAAM,IAAI,EAAe,qCAAsChgB,EAAQkkC,cAAe,CAAElkB,UAAWhgB,EAAQggB,YAG5GvkB,KAAKwoC,WAAajkC,EAAQikC,YAAc,KAEnCjkC,EAAQkkC,cACZzoC,KAAK0oC,UAAYC,GAASC,UAAWrkC,EAAQkkC,eAE7CzoC,KAAK0oC,UAAYC,GAASC,UAAWrkC,EAAQikC,WAAkC,YAArBjkC,EAAQggB,UAA0B,MAAQ,UAGrGvkB,KAAKukB,UAAYhgB,EAAQggB,WAAa,UACtCvkB,KAAK6oC,mBAAqBtkC,EAAQskC,iBAClC7oC,KAAK8oC,UAAYvkC,EAAQukC,QACzB9oC,KAAK+oC,mBAAqBxkC,EAAQwkC,iBAElC/oC,KAAKgpC,qBAAuBhpC,KAAKwoC,WAAaxoC,KAAKwoC,WAAW9H,MAAM9hC,OAAS,KAC7EoB,KAAKipC,mBAAqBjpC,KAAKwoC,WAAaxoC,KAAKwoC,WAAW7H,IAAI/hC,OAAS,IAC1E,CAKA,CAASkC,OAAOC,YACf,OAAOf,IACR,CAMA,YAAWylB,GACV,OAAOzlB,KAAK0oC,SACb,CAgBO,IAAAQ,CAAMA,GACZ,IAAIC,EACAC,EAEJ,GACCA,EAAeppC,KAAKylB,SACpB0jB,EAAanpC,KAAKsB,cACR6nC,EAAW5nC,MAAQ2nC,EAAMC,EAAW3wC,QAEzC2wC,EAAW5nC,OAChBvB,KAAK0oC,UAAYU,EAEnB,CAQO,IAAA9nC,GACN,MAAuB,WAAlBtB,KAAKukB,UACFvkB,KAAKqpC,QAELrpC,KAAKspC,WAEd,CAKQ,KAAAD,GACP,IAAI5jB,EAAWzlB,KAAKylB,SAAS1C,QAC7B,MAAMwmB,EAAmBvpC,KAAKylB,SACxB7mB,EAAS6mB,EAAS7mB,OAGxB,GAAuB,OAAlBA,EAAOA,QAAmB6mB,EAAS7J,SAAahd,EAAgB08B,WACpE,MAAO,CAAE/5B,MAAM,EAAM/I,WAAO,GAI7B,GAAKoG,IAAWoB,KAAKipC,oBAAsBxjB,EAAS7J,QAAU5b,KAAKwoC,WAAY7H,IAAI/kB,OAClF,MAAO,CAAEra,MAAM,EAAM/I,WAAO,GAI7B,IAAI2G,EAGJ,GAAKP,aAAkBs+B,GAAO,CAC7B,GAAKzX,EAAS+jB,QAIb,OAFAxpC,KAAK0oC,UAAYC,GAASc,aAAc7qC,GAEjCoB,KAAKqpC,QAGblqC,EAAOP,EAAOkE,KAAM2iB,EAAS7J,OAC9B,MACCzc,EAASP,EAAuCi9B,SAAUpW,EAAS7J,QAGpE,GAAKzc,aAAgB,GAAU,CAC9B,GAAMa,KAAK8oC,QAEJ,CAEN,GAAK9oC,KAAKwoC,YAAcxoC,KAAKwoC,WAAW7H,IAAIjE,SAAUjX,GACrD,MAAO,CAAElkB,MAAM,EAAM/I,WAAO,GAG7BitB,EAAS7J,QACV,MARC6J,EAAW,IAAIkjB,GAAUxpC,EAAM,GAYhC,OAFAa,KAAK0oC,UAAYjjB,EAEVzlB,KAAK0pC,mBAAoB,eAAgBvqC,EAAMoqC,EAAkB9jB,EAAU,EACnF,CAEA,GAAKtmB,aAAgB+9B,GAAO,CAC3B,GAAKl9B,KAAK6oC,iBAIT,OAHApjB,EAAW,IAAIkjB,GAAUxpC,EAAM,GAC/Ba,KAAK0oC,UAAYjjB,EAEVzlB,KAAKqpC,QAGb,IACIppC,EADA0pC,EAAkBxqC,EAAK2D,KAAKxK,OAgBhC,OAZK6G,GAAQa,KAAKipC,oBACjBU,EAAkB3pC,KAAKwoC,WAAY7H,IAAI/kB,OACvC3b,EAAO,IAAIu9B,GAAWr+B,EAAM,EAAGwqC,GAC/BlkB,EAAWkjB,GAASc,aAAcxpC,KAElCA,EAAO,IAAIu9B,GAAWr+B,EAAM,EAAGA,EAAK2D,KAAKxK,QAEzCmtB,EAAS7J,UAGV5b,KAAK0oC,UAAYjjB,EAEVzlB,KAAK0pC,mBAAoB,OAAQzpC,EAAMspC,EAAkB9jB,EAAUkkB,EAC3E,CAEA,GAAoB,iBAARxqC,EAAmB,CAC9B,IAAIyqC,EAEJ,GAAK5pC,KAAK6oC,iBACTe,EAAa,MACP,CAINA,GAFkBhrC,IAAWoB,KAAKipC,mBAAqBjpC,KAAKwoC,WAAY7H,IAAI/kB,OAAWhd,EAAiBkE,KAAKxK,QAEpFmtB,EAAS7J,MACnC,CAEA,MAAMiuB,EAAY,IAAIrM,GAAW5+B,EAAgB6mB,EAAS7J,OAAQguB,GAKlE,OAHAnkB,EAAS7J,QAAUguB,EACnB5pC,KAAK0oC,UAAYjjB,EAEVzlB,KAAK0pC,mBAAoB,OAAQG,EAAWN,EAAkB9jB,EAAUmkB,EAChF,CAMA,OAHAnkB,EAAWkjB,GAASc,aAAc7qC,GAClCoB,KAAK0oC,UAAYjjB,EAEZzlB,KAAK+oC,iBACF/oC,KAAKqpC,QAGNrpC,KAAK0pC,mBAAoB,aAAc9qC,EAAe2qC,EAAkB9jB,EAChF,CAKQ,SAAA6jB,GACP,IAAI7jB,EAAWzlB,KAAKylB,SAAS1C,QAC7B,MAAMwmB,EAAmBvpC,KAAKylB,SACxB7mB,EAAS6mB,EAAS7mB,OAGxB,GAAuB,OAAlBA,EAAOA,QAAuC,IAApB6mB,EAAS7J,OACvC,MAAO,CAAEra,MAAM,EAAM/I,WAAO,GAI7B,GAAKoG,GAAUoB,KAAKgpC,sBAAwBvjB,EAAS7J,QAAU5b,KAAKwoC,WAAY9H,MAAM9kB,OACrF,MAAO,CAAEra,MAAM,EAAM/I,WAAO,GAI7B,IAAI2G,EAGJ,GAAKP,aAAkBs+B,GAAO,CAC7B,GAAKzX,EAASqkB,UAIb,OAFA9pC,KAAK0oC,UAAYC,GAASoB,cAAenrC,GAElCoB,KAAKspC,YAGbnqC,EAAOP,EAAOkE,KAAM2iB,EAAS7J,OAAS,EACvC,MACCzc,EAASP,EAAuCi9B,SAAUpW,EAAS7J,OAAS,GAG7E,GAAKzc,aAAgB,GACpB,OAAKa,KAAK8oC,SACTrjB,EAAS7J,SACT5b,KAAK0oC,UAAYjjB,EAEVzlB,KAAK0pC,mBAAoB,eAAgBvqC,EAAMoqC,EAAkB9jB,EAAU,KAGnFA,EAAW,IAAIkjB,GAAUxpC,EAAMA,EAAKm8B,YACpCt7B,KAAK0oC,UAAYjjB,EAEZzlB,KAAK+oC,iBACF/oC,KAAKspC,YAGNtpC,KAAK0pC,mBAAoB,aAAcvqC,EAAMoqC,EAAkB9jB,IAGvE,GAAKtmB,aAAgB+9B,GAAO,CAC3B,GAAKl9B,KAAK6oC,iBAIT,OAHApjB,EAAW,IAAIkjB,GAAUxpC,EAAMA,EAAK2D,KAAKxK,QACzC0H,KAAK0oC,UAAYjjB,EAEVzlB,KAAKspC,YAGb,IACIrpC,EADA0pC,EAAkBxqC,EAAK2D,KAAKxK,OAIhC,GAAK6G,GAAQa,KAAKgpC,qBAAuB,CACxC,MAAMptB,EAAS5b,KAAKwoC,WAAY9H,MAAM9kB,OAEtC3b,EAAO,IAAIu9B,GAAWr+B,EAAMyc,EAAQzc,EAAK2D,KAAKxK,OAASsjB,GACvD+tB,EAAkB1pC,EAAK6C,KAAKxK,OAC5BmtB,EAAWkjB,GAASoB,cAAe9pC,EACpC,MACCA,EAAO,IAAIu9B,GAAWr+B,EAAM,EAAGA,EAAK2D,KAAKxK,QAEzCmtB,EAAS7J,SAKV,OAFA5b,KAAK0oC,UAAYjjB,EAEVzlB,KAAK0pC,mBAAoB,OAAQzpC,EAAMspC,EAAkB9jB,EAAUkkB,EAC3E,CAEA,GAAoB,iBAARxqC,EAAmB,CAC9B,IAAIyqC,EAEJ,GAAM5pC,KAAK6oC,iBAMVe,EAAa,MANgB,CAE7B,MAAMI,EAAcprC,IAAWoB,KAAKgpC,qBAAuBhpC,KAAKwoC,WAAY9H,MAAM9kB,OAAS,EAE3FguB,EAAankB,EAAS7J,OAASouB,CAChC,CAIAvkB,EAAS7J,QAAUguB,EAEnB,MAAMC,EAAY,IAAIrM,GAAW5+B,EAAgB6mB,EAAS7J,OAAQguB,GAIlE,OAFA5pC,KAAK0oC,UAAYjjB,EAEVzlB,KAAK0pC,mBAAoB,OAAQG,EAAWN,EAAkB9jB,EAAUmkB,EAChF,CAMA,OAHAnkB,EAAWkjB,GAASoB,cAAenrC,GACnCoB,KAAK0oC,UAAYjjB,EAEVzlB,KAAK0pC,mBAAoB,eAAgB9qC,EAAmB2qC,EAAkB9jB,EAAU,EAChG,CAWQ,kBAAAikB,CACPz+B,EACAhL,EACAspC,EACAU,EACA3xC,GA8BA,OAxBK2H,aAAgBu9B,KAEfv9B,EAAKy9B,aAAez9B,EAAK6C,KAAKxK,QAAU2H,EAAKw9B,SAAS36B,KAAKxK,SACxC,WAAlB0H,KAAKukB,WAA6BvkB,KAAKwoC,YAAcxoC,KAAKwoC,WAAW7H,IAAI1c,QAASjkB,KAAKylB,UAK3F8jB,EAAmBZ,GAASc,aAAcxpC,EAAKw9B,WAJ/CwM,EAAetB,GAASc,aAAcxpC,EAAKw9B,UAE3Cz9B,KAAK0oC,UAAYuB,IAOQ,IAAtBhqC,EAAKy9B,eACc,YAAlB19B,KAAKukB,WAA8BvkB,KAAKwoC,YAAcxoC,KAAKwoC,WAAW9H,MAAMzc,QAASjkB,KAAKylB,UAK9F8jB,EAAmBZ,GAASoB,cAAe9pC,EAAKw9B,WAJhDwM,EAAetB,GAASoB,cAAe9pC,EAAKw9B,UAE5Cz9B,KAAK0oC,UAAYuB,KAOb,CACN1oC,MAAM,EACN/I,MAAO,CACNyS,OACAhL,OACAspC,mBACAU,eACA3xC,UAGH,EChac,MAAMqwC,WAAiB7M,GAiBrC,WAAA/5B,CAAanD,EAAiCgd,GAC7CpO,QAEAxN,KAAKpB,OAASA,EACdoB,KAAK4b,OAASA,CACf,CAMA,aAAWsuB,GACV,OAAKlqC,KAAKpB,OAAOsP,GAAI,SACb,KAGClO,KAAKpB,OAAoBi9B,SAAU77B,KAAK4b,SAAY,IAC9D,CAMA,cAAWuuB,GACV,OAAKnqC,KAAKpB,OAAOsP,GAAI,SACb,KAGClO,KAAKpB,OAAoBi9B,SAAU77B,KAAK4b,OAAS,IAAO,IAClE,CAKA,aAAWkuB,GACV,OAAuB,IAAhB9pC,KAAK4b,MACb,CAKA,WAAW4tB,GACV,MAAMY,EAAYpqC,KAAKpB,OAAOsP,GAAI,SAAYlO,KAAKpB,OAAOkE,KAAKxK,OAAW0H,KAAKpB,OAAgB08B,WAE/F,OAAOt7B,KAAK4b,SAAWwuB,CACxB,CAKA,QAAW9zC,GACV,OAAO0J,KAAKpB,OAAOtI,IACpB,CAMA,mBAAW6xC,GACV,IAAIkC,EAAWrqC,KAAKpB,OAEpB,OAAWyrC,aAAoBnC,KAAoB,CAClD,IAAKmC,EAASzrC,OAGb,OAAO,KAFPyrC,EAAWA,EAASzrC,MAItB,CAEA,OAAOyrC,CACR,CAQO,YAAAC,CAAcne,GACpB,MAAMoe,EAAU5B,GAASC,UAAW5oC,MAE9B4b,EAAS2uB,EAAQ3uB,OAASuQ,EAGhC,OAFAoe,EAAQ3uB,OAASA,EAAS,EAAI,EAAIA,EAE3B2uB,CACR,CAmBO,uBAAAC,CAAyBtB,EAA6C3kC,EAA6B,CAAC,GAC1GA,EAAQkkC,cAAgBzoC,KAExB,MAAMyqC,EAAa,IAAIlC,GAAYhkC,GAGnC,OAFAkmC,EAAWvB,KAAMA,GAEVuB,EAAWhlB,QACnB,CAOO,YAAA0W,GACN,OAAKn8B,KAAKpB,OAAOsP,GAAI,oBACb,CAAElO,KAAKpB,QAEPoB,KAAKpB,OAAOu9B,aAAc,CAAEE,aAAa,GAElD,CAMO,iBAAAE,CAAmB9W,GACzB,MAAM+W,EAAax8B,KAAKm8B,eAClBM,EAAahX,EAAS0W,eAE5B,IAAI3hC,EAAI,EAER,KAAQgiC,EAAYhiC,IAAOiiC,EAAYjiC,IAAOgiC,EAAYhiC,IACzDA,IAGD,OAAa,IAANA,EAAU,KAAOgiC,EAAYhiC,EAAI,EACzC,CAQO,OAAAypB,CAASymB,GACf,OAAS1qC,KAAKpB,QAAU8rC,EAAc9rC,QAAUoB,KAAK4b,QAAU8uB,EAAc9uB,MAC9E,CAYO,QAAA8gB,CAAUgO,GAChB,MAA4C,UAArC1qC,KAAK2qC,YAAaD,EAC1B,CAYO,OAAA7N,CAAS6N,GACf,MAA4C,SAArC1qC,KAAK2qC,YAAaD,EAC1B,CAQO,WAAAC,CAAaD,GACnB,GAAK1qC,KAAK1J,OAASo0C,EAAcp0C,KAChC,MAAO,YAGR,GAAK0J,KAAKikB,QAASymB,GAClB,MAAO,OAIR,MAAM/N,EAAW38B,KAAKpB,OAAOsP,GAAI,QAAWlO,KAAKpB,OAAOs9B,UAAY,GAC9D0O,EAAYF,EAAc9rC,OAAOsP,GAAI,QAAWw8B,EAAc9rC,OAAOs9B,UAAY,GAGvFS,EAAS/8B,KAAMI,KAAK4b,QACpBgvB,EAAUhrC,KAAM8qC,EAAc9uB,QAG9B,MAAMzd,EAASuY,EAAeimB,EAAUiO,GAExC,OAASzsC,GACR,IAAK,SACJ,MAAO,SAER,IAAK,YACJ,MAAO,QAER,QAEC,OAAOw+B,EAAUx+B,GAAqBysC,EAAWzsC,GAAqB,SAAW,QAEpF,CAOO,SAAA0sC,CAAWtmC,EAA6B,CAAC,GAG/C,OAFAA,EAAQkkC,cAAgBzoC,KAEjB,IAAIuoC,GAAYhkC,EACxB,CAKO,KAAAwe,GACN,OAAO,IAAI4lB,GAAU3oC,KAAKpB,OAAQoB,KAAK4b,OACxC,CAkBA,gBAAcgtB,CAAWkC,EAAiClvB,GACzD,GAAKkvB,aAA0BnC,GAC9B,OAAO,IAAI3oC,KAAM8qC,EAAelsC,OAAQksC,EAAelvB,QACjD,CACN,MAAMzc,EAAO2rC,EAEb,GAAe,OAAVlvB,EACJA,EAASzc,EAAK+O,GAAI,SAAY/O,EAAK2D,KAAKxK,OAAW6G,EAAcm8B,eAC3D,IAAe,UAAV1f,EACX,OAAO5b,KAAK+pC,cAAe5qC,GACrB,GAAe,SAAVyc,EACX,OAAO5b,KAAKypC,aAActqC,GACpB,GAAgB,IAAXyc,IAAiBA,EAO5B,MAAM,IAAI,EAAe,wCAAyCzc,EACnE,CAEA,OAAO,IAAIwpC,GAAUxpC,EAAayc,EACnC,CACD,CAQA,mBAAc6tB,CAAcxpC,GAE3B,GAAKA,EAAKiO,GAAI,cACb,OAAO,IAAIy6B,GAAU1oC,EAAKw9B,SAAUx9B,EAAKy9B,aAAez9B,EAAK6C,KAAKxK,QAGnE,IAAM2H,EAAKrB,OAOV,MAAM,IAAI,EAAe,2BAA4BqB,EAAM,CAAE3J,KAAM2J,IAGpE,OAAO,IAAI0oC,GAAU1oC,EAAKrB,OAAUqB,EAAK2E,MAAoB,EAC9D,CAQA,oBAAcmlC,CAAe9pC,GAE5B,GAAKA,EAAKiO,GAAI,cACb,OAAO,IAAIy6B,GAAU1oC,EAAKw9B,SAAUx9B,EAAKy9B,cAG1C,IAAMz9B,EAAKrB,OAOV,MAAM,IAAI,EAAe,4BAA6BqB,EAAM,CAAE3J,KAAM2J,IAGrE,OAAO,IAAI0oC,GAAU1oC,EAAKrB,OAAQqB,EAAK2E,MACxC,EAKD+jC,GAAS9mC,UAAUqM,GAAK,SAAUjD,GACjC,MAAgB,aAATA,GAAgC,kBAATA,CAC/B,ECtWe,MAAM8/B,WAAcjP,GAmBlC,WAAA/5B,CAAa2+B,EAAiBC,EAAuB,MACpDnzB,QAEAxN,KAAK0gC,MAAQA,EAAM3d,QACnB/iB,KAAK2gC,IAAMA,EAAMA,EAAI5d,QAAU2d,EAAM3d,OACtC,CAaA,EAAUjiB,OAAOC,kBACT,IAAIwnC,GAAY,CAAEC,WAAYxoC,KAAM+oC,kBAAkB,GAC9D,CAKA,eAAWiC,GACV,OAAOhrC,KAAK0gC,MAAMzc,QAASjkB,KAAK2gC,IACjC,CAMA,UAAW7iB,GACV,OAAO9d,KAAK0gC,MAAM9hC,SAAWoB,KAAK2gC,IAAI/hC,MACvC,CAKA,QAAWtI,GACV,OAAO0J,KAAK0gC,MAAMpqC,IACnB,CAqBO,WAAA20C,GACN,IAAIvK,EAAQ1gC,KAAK0gC,MAAM8J,wBAAyBU,GAAiB,CAAE3mB,UAAW,aAC1Eoc,EAAM3gC,KAAK2gC,IAAI6J,wBAAyBU,IAW5C,OARKxK,EAAM9hC,OAAOsP,GAAI,UAAawyB,EAAMoJ,YACxCpJ,EAAQiI,GAASoB,cAAerJ,EAAM9hC,SAGlC+hC,EAAI/hC,OAAOsP,GAAI,UAAayyB,EAAI6I,UACpC7I,EAAMgI,GAASc,aAAc9I,EAAI/hC,SAG3B,IAAImsC,GAAOrK,EAAOC,EAC1B,CAqBO,UAAAwK,GACN,IAAIzK,EAAQ1gC,KAAK0gC,MAAM8J,wBAAyBU,IAEhD,GAAKxK,EAAM7D,QAAS78B,KAAK2gC,MAASD,EAAMzc,QAASjkB,KAAK2gC,KACrD,OAAO,IAAIoK,GAAOrK,EAAOA,GAG1B,IAAIC,EAAM3gC,KAAK2gC,IAAI6J,wBAAyBU,GAAiB,CAAE3mB,UAAW,aAC1E,MAAM6mB,EAAiB1K,EAAMwJ,UACvBmB,EAAgB1K,EAAIwJ,WAW1B,OARKiB,GAAkBA,EAAel9B,GAAI,WACzCwyB,EAAQ,IAAIiI,GAAUyC,EAAgB,IAGlCC,GAAiBA,EAAcn9B,GAAI,WACvCyyB,EAAM,IAAIgI,GAAU0C,EAAeA,EAAcvoC,KAAKxK,SAGhD,IAAIyyC,GAAOrK,EAAOC,EAC1B,CAQO,OAAA1c,CAASqnB,GACf,OAAOtrC,MAAQsrC,GAAgBtrC,KAAK0gC,MAAMzc,QAASqnB,EAAW5K,QAAW1gC,KAAK2gC,IAAI1c,QAASqnB,EAAW3K,IACvG,CAQO,gBAAA4K,CAAkB9lB,GACxB,OAAOA,EAASoX,QAAS78B,KAAK0gC,QAAWjb,EAASiX,SAAU18B,KAAK2gC,IAClE,CAYO,aAAA6K,CAAeF,EAAmBG,GAAiB,GACpDH,EAAWN,cACfS,GAAQ,GAGT,MAAMC,EAAgB1rC,KAAKurC,iBAAkBD,EAAW5K,QAAa+K,GAASzrC,KAAK0gC,MAAMzc,QAASqnB,EAAW5K,OACvGiL,EAAc3rC,KAAKurC,iBAAkBD,EAAW3K,MAAW8K,GAASzrC,KAAK2gC,IAAI1c,QAASqnB,EAAW3K,KAEvG,OAAO+K,GAAiBC,CACzB,CAmCO,aAAAC,CAAeN,GACrB,MAAMO,EAAuB,GAqB7B,OAnBK7rC,KAAK8rC,eAAgBR,IAGpBtrC,KAAKurC,iBAAkBD,EAAW5K,QAGtCmL,EAAOjsC,KAAM,IAAImrC,GAAO/qC,KAAK0gC,MAAO4K,EAAW5K,QAG3C1gC,KAAKurC,iBAAkBD,EAAW3K,MAGtCkL,EAAOjsC,KAAM,IAAImrC,GAAOO,EAAW3K,IAAK3gC,KAAK2gC,OAI9CkL,EAAOjsC,KAAMI,KAAK+iB,SAGZ8oB,CACR,CAyBO,eAAA3oB,CAAiBooB,GACvB,GAAKtrC,KAAK8rC,eAAgBR,GAAe,CAGxC,IAAIS,EAAmB/rC,KAAK0gC,MACxBsL,EAAiBhsC,KAAK2gC,IAc1B,OAZK3gC,KAAKurC,iBAAkBD,EAAW5K,SAGtCqL,EAAmBT,EAAW5K,OAG1B1gC,KAAKurC,iBAAkBD,EAAW3K,OAGtCqL,EAAiBV,EAAW3K,KAGtB,IAAIoK,GAAOgB,EAAkBC,EACrC,CAGA,OAAO,IACR,CAOO,SAAAnB,CAAWtmC,EAA6B,CAAC,GAG/C,OAFAA,EAAQikC,WAAaxoC,KAEd,IAAIuoC,GAAYhkC,EACxB,CAMO,iBAAAg4B,GACN,OAAOv8B,KAAK0gC,MAAMnE,kBAAmBv8B,KAAK2gC,IAC3C,CAOO,mBAAAsL,GACN,GAAKjsC,KAAKgrC,YACT,OAAO,KAGR,IAAII,EAAiBprC,KAAK0gC,MAAMwJ,UAC5BmB,EAAgBrrC,KAAK2gC,IAAIwJ,WAmB7B,OARKnqC,KAAK0gC,MAAM9hC,OAAOsP,GAAI,UAAalO,KAAK0gC,MAAM8I,SAAWxpC,KAAK0gC,MAAM9hC,OAAOq9B,cAC/EmP,EAAiBprC,KAAK0gC,MAAM9hC,OAAOq9B,aAG/Bj8B,KAAK2gC,IAAI/hC,OAAOsP,GAAI,UAAalO,KAAK2gC,IAAImJ,WAAa9pC,KAAK2gC,IAAI/hC,OAAOgoB,kBAC3EykB,EAAgBrrC,KAAK2gC,IAAI/hC,OAAOgoB,iBAG5BwkB,GAAkBA,EAAel9B,GAAI,YAAek9B,IAAmBC,EACpED,EAGD,IACR,CAKO,KAAAroB,GACN,OAAO,IAAIgoB,GAAO/qC,KAAK0gC,MAAO1gC,KAAK2gC,IACpC,CAeA,SAAQuL,CAAU3nC,EAA6B,CAAC,GAC/CA,EAAQikC,WAAaxoC,KACrBuE,EAAQwkC,kBAAmB,EAE3B,MAAM0B,EAAa,IAAIlC,GAAYhkC,GAEnC,UAAY/L,KAASiyC,QACdjyC,EAAMyH,IAEd,CAcA,aAAQksC,CAAc5nC,EAA6B,CAAC,GACnDA,EAAQikC,WAAaxoC,KAErB,MAAMyqC,EAAa,IAAIlC,GAAYhkC,SAE7BkmC,EAAWhlB,SAEjB,UAAYjtB,KAASiyC,QACdjyC,EAAMyxC,YAEd,CAQO,cAAA6B,CAAgBR,GACtB,OAAOtrC,KAAK0gC,MAAMhE,SAAU4O,EAAW3K,MAAS3gC,KAAK2gC,IAAI9D,QAASyO,EAAW5K,MAC9E,CAYA,mCAAc0L,CACbC,EACArC,EACAsC,EACAlC,GAEA,OAAO,IAAIpqC,KACV,IAAI2oC,GAAU0D,EAAcrC,GAC5B,IAAIrB,GAAU2D,EAAYlC,GAE5B,CAUA,kCAAcmC,CAA6B9mB,EAAoB0G,GAC9D,MAAMuU,EAAQjb,EACRkb,EAAMlb,EAAS6kB,aAAcne,GAEnC,OAAOA,EAAQ,EAAI,IAAInsB,KAAM0gC,EAAOC,GAAQ,IAAI3gC,KAAM2gC,EAAKD,EAC5D,CASA,gBAAc8L,CAAW/0B,GACxB,OAAOzX,KAAKosC,6BAA8B30B,EAAS,EAAGA,EAASA,EAAQ6jB,WACxE,CAOA,gBAAcmR,CAAWxsC,GACxB,MAAM4T,EAAO5T,EAAKiO,GAAI,cAAiBjO,EAAK09B,WAAa,EAEzD,OAAO39B,KAAKusC,4BAA6B5D,GAASoB,cAAe9pC,GAAQ4T,EAC1E,EAYD,SAASq3B,GAAiB1yC,GACzB,SAAKA,EAAMyH,KAAKiO,GAAI,sBAAwB1V,EAAMyH,KAAKiO,GAAI,aAK5D,CAbA68B,GAAMlpC,UAAUqM,GAAK,SAAUjD,GAC9B,MAAgB,UAATA,GAA6B,eAATA,CAC5B,EC9ce,MAAMyhC,WAAkBt9B,EAAc0sB,KAsFpD,WAAA/5B,IACI7F,GASHsR,QAEAxN,KAAK2sC,QAAU,GACf3sC,KAAK4sC,oBAAqB,EAC1B5sC,KAAK6sC,SAAU,EACf7sC,KAAK8sC,oBAAsB,GAEtB5wC,EAAK5D,QACT0H,KAAKoiC,SAAUlmC,EAEjB,CAOA,UAAW6wC,GACV,OAAO/sC,KAAK6sC,OACb,CAOA,sBAAWG,GACV,OAAOhtC,KAAK8sC,mBACb,CAUA,UAAWpR,GACV,IAAM17B,KAAK2sC,QAAQr0C,OAClB,OAAO,KAGR,MAAMwsB,EAAQ9kB,KAAK2sC,QAAS3sC,KAAK2sC,QAAQr0C,OAAS,GAGlD,OAFe0H,KAAK4sC,mBAAqB9nB,EAAM6b,IAAM7b,EAAM4b,OAE7C3d,OACf,CAOA,SAAWkqB,GACV,IAAMjtC,KAAK2sC,QAAQr0C,OAClB,OAAO,KAGR,MAAMwsB,EAAQ9kB,KAAK2sC,QAAS3sC,KAAK2sC,QAAQr0C,OAAS,GAGlD,OAFc0H,KAAK4sC,mBAAqB9nB,EAAM4b,MAAQ5b,EAAM6b,KAE/C5d,OACd,CAMA,eAAWioB,GACV,OAA2B,IAApBhrC,KAAKktC,YAAoBltC,KAAK2sC,QAAS,GAAI3B,WACnD,CAKA,cAAWkC,GACV,OAAOltC,KAAK2sC,QAAQr0C,MACrB,CAKA,cAAW60C,GACV,OAAQntC,KAAKgrC,aAAehrC,KAAK4sC,kBAClC,CAMA,mBAAWzE,GACV,OAAKnoC,KAAK07B,OACF17B,KAAK07B,OAAOyM,gBAGb,IACR,CAKA,UAAQiF,GACP,UAAYtoB,KAAS9kB,KAAK2sC,cACnB7nB,EAAM/B,OAEd,CAQO,aAAAsqB,GACN,IAAI5c,EAAQ,KAEZ,UAAY3L,KAAS9kB,KAAK2sC,QACnBlc,IAAS3L,EAAM4b,MAAMhE,SAAUjM,EAAMiQ,SAC1CjQ,EAAQ3L,GAIV,OAAO2L,EAAQA,EAAM1N,QAAU,IAChC,CAOO,YAAAuqB,GACN,IAAI5c,EAAO,KAEX,UAAY5L,KAAS9kB,KAAK2sC,QACnBjc,IAAQ5L,EAAM6b,IAAI9D,QAASnM,EAAKiQ,OACrCjQ,EAAO5L,GAIT,OAAO4L,EAAOA,EAAK3N,QAAU,IAC9B,CAOO,gBAAAoS,GACN,MAAMoY,EAAavtC,KAAKqtC,gBAExB,OAAOE,EAAaA,EAAW7M,MAAM3d,QAAU,IAChD,CAOO,eAAAyqB,GACN,MAAMC,EAAYztC,KAAKstC,eAEvB,OAAOG,EAAYA,EAAU9M,IAAI5d,QAAU,IAC5C,CASO,OAAAkB,CAASypB,GACf,GAAK1tC,KAAK+sC,QAAUW,EAAeX,OAClC,OAAO,EAGR,GAAK/sC,KAAK+sC,QAAU/sC,KAAKgtC,oBAAsBU,EAAeV,mBAC7D,OAAO,EAGR,GAAKhtC,KAAKktC,YAAcQ,EAAeR,WACtC,OAAO,EACD,GAAyB,IAApBltC,KAAKktC,WAChB,OAAO,EAGR,IAAMltC,KAAK07B,OAAQzX,QAASypB,EAAehS,UAAc17B,KAAKitC,MAAOhpB,QAASypB,EAAeT,OAC5F,OAAO,EAGR,UAAYU,KAAa3tC,KAAK2sC,QAAU,CACvC,IAAIiB,GAAQ,EAEZ,UAAYtC,KAAgBoC,EAAwBf,QACnD,GAAKgB,EAAU1pB,QAASqnB,GAAe,CACtCsC,GAAQ,EACR,KACD,CAGD,IAAMA,EACL,OAAO,CAET,CAEA,OAAO,CACR,CAUO,SAAAvQ,CAAWqQ,GACjB,GAAK1tC,KAAKmtC,YAAcO,EAAeP,WACtC,OAAO,EAGR,MAAMU,EAAelpC,EAAO3E,KAAKotC,aAIjC,GAAKS,GAHgBlpC,EAAO+oC,EAAeN,aAI1C,OAAO,EAIR,GAAqB,GAAhBS,EACJ,OAAO,EAIR,QAAUC,KAAU9tC,KAAKotC,YAAc,CACtCU,EAASA,EAAO3C,aAEhB,IAAIyC,GAAQ,EAEZ,QAAUG,KAAUL,EAAeN,YAGlC,GAFAW,EAASA,EAAO5C,aAEX2C,EAAOpN,MAAMzc,QAAS8pB,EAAOrN,QAAWoN,EAAOnN,IAAI1c,QAAS8pB,EAAOpN,KAAQ,CAC/EiN,GAAQ,EACR,KACD,CAID,IAAMA,EACL,OAAO,CAET,CAGA,OAAO,CACR,CAOO,kBAAAI,GACN,OAAyB,IAApBhuC,KAAKktC,WACF,KAGDltC,KAAKqtC,gBAAiBpB,qBAC9B,CAmEO,KAAA7J,IACHlmC,GASH,IAAM+xC,EAAYC,EAAe3pC,GAAYrI,EAO7C,GAL6B,iBAAjBgyC,IACX3pC,EAAU2pC,EACVA,OAAgB,GAGG,OAAfD,EACJjuC,KAAKmuC,WAAY,IACjBnuC,KAAKouC,gBAAiB7pC,QAChB,GAAK0pC,aAAsBvB,IAAauB,aAAsBI,GACpEruC,KAAKmuC,WAAYF,EAAWb,YAAaa,EAAWd,YACpDntC,KAAKouC,gBAAiB,CAAEE,KAAML,EAAWlB,OAAQwB,MAAON,EAAWjB,0BAC7D,GAAKiB,aAAsBlD,GACjC/qC,KAAKmuC,WAAY,CAAEF,GAAc1pC,GAAWA,EAAQiqC,UACpDxuC,KAAKouC,gBAAiB7pC,QAChB,GAAK0pC,aAAsBtF,GACjC3oC,KAAKmuC,WAAY,CAAE,IAAIpD,GAAOkD,KAC9BjuC,KAAKouC,gBAAiB7pC,QAChB,GAAK0pC,aAAsB,GAAO,CACxC,MAAMO,IAAajqC,KAAaA,EAAQiqC,SACxC,IAAI1pB,EAEJ,QAAuB,IAAlBopB,EAMJ,MAAM,IAAI,EAAe,iDAAkDluC,MAE3E8kB,EAD4B,MAAjBopB,EACHnD,GAAMyB,UAAWyB,GACG,MAAjBC,EACHnD,GAAM0B,UAAWwB,GAEjB,IAAIlD,GAAOpC,GAASC,UAAWqF,EAAYC,IAGpDluC,KAAKmuC,WAAY,CAAErpB,GAAS0pB,GAC5BxuC,KAAKouC,gBAAiB7pC,EACvB,KAAO,KAAKoS,EAAYs3B,GAWvB,MAAM,IAAI,EAAe,sCAAuCjuC,MARhEA,KAAKmuC,WAAYF,EAAY1pC,GAAWA,EAAQiqC,UAChDxuC,KAAKouC,gBAAiB7pC,EAQvB,CAEAvE,KAAKwQ,KAAgC,SACtC,CAWO,QAAAi+B,CAAU3D,EAAiClvB,GACjD,GAAqB,OAAhB5b,KAAK07B,OAMT,MAAM,IAAI,EAAe,oCAAqC17B,MAG/D,MAAM0uC,EAAW/F,GAASC,UAAWkC,EAAgBlvB,GAErD,GAA4C,QAAvC8yB,EAAS/D,YAAa3qC,KAAKitC,OAC/B,OAGD,MAAMvR,EAAS17B,KAAK07B,OAEpB17B,KAAK2sC,QAAQ3tC,MAE0B,UAAlC0vC,EAAS/D,YAAajP,GAC1B17B,KAAK2uC,UAAW,IAAI5D,GAAO2D,EAAUhT,IAAU,GAE/C17B,KAAK2uC,UAAW,IAAI5D,GAAOrP,EAAQgT,IAGpC1uC,KAAKwQ,KAAgC,SACtC,CAWQ,UAAA29B,CAAYS,EAA4BC,GAA0B,GAGzED,EAAYjuC,MAAMrB,KAAMsvC,GAExB5uC,KAAK2sC,QAAU,GAEf,UAAY7nB,KAAS8pB,EACpB5uC,KAAK2uC,UAAW7pB,GAGjB9kB,KAAK4sC,qBAAuBiC,CAC7B,CAUQ,eAAAT,CAAiB7pC,EAA4B,CAAC,GACrDvE,KAAK6sC,UAAYtoC,EAAQ+pC,KACzBtuC,KAAK8sC,oBAAsBvoC,EAAQ+pC,MAAO/pC,EAAQgqC,OAAc,EACjE,CAcQ,SAAAI,CAAW7pB,EAAcqoB,GAAsB,GACtD,KAAQroB,aAAiBimB,IAMxB,MAAM,IAAI,EACT,qCACA/qC,MAIFA,KAAK8uC,WAAYhqB,GACjB9kB,KAAK4sC,qBAAuBO,CAC7B,CAQQ,UAAA2B,CAAYhqB,GACnB,UAAYiqB,KAAe/uC,KAAK2sC,QAC/B,GAAK7nB,EAAMgnB,eAAgBiD,GAQ1B,MAAM,IAAI,EACT,kCACA/uC,KACA,CAAEgvC,WAAYlqB,EAAOmqB,kBAAmBF,IAK3C/uC,KAAK2sC,QAAQ/sC,KAAM,IAAImrC,GAAOjmB,EAAM4b,MAAO5b,EAAM6b,KAClD,EAKD+L,GAAU7qC,UAAUqM,GAAK,SAAUjD,GAClC,MAAgB,cAATA,GAAiC,mBAATA,CAChC,EC9nBe,MAAMojC,WAA0Bj/B,EAAc0sB,KAiG5D,WAAA/5B,IAAgB7F,GACfsR,QAEAxN,KAAKkvC,WAAa,IAAIxC,GAGtB1sC,KAAKkvC,WAAW99B,SAAU,UAAW7R,GAAIS,MAGpC9D,EAAK5D,QACT0H,KAAKkvC,WAAW9M,SAAUlmC,EAE5B,CAOA,UAAW6wC,GACV,OAAO/sC,KAAKkvC,WAAWnC,MACxB,CAOA,sBAAWC,GACV,OAAOhtC,KAAKkvC,WAAWlC,kBACxB,CAUA,UAAWtR,GACV,OAAO17B,KAAKkvC,WAAWxT,MACxB,CAOA,SAAWuR,GACV,OAAOjtC,KAAKkvC,WAAWjC,KACxB,CAMA,eAAWjC,GACV,OAAOhrC,KAAKkvC,WAAWlE,WACxB,CAKA,cAAWkC,GACV,OAAOltC,KAAKkvC,WAAWhC,UACxB,CAKA,cAAWC,GACV,OAAOntC,KAAKkvC,WAAW/B,UACxB,CAMA,mBAAWhF,GACV,OAAOnoC,KAAKkvC,WAAW/G,eACxB,CAOA,WAAWwE,GACV,OAAS3sC,KAAKkvC,WAAoBvC,OACnC,CAKA,UAAQS,SACAptC,KAAKkvC,WAAW9B,WACxB,CAQO,aAAAC,GACN,OAAOrtC,KAAKkvC,WAAW7B,eACxB,CAOO,YAAAC,GACN,OAAOttC,KAAKkvC,WAAW5B,cACxB,CAOO,gBAAAnY,GACN,OAAOn1B,KAAKkvC,WAAW/Z,kBACxB,CAOO,eAAAqY,GACN,OAAOxtC,KAAKkvC,WAAW1B,iBACxB,CAOO,kBAAAQ,GACN,OAAOhuC,KAAKkvC,WAAWlB,oBACxB,CASO,OAAA/pB,CAASypB,GACf,OAAO1tC,KAAKkvC,WAAWjrB,QAASypB,EACjC,CAUO,SAAArQ,CAAWqQ,GACjB,OAAO1tC,KAAKkvC,WAAW7R,UAAWqQ,EACnC,CAiEO,MAAAyB,IAAWjzC,GACjB8D,KAAKkvC,WAAW9M,SAAUlmC,EAC3B,CAYO,SAAAkzC,CAAWtE,EAAiClvB,GAClD5b,KAAKkvC,WAAWT,SAAU3D,EAAgBlvB,EAC3C,EAKDyyB,GAAkBxsC,UAAUqM,GAAK,SAAUjD,GAC1C,MAAgB,cAATA,GACE,qBAARA,GACQ,kBAARA,GACQ,0BAARA,CACF,ECpXe,MAAMokC,WAA4EnjC,EAqBhG,WAAAnK,CAAamB,EAAgBlB,EAAastC,GACzC9hC,MAAOtK,EAAQlB,GAEfhC,KAAKsvC,WAAaA,EAClBtvC,KAAKuvC,YAAc,OACnBvvC,KAAKwvC,eAAiB,IACvB,CAKA,cAAWC,GACV,OAAOzvC,KAAKuvC,WACb,CAKA,iBAAWG,GACV,OAAO1vC,KAAKwvC,cACb,EC5BD,MAAMG,GAAiB7uC,OAAQ,qBAkBhB,SAAS8uC,GAAyDlrC,GA4GhF,OA3GA,cAA6BA,EAGZ,IAAA8L,CACfC,KACGo/B,GAEH,IACC,MAAMn/B,EAAYD,aAAuBvE,EAAYuE,EAAc,IAAIvE,EAAWlM,KAAMyQ,GAClFq/B,EAAgBC,GAAqB/vC,MAE3C,IAAM8vC,EAAcj8B,KACnB,OAMD,GAHAm8B,GAAiBt/B,EAAW,YAAa1Q,MAGpCiwC,GAAiBH,EAAe,WAAYp/B,KAAcm/B,GAC9D,OAAOn/B,EAAUS,OAGlB,MAAMm+B,EAAe5+B,EAAiC4+B,YAActvC,KAAKi1B,UAAUoY,gBAC7E6C,EAAkBZ,EAAaA,EAAWrD,sBAAwB,KAClEkE,IAAkBD,GAAkB5sC,QAAS8sC,GAAkBN,EAAeI,IAEpF,IAAI/wC,EAAoB+wC,GAkK5B,SAA+BprB,GAC9B,IAAMA,EACL,OAAO,KAGR,MAAMurB,EAAcvrB,EAAM4b,MAAM9hC,OAC1B0xC,EAAYxrB,EAAM6b,IAAI/hC,OAEtB2xC,EAAYF,EAAYnU,UACxBsU,EAAUF,EAAUpU,UAE1B,OAAOqU,EAAUj4C,OAASk4C,EAAQl4C,OAAS+3C,EAAcC,CAC1D,CA9K+CG,CAAsBnB,GAKjE,GAHAU,GAAiBt/B,EAAW,WAAYvR,IAGlCgxC,EAAkB,CACvB,GAAKF,GAAiBH,EAAe,QAASp/B,KAAcm/B,GAC3D,OAAOn/B,EAAUS,OAGlB6+B,GAAiBt/B,EAAW,WAAYvR,EACzC,CAEA,KAAQA,GAAO,CAEd,GAAKA,EAAK+O,GAAI,gBACb,GAAK+hC,GAAiBH,EAAe,QAASp/B,KAAcm/B,GAC3D,OAAOn/B,EAAUS,YAEnB,GAGUhS,EAAK+O,GAAI,YACb+hC,GAAiBH,EAAe3wC,EAAK6C,KAAM0O,KAAcm/B,GAC7D,OAAOn/B,EAAUS,OAKnB,GAAK8+B,GAAiBH,EAAe3wC,EAAMuR,KAAcm/B,GACxD,OAAOn/B,EAAUS,OAGlBhS,EAAOA,EAAKP,OAEZoxC,GAAiBt/B,EAAW,WAAYvR,EACzC,CAOA,OALA6wC,GAAiBt/B,EAAW,WAAY1Q,MAGxCiwC,GAAiBH,EAAe,YAAap/B,KAAcm/B,GAEpDn/B,EAAUS,MAClB,CAAE,MAAQ3P,GAGT,EAAc2M,uBAAwB3M,EAAKxB,KAC5C,CACD,CAEO,iBAAAqQ,CAENf,EACAC,EACAhL,GAEA,MAAMmsC,EAAW,GAASnsC,EAAQgJ,SAAW,aACvCuiC,EAAgBC,GAAqB/vC,MAE3C,UAAYuN,KAAWmjC,EAAW,CACjC,IAAI9gC,EAAUkgC,EAAc1nC,IAAKmF,GAE3BqC,IACLA,EAAU,IAAMR,KAChB0gC,EAAcr+B,IAAKlE,EAASqC,IAG7B5P,KAAKwP,SAAUI,EAAUN,EAAOC,EAAUhL,EAC3C,CACD,CAEO,oBAAAuM,CAAsCxB,EAAeC,GAC3D,MAAMugC,EAAgBC,GAAqB/vC,MAE3C,UAAY4P,KAAWkgC,EAAc5kC,SACpClL,KAAK2P,cAAeC,EAASN,EAAOC,EAEtC,EAIF,CAGA,CACC,MAAMohC,EAAUf,GAA+B54C,QAE/C,CAAE,OAAQ,oBAAqB,wBAAyB4G,SAAS7G,IAC9D64C,GAA+B74C,GAAQ45C,EAAM9uC,UAAW9K,EAAK,GAEjE,CASA,SAASi5C,GACRt/B,EACA++B,EACAC,GAEKh/B,aAAqB2+B,KACvB3+B,EAAmB6+B,YAAcE,EACjC/+B,EAAmB8+B,eAAiBE,EAExC,CASA,SAASO,GACRH,EACAviC,EACAmD,KACGm/B,GAEH,MAAMjgC,EAA4B,iBAAXrC,EAAsBuiC,EAAc1nC,IAAKmF,GAAY6iC,GAAkBN,EAAeviC,GAE7G,QAAMqC,IAINA,EAAQY,KAAME,KAAcm/B,GAErBn/B,EAAUvE,KAAKF,OACvB,CAKA,SAASmkC,GAAkBN,EAAsC3wC,GAChE,UAAcoO,EAASqC,KAAakgC,EACnC,GAAuB,mBAAXviC,GAAyBA,EAASpO,GAC7C,OAAOyQ,EAIT,OAAO,IACR,CAKA,SAASmgC,GAAqB7sC,GAK7B,OAJMA,EAAQysC,MACbzsC,EAAQysC,IAAmB,IAAIp+B,KAGzBrO,EAAQysC,GAChB,CCjNe,MAAMxwB,WAAiBywB,GAAsBz8B,MAyE3D,WAAApR,CAAagkC,GACZv4B,QARD,KAAiBojC,YAAc,IAAIh9B,IAUlC5T,KAAKi1B,UAAY,IAAIoZ,GACrBruC,KAAK6wC,MAAQ,IAAI,GAAY,CAAE1gB,WAAY,aAC3CnwB,KAAK+lC,gBAAkBA,EAEvB/lC,KAAKyR,IAAK,cAAc,GACxBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,eAAe,GACzBzR,KAAKyR,IAAK,eAAe,EAC1B,CASO,OAAAq/B,CAAS9uC,EAAe,QAC9B,OAAOhC,KAAK6wC,MAAMzoC,IAAKpG,EACxB,CAkDO,iBAAA04B,CAAmBqW,GACzB/wC,KAAK4wC,YAAY/iC,IAAKkjC,EACvB,CAKO,OAAA/qB,GACNhmB,KAAK6wC,MAAM7zC,KAAK1G,GAAQA,EAAK0vB,YAC7BhmB,KAAK2P,eACN,CAOO,eAAAqhC,CAAiBxW,GACvB,IAAIyW,GAAW,EAEf,GACC,UAAY1hC,KAAYvP,KAAK4wC,YAG5B,GAFAK,EAAW1hC,EAAUirB,GAEhByW,EACJ,YAGOA,EACX,ECxKc,MAAMC,WAAyB,GAuC7C,WAAAnvC,CACCyB,EACAxB,EACAojC,EACA9tB,GAEA9J,MAAOhK,EAAUxB,EAAMojC,EAAO9tB,GApC/B,KAAO65B,UAtBiB,GA+BxB,KAAOC,IAA8B,KAQrC,KAAiBC,aAA6C,KAqB7DrxC,KAAKgnC,gBAAkB,EACxB,CAKA,YAAWp6B,GACV,OAAO5M,KAAKmxC,SACb,CAMA,MAAWtxC,GACV,OAAOG,KAAKoxC,GACb,CAcO,qBAAAE,GACN,GAAiB,OAAZtxC,KAAKH,GAMT,MAAM,IAAI,EACT,oDACAG,MAIF,OAAO,IAAI4T,IAAK5T,KAAKqxC,aACtB,CAoBgB,SAAAhU,CAAW6I,GAE1B,OAAiB,OAAZlmC,KAAKH,IAA8C,OAA7BqmC,EAAsBrmC,GACzCG,KAAKH,KAASqmC,EAAsBrmC,GAGrC2N,MAAM6vB,UAAW6I,IAAkBlmC,KAAK4M,UAAcs5B,EAAsBt5B,QACpF,CAUgB,MAAA2wB,CAAQsJ,GAAgB,GACvC,MAAME,EAASv5B,MAAM+vB,OAAQsJ,GAQ7B,OALAE,EAAOoK,UAAYnxC,KAAKmxC,UAGxBpK,EAAOqK,IAAMpxC,KAAKoxC,IAEXrK,CACR,EAyBD,SAAS,KAER,GAAKwK,GAAoBvxC,MACxB,OAAO,KAGR,IAAIyX,EAAUzX,KAAKpB,OAGnB,KAAQ6Y,GAAWA,EAAQvJ,GAAI,qBAAuB,CACrD,GAAKqjC,GAAoB95B,GAAY,EACpC,OAAO,KAGRA,EAAUA,EAAQ7Y,MACnB,CAEA,OAAM6Y,GAAW85B,GAAoB95B,GAAY,EACzC,KAIDzX,KAAKs7B,UACb,CAKA,SAASiW,GAAoB95B,GAC5B,OAAO9W,MAAMrB,KAAMmY,EAAQ+jB,eAAgBz1B,QAAQ0R,IAAYA,EAAQvJ,GAAI,eAAgB5V,MAC5F,CAjMqB44C,GACGM,iBAdC,GA4JzBN,GAAiBrvC,UAAUqM,GAAK,SAAUjD,EAAcjJ,GACvD,OAAMA,EAMEA,IAAShC,KAAKgC,OACX,qBAATiJ,GAAwC,0BAATA,GAEtB,YAATA,GAA+B,iBAATA,GARP,qBAATA,GAAwC,0BAATA,GAE5B,YAATA,GAA+B,iBAATA,GACb,SAATA,GAA4B,cAATA,CAQtB,ECpKe,MAAMwmC,WAAqB,GAczC,WAAA1vC,CACCyB,EACAxB,EACAuD,EACA+R,GAEA9J,MAAOhK,EAAUxB,EAAMuD,EAAY+R,GAEnCtX,KAAKgnC,gBAAkB,EACxB,CASgB,YAAArB,CAAc/gC,EAAegsB,GAC5C,GAAKA,IAAWA,aAAiB,IAAQjwB,MAAMrB,KAAMsxB,GAA0Bt4B,OAAS,GAMvF,MAAM,IAAI,EACT,+BACA,CAAE0H,KAAM4wB,IAIV,OAAO,CACR,EAsBD,SAAS,KACR,OAAO,IACR,CAnBA6gB,GAAa5vC,UAAUqM,GAAK,SAAUjD,EAAcjJ,GACnD,OAAMA,EAMEA,IAAShC,KAAKgC,OACX,iBAATiJ,GAAoC,sBAATA,GAClB,YAATA,GAA+B,iBAATA,GAPP,iBAATA,GAAoC,sBAATA,GAExB,YAATA,GAA+B,iBAATA,GACb,SAATA,GAA4B,cAATA,CAOtB,EC5Ce,MAAMymC,WAAkB,GActC,WAAA3vC,CACCyB,EACAxB,EACAojC,EACA9tB,GAEA9J,MAAOhK,EAAUxB,EAAMojC,EAAO9tB,GAE9BtX,KAAKgnC,gBAAkB,EACxB,CASgB,YAAArB,CAAc/gC,EAAegsB,GAC5C,GAAKA,IAAWA,aAAiB,IAAQjwB,MAAMrB,KAAMsxB,GAA0Bt4B,OAAS,GAMvF,MAAM,IAAI,EAAe,4BAA6B,CAAE0H,KAAM4wB,IAG/D,OAAO,CACR,CAwBO,MAAA+gB,CAAQC,EAA0BC,GAExC,OAAO7xC,KAAK8xC,aAAcF,EAC3B,CAMO,YAAAE,CAAcF,GACpB,MAAMG,EAAaH,EAAYtsC,cAAetF,KAAKgC,MAEnD,UAAYjL,KAAOiJ,KAAKs+B,mBACvByT,EAAWtsC,aAAc1O,EAAKiJ,KAAKq7B,aAActkC,IAGlD,OAAOg7C,CACR,EA4BM,SAASC,GAAyB/X,GACxCA,EAAKz2B,SAAS6L,GAA+B,YAAY,CAAEgG,EAAKvS,IAgBjE,SAA4BuS,EAAcvS,EAAoB+uC,GAC7D,GAAK/uC,EAAKwqB,SAAWjB,GAASG,WAAa,CAC1C,MAAMylB,EAAenvC,EAAKovC,UAAU9yB,cAAcF,YAAaizB,eACzDC,EAAmD,GAA3BH,EAAa/E,YAAmB+E,EAAaI,WAAY,GAAIC,UAG3F,GAAKF,GAAyBtvC,EAAK2qB,SAAW,CAC7C,MAAM8kB,EAAYN,EAAaO,UACzBC,EAAYR,EAAaS,YAEzBC,EAAed,EAAae,kBAAmBL,EAAWE,GAGhE,GAAsB,OAAjBE,EACJ,OAID,IAAIE,GAAyB,EAE7B,MAAMC,EAAmBH,EAAanI,yBAAyBhyC,IACzDA,EAAMyH,KAAKiO,GAAI,eAEnB2kC,GAAyB,MAIrBr6C,EAAMyH,KAAKiO,GAAI,eAAiB1V,EAAMyH,KAAKiO,GAAI,wBAUrD,GAAK2kC,EAAyB,CAC7B,MAAME,EAAiBlB,EAAamB,kBAAmBF,GAElDV,EAEJH,EAAagB,SAAUF,EAAen0C,OAAQm0C,EAAen3B,QAG7Dq2B,EAAaiB,OAAQH,EAAen0C,OAAQm0C,EAAen3B,OAE7D,CACD,CACD,CACD,CAjEEu3B,CAAmB99B,EAAKvS,EAAMm3B,EAAK4X,eAAgB,CAAEjlC,SAAU,OACjE,CAKA,SAAS,KACR,OAAO,IACR,CAjCA8kC,GAAU7vC,UAAUqM,GAAK,SAAUjD,EAAcjJ,GAChD,OAAMA,EAMEA,IAAShC,KAAKgC,OACX,cAATiJ,GAAiC,mBAATA,GACf,YAATA,GAA+B,iBAATA,GAPP,cAATA,GAAiC,mBAATA,GAErB,YAATA,GAA+B,iBAATA,GACb,SAATA,GAA4B,cAATA,CAOtB,ECzGe,MAAMmoC,WAAmB,GAcvC,WAAArxC,CACCyB,EACAxB,EACAojC,EACA9tB,GAEA9J,MAAOhK,EAAUxB,EAAMojC,EAAO9tB,GAG9BtX,KAAKgnC,gBAAkB,EACxB,CASgB,YAAArB,CAAc/gC,EAAegsB,GAC5C,GAAKA,IAAWA,aAAiB,IAAQjwB,MAAMrB,KAAMsxB,GAA0Bt4B,OAAS,GAMvF,MAAM,IAAI,EACT,6BACA,CAAE0H,KAAM4wB,IAIV,OAAO,CACR,CAoBO,MAAA+gB,CAAQI,EAAwBF,GAAoC,EAuB5E,SAAS,KACR,OAAO,IACR,CApBAuB,GAAWvxC,UAAUqM,GAAK,SAAUjD,EAAcjJ,GACjD,OAAMA,EAOEA,IAAShC,KAAKgC,OACX,eAATiJ,GAAkC,oBAATA,GAChB,YAATA,GAA+B,iBAATA,GARP,eAATA,GAAkC,oBAATA,GAE/BA,IAASjL,KAAKgC,MAAQiJ,IAAS,QAAUjL,KAAKgC,MACrC,YAATiJ,GAA+B,iBAATA,GACb,SAATA,GAA4B,cAATA,CAOtB,EC/Fe,MAAMooC,WAAyBjkC,EAAc0sB,KAwB3D,WAAA/5B,CAAayB,EAAoB8T,GAChC9J,QAhBD,KAAiBk4B,UAAyB,GAM1C,KAAiBJ,kBAAoB,IAAI/zB,IAYxCvR,KAAKwD,SAAWA,EAEX8T,GACJtX,KAAK2lC,aAAc,EAAGruB,EAExB,CAOA,CAASxW,OAAOC,YACf,OAAOf,KAAK0lC,UAAW5kC,OAAOC,WAC/B,CAKA,cAAWu6B,GACV,OAAOt7B,KAAK0lC,UAAUptC,MACvB,CAKA,WAAW6pC,GACV,OAA2B,IAApBniC,KAAKs7B,UACb,CAKA,QAAWhlC,GACV,OAAO0J,IACR,CAKA,UAAWpB,GACV,OAAO,IACR,CAKA,QAAWoD,GAEX,CAKO,iBAAAwkC,CAAmBzvC,GACzB,OAAOiJ,KAAKslC,kBAAkBl9B,IAAKrR,EACpC,CAMA,oBAAQ0vC,SACAzmC,KAAKslC,kBAAkB7sB,SAC/B,CAUO,YAAAwuB,CAAcrW,GACpB,OAAO5wB,KAAK2lC,aAAc3lC,KAAKs7B,WAAY1K,EAC5C,CAQO,QAAAiL,CAAUj3B,GAChB,OAAO5E,KAAK0lC,UAAW9gC,EACxB,CAQO,aAAAo3B,CAAe78B,GACrB,OAAOa,KAAK0lC,UAAU18B,QAAS7J,EAChC,CAOO,WAAAq8B,GACN,OAAOx7B,KAAK0lC,UAAW5kC,OAAOC,WAC/B,CAWO,YAAA4kC,CAAc/gC,EAAegsB,GACnC5wB,KAAK+8B,YAAa,WAAY/8B,MAC9B,IAAI2E,EAAQ,EAEZ,MAAMuiC,EAkGR,SAAoB1jC,EAAoB0jC,GAEvC,GAAqB,iBAATA,EACX,MAAO,CAAE,IAAIhK,GAAM15B,EAAU0jC,IAGxBvwB,EAAYuwB,KACjBA,EAAQ,CAAEA,IAIX,OAAOvmC,MAAMrB,KAAM4nC,GACjBlqC,KAAKmC,GACe,iBAARA,EACJ,IAAI+9B,GAAM15B,EAAUrE,GAGvBA,aAAgBq+B,GACb,IAAIN,GAAM15B,EAAUrE,EAAK2D,MAG1B3D,GAEV,CAzHgB,CAAWa,KAAKwD,SAAUotB,GAExC,UAAYzxB,KAAQ+nC,EAEE,OAAhB/nC,EAAKP,QACTO,EAAKkyB,UAGJlyB,EAAcP,OAASoB,KAEzBA,KAAK0lC,UAAUl+B,OAAQ5C,EAAO,EAAGzF,GACjCyF,IACAD,IAGD,OAAOA,CACR,CAUO,eAAAm4B,CAAiBl4B,EAAeuG,EAAkB,GACxDnL,KAAK+8B,YAAa,WAAY/8B,MAE9B,QAAUxF,EAAIoK,EAAOpK,EAAIoK,EAAQuG,EAAS3Q,IACvCwF,KAAK0lC,UAAWlrC,GAAaoE,OAAS,KAGzC,OAAOoB,KAAK0lC,UAAUl+B,OAAQ5C,EAAOuG,EACtC,CASO,WAAA4xB,CAAa9xB,EAAkB9L,GACrCa,KAAKwQ,KAAM,UAAYvF,EAAM9L,EAC9B,CAQO,kBAAAwoC,CAAoB5wC,EAAsByB,GAChDwH,KAAKslC,kBAAkB7zB,IAAK1a,EAAKyB,EAClC,CASO,qBAAAovC,CAAuB7wC,GAC7B,OAAOiJ,KAAKslC,kBAAkB3zB,OAAQ5a,EACvC,EAyBDs8C,GAAiBxxC,UAAUqM,GAAK,SAAUjD,GACzC,MAAgB,qBAATA,GAAwC,0BAATA,CACvC,ECvNe,MAAM,GAoBpB,WAAAlJ,CAAayB,GAVb,KAAiB8vC,aAAe,IAAI/hC,IAKpC,KAAQgiC,aAAoG,KAM3GvzC,KAAKwD,SAAWA,CACjB,CAwGO,YAAAgwC,IAAiBt3C,GACvB8D,KAAKwD,SAASyxB,UAAUka,UAAWjzC,EACpC,CAUO,iBAAAu3C,CAAmB3I,EAAiClvB,GAC1D5b,KAAKwD,SAASyxB,UAAUma,UAAWtE,EAAgBlvB,EACpD,CAQO,sBAAA83B,CAAwBp8B,GAC9B,OAAO,IAAI+7B,GAAkBrzC,KAAKwD,SAAU8T,EAC7C,CAYO,UAAAq8B,CAAY7wC,GAClB,OAAO,IAAIo6B,GAAMl9B,KAAKwD,SAAUV,EACjC,CAyBO,sBAAA8wC,CACN5xC,EACAuD,EACAhB,EAII,CAAC,GAEL,MAAMsvC,EAAmB,IAAI3C,GAAkBlxC,KAAKwD,SAAUxB,EAAMuD,GAcpE,MAZiC,iBAArBhB,EAAQqI,WACnBinC,EAAiB1C,UAAY5sC,EAAQqI,UAGjCrI,EAAQ1E,KACZg0C,EAAiBzC,IAAM7sC,EAAQ1E,IAG3B0E,EAAQuvC,wBACZD,EAAiBxO,0BAA0BzlC,QAAS2E,EAAQuvC,wBAGtDD,CACR,CAqEO,sBAAAE,CACN/xC,EACAuD,EACAyuC,EAAwF,CAAC,EACzFzvC,EAAsD,CAAC,GAEvD,IAAI+S,EAAW,KAEV,GAAe08B,GACnBzvC,EAAUyvC,EAEV18B,EAAW08B,EAGZ,MAAMC,EAAmB,IAAIjM,GAAkBhoC,KAAKwD,SAAUxB,EAAMuD,EAAY+R,GAMhF,OAJK/S,EAAQuvC,wBACZG,EAAiB5O,0BAA0BzlC,QAAS2E,EAAQuvC,wBAGtDG,CACR,CAoBO,qBAAAC,CACNlyC,EACAuD,EACAhB,EAEI,CAAC,GAEL,MAAM4jC,EAAkB,IAAID,GAAiBloC,KAAKwD,SAAUxB,EAAMuD,GAMlE,OAJKhB,EAAQuvC,wBACZ3L,EAAgB9C,0BAA0BzlC,QAAS2E,EAAQuvC,wBAGrD3L,CACR,CAiBO,kBAAAgM,CACNnyC,EACAuD,EACAhB,EAEI,CAAC,GAEL,MAAM6vC,EAAe,IAAI3C,GAAczxC,KAAKwD,SAAUxB,EAAMuD,GAM5D,OAJKhB,EAAQuvC,wBACZM,EAAa/O,0BAA0BzlC,QAAS2E,EAAQuvC,wBAGlDM,CACR,CA+BO,eAAAC,CACNryC,EACAuD,EACA+uC,GAEA,MAAMC,EAAY,IAAI7C,GAAW1xC,KAAKwD,SAAUxB,EAAMuD,GAMtD,OAJK+uC,IACJC,EAAU5C,OAAS2C,GAGbC,CACR,CA+BO,gBAAAC,CACNxyC,EACAuD,EACA+uC,EACA/vC,EAEI,CAAC,GAEL,MAAMkwC,EAAa,IAAIrB,GAAYpzC,KAAKwD,SAAUxB,EAAMuD,GAUxD,OARK+uC,IACJG,EAAW9C,OAAS2C,GAGhB/vC,EAAQuvC,wBACZW,EAAWpP,0BAA0BzlC,QAAS2E,EAAQuvC,wBAGhDW,CACR,CAYO,YAAAhvC,CAAc1O,EAAayB,EAAgBif,GACjDA,EAAQ2vB,cAAerwC,EAAKyB,EAC7B,CAWO,eAAAkO,CAAiB3P,EAAa0gB,GACpCA,EAAQ6vB,iBAAkBvwC,EAC3B,CAUO,QAAA6kC,CAAUuK,EAAmC1uB,GACnDA,EAAQ8vB,UAAWpB,EACpB,CAUO,WAAApL,CAAaoL,EAAmC1uB,GACtDA,EAAQ+vB,aAAcrB,EACvB,CAwCO,QAAAuO,CACNthC,EACA5a,EACAif,GAGK,GAAerE,SAA0B,IAAZqE,EAC/Bjf,EAAmBivC,UAAWr0B,GAEhCqE,EAASgwB,UAAWr0B,EAAoB5a,EAE1C,CAcO,WAAAm8C,CAAavhC,EAAkCqE,GACrDA,EAAQiwB,aAAct0B,EACvB,CAMO,iBAAAwhC,CAAmB79C,EAAsByB,EAAgBif,GAC/DA,EAAQkwB,mBAAoB5wC,EAAKyB,EAClC,CAOO,oBAAAq8C,CAAsB99C,EAAsB0gB,GAClD,OAAOA,EAAQmwB,sBAAuB7wC,EACvC,CAuCO,eAAA+9C,CAAiBC,GACvB,OAAKA,aAA2BpM,GACxB3oC,KAAKg1C,iBAAkBD,GAEvB/0C,KAAKi1C,sBAAuBF,EAErC,CA2BO,cAAAG,CAAgBzvB,GACtB,MAAMhO,EAAUgO,EAAS7mB,OAEzB,IAAQ6Y,EAAQvJ,GAAI,oBAMnB,MAAM,IAAI,EAAe,0CAA2ClO,KAAKwD,UAG1E,IAAMiU,EAAQ7Y,OAMb,MAAM,IAAI,EAAe,yBAA0BoB,KAAKwD,UAGzD,GAAKiiB,EAASqkB,UACb,OAAOnB,GAASoB,cAAetyB,GACzB,IAAMgO,EAAS+jB,QAAU,CAC/B,MAAM2L,EAAa19B,EAAQ8lB,QAAQ,GAEnCv9B,KAAK0F,OAAQijC,GAASc,aAAchyB,GAAW09B,GAE/C,MAAMC,EAAc,IAAIrK,GAAOtlB,EAAUkjB,GAASC,UAAWnxB,EAAS,QAChE49B,EAAiB,IAAI1M,GAAUwM,EAAY,GAEjDn1C,KAAKs1C,KAAMF,EAAaC,EACzB,CAEA,OAAO1M,GAASc,aAAchyB,EAC/B,CAgCO,eAAA89B,CAAiB9vB,GACvB,MAAM+vB,EAAiB/vB,EAAS7J,OAC1B65B,EAAiBhwB,EAAS7mB,OAGhC,GAAK62C,EAAevnC,GAAI,SACvB,OAAOuX,EAIR,GAAKgwB,EAAevnC,GAAI,qBAAsD,IAA9BunC,EAAena,WAAmB,CACjF,MAAM18B,EAAS62C,EAAe72C,OACxBgd,EAAS65B,EAAe7wC,MAK9B,OAHA6wC,EAAepkB,UACfrxB,KAAK01C,+BAAgCD,GAE9Bz1C,KAAKu1C,gBAAiB,IAAI5M,GAAU/pC,EAASgd,GACrD,CAEA,MAAMuuB,EAAesL,EAA4B5Z,SAAU2Z,EAAiB,GACtEtL,EAAcuL,EAA4B5Z,SAAU2Z,GAG1D,IAAMrL,IAAeD,EACpB,OAAOzkB,EAIR,GAAK0kB,EAAWj8B,GAAI,UAAag8B,EAAUh8B,GAAI,SAC9C,OAAOynC,GAAgBxL,EAAYD,GACpC,GAEUC,EAAWj8B,GAAI,qBAAwBg8B,EAAUh8B,GAAI,qBAAwBi8B,EAAW9M,UAAW6M,GAAc,CAE1H,MAAMvlC,EAAQwlC,EAAW7O,WAQzB,OAPA6O,EAAWlD,aAAciD,EAAU1O,eAEnC0O,EAAU7Y,UACVrxB,KAAK01C,+BAAgCxL,GAI9BlqC,KAAKu1C,gBAAiB,IAAI5M,GAAUwB,EAAYxlC,GACxD,CAEA,OAAO8gB,CACR,CAsBO,eAAAmwB,CAAiBnwB,GACvB,MAAMowB,EAAOpwB,EAAS0kB,WAChB7oC,EAAOmkB,EAASykB,UAEtB,KAAM2L,GAASv0C,GAASu0C,EAAK3nC,GAAI,qBAAyB5M,EAAK4M,GAAI,qBAMlE,MAAM,IAAI,EAAe,gDAAiDlO,KAAKwD,UAGhF,MAAMykC,EAAY4N,EAAKha,SAAUga,EAAKva,WAAa,GAC7Cwa,EAAc7N,aAAqB/K,GAAOyL,GAASC,UAAWX,EAAW,OAAUU,GAASC,UAAWiN,EAAM,OAKnH,OAHA71C,KAAKs1C,KAAMvK,GAAMyB,UAAWlrC,GAAQqnC,GAASC,UAAWiN,EAAM,QAC9D71C,KAAKiG,OAAQ8kC,GAAM0B,UAAWnrC,IAEvBw0C,CACR,CAkBO,MAAApwC,CAAQ+f,EAAoByhB,GAIlC6O,GAHA7O,EAAQvwB,EAAYuwB,GAAU,IAAKA,GAAU,CAAEA,GAGjBlnC,KAAKwD,UAGnC,MAAMwyC,EAAe9O,EAAuBjZ,QAAQ,CAAEgoB,EAAiE92C,KACtH,MAAM+2C,EAAYD,EAAQA,EAAO39C,OAAS,GAIpCw8C,GAAmB31C,EAAK+O,GAAI,aAWlC,OATMgoC,GAAaA,EAAUpB,iBAAmBA,EAM/CoB,EAAUhP,MAAMtnC,KAAMT,GALtB82C,EAAOr2C,KAAM,CACZk1C,kBACA5N,MAAO,CAAE/nC,KAMJ82C,CAAM,GACX,IAGH,IAAIvV,EAAQ,KACRC,EAAMlb,EAEV,UAAcyhB,MAAAA,EAAAA,gBAAO4N,KAAqBkB,EAAa,CACtD,MAAMlxB,EAAQ9kB,KAAKm2C,aAAcxV,EAAKuG,EAAO4N,GAEvCpU,IACLA,EAAQ5b,EAAM4b,OAGfC,EAAM7b,EAAM6b,GACb,CAGA,OAAMD,EAIC,IAAIqK,GAAOrK,EAAOC,GAHjB,IAAIoK,GAAOtlB,EAIpB,CAcO,MAAAxf,CAAQmwC,GACd,MAAMtxB,EAAQsxB,aAAuBrL,GAAQqL,EAAcrL,GAAM0B,UAAW2J,GAK5E,GAHAC,GAAwBvxB,EAAO9kB,KAAKwD,UAG/BshB,EAAMkmB,YACV,OAAO,IAAIqI,GAAkBrzC,KAAKwD,UAInC,MAAQk9B,MAAO4V,EAAY3V,IAAK4V,GAAav2C,KAAKi1C,sBAAuBnwB,GAAO,GAC1E0xB,EAAkBF,EAAW13C,OAE7B+F,EAAQ4xC,EAAS36B,OAAS06B,EAAW16B,OAGrCoV,EAAUwlB,EAAgB1Z,gBAAiBwZ,EAAW16B,OAAQjX,GAEpE,UAAYxF,KAAQ6xB,EACnBhxB,KAAK01C,+BAAgCv2C,GAItC,MAAMs3C,EAAgBz2C,KAAKu1C,gBAAiBe,GAK5C,OAJExxB,EAAe4b,MAAQ+V,EACvB3xB,EAAe6b,IAAM8V,EAAc1zB,QAG9B,IAAIswB,GAAkBrzC,KAAKwD,SAAUwtB,EAC7C,CAYO,KAAApf,CAAOkT,EAAcrN,GAC3B4+B,GAAwBvxB,EAAO9kB,KAAKwD,UAIpC,MAAMkzC,EAAS5xB,EAAM+lB,UAAW,CAC/BtmB,UAAW,WACXwkB,kBAAkB,IAInB,UAAYhqC,KAAW23C,EAAS,CAC/B,MAAMz2C,EAAOlB,EAAQkB,KACrB,IAAI02C,EAGJ,GAAK12C,EAAKiO,GAAI,YAAeuJ,EAAQ4lB,UAAWp9B,GAE/C02C,EAAgB5L,GAAM0B,UAAWxsC,QAE3B,IAAMlB,EAAQkrC,aAAapN,QAAS/X,EAAM4b,QAAWzgC,EAAKiO,GAAI,cAAiB,CAErF,MAAM4Y,EAAgB7mB,EAAKk8B,eAAe5K,MAAMqlB,GACxCA,EAAS1oC,GAAI,YAAeuJ,EAAQ4lB,UAAWuZ,KAIlD9vB,IACJ6vB,EAAgB5L,GAAMyB,UAAW1lB,GAEnC,CAGK6vB,IAECA,EAAchW,IAAI9D,QAAS/X,EAAM6b,OACnCgW,EAAuBhW,IAAM7b,EAAM6b,KAGjCgW,EAAcjW,MAAMhE,SAAU5X,EAAM4b,SACtCiW,EAAuBjW,MAAQ5b,EAAM4b,OAIxC1gC,KAAKiG,OAAQ0wC,GAEf,CACD,CAcO,IAAArB,CAAMF,EAAoBC,GAChC,IAAInO,EAEJ,GAAKmO,EAAexY,QAASuY,EAAYzU,KAAQ,CAGhD,MAAM/hC,GAFNy2C,EAAiBr1C,KAAKg1C,iBAAkBK,GAAgB,IAE1Bz2C,OACxBi4C,EAAcj4C,EAAO08B,WAE3B8Z,EAAcp1C,KAAKi1C,sBAAuBG,GAAa,GAEvDlO,EAAQlnC,KAAKiG,OAAQmvC,GAErBC,EAAez5B,QAAYhd,EAAO08B,WAAaub,CAChD,MACC3P,EAAQlnC,KAAKiG,OAAQmvC,GAGtB,OAAOp1C,KAAK0F,OAAQ2vC,EAAgBnO,EACrC,CAuBO,IAAA4P,CAAMhyB,EAAc1O,GAC1B,KAAQA,aAAqB86B,IAC5B,MAAM,IAAI,EACT,qCACAlxC,KAAKwD,UAMP,GAFA6yC,GAAwBvxB,EAAO9kB,KAAKwD,UAE9BshB,EAAMkmB,YAGL,CAEN,IAAIvlB,EAAWX,EAAM4b,MAEhBjb,EAAS7mB,OAAOsP,GAAI,aAq/BAtP,EAr/BmC6mB,EAAS7mB,QAs/BhE+B,MAAMrB,KAAMV,EAAO48B,eAAgBnH,MAAM1c,IAAUA,EAAMzJ,GAAI,kBAr/BjEuX,EAAWA,EAAS+kB,yBAAyBhyC,GAASA,EAAMyH,KAAKiO,GAAI,gBAGtEuX,EAAWzlB,KAAK+2C,cAAetxB,EAAUrP,GACzC,MAAM4gC,EAAgBh3C,KAAKwD,SAASyxB,UAOpC,OAJK+hB,EAAchM,aAAegM,EAAc7hB,mBAAoBlR,QAASa,EAAM4b,QAClF1gC,KAAKwzC,aAAc/tB,GAGb,IAAIslB,GAAOtlB,EACnB,CAlBC,OAAOzlB,KAAKi3C,WAAYnyB,EAAO1O,GA0/BlC,IAA4BxX,CAv+B3B,CASO,MAAAs4C,CAAQpyB,EAAc1O,GAC5B,KAAQA,aAAqB86B,IAO5B,MAAM,IAAI,EACT,uCACAlxC,KAAKwD,UAOP,GAHA6yC,GAAwBvxB,EAAO9kB,KAAKwD,UAG/BshB,EAAMkmB,YACV,OAAOlmB,EAIR,MAAQ4b,MAAO4V,EAAY3V,IAAK4V,GAAav2C,KAAKi1C,sBAAuBnwB,GAAO,GAC1E0xB,EAAkBF,EAAW13C,OAG7Bu4C,EAAWn3C,KAAKo3C,gBAAiBZ,EAAiBF,EAAW16B,OAAQ26B,EAAS36B,OAAQxF,GAGtFsqB,EAAQ1gC,KAAKu1C,gBAAiB4B,EAASzW,OAGvCA,EAAMzc,QAASkzB,EAASzW,QAC7ByW,EAASxW,IAAI/kB,SAGd,MAAM+kB,EAAM3gC,KAAKu1C,gBAAiB4B,EAASxW,KAE3C,OAAO,IAAIoK,GAAOrK,EAAOC,EAC1B,CAeO,MAAA0W,CAAQC,EAAiBC,GAC/B,MAAMpC,EAAa,IAAInN,GAAkBhoC,KAAKwD,SAAU8zC,EAASC,EAAYvR,iBAM7E,OAJAhmC,KAAK0F,OAAQijC,GAASc,aAAc8N,GAAepC,GACnDn1C,KAAKs1C,KAAMvK,GAAMyB,UAAW+K,GAAe5O,GAASC,UAAWuM,EAAY,IAC3En1C,KAAKiG,OAAQ8kC,GAAM0B,UAAW8K,IAEvBpC,CACR,CAgBO,wBAAAqC,CAA0BC,GAChCz3C,KAAKszC,aAAa3hC,OAAQ8lC,EAC3B,CAiBO,gBAAAC,CAAkB5M,EAAiClvB,GACzD,OAAO+sB,GAASC,UAAWkC,EAAgBlvB,EAC5C,CAOO,mBAAA+7B,CAAqB13C,GAC3B,OAAO0oC,GAASc,aAAcxpC,EAC/B,CAOO,oBAAA23C,CAAsB33C,GAC5B,OAAO0oC,GAASoB,cAAe9pC,EAChC,CAUO,WAAA43C,CAAanX,EAAiBC,GACpC,OAAO,IAAIoK,GAAOrK,EAAOC,EAC1B,CAKO,aAAAmX,CAAe73C,GACrB,OAAO8qC,GAAM0B,UAAWxsC,EACzB,CAQO,aAAA83C,CAAetgC,GACrB,OAAOszB,GAAMyB,UAAW/0B,EACzB,CAgGO,eAAAugC,IAAoB97C,GAC1B,OAAO,IAAIwwC,MAAcxwC,EAC1B,CA+BO,UAAA+7C,CAAYC,EAAwC,YAC1D,IAAMl4C,KAAKuzC,aAMV,MAAM,IAAI,EAAe,0CAA2CvzC,KAAKwD,UAG1E,OAAOxD,KAAKuzC,aAAcvzC,KAAMk4C,EACjC,CAQO,oBAAAC,CAAsBC,GAC5Bp4C,KAAKuzC,aAAe6E,CACrB,CAOO,iBAAAC,GACNr4C,KAAKuzC,aAAe,IACrB,CAWQ,YAAA4C,CAAc1wB,EAAoByhB,EAAuB4N,GAChE,IAAIhuB,EAsBAwxB,EAZJ,GALCxxB,EADIguB,EACYyD,GAAoB9yB,GAEpBA,EAAS7mB,OAAOsP,GAAI,SAAYuX,EAAS7mB,OAAOA,OAAS6mB,EAAS7mB,QAG7EkoB,EAML,MAAM,IAAI,EACT,yCACA9mB,KAAKwD,UAON80C,EADIxD,EACgB90C,KAAKg1C,iBAAkBvvB,GAAU,GAEjCA,EAAS7mB,OAAOsP,GAAI,SAAYsqC,GAAe/yB,GAAaA,EAGjF,MAAMntB,EAAWwuB,EAA8C6e,aAAc2S,EAAkB18B,OAAQsrB,GAEvG,UAAY/nC,KAAQ+nC,EACnBlnC,KAAKy4C,0BAA2Bt5C,GAGjC,MAAMu5C,EAAcJ,EAAkBhO,aAAchyC,GAC9CooC,EAAQ1gC,KAAKu1C,gBAAiB+C,GAG9B5X,EAAMzc,QAASq0B,IACpBI,EAAY98B,SAGb,MAAM+kB,EAAM3gC,KAAKu1C,gBAAiBmD,GAElC,OAAO,IAAI3N,GAAOrK,EAAOC,EAC1B,CAMQ,aAAAgY,CAAe/5C,EAAiBorC,EAAqBI,EAAmBwO,GAC/E,IAAIp+C,EAAIwvC,EACR,MAAM6O,EAAiC,GAEvC,KAAQr+C,EAAI4vC,GAAY,CACvB,MAAMzyB,EAAQ/Y,EAAOi9B,SAAUrhC,GACzB6mB,EAAS1J,EAAMzJ,GAAI,SACnB4qC,EAAcnhC,EAAMzJ,GAAI,oBAU9B,GAAK4qC,GAAe94C,KAAK+4C,sBAAuBH,EAAajhC,GAC5DkhC,EAAcj5C,KAAM,IAAI+oC,GAAU/pC,EAAQpE,SAC3C,GAOU6mB,IAAWy3B,GAAeE,GAAmBJ,EAAajhC,GAAU,CAE7E,MAAMshC,EAAeL,EAAYrb,SAGjC5lB,EAAM0Z,UACN4nB,EAAahS,aAActvB,GAE3B/Y,EAAO+mC,aAAcnrC,EAAGy+C,GACxBj5C,KAAKy4C,0BAA2BQ,GAEhCJ,EAAcj5C,KAAM,IAAI+oC,GAAU/pC,EAAQpE,GAC3C,MAOCwF,KAAK24C,cAAehhC,EAAO,EAAGA,EAAM2jB,WAAYsd,GAGjDp+C,GACD,CAGA,IAAI0+C,EAAe,EAEnB,UAAYzzB,KAAYozB,EAAgB,CAIvC,GAHApzB,EAAS7J,QAAUs9B,EAGdzzB,EAAS7J,QAAUouB,EACvB,SAGmBhqC,KAAKu1C,gBAAiB9vB,GAGxBxB,QAASwB,KAC1ByzB,IACA9O,IAEF,CAEA,OAAOW,GAAMqB,6BAA8BxtC,EAAQorC,EAAaprC,EAAQwrC,EACzE,CAMQ,eAAAgN,CAAiBx4C,EAAiBorC,EAAqBI,EAAmB+O,GACjF,IAAI3+C,EAAIwvC,EACR,MAAMoP,EAAmC,GAKzC,KAAQ5+C,EAAI4vC,GAAY,CACvB,MAAMzyB,EAAQ/Y,EAAOi9B,SAAUrhC,GAG/B,GAAMmd,EAAMzJ,GAAI,oBAahB,GAAKyJ,EAAM0lB,UAAW8b,GAAtB,CACC,MAAME,EAAY1hC,EAAM6jB,cAClB72B,EAAQgT,EAAM2jB,WAGpB3jB,EAAM0Z,UACNzyB,EAAO+mC,aAAcnrC,EAAG6+C,GAExBr5C,KAAK01C,+BAAgC/9B,GAGrCyhC,EAAgBx5C,KACf,IAAI+oC,GAAU/pC,EAAQpE,GACtB,IAAImuC,GAAU/pC,EAAQpE,EAAImK,IAI3BnK,GAAKmK,EACLylC,GAAazlC,EAAQ,CAGtB,MASK3E,KAAKs5C,wBAAyBH,EAAexhC,IACjDyhC,EAAgBx5C,KACf,IAAI+oC,GAAU/pC,EAAQpE,GACtB,IAAImuC,GAAU/pC,EAAQpE,EAAI,IAG3BA,MAUDwF,KAAKo3C,gBAAiBz/B,EAAO,EAAGA,EAAM2jB,WAAY6d,GAElD3+C,UA5DCA,GA6DF,CAGA,IAAI0+C,EAAe,EAEnB,UAAYzzB,KAAY2zB,EAAkB,CAIzC,GAHA3zB,EAAS7J,QAAUs9B,EAGdzzB,EAAS7J,QAAUouB,GAAevkB,EAAS7J,QAAUwuB,EACzD,SAGmBpqC,KAAKu1C,gBAAiB9vB,GAGxBxB,QAASwB,KAC1ByzB,IACA9O,IAEF,CAEA,OAAOW,GAAMqB,6BAA8BxtC,EAAQorC,EAAaprC,EAAQwrC,EACzE,CAWQ,UAAA6M,CAAYnyB,EAAc1O,GAEjC,MAAQsqB,MAAO4V,EAAY3V,IAAK4V,GAAav2C,KAAKi1C,sBAAuBnwB,GAAO,GAC1E0xB,EAAkBF,EAAW13C,OAG7Bu4C,EAAWn3C,KAAK24C,cAAenC,EAAiBF,EAAW16B,OAAQ26B,EAAS36B,OAAQxF,GAGpFsqB,EAAQ1gC,KAAKu1C,gBAAiB4B,EAASzW,OAGvCA,EAAMzc,QAASkzB,EAASzW,QAC7ByW,EAASxW,IAAI/kB,SAEd,MAAM+kB,EAAM3gC,KAAKu1C,gBAAiB4B,EAASxW,KAE3C,OAAO,IAAIoK,GAAOrK,EAAOC,EAC1B,CAWQ,aAAAoW,CAAetxB,EAAoBrP,GAE1C,GAAKA,EAAUinB,UAAW5X,EAAS7mB,QAClC,OAAO26C,GAAwB9zB,EAAS1C,SAIpC0C,EAAS7mB,OAAOsP,GAAI,WACxBuX,EAAW+yB,GAAe/yB,IAI3B,MAAM+zB,EAAcx5C,KAAK4zC,uBAAwB,8BAC/C4F,EAAqBrI,UAAY/rB,OAAOC,kBAC1Cm0B,EAAYnc,UAAY,KAAM,EAG5B5X,EAAS7mB,OAAoB+mC,aAAclgB,EAAS7J,OAAQ49B,GAG9D,MAAMC,EAAY,IAAI1O,GAAOtlB,EAAUA,EAAS6kB,aAAc,IAG9DtqC,KAAK82C,KAAM2C,EAAWrjC,GAGtB,MAAM0/B,EAAc,IAAInN,GAAU6Q,EAAY56C,OAAS46C,EAAY50C,OACnE40C,EAAYnoB,UAGZ,MAAM8Y,EAAa2L,EAAY3L,WACzBD,EAAY4L,EAAY5L,UAE9B,OAAKC,aAAsBjN,IAAQgN,aAAqBhN,GAChDyY,GAAgBxL,EAAYD,GAI7BqP,GAAwBzD,EAChC,CAWQ,qBAAAiD,CAAuBW,EAA2BC,GACzD,IAAMC,GAAaF,EAASC,GAC3B,OAAO,EAIR,GAAKD,EAAQ13C,OAAS23C,EAAO33C,MAAQ03C,EAAQ9sC,WAAa+sC,EAAO/sC,SAChE,OAAO,EAIR,UAAY7V,KAAO2iD,EAAQpb,mBAE1B,GAAa,UAARvnC,GAA2B,UAARA,GAKnB4iD,EAAO1T,aAAclvC,IAAS4iD,EAAOte,aAActkC,KAAU2iD,EAAQre,aAActkC,GACvF,OAAO,EAKT,UAAYA,KAAO2iD,EAAQ5a,gBAC1B,GAAK6a,EAAOtT,SAAUtvC,IAAS4iD,EAAO5a,SAAUhoC,KAAU2iD,EAAQ3a,SAAUhoC,GAC3E,OAAO,EAKT,UAAYA,KAAO2iD,EAAQpb,mBAEb,UAARvnC,GAA2B,UAARA,IAKlB4iD,EAAO1T,aAAclvC,IAC1BiJ,KAAKyF,aAAc1O,EAAK2iD,EAAQre,aAActkC,GAAQ4iD,IAIxD,UAAY5iD,KAAO2iD,EAAQ5a,gBACpB6a,EAAOtT,SAAUtvC,IACtBiJ,KAAK00C,SAAU39C,EAAK2iD,EAAQ3a,SAAUhoC,GAAQ4iD,GAIhD,UAAY5iD,KAAO2iD,EAAQ/a,gBACpBgb,EAAO7e,SAAU/jC,IACtBiJ,KAAK47B,SAAU7kC,EAAK4iD,GAItB,OAAO,CACR,CAWQ,uBAAAL,CAAyBI,EAA2BG,GAC3D,IAAMD,GAAaF,EAASG,GAC3B,OAAO,EAIR,GAAKH,EAAQ13C,OAAS63C,EAAS73C,MAAQ03C,EAAQ9sC,WAAaitC,EAASjtC,SACpE,OAAO,EAIR,UAAY7V,KAAO2iD,EAAQpb,mBAE1B,GAAa,UAARvnC,GAA2B,UAARA,KAKlB8iD,EAAS5T,aAAclvC,IAAS8iD,EAASxe,aAActkC,KAAU2iD,EAAQre,aAActkC,IAC5F,OAAO,EAKT,IAAM8iD,EAAS/e,YAAa4e,EAAQ/a,iBACnC,OAAO,EAIR,UAAY5nC,KAAO2iD,EAAQ5a,gBAE1B,IAAM+a,EAASxT,SAAUtvC,IAAS8iD,EAAS9a,SAAUhoC,KAAU2iD,EAAQ3a,SAAUhoC,GAChF,OAAO,EAKT,UAAYA,KAAO2iD,EAAQpb,mBAEb,UAARvnC,GAA2B,UAARA,GAIxBiJ,KAAK0G,gBAAiB3P,EAAK8iD,GAS5B,OALA75C,KAAK+6B,YAAap6B,MAAMrB,KAAMo6C,EAAQ/a,iBAAmBkb,GAGzD75C,KAAK20C,YAAah0C,MAAMrB,KAAMo6C,EAAQ5a,iBAAmB+a,IAElD,CACR,CAUQ,qBAAA5E,CAAuBnwB,EAAcg1B,GAA0B,GACtE,MAAMC,EAAaj1B,EAAM4b,MACnBsZ,EAAWl1B,EAAM6b,IAKvB,GAHA0V,GAAwBvxB,EAAO9kB,KAAKwD,UAG/BshB,EAAMkmB,YAAc,CACxB,MAAMvlB,EAAWzlB,KAAKg1C,iBAAkBlwB,EAAM4b,MAAOoZ,GAErD,OAAO,IAAI/O,GAAOtlB,EAAUA,EAC7B,CAEA,MAAM8wB,EAAWv2C,KAAKg1C,iBAAkBgF,EAAUF,GAC5Cn1C,EAAU4xC,EAAS33C,OAAoB08B,WACvCgb,EAAat2C,KAAKg1C,iBAAkB+E,EAAYD,GAKtD,OAFAvD,EAAS36B,QAAY26B,EAAS33C,OAAoB08B,WAAa32B,EAExD,IAAIomC,GAAOuL,EAAYC,EAC/B,CAgBQ,gBAAAvB,CAAkBvvB,EAAoBq0B,GAA0B,GACvE,MAAMtE,EAAiB/vB,EAAS7J,OAC1B65B,EAAiBhwB,EAAS7mB,OAGhC,GAAK6mB,EAAS7mB,OAAOsP,GAAI,gBAUxB,MAAM,IAAI,EAAe,yCAA0ClO,KAAKwD,UAIzE,GAAKiiB,EAAS7mB,OAAOsP,GAAI,aAUxB,MAAM,IAAI,EAAe,sCAAuClO,KAAKwD,UAItE,GAAKiiB,EAAS7mB,OAAOsP,GAAI,cAUxB,MAAM,IAAI,EAAe,uCAAwClO,KAAKwD,UAIvE,IAAMs2C,GAAkBrE,EAAevnC,GAAI,UAAa+rC,GAAuBxE,EAAe72C,QAC7F,OAAO6mB,EAAS1C,QAIjB,GAAKk3B,GAAuBxE,GAC3B,OAAOhwB,EAAS1C,QAIjB,GAAK0yB,EAAevnC,GAAI,SACvB,OAAOlO,KAAKg1C,iBAAkBwD,GAAe/yB,GAAYq0B,GAQ1D,GAAKtE,GALYC,EAAwBna,WAKT,CAC/B,MAAMwa,EAAc,IAAInN,GAAU8M,EAAe72C,OAAiB62C,EAAwB7wC,MAAQ,GAElG,OAAO5E,KAAKg1C,iBAAkBc,EAAagE,EAC5C,CAIC,GAAwB,IAAnBtE,EAAuB,CAC3B,MAAMM,EAAc,IAAInN,GAAU8M,EAAe72C,OAAqB62C,EAAwB7wC,OAE9F,OAAO5E,KAAKg1C,iBAAkBc,EAAagE,EAC5C,CAKK,CACJ,MAAMI,EAAgBzE,EAAwB7wC,MAAQ,EAGhDu1C,EAAe1E,EAAwBlY,SAG3CkY,EAAe72C,OAAgB+mC,aAAcuU,EAAaC,GAC5Dn6C,KAAKy4C,0BAA2B0B,GAGhC,MAAMx1C,EAAU8wC,EAAwBna,WAAaka,EAC/C4E,EAAgB3E,EAAwB3Y,gBAAiB0Y,EAAgB7wC,GAG/Ew1C,EAAWlT,aAAcmT,GAGzB,MAAMtE,EAAc,IAAInN,GAAY8M,EAAwB72C,OAAQs7C,GAEpE,OAAOl6C,KAAKg1C,iBAAkBc,EAAagE,EAC5C,CAEF,CAaQ,yBAAArB,CAA2BhhC,GAElC,IAAMA,EAAQnhB,KAAK4X,GAAI,eACtB,OAKD,GAAKuJ,EAAQvJ,GAAI,WAChB,UAAYyJ,KAASF,EAAQ+jB,cAC5Bx7B,KAAKy4C,0BAA2B9gC,GAIlC,MAAM9X,EAAO4X,EAAiB5X,GAE9B,IAAMA,EACL,OAGD,IAAIw6C,EAAQr6C,KAAKszC,aAAalrC,IAAKvI,GAE7Bw6C,IACLA,EAAQ,IAAIzmC,IACZ5T,KAAKszC,aAAa7hC,IAAK5R,EAAIw6C,IAG5BA,EAAMxsC,IAAK4J,GACTA,EAAiB45B,aAAegJ,CACnC,CAaQ,8BAAA3E,CAAgCj+B,GAGvC,GAAKA,EAAQvJ,GAAI,WAChB,UAAYyJ,KAASF,EAAQ+jB,cAC5Bx7B,KAAK01C,+BAAgC/9B,GAIvC,MAAM9X,EAAO4X,EAAiB5X,GAE9B,IAAMA,EACL,OAGD,MAAMw6C,EAAQr6C,KAAKszC,aAAalrC,IAAKvI,GAE/Bw6C,GAINA,EAAM1oC,OAAQ8F,EAGf,EAuBD,SAAS8gC,GAAoB9yB,GAC5B,IAAI7mB,EAAS6mB,EAAS7mB,OAEtB,MAASq7C,GAAuBr7C,IAAW,CAC1C,IAAMA,EACL,OAGDA,EAASA,EAAOA,MACjB,CAEA,OAASA,CACV,CAQA,SAASo6C,GAAmBr9C,EAAqBhD,GAChD,OAAKgD,EAAEiR,SAAWjU,EAAEiU,YAERjR,EAAEiR,SAAWjU,EAAEiU,WAKpBjR,EAAE+qC,cAAgB/tC,EAAE+tC,aAC5B,CAcA,SAAS6S,GAAwB9zB,GAChC,MAAM0kB,EAAa1kB,EAAS0kB,WAE5B,GAAKA,GAAcA,EAAWj8B,GAAI,SACjC,OAAO,IAAIy6B,GAAUwB,EAAYA,EAAWrnC,KAAKxK,QAGlD,MAAM4xC,EAAYzkB,EAASykB,UAE3B,OAAKA,GAAaA,EAAUh8B,GAAI,SACxB,IAAIy6B,GAAUuB,EAAW,GAG1BzkB,CACR,CAcA,SAAS+yB,GAAe/yB,GACvB,GAAKA,EAAS7J,QAAY6J,EAAS7mB,OAAiBkE,KAAKxK,OACxD,OAAO,IAAIqwC,GAAUljB,EAAS7mB,OAAOA,OAAiB6mB,EAAS7mB,OAAiBgG,MAAS,GAG1F,GAAyB,IAApB6gB,EAAS7J,OACb,OAAO,IAAI+sB,GAAUljB,EAAS7mB,OAAOA,OAAiB6mB,EAAS7mB,OAAiBgG,OAIjF,MAAM01C,EAAe70B,EAAS7mB,OAAiBkE,KAAKhB,MAAO2jB,EAAS7J,QAYpE,OATE6J,EAAS7mB,OAAiBw+B,MAAU3X,EAAS7mB,OAAiBkE,KAAKhB,MAAO,EAAG2jB,EAAS7J,QAGtF6J,EAAS7mB,OAAOA,OAAgB+mC,aAC/BlgB,EAAS7mB,OAAiBgG,MAAS,EACrC,IAAIs4B,GAAMzX,EAASnvB,KAAKkN,SAAU82C,IAI5B,IAAI3R,GAAUljB,EAAS7mB,OAAOA,OAAiB6mB,EAAS7mB,OAAiBgG,MAAS,EAC1F,CASA,SAAS+wC,GAAgBp7C,EAAUH,GAElC,MAAMmgD,EAAmBhgD,EAAGuI,KAAKxK,OAIjC,OAHAiC,EAAG6iC,OAAShjC,EAAG0I,KACf1I,EAAGi3B,UAEI,IAAIsX,GAAUpuC,EAAIggD,EAC1B,CAEA,MAAMC,GAAqB,CAAEtd,GAAMgU,GAAkBlJ,GAAkByJ,GAAc2B,GAAY1B,IAQjG,SAASqE,GAAuB7O,EAAuBuT,GACtD,UAAYt7C,KAAQ+nC,EAAQ,CAC3B,IAAMsT,GAAmBnmB,MAAQqmB,GAAav7C,aAAgBu7C,IAgB7D,MAAM,IAAI,EAAe,uCAAwCD,GAG5Dt7C,EAAK+O,GAAI,UACd6nC,GAAyB52C,EAAkBq8B,cAAeif,EAE5D,CACD,CAOA,SAASR,GAAuB96C,GAC/B,OAAOA,IAAUA,EAAK+O,GAAI,qBAAwB/O,EAAK+O,GAAI,oBAC5D,CAOA,SAASmoC,GAAwBvxB,EAAc21B,GAC9C,MAAMv1B,EAAiBqzB,GAAoBzzB,EAAM4b,OAC3Cia,EAAepC,GAAoBzzB,EAAM6b,KAE/C,IAAMzb,IAAmBy1B,GAAgBz1B,IAAmBy1B,EAiB3D,MAAM,IAAI,EAAe,sCAAuCF,EAElE,CAMA,SAASb,GAAaj+C,EAAqBhD,GAC1C,OAAgB,OAATgD,EAAEkE,IAAwB,OAATlH,EAAEkH,EAC3B,CCtvEO,MAAM+6C,GAAgBhJ,GAAiCA,EAAYvrC,eAAgB,KAS7Ew0C,GAAuBjJ,IACnC,MAAMkJ,EAAOlJ,EAAYtsC,cAAe,QAIxC,OAHAw1C,EAAKC,QAAQC,UAAY,OACzBF,EAAKG,UAAY,IAEVH,CAAI,EAUCI,GAActJ,IAC1B,MAAMuJ,EAAWvJ,EAAYtsC,cAAe,MAG5C,OAFA61C,EAASJ,QAAQC,UAAY,OAEtBG,CAAQ,EAMHC,GAAuB,EAKvBC,GAAgB,IAASC,OAAQF,IAevC,SAASG,GAAkBC,GACjC,MAAuB,iBAAXA,EACJA,EAAQvpC,OAAQ,EAAGmpC,MAA2BC,GAG/Ch6B,GAAQm6B,IAAeA,EAAQ14C,KAAKmP,OAAQ,EAAGmpC,MAA2BC,EAClF,CAaO,SAASI,GAAgBC,GAC/B,OAAOA,EAAQ54C,KAAKxK,QAAU8iD,IAAwBG,GAAkBG,EACzE,CAcO,SAASC,GAAsBD,GACrC,MAAM54C,EAAyB,iBAAX44C,EAAsBA,EAAUA,EAAQ54C,KAE5D,OAAKy4C,GAAkBG,GACf54C,EAAKhB,MAAOs5C,IAGbt4C,CACR,CAeA,SAAS84C,GAAsBvmC,EAAcvS,GAC5C,GAAKA,EAAKwqB,SAAWjB,GAASC,UAAY,CACzC,MAAM2lB,EAAenvC,EAAKovC,UAAU9yB,cAAcF,YAAaizB,eAE/D,GAAgC,GAA3BF,EAAa/E,YAAmB+E,EAAaI,WAAY,GAAIC,UAAY,CAC7E,MAAMC,EAAYN,EAAaI,WAAY,GAAIntB,eACzCutB,EAAYR,EAAaI,WAAY,GAAIrI,YAE1CuR,GAAkBhJ,IAAeE,GAAa2I,IAClDnJ,EAAagB,SAAUV,EAAW,EAEpC,CACD,CACD,C,eCxKI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQzY,OC2CR,MAAM+hB,WAAiB1oC,KA4ErC,WAAApR,CAAa8vC,EAA4B5c,GACxCznB,QAzED,KAAgBsuC,aAAiC,IAAIloC,IAUrD,KAAgBmoC,iBAAqC,IAAInoC,IAKzD,KAAgBooC,eAAmC,IAAIpoC,IAKvD,KAAgBqoC,YAA6B,IAAIroC,IAuCjD,KAAQsoC,cAAgC,KAKxC,KAAQC,wBAA6C,KAWpDn8C,KAAK6xC,aAAeA,EACpB7xC,KAAKi1B,UAAYA,EAEjBj1B,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,eAAe,GAMpB,EAAInI,UAAY,EAAID,WACxBrJ,KAAKqP,GAA2B,sBAAsB,KAC/CrP,KAAKo8C,aACVp8C,KAAK2xC,QACN,IAIF3xC,KAAKyR,IAAK,eAAe,GAEzBzR,KAAKqP,GAA2B,sBAAsB,KAC/CrP,KAAK27B,aACV37B,KAAK2xC,QACN,GAEF,CAcO,UAAA0K,CAAYpxC,EAAkB9L,GACpC,GAAc,SAAT8L,EACCjL,KAAK6xC,aAAayK,aAAcn9C,EAAKP,SACzCoB,KAAKi8C,YAAYpuC,IAAK1O,OAEjB,CAGN,IAAMa,KAAK6xC,aAAayK,aAAcn9C,GACrC,OAGD,GAAc,eAAT8L,EACJjL,KAAK+7C,iBAAiBluC,IAAK1O,OACrB,IAAc,aAAT8L,EAEL,CASN,MAAM,IAAI,EAAe,6BAA8BjL,KACxD,CAXCA,KAAKg8C,eAAenuC,IAAK1O,EAW1B,CACD,CACD,CAaO,MAAAwyC,GAKN,GAAK3xC,KAAK27B,cAAgB,EAAItyB,UAO7B,OASD,IAAIkzC,EAA4C,KAChD,MAAMC,IAAkC,EAAIlzC,UAAY,EAAID,aAAarJ,KAAKo8C,YAG9E,UAAY3kC,KAAWzX,KAAKg8C,eAC3Bh8C,KAAKy8C,wBAAyBhlC,GAM1B+kC,GAICx8C,KAAKk8C,gBAAkBl8C,KAAK08C,8BAChC18C,KAAK28C,sBAID38C,KAAKk8C,cACTK,EAAuBv8C,KAAK48C,2BAGnB58C,KAAK68C,kCACdN,EAAuBv8C,KAAKi1B,UAAUE,mBAGtCn1B,KAAKg8C,eAAenuC,IAAK0uC,EAAqB39C,UAItCoB,KAAKk8C,eAAiBl8C,KAAKk8C,cAAcj1C,aAElDs1C,EAAuBv8C,KAAK6xC,aAAae,kBAAmB5yC,KAAKk8C,eAM5DK,GAAwBA,EAAqB39C,OAAOsP,GAAI,WAE5DquC,EAAuB,GAAaxS,cAAewS,EAAqB39C,UAI1E,UAAY6Y,KAAWzX,KAAK+7C,iBAC3B/7C,KAAK88C,aAAcrlC,GAGpB,UAAYA,KAAWzX,KAAKg8C,eAC3Bh8C,KAAK+8C,gBAAiBtlC,EAAS,CAAE8kC,yBAGlC,UAAYp9C,KAAQa,KAAKi8C,aAClBj8C,KAAKg8C,eAAepuC,IAAKzO,EAAKP,SAA2BoB,KAAK6xC,aAAayK,aAAcn9C,EAAKP,SACnGoB,KAAKg9C,YAAa79C,EAAkB,CAAEo9C,yBAYxC,GAAKC,EACJ,GAAKD,EAAuB,CAC3B,MAAMU,EAAoBj9C,KAAK6xC,aAAamB,kBAAmBuJ,GACzD3K,EAAcqL,EAAkBr+C,OAAOwgB,cAEvCm8B,GAAkB0B,EAAkBr+C,QAKzCoB,KAAKk8C,cAAgBe,EAAkBr+C,OAHvCoB,KAAKk8C,cAAgBgB,GAAiBtL,EAAaqL,EAAkBr+C,OAAQq+C,EAAkBrhC,OAKjG,MAEC5b,KAAKk8C,cAAgB,KAMvBl8C,KAAKm9C,eACLn9C,KAAKo9C,mBAELp9C,KAAKi8C,YAAYrqC,QACjB5R,KAAK+7C,iBAAiBnqC,QACtB5R,KAAKg8C,eAAepqC,OAKrB,CAWQ,uBAAA6qC,CAAyBlF,GAGhC,IAFmBv3C,KAAK6xC,aAAayK,aAAc/E,GAIlD,OASD,MAAM8F,EAAoB18C,MAAMrB,KAC/BU,KAAK6xC,aAAayK,aAAc/E,GAAejxC,YAE1Cg3C,EAAsB38C,MAAMrB,KACjCU,KAAK6xC,aAAa0L,kBAAmBhG,EAAa,CAAEiG,cAAc,KAE7DjkD,EAAOyG,KAAKy9C,eAAgBJ,EAAmBC,GAC/CI,EAAU19C,KAAK29C,mBAAoBpkD,EAAM8jD,EAAmBC,EAAqBM,IAEvF,IAAsC,IAAjCF,EAAQ10C,QAAS,UAAoB,CACzC,MAAM60C,EAAU,CAAEC,MAAO,EAAGp4C,OAAQ,EAAGiM,OAAQ,GAE/C,UAAYosC,KAAUL,EACrB,GAAgB,WAAXK,EAAsB,CAC1B,MAAMC,EAAcH,EAAQC,MAAQD,EAAQn4C,OACtCu4C,EAAcJ,EAAQC,MAAQD,EAAQlsC,OACtCusC,EAAY3G,EAAY1b,SAAUmiB,IAKnCE,GAAgBA,EAAUhwC,GAAI,cAAiBgwC,EAAUhwC,GAAI,eACjElO,KAAKm+C,uBAAwBD,EAA0Bb,EAAmBY,IAG3Eh4C,GAAQq3C,EAAqBU,IAC7BH,EAAQC,OACT,MACCD,EAASE,IAGZ,CACD,CAQQ,sBAAAI,CAAwB5G,EAA0BxF,GAEzD/xC,KAAK6xC,aAAauM,iBAAkBrM,GACpC/xC,KAAK6xC,aAAawM,aAActM,EAAYwF,GAG5Cv3C,KAAKg8C,eAAenuC,IAAK0pC,GAWzBv3C,KAAK+7C,iBAAiBluC,IAAK0pC,EAC5B,CAYQ,wBAAAqF,GACP,MAAM0B,EAAWt+C,KAAKi1B,UAAUE,mBAEhC,OAAKmpB,EAAS1/C,OAAOsP,GAAI,SACjB,GAAa67B,cAAeuU,EAAS1/C,QAErC0/C,CAET,CASQ,0BAAA5B,GACP,GAAkC,GAA7B18C,KAAKi1B,UAAUiY,aAAoBltC,KAAKi1B,UAAU+V,YACtD,OAAO,EAYR,MAAMuT,EAAoBv+C,KAAKi1B,UAAUE,mBACnC1P,EAAWzlB,KAAK6xC,aAAamB,kBAAmBuL,GAEtD,SAAK94B,GAAYpE,GAAQoE,EAAS7mB,SAAY28C,GAAkB91B,EAAS7mB,QAK1E,CAKQ,mBAAA+9C,GACP,MAAM6B,EAAgBx+C,KAAKk8C,cAG3B,IAAMX,GAAkBiD,GAOvB,MAAM,IAAI,EAAe,gCAAiCx+C,MAGtDy7C,GAAgB+C,GACpBA,EAAcv4C,SAEdu4C,EAAc17C,KAAO07C,EAAc17C,KAAKmP,OAAQmpC,IAGjDp7C,KAAKk8C,cAAgB,IACtB,CAOQ,6BAAAW,GACP,GAAkC,GAA7B78C,KAAKi1B,UAAUiY,aAAoBltC,KAAKi1B,UAAU+V,YACtD,OAAO,EAGR,MAAMuT,EAAoBv+C,KAAKi1B,UAAUE,mBACnCspB,EAAkBF,EAAkB3/C,OACpC8/C,EAAkBH,EAAkB3iC,OAG1C,IAAM5b,KAAK6xC,aAAayK,aAAcmC,EAAgBnoD,MACrD,OAAO,EAGR,IAAQmoD,EAAgBvwC,GAAI,WAC3B,OAAO,EAKR,IAghBF,SAAqBuJ,GACpB,GAAkD,SAA7CA,EAAQ4jB,aAAc,mBAC1B,OAAO,EAGR,MAAMz8B,EAAS6Y,EAAQ6uB,cAAc7uB,GAAWA,EAAQwuB,aAAc,qBAEtE,OAAQrnC,GAAsD,QAA5CA,EAAOy8B,aAAc,kBACxC,CAxhBQsjB,CAAYF,GACjB,OAAO,EAIR,GAAKC,IAAoBD,EAAgBzX,kBACxC,OAAO,EAGR,MAAMmD,EAAaoU,EAAkBpU,WAC/BD,EAAYqU,EAAkBrU,UAEpC,QAAKC,aAAsB,IAAYD,aAAqB,OAMvD,EAAI7gC,YAAe8gC,IAAcD,EAKvC,CAQQ,WAAA8S,CAAa4B,EAAoBr6C,GACxC,MAAMm3C,EAAU17C,KAAK6xC,aAAagN,yBAA0BD,GAG5D,IAAIE,EAFe9+C,KAAK6xC,aAAakN,UAAWH,GAElB97C,KAC9B,MAAMk8C,EAASz6C,EAAQg4C,qBAElByC,GAAUA,EAAOpgD,QAAUggD,EAAShgD,QAAUogD,EAAOpjC,QAAUgjC,EAASh6C,QAC5Ek6C,EAAezD,GAAgByD,GAShCG,GAAgBvD,EAASoD,EAK1B,CAOQ,YAAAhC,CAAcvF,GACrB,MAAMxF,EAAa/xC,KAAK6xC,aAAayK,aAAc/E,GAEnD,IAAMxF,EAKL,OAGD,MAAMmN,EAAcv+C,MAAMrB,KAAQyyC,EAA2BxsC,YAAavI,KAAKmiD,GAAQA,EAAKn9C,OACtFo9C,EAAe7H,EAAYjZ,mBAGjC,UAAYvnC,KAAOqoD,EAClBp/C,KAAK6xC,aAAawN,uBAAwBtN,EAA0Bh7C,EAAKwgD,EAAYlc,aAActkC,GAAQwgD,GAI5G,UAAYxgD,KAAOmoD,EAEZ3H,EAAYtR,aAAclvC,IAC/BiJ,KAAK6xC,aAAayN,0BAA2BvN,EAA0Bh7C,EAG1E,CAWQ,eAAAgmD,CAAiBxF,EAA0BhzC,GAClD,MAAMwtC,EAAa/xC,KAAK6xC,aAAayK,aAAc/E,GAEnD,IAAMxF,EAGL,OAaD,GAAK,EAAI1oC,UAAY,CACpB,IAAIk2C,EAAkB,KAEtB,UAAY/D,KAAW76C,MAAMrB,KAAMyyC,EAAWzrC,YAAe,CAC5D,GAAKi5C,GAAmBl+B,GAAQk+B,IAAqBl+B,GAAQm6B,GAAY,CACxEzJ,EAAW5K,YAEX,KACD,CAEAoY,EAAkB/D,CACnB,CACD,CAEA,MAAMe,EAAuBh4C,EAAQg4C,qBAC/Bc,EAAoBtL,EAAWzrC,WAC/Bg3C,EAAsB38C,MAAMrB,KACjCU,KAAK6xC,aAAa0L,kBAAmBhG,EAAa,CAAEvwC,MAAM,KAMtDu1C,GAAwBA,EAAqB39C,SAAW24C,GAC5D2F,GAAmBnL,EAA2B3yB,cAAek+B,EAAqBf,EAAqB3gC,QAGxG,MAAMriB,EAAOyG,KAAKy9C,eAAgBJ,EAAmBC,GAI/CI,EAAU19C,KAAK29C,mBAAoBpkD,EAAM8jD,EAAmBC,EAAqBkC,IAEvF,IAAIhlD,EAAI,EACR,MAAMilD,EAA8B,IAAI7rC,IAQxC,UAAYmqC,KAAUL,EACL,WAAXK,GAMJ0B,EAAc5xC,IAAKwvC,EAAmB7iD,IACtCyL,GAAQo3C,EAAmB7iD,KACL,UAAXujD,GAAiC,WAAXA,GACjCvjD,IAIFA,EAAI,EAEJ,UAAYujD,KAAUL,EACL,WAAXK,GAOJl3B,GAAUkrB,EAA0Bv3C,EAAG8iD,EAAqB9iD,IAC5DA,KAGoB,WAAXujD,GAOTkB,GAAgB5B,EAAmB7iD,GAAkB8iD,EAAqB9iD,GAAiBsI,MAC3FtI,KAKsB,UAAXujD,IAGX/9C,KAAK0/C,0BAA2B1/C,KAAK6xC,aAAa8N,UAAWrC,EAAqB9iD,KAClFA,KAOF,UAAY2E,KAAQsgD,EACbtgD,EAAK8H,YACVjH,KAAK6xC,aAAauM,iBAAkBj/C,EAOvC,CASQ,cAAAs+C,CAAgBJ,EAA8CC,GAGrE,OAFAD,EA+ZF,SAA0CuC,EAAyCC,GAClF,MAAMC,EAAYn/C,MAAMrB,KAAMsgD,GAE9B,GAAyB,GAApBE,EAAUxnD,SAAgBunD,EAC9B,OAAOC,EAGR,MAAMpvB,EAAOovB,EAAWA,EAAUxnD,OAAS,GAEtCo4B,GAAQmvB,GACZC,EAAU9gD,MAGX,OAAO8gD,CACR,CA7asBC,CAAiC1C,EAAmBr9C,KAAKm8C,yBAEtE5iD,EAAM8jD,EAAmBC,EAAqB0C,GAAUh5C,KAAM,KAAMhH,KAAK6xC,cACjF,CAoBQ,kBAAA8L,CACPD,EACAuC,EACAC,EACAC,GAGA,IAAsC,IAAjCzC,EAAQ10C,QAAS,YAAsD,IAAjC00C,EAAQ10C,QAAS,UAC3D,OAAO00C,EAGR,IAAI0C,EAA2C,GAC3CC,EAAc,GACdC,EAAgB,GAEpB,MAAMzC,EAAU,CAAEC,MAAO,EAAGp4C,OAAQ,EAAGiM,OAAQ,GAE/C,UAAYosC,KAAUL,EACL,WAAXK,EACJuC,EAAc1gD,KAAMsgD,EAAarC,EAAQC,MAAQD,EAAQn4C,SACnC,WAAXq4C,EACXsC,EAAYzgD,KAAMqgD,EAAWpC,EAAQC,MAAQD,EAAQlsC,UAErDyuC,EAAaA,EAAWjgD,OACvB5G,EAAM8mD,EAAaC,EAAeH,GAChCnjD,KAAK+gD,GAAqB,UAAXA,EAAqB,SAAWA,KAGlDqC,EAAWxgD,KAAM,SAGjBygD,EAAc,GACdC,EAAgB,IAEjBzC,EAASE,KAGV,OAAOqC,EAAWjgD,OACjB5G,EAAM8mD,EAAaC,EAAeH,GAChCnjD,KAAK+gD,GAAqB,UAAXA,EAAqB,SAAWA,IAEnD,CASQ,yBAAA2B,CAA2Ba,GAClC,GAAMA,EAIN,GAAKA,EAASryC,GAAI,SACjBlO,KAAKi8C,YAAYpuC,IAAK0yC,QAChB,GAAKA,EAASryC,GAAI,WACxB,UAAYyJ,KAAS4oC,EAAS/kB,cAC7Bx7B,KAAK0/C,0BAA2B/nC,EAGnC,CAKQ,gBAAAylC,GAKP,GAAK,EAAI9zC,UAAY,EAAID,WAAarJ,KAAKo8C,cAAgBp8C,KAAKg8C,eAAenoC,KAC9E,OAID,GAAmC,IAA9B7T,KAAKi1B,UAAUiY,WAInB,OAHAltC,KAAKwgD,2BACLxgD,KAAKygD,uBAKN,MAAMC,EAAU1gD,KAAK6xC,aAAayK,aAAct8C,KAAKi1B,UAAUkT,iBAGzDnoC,KAAK8yB,WAAc4tB,IAKpB1gD,KAAKi1B,UAAU8X,OACnB/sC,KAAK2gD,qBAAsBD,GAIlB1gD,KAAKm8C,yBAA2Bn8C,KAAKm8C,wBAAwByE,aACtE5gD,KAAKygD,uBACLzgD,KAAK6gD,oBAAqBH,IAMd1gD,KAAK27B,aAAe,EAAItyB,WACpCrJ,KAAK6gD,oBAAqBH,GAE5B,CAOQ,oBAAAC,CAAsBD,GAC7B,MAAM9O,EAAc8O,EAAQthC,cAEtBpf,KAAKm8C,0BACVn8C,KAAKm8C,wBAoSR,SAAuCvK,GACtC,MAAMkP,EAAYlP,EAAYtsC,cAAe,OAe7C,OAbAw7C,EAAU3a,UAAY,8BAEtBnvC,OAAO+wB,OAAQ+4B,EAAUz7C,MAAO,CAC/BogB,SAAU,QACVhE,IAAK,EACLM,KAAM,UAENc,MAAO,SAIRi+B,EAAUC,YAAc,IAEjBD,CACR,CArTkCE,CAA8BpP,IAG9D,MAAMkP,EAAY9gD,KAAKm8C,wBAKvB,GAFAn8C,KAAK6xC,aAAaoP,kBAAmBH,EAAW9gD,KAAKi1B,YAE/Cj1B,KAAKkhD,0BAA2BR,GACrC,OAGKI,EAAUh6B,eAAiBg6B,EAAUh6B,eAAiB45B,GAC3DA,EAAQ/6C,YAAam7C,GAGtBA,EAAUC,YAAc/gD,KAAKi1B,UAAU+X,oBAAsB,IAE7D,MAAMiF,EAAeL,EAAYO,eAC3BgP,EAAWvP,EAAYiG,cAE7B5F,EAAamP,kBACbD,EAASE,mBAAoBP,GAC7B7O,EAAaqP,SAAUH,EACxB,CAOQ,mBAAAN,CAAqBH,GAC5B,MAAMzO,EAAeyO,EAAQthC,cAAcF,YAAaizB,eAGxD,IAAMnyC,KAAKuhD,yBAA0BtP,GACpC,OAQD,MAAMvW,EAAS17B,KAAK6xC,aAAamB,kBAAmBhzC,KAAKi1B,UAAUyG,QAC7DuR,EAAQjtC,KAAK6xC,aAAamB,kBAAmBhzC,KAAKi1B,UAAUgY,OAQlEgF,EAAauP,iBAAkB9lB,EAAO98B,OAAQ88B,EAAO9f,OAAQqxB,EAAMruC,OAAQquC,EAAMrxB,QAG5E,EAAI3S,SAuMX,SAAmCgkC,EAAwDgF,GAC1F,MAAMrzC,EAASquC,EAAOruC,OAItB,GAAKA,EAAOmb,UAAYsF,KAAKoiC,cAAgBxU,EAAOrxB,QAAUhd,EAAO0H,WAAWhO,OAAS,EACxF,OAGD,MAAMopD,EAAgB9iD,EAAO0H,WAAY2mC,EAAOrxB,QAI3C8lC,GAA4D,MAAzCA,EAA8BC,SACrD1P,EAAaqP,SAAUrP,EAAaI,WAAY,GAElD,CAtNGuP,CAA0B3U,EAAOgF,EAEnC,CAOQ,wBAAAsP,CAA0BtP,GACjC,IAAMjyC,KAAK6xC,aAAagQ,sBAAuB5P,GAE9C,OAAO,EAGR,MAAM6P,EAAmB7P,GAAgBjyC,KAAK6xC,aAAakQ,mBAAoB9P,GAE/E,QAAK6P,IAAoB9hD,KAAKi1B,UAAUhR,QAAS69B,QAK3C9hD,KAAKi1B,UAAU+V,aAAehrC,KAAKi1B,UAAUoI,UAAWykB,GAO/D,CAOQ,yBAAAZ,CAA2BR,GAClC,MAAMI,EAAY9gD,KAAKm8C,wBACjBlK,EAAeyO,EAAQthC,cAAc+yB,eAI3C,OAAM2O,GAAaA,EAAUh6B,gBAAkB45B,IAK1CzO,EAAa+P,aAAelB,IAAcA,EAAU58B,SAAU+tB,EAAa+P,aAIzElB,EAAUC,cAAgB/gD,KAAKi1B,UAAU+X,mBACjD,CAKQ,mBAAAwT,GACP,UAAYnpC,KAAOrX,KAAK87C,aAAe,CACtC,MAAM7J,EAAe56B,EAAI86B,eAEzB,GAAKF,EAAa/E,WAAa,CAC9B,MAAM+U,EAAmB5qC,EAAI6qC,cACvB3K,EAAcv3C,KAAK6xC,aAAasQ,aAAcF,GAE/CA,GAAoB1K,GACxBtF,EAAamP,iBAEf,CACD,CACD,CAKQ,oBAAAX,GACP,MAAMK,EAAY9gD,KAAKm8C,wBAElB2E,GACJA,EAAU76C,QAEZ,CAKQ,YAAAk3C,GACP,GAAKn9C,KAAK8yB,UAAY,CACrB,MAAMuX,EAAWrqC,KAAKi1B,UAAUkT,gBAE3BkC,GACJrqC,KAAK6xC,aAAa5E,MAAO5C,EAE3B,CACD,EAwBD,SAAS6S,GAAiBtL,EAA0BwQ,EAA4CxmC,GAC/F,MAAMtV,EAAa87C,aAA4BzhD,MAAQyhD,EAAmBA,EAAiB97C,WACrF+7C,EAAkB/7C,EAAYsV,GAEpC,GAAKyF,GAAQghC,GAGZ,OAFAA,EAAgBv/C,KAAOu4C,GAAgBgH,EAAgBv/C,KAEhDu/C,EACD,CACN,MAAMC,EAAa1Q,EAAYvrC,eAAgBg1C,IAQ/C,OANK16C,MAAMC,QAASwhD,GACjB97C,EAA+BkB,OAAQoU,EAAQ,EAAG0mC,GAEpDz7B,GAAUu7B,EAAgCxmC,EAAQ0mC,GAG5CA,CACR,CACD,CAMA,SAAS1E,GAAoB2E,EAAgBC,GAC5C,OAAOvjC,GAAQsjC,IAAWtjC,GAAQujC,KAChCnhC,GAAQkhC,KAAYlhC,GAAQmhC,KAC5Bx7B,GAAWu7B,KAAYv7B,GAAWw7B,IACjCD,EAAsBZ,QAAQ/4C,gBAAoB45C,EAAsBb,QAAQ/4C,aACpF,CAKA,SAAS42C,GAAc+C,EAAgBC,GACtC,OAAOvjC,GAAQsjC,IAAWtjC,GAAQujC,IACjCnhC,GAAQkhC,IAAWlhC,GAAQmhC,EAC7B,CAYA,SAASxC,GAAWnO,EAA4B4Q,EAAyBC,GAExE,OAAKD,IAAmBC,IAIdrhC,GAAQohC,IAAoBphC,GAAQqhC,GACtCD,EAAe3/C,OAAS4/C,EAAiB5/C,QAGvC+uC,EAAa8Q,cAAeF,KACrC5Q,EAAa8Q,cAAeD,IAM9B,CA4EA,SAASzD,GAAgBvD,EAAkBoD,GAC1C,MAAM8D,EAAalH,EAAQ54C,KAE3B,GAAK8/C,GAAc9D,EAQlB,OAUD,MAAMpB,EAAU7zC,EAAU+4C,EAAY9D,GAEtC,UAAYf,KAAUL,EACA,WAAhBK,EAAO9yC,KACXywC,EAAQmH,WAAY9E,EAAOn5C,MAAOm5C,EAAO7yC,OAAOhO,KAAM,KAEtDw+C,EAAQoH,WAAY/E,EAAOn5C,MAAOm5C,EAAO5yC,QAG5C,CCnqCA,MAAM43C,GAAgB7H,GAAW,GAAO13C,UAClCw/C,GAAkBpI,GAAa,GAAOp3C,UACtCy/C,GAAyBpI,GAAoB,GAAOr3C,UACpD0/C,GAA+B,4BAC/BC,GAAuC,yBAe9B,MAAMC,GAyFpB,WAAArhD,CACCyB,GACA,gBAAE6/C,EAAe,cAAEC,EAAgB,WAG/B,CAAC,GAxCN,KAAiBC,kBAAoB,IAAIlzB,QAKzC,KAAiBmzB,kBAAoB,IAAInzB,QAKzC,KAAiBozB,sBAAwB,IAAIpzB,QAM7C,KAAiBqzB,0BAA4B,IAAI,GAKjD,KAAiBC,4BAA8B,IAAI,GAqBlD3jD,KAAKwD,SAAWA,EAChBxD,KAAKsjD,cAAgBA,EACrBtjD,KAAKqjD,gBAAkBA,IAAuC,YAAlBC,EAA8B,KAAO,QACjFtjD,KAAK4jD,YAAc,CAAE,OACrB5jD,KAAK6jD,cAAgB,CACpB,UAAW,UAAW,QAAS,aAAc,UAAW,SAAU,KAAM,UAAW,MAAO,MAC1F,KAAM,KAAM,WAAY,aAAc,SAAU,SAAU,OAAQ,KAAM,KAAM,KAAM,KAAM,KAAM,KAAM,SACtG,SAAU,SAAU,KAAM,OAAQ,OAAQ,MAAO,KAAM,IAAK,MAAO,UAAW,UAAW,QAAS,QAClG,KAAM,QAAS,KAAM,QAAS,KAAM,MAErC7jD,KAAK8jD,qBAAuB,CAC3B,SAAU,SAAU,QAAS,SAAU,WAAY,SAAU,SAAU,QAAS,QAAS,QAAS,MAAO,UAE1G9jD,KAAK+jD,eAAiB,CAAE,SAAU,SAElC/jD,KAAKgkD,aAAsC,YAAvBhkD,KAAKsjD,cAA8B,GAAO9/C,SAAW,GAAOA,SAASygD,eAAeC,mBAAoB,GAC7H,CAQO,iBAAAjD,CAAmBlP,EAAwBoS,GACjDnkD,KAAKyjD,sBAAsBhyC,IAAKsgC,EAAY,IAAI,GAAeoS,GAChE,CAMO,mBAAAC,CAAqBrS,GAC3B,OAAO/xC,KAAKyjD,sBAAsBr7C,IAAK2pC,EACxC,CAUO,YAAAsM,CAActM,EAAwBwF,GAC5Cv3C,KAAKujD,kBAAkB9xC,IAAKsgC,EAAYwF,GACxCv3C,KAAKwjD,kBAAkB/xC,IAAK8lC,EAAaxF,EAC1C,CAQO,gBAAAqM,CAAkBrM,GACxB,MAAMwF,EAAcv3C,KAAKujD,kBAAkBn7C,IAAK2pC,GAEhD,GAAKwF,EAAc,CAClBv3C,KAAKujD,kBAAkB5xC,OAAQogC,GAC/B/xC,KAAKwjD,kBAAkB7xC,OAAQ4lC,GAE/B,UAAY5/B,KAAShX,MAAMrB,KAAMyyC,EAAWz6B,UAC3CtX,KAAKo+C,iBAAkBzmC,EAEzB,CACD,CAUO,qBAAA0sC,CAAuBC,EAAkCC,GAC/DvkD,KAAKujD,kBAAkB9xC,IAAK6yC,EAAaC,GACzCvkD,KAAKwjD,kBAAkB/xC,IAAK8yC,EAAcD,EAC3C,CAOO,qBAAAE,CAAuBC,EAAsBC,EAAwBC,GAC3E,MAA4B,SAAvB3kD,KAAKsjD,iBAIVmB,EAAeA,EAAa77C,eAEVg8C,WAAY,SAKZ,WAAjBH,IACAC,EAAe7nD,MAAO,6CAMN,QAAhB8nD,IACmB,QAAjBF,GAA2C,WAAjBA,KAKR,WAAhBE,GAA6C,WAAjBF,IAI5BC,EAAe7nD,MAAO,uDAK5B,CAQO,YAAAgoD,CAAc9S,EAAwB+S,GAE5C,GAA4B,SAAvB9kD,KAAKsjD,cAGT,YAFAvR,EAAWtrB,UAAYq+B,GAKxB,MAAMthD,GAAW,IAAIuhD,WAAYC,gBAAiBF,EAAM,aAClDG,EAAWzhD,EAASkwC,yBACpBwR,EAAiB1hD,EAASgiB,KAAKlf,WAErC,KAAQ4+C,EAAe5sD,OAAS,GAC/B2sD,EAASt/C,YAAau/C,EAAgB,IAGvC,MAAMza,EAAajnC,EAAS2hD,iBAAkBF,EAAUG,WAAWC,cAC7Dne,EAA2B,GAEjC,IAAIoe,EAGJ,KAAQA,EAAc7a,EAAW8a,YAChCre,EAAMtnC,KAAM0lD,GAGb,UAAYA,KAAepe,EAAQ,CAElC,UAAYN,KAAiB0e,EAAYE,oBACxCxlD,KAAKq/C,uBAAwBiG,EAAa1e,EAAe0e,EAAYjqB,aAAcuL,IAGpF,MAAM+d,EAAcW,EAAY3D,QAAQ/4C,cAGnC5I,KAAKylD,qBAAsBd,KAC/Be,GAAmBf,GAEnBW,EAAYK,YAAa3lD,KAAK4lD,6BAA8BjB,EAAaW,IAE3E,CAGA,KAAQvT,EAAWprC,YAClBorC,EAAWprC,WAAWV,SAGvB8rC,EAAW8T,OAAQZ,EACpB,CAYO,SAAAlG,CACNwB,EACAh8C,EAAsD,CAAC,GAEvD,GAAKg8C,EAASryC,GAAI,SAAY,CAC7B,MAAM43C,EAAW9lD,KAAK+lD,yBAA0BxF,GAEhD,OAAOvgD,KAAKgkD,aAAa39C,eAAgBy/C,EAC1C,CAAO,CACN,GAAK9lD,KAAKs8C,aAAciE,GACvB,OAAOvgD,KAAKs8C,aAAciE,GAG3B,IAAIxO,EAEJ,GAAKwO,EAASryC,GAAI,oBAEjB6jC,EAAa/xC,KAAKgkD,aAAatQ,yBAE1BnvC,EAAQyC,MACZhH,KAAKqkD,sBAAuBtS,EAAYwO,OAEnC,IAAKA,EAASryC,GAAI,aAYxB,OAVC6jC,EADsB,aAAlBwO,EAASv+C,KACAhC,KAAKgkD,aAAagC,cAAezF,EAAS/Z,kBAAmB,gBAG7D+Z,EAAS5O,OAAQ3xC,KAAKgkD,aAAchkD,MAG7CuE,EAAQyC,MACZhH,KAAKq+C,aAActM,EAA0BwO,GAGvCxO,EAGF/xC,KAAKylD,qBAAwBlF,EAA0Bv+C,OAC3D0jD,GAAqBnF,EAA0Bv+C,MAE/C+vC,EAAa/xC,KAAK4lD,6BAAgCrF,EAA0Bv+C,OAE5E+vC,EADawO,EAA0Bta,aAAc,SACxCjmC,KAAKgkD,aAAatsC,gBAC5B6oC,EAA0BllB,aAAc,SACxCklB,EAA0Bv+C,MAGhBhC,KAAKgkD,aAAa1+C,cAAiBi7C,EAA0Bv+C,MAKtEu+C,EAASryC,GAAI,eACjBqyC,EAAS5O,OAAQI,EAAY/xC,MAGzBuE,EAAQyC,MACZhH,KAAKq+C,aAActM,EAAcwO,GAIlC,UAAYxpD,KAASwpD,EAA0BjiB,mBAC9Ct+B,KAAKq/C,uBACJtN,EACAh7C,EACEwpD,EAA0BllB,aAActkC,GACxCwpD,EAGL,CAEA,IAA8B,IAAzBh8C,EAAQi5C,aACZ,UAAY7lC,KAAS3X,KAAKu9C,kBAAmBgD,EAAyBh8C,GACrEwtC,EAAYpsC,YAAagS,GAI3B,OAAOo6B,CACR,CACD,CAcO,sBAAAsN,CAAwBtN,EAAwBh7C,EAAayB,EAAeytD,GAClF,MAAMzB,EAAwBxkD,KAAKwkD,sBAAuBztD,EAAKyB,EAAOu5C,EAAW4P,QAAQ/4C,gBACxFq9C,GAAsBA,EAAmBtf,4BAA6B5vC,GAEjEytD,GACL,EAAY,yCAA0C,CAAEzS,aAAYh7C,MAAKyB,UCrb7D,SAA+BwJ,GAC7C,IACC,GAAOwB,SAAS0iD,gBAAiBlkD,EAClC,CAAE,MAAQ4H,GACT,OAAO,CACR,CAEA,OAAO,CACR,CDgbQu8C,CAAsBpvD,IAYvBg7C,EAAW9L,aAAclvC,KAAUytD,EACvCzS,EAAWrrC,gBAAiB3P,GAGnBg7C,EAAW9L,aAAcid,GAA+BnsD,IAASytD,GAC1EzS,EAAWrrC,gBAAiBw8C,GAA+BnsD,GAK5Dg7C,EAAWtsC,aAAc++C,EAAwBztD,EAAMmsD,GAA+BnsD,EAAKyB,IAhB1F,EAAY,0CAA2C,CAAEu5C,aAAYh7C,MAAKyB,SAiB5E,CAUO,yBAAA8mD,CAA2BvN,EAAwBh7C,GAEpDA,GAAOosD,KAIZpR,EAAWrrC,gBAAiB3P,GAG5Bg7C,EAAWrrC,gBAAiBw8C,GAA+BnsD,GAC5D,CAWA,kBAAQwmD,CACPhG,EACAhzC,EAAsD,CAAC,GAEvD,MAAM6hD,EAAuB7O,EAAYvQ,iBAAmBuQ,EAAYvQ,kBACxE,IAAIprB,EAAS,EAEb,UAAYyqC,KAAa9O,EAAY/b,cAAgB,CAC/C4qB,IAAyBxqC,UACvB5b,KAAKsmD,mBAGZ,MAAMC,EAAuBF,EAAUn4C,GAAI,cACxCm4C,EAAU7f,kBAAmB,uCAC9B,GAAO6f,EAAUrgB,iBAEdugB,GAA8C,QAAtBvmD,KAAKsjD,oBAC1BtjD,KAAKu9C,kBAAmB8I,EAAW9hD,IAErCgiD,GAMJ,EAAY,qEAAsE,CAAEhP,YAAa8O,UAG5FrmD,KAAK++C,UAAWsH,EAAW9hD,IAGlCqX,GACD,CAEKwqC,IAAyBxqC,UACvB5b,KAAKsmD,kBAEb,CASO,cAAAE,CAAgBC,GACtB,MAAMC,EAAW1mD,KAAKgzC,kBAAmByT,EAAU/lB,OAC7CimB,EAAS3mD,KAAKgzC,kBAAmByT,EAAU9lB,KAE3CwgB,EAAWnhD,KAAKgkD,aAAanM,cAInC,OAHAsJ,EAASyF,SAAUF,EAAS9nD,OAAQ8nD,EAAS9qC,QAC7CulC,EAAS0F,OAAQF,EAAO/nD,OAAQ+nD,EAAO/qC,QAEhCulC,CACR,CAcO,iBAAAnO,CAAmBL,GACzB,MAAMmU,EAAanU,EAAa/zC,OAEhC,GAAKkoD,EAAW54C,GAAI,SAAY,CAC/B,MAAMqkC,EAAYvyC,KAAK6+C,yBAA0BiI,GAEjD,IAAMvU,EAEL,OAAO,KAGR,IAAI32B,EAAS+2B,EAAa/2B,OAM1B,OAJK2/B,GAAkBhJ,KACtB32B,GAAUw/B,IAGJ,CAAEx8C,OAAQ2zC,EAAW32B,SAC7B,CAAO,CAEN,IAAI22B,EAAWwU,EAAWC,EAE1B,GAA6B,IAAxBrU,EAAa/2B,OAAe,CAGhC,GAFA22B,EAAYvyC,KAAKs8C,aAAcwK,IAEzBvU,EAEL,OAAO,KAGRyU,EAAWzU,EAAUjsC,WAAY,EAClC,KAAO,CACN,MAAM6jC,EAAawI,EAAaxI,WAMhC,GAJA4c,EAAY5c,EAAWj8B,GAAI,SAC1BlO,KAAK6+C,yBAA0B1U,GAC/BnqC,KAAKs8C,aAAcnS,IAEd4c,EAEL,OAAO,KAGRxU,EAAYwU,EAAU9/C,WACtB+/C,EAAWD,EAAU9qB,WACtB,CAIA,GAAK5a,GAAQ2lC,IAAczL,GAAkByL,GAC5C,MAAO,CAAEpoD,OAAQooD,EAAUprC,OAAQw/B,IAKpC,MAAO,CAAEx8C,OAAQ2zC,EAAY32B,OAFdmrC,EAAY/9C,GAAS+9C,GAAc,EAAI,EAGvD,CACD,CAkBO,SAAApH,CACNnE,EACAj3C,EAKI,CAAC,GAEL,MAAM0iD,EAA+B,GAC/BC,EAAYlnD,KAAKmnD,WAAY3L,EAASj3C,EAAS0iD,GAG/C9nD,EAAO+nD,EAAU5lD,OAAO9I,MAE9B,OAAM2G,GAKN+nD,EAAU5lD,OAGVtB,KAAKonD,uBAAwB,KAAMH,EAAa1iD,GAG3CpF,EAAK+O,GAAI,UAAiC,GAApB/O,EAAK2D,KAAKxK,OAC7B,KAGD6G,GAdC,IAeT,CAYA,kBAAQkoD,CACPtV,EACAxtC,EAAwD,CAAC,EACzD0iD,EAA+B,IAE/B,QAAUzsD,EAAI,EAAGA,EAAIu3C,EAAWzrC,WAAWhO,OAAQkC,IAAM,CACxD,MAAM8sD,EAAWvV,EAAWzrC,WAAY9L,GAClC0sD,EAAYlnD,KAAKmnD,WAAYG,EAAU/iD,EAAS0iD,GAGhD/I,EAAYgJ,EAAU5lD,OAAO9I,MAEhB,OAAd0lD,IAECl+C,KAAKunD,oBAAqBrJ,IAC9Bl+C,KAAKonD,uBAAwBrV,EAAYkV,EAAa1iD,SAGjD25C,EAGNgJ,EAAU5lD,OAEZ,CAGAtB,KAAKonD,uBAAwBrV,EAAYkV,EAAa1iD,EACvD,CASO,kBAAAw9C,CAAoB9P,GAG1B,GAAiC,IAA5BA,EAAa/E,WAAmB,CACpC,IAAI4T,EAAyB7O,EAAaI,WAAY,GAAIntB,eAGrD7D,GAAQy/B,KACZA,EAAYA,EAAU75C,YAGvB,MAAM+vC,EAAgBh3C,KAAKokD,oBAAqBtD,GAEhD,GAAK9J,EACJ,OAAOA,CAET,CAEA,MAAM7J,EAAantC,KAAKwnD,uBAAwBvV,GAE1CwV,EAA+B,GAErC,QAAUjtD,EAAI,EAAGA,EAAIy3C,EAAa/E,WAAY1yC,IAAM,CAEnD,MAAM2mD,EAAWlP,EAAaI,WAAY73C,GACpCisD,EAAYzmD,KAAK0nD,eAAgBvG,GAElCsF,GACJgB,EAAW7nD,KAAM6mD,EAEnB,CAEA,OAAO,IAAI,GAAegB,EAAY,CAAEjZ,SAAUrB,GACnD,CASO,cAAAua,CAAgBvG,GACtB,MAAMwG,EAAY3nD,KAAK4yC,kBAAmBuO,EAASj8B,eAAgBi8B,EAASnX,aACtE4d,EAAU5nD,KAAK4yC,kBAAmBuO,EAASxG,aAAcwG,EAAS/W,WAExE,OAAKud,GAAaC,EACV,IAAI,GAAWD,EAAWC,GAG3B,IACR,CAiBO,iBAAAhV,CAAmBL,EAAoBE,EAAoB,GACjE,GAAKzyC,KAAK2iD,cAAepQ,GACxB,OAAOvyC,KAAK4yC,kBAAmBL,EAAUtrC,WAAa+B,GAASupC,IAIhE,MAAMgF,EAAcv3C,KAAKmiD,aAAc5P,GAEvC,GAAKgF,IAAiBA,EAAYrpC,GAAI,cAAiBqpC,EAAYrpC,GAAI,eACtE,OAAO,GAAa67B,cAAewN,GAGpC,GAAKl2B,GAAQkxB,GAAc,CAC1B,GAAKkJ,GAAgBlJ,GACpB,OAAOvyC,KAAK4yC,kBAAmBL,EAAUtrC,WAAa+B,GAASupC,IAGhE,MAAMuU,EAAa9mD,KAAK6nD,0BAA2BtV,GACnD,IAAI32B,EAAS62B,EAEb,OAAMqU,GAIDvL,GAAkBhJ,KACtB32B,GAAUw/B,GACVx/B,EAASA,EAAS,EAAI,EAAIA,GAGpB,IAAI,GAAckrC,EAAYlrC,IAR7B,IAST,CAGC,GAAmB,IAAd62B,EAAkB,CACtB,MAAMqU,EAAa9mD,KAAKmiD,aAAc5P,GAEtC,GAAKuU,EACJ,OAAO,IAAI,GAAcA,EAAY,EAEvC,KAAO,CACN,MAAMC,EAAYxU,EAAUjsC,WAAYmsC,EAAY,GAGpD,GAAKpxB,GAAQ0lC,IAAetL,GAAgBsL,IAAeA,GAAa/mD,KAAK2iD,cAAeoE,GAC3F,OAAO/mD,KAAK4yC,kBAAmBmU,EAAU9/C,WAAa+B,GAAS+9C,IAGhE,MAAMe,EAAazmC,GAAQ0lC,GAC1B/mD,KAAK6nD,0BAA2Bd,GAChC/mD,KAAKmiD,aAAc4E,GAGpB,GAAKe,GAAcA,EAAWlpD,OAC7B,OAAO,IAAI,GAAckpD,EAAWlpD,OAAQkpD,EAAWljD,MAAS,EAElE,CAEA,OAAO,IAET,CAcO,YAAAu9C,CAAc4F,GAGpB,OAFoB/nD,KAAKgoD,mBAAoBD,IAEvB/nD,KAAKujD,kBAAkBn7C,IAAK2/C,EACnD,CAsBO,yBAAAF,CAA2BnM,GACjC,GAAKD,GAAgBC,GACpB,OAAO,KAIR,MAAMphB,EAAct6B,KAAKgoD,mBAAoBtM,GAE7C,GAAKphB,EACJ,OAAOA,EAGR,MAAM1T,EAAkB80B,EAAQ90B,gBAGhC,GAAKA,EAAkB,CACtB,IAAQ5mB,KAAKioD,UAAWrhC,GAEvB,OAAO,KAGR,MAAM2wB,EAAcv3C,KAAKmiD,aAAcv7B,GAEvC,GAAK2wB,EAAc,CAClB,MAAMtb,EAAgBsb,EAA6Btb,YAGnD,OAAKA,aAAuB,GACpBA,EAEA,IAET,CACD,KAEK,CACJ,MAAMsb,EAAcv3C,KAAKmiD,aAAczG,EAAQz0C,YAE/C,GAAKswC,EAAc,CAClB,MAAM5wC,EAAe4wC,EAA6B1b,SAAU,GAG5D,OAAKl1B,aAAsB,GACnBA,EAEA,IAET,CACD,CAEA,OAAO,IACR,CAsCO,YAAA21C,CAAc4L,GACpB,OAAOloD,KAAKwjD,kBAAkBp7C,IAAK8/C,EACpC,CAiBO,wBAAArJ,CAA0BD,GAChC,MAAMh4B,EAAkBg4B,EAASh4B,gBAGjC,OAAKA,GAAmB5mB,KAAKs8C,aAAc11B,GACnC5mB,KAAKs8C,aAAc11B,GAAkCqV,aAIvDrV,GAAmBg4B,EAAShgD,QAAUoB,KAAKs8C,aAAcsC,EAAShgD,QAChEoB,KAAKs8C,aAAcsC,EAAShgD,QAAU0H,WAAY,GAGnD,IACR,CAKO,KAAA2mC,CAAOkb,GACb,MAAMC,EAAcpoD,KAAKs8C,aAAc6L,GAEvC,GAAKC,GAAeA,EAAYhpC,cAAc8iC,gBAAkBkG,EAAc,CAE7E,MAAM,QAAEv/B,EAAO,QAAEC,GAAY,GAAOvlB,OAC9B8kD,EAA6C,GAInDC,GAA2BF,GAAajpD,IACvC,MAAM,WAAEwqB,EAAU,UAAEC,GAAgBzqB,EAEpCkpD,EAAgBzoD,KAAM,CAAE+pB,EAAYC,GAAa,IAGlDw+B,EAAYnb,QAMZqb,GAA2BF,GAAajpD,IACvC,MAAQwqB,EAAYC,GAAcy+B,EAAgBl8B,QAElDhtB,EAAKwqB,WAAaA,EAClBxqB,EAAKyqB,UAAYA,CAAS,IAK3B,GAAOrmB,OAAOqnB,SAAU/B,EAASC,EAClC,CACD,CAOO,kBAAAy/B,GACN,MAAMH,EAAcpoD,KAAKs8C,aAAct8C,KAAKwD,SAASyxB,UAAUkT,iBAE/D,IAAMigB,EACL,OAID,MAAMnW,EAAemW,EAAYhpC,cAAcF,YAAaizB,eACtDqW,EAAmBxoD,KAAK+hD,mBAAoB9P,GACtBuW,GAAoBA,EAAiBtb,WAAa,GAG7E+E,EAAamP,iBAEf,CAOO,SAAA6G,CAAW9oD,GACjB,OAAOA,GAAQA,EAAK4a,UAAYsF,KAAKoiC,YACtC,CAOO,kBAAAgH,CAAoBtpD,GAC1B,OAAOA,GAAQA,EAAK4a,UAAYsF,KAAKqpC,sBACtC,CAmBO,aAAA/F,CAAenH,GACrB,MAA6B,MAAxBx7C,KAAKqjD,gBACF7H,EAAQmN,YAAa5F,MAKU,OAApCvH,EAAwBmG,UAC1BiH,GAAgBpN,EAASx7C,KAAK6jD,gBAC8B,IAA1DrI,EAAwBv0C,WAAYX,WAAWhO,UAM3CkjD,EAAQmN,YAAa1F,KA8mB9B,SAA4BzH,EAAkBqI,GAC7C,MAAMgF,EAASrN,EAAQmN,YAAa3F,IAEpC,OAAO6F,GAAUD,GAAgBpN,EAASqI,IAA+E,IAA1DrI,EAAwBv0C,WAAYX,WAAWhO,MAC/G,CAlnB0DwwD,CAAmBtN,EAASx7C,KAAK6jD,eAC1F,CAOO,sBAAA2D,CAAwBvyB,GAC9B,GAAKA,EAAU+V,YACd,OAAO,EAKR,MAAMlmB,EAAQ9kB,KAAKgkD,aAAanM,cAEhC,IACC/yB,EAAM8hC,SAAU3xB,EAAU+sB,WAAa/sB,EAAU8zB,cACjDjkC,EAAM+hC,OAAQ5xB,EAAUud,UAAYvd,EAAUyd,YAC/C,CAAE,MAAQxuC,GAGT,OAAO,CACR,CAEA,MAAMsqC,EAAW1pB,EAAMwtB,UAIvB,OAFAxtB,EAAMhE,SAEC0tB,CACR,CAMO,kBAAAwZ,CAAoBxM,GAC1B,MAAMpf,EEhoCO,SAAuBj9B,GACrC,MAAM+nC,EAAqB,GAC3B,IAAIoe,EAA2BnmD,EAG/B,KAAQmmD,GAAeA,EAAYvrC,UAAYsF,KAAK2pC,eACnD9hB,EAAM9nC,QAASkmD,GACfA,EAAcA,EAAYr+C,WAG3B,OAAOigC,CACR,CFqnCoB/K,CAAcqf,GAKhC,IAFApf,EAAUp9B,MAEFo9B,EAAU9jC,QAAS,CAC1B,MAAMkjD,EAAUpf,EAAUp9B,MACpBuhD,EAAWvgD,KAAKujD,kBAAkBn7C,IAAKozC,GAE7C,GAAK+E,IAAcA,EAASryC,GAAI,cAAiBqyC,EAASryC,GAAI,eAC7D,OAAOqyC,CAET,CAEA,OAAO,IACR,CAcO,qBAAAsB,CAAuB5P,GAC7B,OAAOjyC,KAAKipD,+BAAgChX,EAAa+P,WAAa/P,EAAa8W,eAClF/oD,KAAKipD,+BAAgChX,EAAaO,UAAYP,EAAaS,YAC7E,CAeO,yBAAAwW,CAA2BrrB,GACjC79B,KAAK0jD,0BAA0B71C,IAAKgwB,EACrC,CAYO,2BAAAsrB,CAA6BtrB,GACnC79B,KAAK2jD,4BAA4B91C,IAAKgwB,EACvC,CAKQ,eAAAyoB,GACP,OAAStmD,KAAKqjD,iBACb,IAAK,OACJ,OAAOzI,GAAa56C,KAAKgkD,cAC1B,IAAK,aACJ,OAAOnJ,GAAoB76C,KAAKgkD,cACjC,IAAK,KACJ,OAAO9I,GAAWl7C,KAAKgkD,cAE1B,CASQ,8BAAAiF,CAAgC1W,EAAoB32B,GAE3D,GAAKyF,GAAQkxB,IAAegJ,GAAkBhJ,IAAe32B,EAASw/B,GAErE,OAAO,EAGR,GAAKp7C,KAAKioD,UAAW1V,IAAegJ,GAAkBhJ,EAAUjsC,WAAYsV,IAE3E,OAAO,EAGR,MAAMkrC,EAAa9mD,KAAKmiD,aAAc5P,GAKtC,OAAKuU,IAAgBA,EAAW54C,GAAI,eAAiB44C,EAAW54C,GAAI,aAKrE,CAUA,WAASi5C,CACR3L,EACAj3C,EAMA0iD,GAEA,GAAKjnD,KAAK2iD,cAAenH,GACxB,OAAO,KAIR,MAAMlhB,EAAct6B,KAAKgoD,mBAAoBxM,GAE7C,GAAKlhB,EACJ,OAAOA,EAGR,GAAKtT,GAAWw0B,IAAaj3C,EAAQ6kD,aACpC,OAAO,KAGR,GAAK/nC,GAAQm6B,GAAY,CACxB,GAAKC,GAAgBD,GACpB,OAAO,KACD,CACN,MAAMsK,EAAWtK,EAAQ14C,KAEzB,GAAkB,KAAbgjD,EACJ,OAAO,KAGR,MAAMroB,EAAW,IAAI,GAAUz9B,KAAKwD,SAAUsiD,GAI9C,OAFAmB,EAAYrnD,KAAM69B,GAEXA,CACR,CACD,CAAO,CACN,IAAI8Z,EAAcv3C,KAAKmiD,aAAc3G,GAErC,GAAKjE,EAKJ,OAJKv3C,KAAKqpD,uBAAwB9R,IACjC0P,EAAYrnD,KAAM23C,GAGZA,EAGR,GAAKv3C,KAAKyoD,mBAAoBjN,GAE7BjE,EAAc,IAAI,GAAsBv3C,KAAKwD,UAExCe,EAAQyC,MACZhH,KAAKqkD,sBAAuB7I,EAASjE,OAEhC,CAENA,EAAcv3C,KAAKspD,mBAAoB9N,EAASj3C,GAE3CA,EAAQyC,MACZhH,KAAKq+C,aAAc7C,EAAuBjE,GAI3C,MAAMnS,EAAUoW,EAAwBj2C,WAExC,GAAK6/B,EACJ,QAAUlsC,EAAIksC,EAAM9sC,OAAQkC,EAAI,EAAGA,EAAItB,EAAGsB,IACzC+8C,EAAYnQ,cAAehC,EAAO5qC,GAAIwH,KAAMojC,EAAO5qC,GAAIhC,OAKzD,GAAKwH,KAAKupD,6BAA8BhS,EAAahzC,GAOpD,OANAgzC,EAAY5P,mBAAoB,cAAiB6T,EAAwB/0B,WAEnEzmB,KAAKunD,oBAAqBhQ,IAC/B0P,EAAYrnD,KAAM23C,GAGZA,EAIR,GAAKvwB,GAAWw0B,GAGf,OAFAjE,EAAY5P,mBAAoB,cAAe6T,EAAQ14C,MAEhDy0C,CAET,OAGMA,EAEN,MAAMiS,EAAqC,GAE3C,IAA8B,IAAzBjlD,EAAQi5C,aACZ,UAAY7lC,KAAS3X,KAAKqnD,kBAAmB7L,EAAuBj3C,EAASilD,GAC5EjS,EAAYtQ,aAActvB,GAM5B,GAAK3X,KAAKqpD,uBAAwB9R,GACjC0P,EAAYrnD,KAAM23C,QAGlB,UAAYkS,KAAcD,EACzBvC,EAAYrnD,KAAM6pD,EAGrB,CACD,CAUQ,sBAAArC,CACP7U,EACA0U,EACA1iD,GAEA,IAAM0iD,EAAY3uD,OACjB,OAKD,GAAKi6C,IAAcvyC,KAAKyoD,mBAAoBlW,KAAgBvyC,KAAK0pD,mBAAoBnX,GACpF,OAGD,IAAIoX,GAAwB,EAE5B,QAAUnvD,EAAI,EAAGA,EAAIysD,EAAY3uD,OAAQkC,IAAM,CAC9C,MAAM2E,EAAO8nD,EAAazsD,GAE1B,IAAM2E,EAAK+O,GAAI,SAAY,CAC1By7C,GAAwB,EACxB,QACD,CAEA,IAAI7mD,EACA8mD,GAA6B,EAEjC,GAAKC,GAAsB1qD,EAAMa,KAAK4jD,aACrC9gD,EAAO64C,GAAsBx8C,EAAK2D,UAC5B,CAKNA,EAAO3D,EAAK2D,KAAKmW,QAAS,iBAAkB,KAC5C2wC,EAAoB,cAAc3nD,KAAMa,EAAKuqB,OAAQvqB,EAAKxK,OAAS,IAEnE,MAAMwxD,EAAWtvD,EAAI,EAAIysD,EAAazsD,EAAI,GAAM,KAC1C+qD,EAAW/qD,EAAI,EAAIysD,EAAY3uD,OAAS2uD,EAAazsD,EAAI,GAAM,KAE/DuvD,GAAkBD,GAAYA,EAAS57C,GAAI,YAAgC,MAAjB47C,EAAS9nD,MAAgB2nD,EACnFK,GAAkBzE,IAAoBhK,GAAkBp8C,EAAK2D,OAGrC,IAAzByB,EAAQi5C,eAGPuM,IACJjnD,EAAOA,EAAKmW,QAAS,KAAM,KAIvB+wC,IACJlnD,EAAOA,EAAKmW,QAAS,KAAM,MAQ7BnW,EAAO64C,GAAsB74C,GAS7BA,EAAOA,EAAKmW,QAAS,WAAY,MAEjC,MAAMgxC,EAAgC1E,GAAYA,EAASr3C,GAAI,YAAgC,MAAjBq3C,EAASvjD,KACjFkoD,EAA8B3E,GAAYA,EAASr3C,GAAI,UAA0C,KAA7Bq3C,EAASziD,KAAKuqB,OAAQ,IAG3F,mBAAmBprB,KAAMa,KAAWyiD,GAAY0E,GAAiCC,KACrFpnD,EAAOA,EAAKmW,QAAS,UAAW,OAK5B8wC,GAAkBD,GAAYA,EAAS57C,GAAI,YAAgC,MAAjB47C,EAAS9nD,QACvEc,EAAOA,EAAKmW,QAAS,UAAW,KAElC,CAKoB,GAAfnW,EAAKxK,QAAe6G,EAAKP,QAC7BO,EAAKkyB,UACL41B,EAAYz/C,OAAQhN,EAAG,GACvBA,MAEA2E,EAAKi+B,MAAQt6B,EACb6mD,EAAwBC,EAE1B,CAEA3C,EAAY3uD,OAAS,CACtB,CAmBQ,wBAAAytD,CAA0B5mD,GACjC,IAAI2D,EAAO3D,EAAK2D,KAIhB,GAAK3D,EAAKg9B,eAAe9H,MAAMz1B,GAAUoB,KAAK4jD,YAAYr0B,SAAY3wB,EAAwBoD,QAC7F,OAAOc,EAKR,GAAyB,KAApBA,EAAKuqB,OAAQ,GAAa,CAC9B,MAAMy8B,EAAW9pD,KAAKmqD,2BAA4BhrD,GAAkB,KAC1C2qD,GAAYA,EAAS57C,GAAI,eAAkBlO,KAAKoqD,mBAAoBN,KAEnEA,IAC1BhnD,EAAO,IAAWA,EAAKmP,OAAQ,GAEjC,CAWA,GAAuC,KAAlCnP,EAAKuqB,OAAQvqB,EAAKxK,OAAS,GAAa,CAC5C,MAAMitD,EAAWvlD,KAAKmqD,2BAA4BhrD,GAAkB,GAC9DkrD,EAAsB9E,GAAYA,EAASr3C,GAAI,eAA+C,KAA7Bq3C,EAASziD,KAAKuqB,OAAQ,GAEtD,KAAlCvqB,EAAKuqB,OAAQvqB,EAAKxK,OAAS,IAAeitD,IAAY8E,IAC1DvnD,EAAOA,EAAKmP,OAAQ,EAAGnP,EAAKxK,OAAS,GAAM,IAE7C,CAGA,OAAOwK,EAAKmW,QAAS,QAAS,KAC/B,CAQQ,kBAAAmxC,CAAoBjrD,GAC3B,GAAKA,EAAKg9B,eAAe9H,MAAMz1B,GAAUoB,KAAK4jD,YAAYr0B,SAAY3wB,EAAwBoD,QAC7F,OAAO,EAGR,MAAMc,EAAO9C,KAAK+lD,yBAA0B5mD,GAE5C,MAAyC,KAAlC2D,EAAKuqB,OAAQvqB,EAAKxK,OAAS,EACnC,CAUQ,0BAAA6xD,CAA4BhrD,EAAgBmrD,GACnD,MAAM7f,EAAa,IAAI,GAAgB,CACtChC,cAAe6hB,EAAU,GAAa7gB,aAActqC,GAAS,GAAa4qC,cAAe5qC,GACzFolB,UAAW+lC,EAAU,UAAY,aAGlC,UAAY9xD,KAASiyC,EAAa,CAEjC,GAAKjyC,EAAMyH,KAAKiO,GAAI,UAAW,MAC9B,OAAO,KACR,GAEUlO,KAAKqpD,uBAAwB7wD,EAAMyH,MAC5C,OAAOzH,EAAMyH,KACd,GAGUzH,EAAMyH,KAAKiO,GAAI,oBACxB,OAAO,KACR,GAEU1V,EAAMyH,KAAKiO,GAAI,cACxB,OAAO1V,EAAMyH,IAEf,CAEA,OAAO,IACR,CAKQ,kBAAAypD,CAAoBvqD,GAC3B,OAAOa,KAAKioD,UAAW9oD,IAAUa,KAAK6jD,cAAct0B,SAAUpwB,EAAKwiD,QAAQ/4C,cAC5E,CAKQ,mBAAA2+C,CAAqBpoD,GAC5B,OAAOA,EAAK+O,GAAI,YAAelO,KAAK6jD,cAAct0B,SAAUpwB,EAAK6C,KAClE,CAKQ,sBAAAqnD,CAAwBlqD,GAC/B,QAAMA,EAAK+O,GAAI,aAIK,MAAb/O,EAAK6C,MACXhC,KAAK8jD,qBAAqBv0B,SAAUpwB,EAAK6C,SACvChC,KAAK2jD,4BAA4B9mD,MAAOsC,GAC5C,CAQQ,kBAAAmqD,CAAoBnqD,EAAeoF,GAC1C,GAAKyiB,GAAW7nB,GACf,OAAO,IAAI,GAAea,KAAKwD,SAAU,YAG1C,MAAM+mD,EAAWhmD,EAAQimD,iBAAqBrrD,EAAqBwiD,QAAYxiD,EAAqBwiD,QAAQ/4C,cAE5G,OAAO,IAAI,GAAa5I,KAAKwD,SAAU+mD,EACxC,CAQQ,4BAAAhB,CAA8BhS,EAAiDhzC,GACtF,OAAgC,IAAzBA,EAAQi5C,cAA0BjG,EAAYrpC,GAAI,cAAiBlO,KAAK0jD,0BAA0B7mD,MAAO06C,EACjH,CAOQ,oBAAAkO,CAAsBd,GAC7B,MAAM3iD,EAAO2iD,EAAY/7C,cAEzB,MAA8B,YAAvB5I,KAAKsjD,eAA+BtjD,KAAK+jD,eAAex0B,SAAUvtB,EAC1E,CASQ,4BAAA4jD,CAA8BjB,EAAqB8F,GAC1D,MAAMC,EAAgB1qD,KAAKgkD,aAAa1+C,cAAe,QAKvD,GAFAolD,EAAcjlD,aAAc09C,GAAsCwB,GAE7D8F,EAAqB,CACzB,KAAQA,EAAmB9jD,YAC1B+jD,EAAc/kD,YAAa8kD,EAAmB9jD,YAG/C,UAAYigC,KAAiB6jB,EAAmBjF,oBAC/CkF,EAAcjlD,aAAcmhC,EAAe6jB,EAAmBpvB,aAAcuL,GAE9E,CAEA,OAAO8jB,CACR,EASD,SAASb,GAAsB1qD,EAAgBob,GAC9C,OAAOpb,EAAKg9B,eAAe9H,MAAMz1B,GAAUA,EAAOsP,GAAI,YAAeqM,EAAMgV,SAAU3wB,EAAOoD,OAC7F,CAQA,SAASsmD,GAA2B7wC,EAAqBlI,GACxD,IAAIpQ,EAA0BsY,EAE9B,KAAQtY,GACPoQ,EAAUpQ,GACVA,EAAOA,EAAK2nB,aAEd,CAoBA,SAAS8hC,GAAgBpN,EAAkBqI,GAC1C,MAAMjlD,EAAS48C,EAAQv0C,WAEvB,QAASrI,KAAcA,EAAuB+iD,SAAWkC,EAAct0B,SAAY3wB,EAAuB+iD,QAAQ/4C,cACnH,CAQA,SAAS88C,GAAmBf,GACN,WAAhBA,GACJ,EAAY,+CAGQ,UAAhBA,GACJ,EAAY,6CAEd,CGvuDe,MAAegG,WAAiB,MAmB9C,WAAA5oD,CAAak4B,GACZzsB,QAND,KAAQo9C,YAAsB,EAQ7B5qD,KAAKi6B,KAAOA,EACZj6B,KAAKwD,SAAWy2B,EAAKz2B,QACtB,CAKA,aAAWkxB,GACV,OAAO10B,KAAK4qD,UACb,CAYO,MAAAC,GACN7qD,KAAK4qD,YAAa,CACnB,CAQO,OAAAE,GACN9qD,KAAK4qD,YAAa,CACnB,CAKO,OAAA5kC,GACNhmB,KAAK8qD,UACL9qD,KAAK2P,eACN,CAcO,gCAAAo7C,CAAkC7Y,GAKxC,OAJKA,GAAoC,IAAvBA,EAAUn4B,WAC3Bm4B,EAAYA,EAAUjrC,eAGjBirC,GAAoC,IAAvBA,EAAUn4B,WAIpBm4B,EAAmB/d,QAAS,uDACtC,ECpED,SAJe,IAAe,SAAS/e,EAAQlS,GAC7C,GAAWA,EAAQ,GAAOA,GAASkS,EACrC,ICfe,MAAM41C,GA0BpB,WAAAjpD,CAAak4B,EAAYgxB,EAAkBC,GAC1ClrD,KAAKi6B,KAAOA,EACZj6B,KAAKwD,SAAWy2B,EAAKz2B,SACrBxD,KAAKirD,SAAWA,EAChBjrD,KAAKkyC,UAAY+Y,EAASrnD,OAE1B,GAAQ5D,KAAMkrD,EACf,CAKA,UAAWtnD,GACV,OAAO5D,KAAKi6B,KAAK4X,aAAasQ,aAAcniD,KAAKkyC,UAClD,CAKO,cAAAhf,GACNlzB,KAAKirD,SAAS/3B,gBACf,CAKO,eAAAC,GACNnzB,KAAKirD,SAAS93B,iBACf,EClCc,MAAeg4B,WAGpBR,GAHV,kCAcC,KAAOjrC,YAAsB,EAab,OAAAqG,CAASgsB,IACkB,iBAArB/xC,KAAKorD,aAA2B,CAAEprD,KAAKorD,cAAiBprD,KAAKorD,cAE5ExtD,SAASqN,IACdjL,KAAKwP,SAAUuiC,EAAY9mC,GAAM,CAAEyF,EAAWu6C,KACxCjrD,KAAK00B,YAAc10B,KAAK+qD,iCAAkCE,EAASrnD,SACvE5D,KAAKqrD,WAAYJ,EAClB,GACE,CAAEvrC,WAAY1f,KAAK0f,YAAc,GAEtC,CAKgB,aAAA4rC,CAAevZ,GAC9B/xC,KAAK2P,cAAeoiC,EACrB,CAWgB,IAAAvhC,CAAM+6C,EAA+BN,EAAiBC,GAChElrD,KAAK00B,WACT10B,KAAKwD,SAASgN,KAAM+6C,EAAW,IAAIP,GAAchrD,KAAKi6B,KAAMgxB,EAAUC,GAExE,ECnFc,MAAMM,WAAoBL,GAAzC,kCAIC,KAAgBC,aAAe,CAAE,UAAW,QAAQ,CAK7C,UAAAC,CAAYrqC,GAClB,MAAMle,EAAO,CACZwqB,QAAStM,EAAOsM,QAEhBC,OAAQvM,EAAOuM,OACfC,QAASxM,EAAOwM,QAChBC,SAAUzM,EAAOyM,SACjBC,QAAS1M,EAAO0M,QAEhB,aAAIE,GACH,OAAO,GAAS5tB,KACjB,GAGDA,KAAKwQ,KAAMwQ,EAAO/V,KAAM+V,EAAQle,EACjC,ECpBD,SAJU,WACR,OAAO,GAAK+L,KAAK2yB,KACnB,ECnBA,IAAIiqB,GAAe,KAiBnB,SAPA,SAAyBhvD,GAGvB,IAFA,IAAImI,EAAQnI,EAAOnE,OAEZsM,KAAW6mD,GAAaxpD,KAAKxF,EAAO4wB,OAAOzoB,MAClD,OAAOA,CACT,ECbA,IAAI8mD,GAAc,OAelB,SANA,SAAkBjvD,GAChB,OAAOA,EACHA,EAAOqF,MAAM,EAAG,GAAgBrF,GAAU,GAAGwc,QAAQyyC,GAAa,IAClEjvD,CACN,ECXA,IAGIkvD,GAAa,qBAGbC,GAAa,aAGbC,GAAY,cAGZC,GAAe1uD,SA8CnB,SArBA,SAAkB5E,GAChB,GAAoB,iBAATA,EACT,OAAOA,EAET,GAAI,GAASA,GACX,OA1CM,IA4CR,GAAI,EAASA,GAAQ,CACnB,IAAI8f,EAAgC,mBAAjB9f,EAAM2kB,QAAwB3kB,EAAM2kB,UAAY3kB,EACnEA,EAAQ,EAAS8f,GAAUA,EAAQ,GAAMA,CAC3C,CACA,GAAoB,iBAAT9f,EACT,OAAiB,IAAVA,EAAcA,GAASA,EAEhCA,EAAQ,GAASA,GACjB,IAAIuzD,EAAWH,GAAW3pD,KAAKzJ,GAC/B,OAAQuzD,GAAYF,GAAU5pD,KAAKzJ,GAC/BszD,GAAatzD,EAAMsJ,MAAM,GAAIiqD,EAAW,EAAI,GAC3CJ,GAAW1pD,KAAKzJ,GAvDb,KAuD6BA,CACvC,ECxDA,IAGI,GAAYK,KAAKC,IACjBkzD,GAAYnzD,KAAKD,IAqLrB,SA7HA,SAAkBgf,EAAM0b,EAAM/uB,GAC5B,IAAI0nD,EACAC,EACAC,EACAhuD,EACAiuD,EACAC,EACAC,EAAiB,EACjBC,GAAU,EACVC,GAAS,EACTC,GAAW,EAEf,GAAmB,mBAAR70C,EACT,MAAM,IAAIzV,UAzEQ,uBAmFpB,SAASuqD,EAAWC,GAClB,IAAIzwD,EAAO+vD,EACP7qB,EAAU8qB,EAKd,OAHAD,EAAWC,OAAWrkD,EACtBykD,EAAiBK,EACjBxuD,EAASyZ,EAAK/G,MAAMuwB,EAASllC,EAE/B,CAqBA,SAAS0wD,EAAaD,GACpB,IAAIE,EAAoBF,EAAON,EAM/B,YAAyBxkD,IAAjBwkD,GAA+BQ,GAAqBv5B,GACzDu5B,EAAoB,GAAOL,GANJG,EAAOL,GAM8BH,CACjE,CAEA,SAASW,IACP,IAAIH,EAAO,KACX,GAAIC,EAAaD,GACf,OAAOI,EAAaJ,GAGtBP,EAAUr5B,WAAW+5B,EA3BvB,SAAuBH,GACrB,IAEIK,EAAc15B,GAFMq5B,EAAON,GAI/B,OAAOG,EACHR,GAAUgB,EAAab,GAJDQ,EAAOL,IAK7BU,CACN,CAmBqCC,CAAcN,GACnD,CAEA,SAASI,EAAaJ,GAKpB,OAJAP,OAAUvkD,EAIN4kD,GAAYR,EACPS,EAAWC,IAEpBV,EAAWC,OAAWrkD,EACf1J,EACT,CAcA,SAAS+uD,IACP,IAAIP,EAAO,KACPQ,EAAaP,EAAaD,GAM9B,GAJAV,EAAWpyC,UACXqyC,EAAWlsD,KACXqsD,EAAeM,EAEXQ,EAAY,CACd,QAAgBtlD,IAAZukD,EACF,OAzEN,SAAqBO,GAMnB,OAJAL,EAAiBK,EAEjBP,EAAUr5B,WAAW+5B,EAAcx5B,GAE5Bi5B,EAAUG,EAAWC,GAAQxuD,CACtC,CAkEaivD,CAAYf,GAErB,GAAIG,EAIF,OAFA35B,aAAau5B,GACbA,EAAUr5B,WAAW+5B,EAAcx5B,GAC5Bo5B,EAAWL,EAEtB,CAIA,YAHgBxkD,IAAZukD,IACFA,EAAUr5B,WAAW+5B,EAAcx5B,IAE9Bn1B,CACT,CAGA,OA3GAm1B,EAAO,GAASA,IAAS,EACrB,EAAS/uB,KACXgoD,IAAYhoD,EAAQgoD,QAEpBJ,GADAK,EAAS,YAAajoD,GACH,GAAU,GAASA,EAAQ4nD,UAAY,EAAG74B,GAAQ64B,EACrEM,EAAW,aAAcloD,IAAYA,EAAQkoD,SAAWA,GAoG1DS,EAAUz5B,OApCV,gBACkB5rB,IAAZukD,GACFv5B,aAAau5B,GAEfE,EAAiB,EACjBL,EAAWI,EAAeH,EAAWE,OAAUvkD,CACjD,EA+BAqlD,EAAUG,MA7BV,WACE,YAAmBxlD,IAAZukD,EAAwBjuD,EAAS4uD,EAAa,KACvD,EA4BOG,CACT,EChKe,MAAMI,WAA8B3C,GASlD,WAAA5oD,CAAak4B,GACZzsB,MAAOysB,GAEPj6B,KAAKutD,kCAAoC,IAAUzqD,IAClD9C,KAAKwD,SAASgN,KAA4C,sBAAuB1N,EAAM,GACrF,IACJ,CAKgB,OAAAijB,GACf,MAAMviB,EAAWxD,KAAKwD,SAEtBA,EAAS6L,GAA+B,YAAY,CAAEqB,EAAW5N,KAC9CU,EAASyxB,UAEZ8X,QAAU/sC,KAAK00B,WAE7B5xB,EAAKowB,gBACN,GACE,CAAE3lB,QAAS,aAEd/J,EAAS6L,GAA+B,YAAY,CAAEqB,EAAW5N,KAC9CU,EAASyxB,UAEZ8X,QAAU/sC,KAAK00B,WAC7B10B,KAAKwtD,qBAAsB1qD,EAAKwqB,QACjC,GACE,CAAE1gB,SAAU,UAChB,CAKgB,aAAA0+C,GAAuB,CAKvB,OAAAtlC,GACfxY,MAAMwY,UAENhmB,KAAKutD,kCAAkC95B,QACxC,CAUQ,oBAAA+5B,CAAsBlgC,GAC7B,MAAM2H,EAAYj1B,KAAKwD,SAASyxB,UAC1Bw4B,EAAe,IAAI,GAAex4B,EAAUmY,YAAa,CAAEoB,SAAUvZ,EAAUkY,WAAYmB,MAAM,IAGlGhhB,GAAWjB,GAASC,WAAagB,GAAWjB,GAASE,SACzDkhC,EAAarrB,MAAOqrB,EAAat4B,oBAI7B7H,GAAWjB,GAASG,YAAcc,GAAWjB,GAASI,WAC1DghC,EAAarrB,MAAOqrB,EAAajgB,mBAGlC,MAAM1qC,EAAO,CACZ4qD,aAAcz4B,EACdw4B,eACAxb,aAAc,MAIfjyC,KAAKwD,SAASgN,KAAwC,kBAAmB1N,GAMzE9C,KAAKutD,kCAAmCzqD,EACzC,ECrGD,SALA,SAAqBtK,GAEnB,OADAwH,KAAKqY,SAAS5G,IAAIjZ,EAbC,6BAcZwH,IACT,ECHA,SAJA,SAAqBxH,GACnB,OAAOwH,KAAKqY,SAASzK,IAAIpV,EAC3B,ECCA,SAASm1D,GAASziD,GAChB,IAAItG,GAAS,EACTtM,EAAmB,MAAV4S,EAAiB,EAAIA,EAAO5S,OAGzC,IADA0H,KAAKqY,SAAW,IAAI,KACXzT,EAAQtM,GACf0H,KAAK6N,IAAI3C,EAAOtG,GAEpB,CAGA+oD,GAAS9rD,UAAUgM,IAAM8/C,GAAS9rD,UAAUjC,KAAO,GACnD+tD,GAAS9rD,UAAU+L,IAAM,GAEzB,YCJA,SAZA,SAAmB2K,EAAOgD,GAIxB,IAHA,IAAI3W,GAAS,EACTtM,EAAkB,MAATigB,EAAgB,EAAIA,EAAMjgB,SAE9BsM,EAAQtM,GACf,GAAIijB,EAAUhD,EAAM3T,GAAQA,EAAO2T,GACjC,OAAO,EAGX,OAAO,CACT,ECRA,SAJA,SAAkBynB,EAAOjpC,GACvB,OAAOipC,EAAMpyB,IAAI7W,EACnB,ECyEA,SA9DA,SAAqBwhB,EAAOD,EAAOuF,EAASnE,EAAYk0C,EAAWv/C,GACjE,IAAIuvB,EAjBqB,EAiBT/f,EACZgwC,EAAYt1C,EAAMjgB,OAClBw1D,EAAYx1C,EAAMhgB,OAEtB,GAAIu1D,GAAaC,KAAelwB,GAAakwB,EAAYD,GACvD,OAAO,EAGT,IAAIE,EAAa1/C,EAAMjG,IAAImQ,GACvBy1C,EAAa3/C,EAAMjG,IAAIkQ,GAC3B,GAAIy1C,GAAcC,EAChB,OAAOD,GAAcz1C,GAAS01C,GAAcz1C,EAE9C,IAAI3T,GAAS,EACTzG,GAAS,EACT8vD,EA/BuB,EA+BfpwC,EAAoC,IAAI,QAAWhW,EAM/D,IAJAwG,EAAMoD,IAAI8G,EAAOD,GACjBjK,EAAMoD,IAAI6G,EAAOC,KAGR3T,EAAQipD,GAAW,CAC1B,IAAIK,EAAW31C,EAAM3T,GACjBupD,EAAW71C,EAAM1T,GAErB,GAAI8U,EACF,IAAI00C,EAAWxwB,EACXlkB,EAAWy0C,EAAUD,EAAUtpD,EAAO0T,EAAOC,EAAOlK,GACpDqL,EAAWw0C,EAAUC,EAAUvpD,EAAO2T,EAAOD,EAAOjK,GAE1D,QAAiBxG,IAAbumD,EAAwB,CAC1B,GAAIA,EACF,SAEFjwD,GAAS,EACT,KACF,CAEA,GAAI8vD,GACF,IAAK,GAAU31C,GAAO,SAAS61C,EAAUE,GACnC,IAAK,GAASJ,EAAMI,KACfH,IAAaC,GAAYP,EAAUM,EAAUC,EAAUtwC,EAASnE,EAAYrL,IAC/E,OAAO4/C,EAAKruD,KAAKyuD,EAErB,IAAI,CACNlwD,GAAS,EACT,KACF,OACK,GACD+vD,IAAaC,IACXP,EAAUM,EAAUC,EAAUtwC,EAASnE,EAAYrL,GACpD,CACLlQ,GAAS,EACT,KACF,CACF,CAGA,OAFAkQ,EAAc,OAAEkK,GAChBlK,EAAc,OAAEiK,GACTna,CACT,EChEA,SAVA,SAAoBnB,GAClB,IAAI4H,GAAS,EACTzG,EAASwC,MAAM3D,EAAI6W,MAKvB,OAHA7W,EAAIY,SAAQ,SAASpF,EAAOzB,GAC1BoH,IAASyG,GAAS,CAAC7N,EAAKyB,EAC1B,IACO2F,CACT,ECEA,SAVA,SAAoBsT,GAClB,IAAI7M,GAAS,EACTzG,EAASwC,MAAM8Q,EAAIoC,MAKvB,OAHApC,EAAI7T,SAAQ,SAASpF,GACnB2F,IAASyG,GAASpM,CACpB,IACO2F,CACT,ECPA,IAkBI,GAAc,GAAS,GAAO0D,eAAYgG,EAC1C,GAAgB,GAAc,GAAYsV,aAAUtV,EAoFxD,SAjEA,SAAoBuN,EAAQkD,EAAOnB,EAAK0G,EAASnE,EAAYk0C,EAAWv/C,GACtE,OAAQ8I,GACN,IAzBc,oBA0BZ,GAAK/B,EAAOwH,YAActE,EAAMsE,YAC3BxH,EAAO0H,YAAcxE,EAAMwE,WAC9B,OAAO,EAET1H,EAASA,EAAOgG,OAChB9C,EAAQA,EAAM8C,OAEhB,IAlCiB,uBAmCf,QAAKhG,EAAOwH,YAActE,EAAMsE,aAC3BgxC,EAAU,IAAI,GAAWx4C,GAAS,IAAI,GAAWkD,KAKxD,IAnDU,mBAoDV,IAnDU,gBAoDV,IAjDY,kBAoDV,OAAO,IAAIlD,GAASkD,GAEtB,IAxDW,iBAyDT,OAAOlD,EAAOpT,MAAQsW,EAAMtW,MAAQoT,EAAOhH,SAAWkK,EAAMlK,QAE9D,IAxDY,kBAyDZ,IAvDY,kBA2DV,OAAOgH,GAAWkD,EAAQ,GAE5B,IAjES,eAkEP,IAAIphB,EAAU,GAEhB,IAjES,eAkEP,IAAI0mC,EA5EiB,EA4EL/f,EAGhB,GAFA3mB,IAAYA,EAAU,IAElBke,EAAOvB,MAAQyE,EAAMzE,OAAS+pB,EAChC,OAAO,EAGT,IAAI3f,EAAU5P,EAAMjG,IAAIgN,GACxB,GAAI6I,EACF,OAAOA,GAAW3F,EAEpBuF,GAtFuB,EAyFvBxP,EAAMoD,IAAI2D,EAAQkD,GAClB,IAAIna,EAAS,GAAYjH,EAAQke,GAASle,EAAQohB,GAAQuF,EAASnE,EAAYk0C,EAAWv/C,GAE1F,OADAA,EAAc,OAAE+G,GACTjX,EAET,IAnFY,kBAoFV,GAAI,GACF,OAAO,GAAckD,KAAK+T,IAAW,GAAc/T,KAAKiX,GAG9D,OAAO,CACT,EC1GA,IAMI,GAHcthB,OAAO6K,UAGQ2G,eAgFjC,SAjEA,SAAsB4M,EAAQkD,EAAOuF,EAASnE,EAAYk0C,EAAWv/C,GACnE,IAAIuvB,EAtBqB,EAsBT/f,EACZywC,EAAW,GAAWl5C,GACtBm5C,EAAYD,EAASh2D,OAIzB,GAAIi2D,GAHW,GAAWj2C,GACDhgB,SAEMslC,EAC7B,OAAO,EAGT,IADA,IAAIh5B,EAAQ2pD,EACL3pD,KAAS,CACd,IAAI7N,EAAMu3D,EAAS1pD,GACnB,KAAMg5B,EAAY7mC,KAAOuhB,EAAQ,GAAejX,KAAKiX,EAAOvhB,IAC1D,OAAO,CAEX,CAEA,IAAIy3D,EAAangD,EAAMjG,IAAIgN,GACvB44C,EAAa3/C,EAAMjG,IAAIkQ,GAC3B,GAAIk2C,GAAcR,EAChB,OAAOQ,GAAcl2C,GAAS01C,GAAc54C,EAE9C,IAAIjX,GAAS,EACbkQ,EAAMoD,IAAI2D,EAAQkD,GAClBjK,EAAMoD,IAAI6G,EAAOlD,GAGjB,IADA,IAAIq5C,EAAW7wB,IACNh5B,EAAQ2pD,GAAW,CAE1B,IAAI/0C,EAAWpE,EADfre,EAAMu3D,EAAS1pD,IAEXupD,EAAW71C,EAAMvhB,GAErB,GAAI2iB,EACF,IAAI00C,EAAWxwB,EACXlkB,EAAWy0C,EAAU30C,EAAUziB,EAAKuhB,EAAOlD,EAAQ/G,GACnDqL,EAAWF,EAAU20C,EAAUp3D,EAAKqe,EAAQkD,EAAOjK,GAGzD,UAAmBxG,IAAbumD,EACG50C,IAAa20C,GAAYP,EAAUp0C,EAAU20C,EAAUtwC,EAASnE,EAAYrL,GAC7E+/C,GACD,CACLjwD,GAAS,EACT,KACF,CACAswD,IAAaA,EAAkB,eAAP13D,EAC1B,CACA,GAAIoH,IAAWswD,EAAU,CACvB,IAAIC,EAAUt5C,EAAOrT,YACjB4sD,EAAUr2C,EAAMvW,YAGhB2sD,GAAWC,KACV,gBAAiBv5C,MAAU,gBAAiBkD,IACzB,mBAAXo2C,GAAyBA,aAAmBA,GACjC,mBAAXC,GAAyBA,aAAmBA,IACvDxwD,GAAS,EAEb,CAGA,OAFAkQ,EAAc,OAAE+G,GAChB/G,EAAc,OAAEiK,GACTna,CACT,EC7EA,IAGI,GAAU,qBACV,GAAW,iBACX,GAAY,kBAMZ,GAHcnH,OAAO6K,UAGQ2G,eA6DjC,SA7CA,SAAyB4M,EAAQkD,EAAOuF,EAASnE,EAAYk0C,EAAWv/C,GACtE,IAAIugD,EAAW,GAAQx5C,GACnBy5C,EAAW,GAAQv2C,GACnBw2C,EAASF,EAAW,GAAW,GAAOx5C,GACtC25C,EAASF,EAAW,GAAW,GAAOv2C,GAKtC02C,GAHJF,EAASA,GAAU,GAAU,GAAYA,IAGhB,GACrBG,GAHJF,EAASA,GAAU,GAAU,GAAYA,IAGhB,GACrBG,EAAYJ,GAAUC,EAE1B,GAAIG,GAAa,GAAS95C,GAAS,CACjC,IAAK,GAASkD,GACZ,OAAO,EAETs2C,GAAW,EACXI,GAAW,CACb,CACA,GAAIE,IAAcF,EAEhB,OADA3gD,IAAUA,EAAQ,IAAI,IACdugD,GAAY,GAAax5C,GAC7B,GAAYA,EAAQkD,EAAOuF,EAASnE,EAAYk0C,EAAWv/C,GAC3D,GAAW+G,EAAQkD,EAAOw2C,EAAQjxC,EAASnE,EAAYk0C,EAAWv/C,GAExE,KArDyB,EAqDnBwP,GAAiC,CACrC,IAAIsxC,EAAeH,GAAY,GAAe3tD,KAAK+T,EAAQ,eACvDg6C,EAAeH,GAAY,GAAe5tD,KAAKiX,EAAO,eAE1D,GAAI62C,GAAgBC,EAAc,CAChC,IAAIC,EAAeF,EAAe/5C,EAAO5c,QAAU4c,EAC/Ck6C,EAAeF,EAAe92C,EAAM9f,QAAU8f,EAGlD,OADAjK,IAAUA,EAAQ,IAAI,IACfu/C,EAAUyB,EAAcC,EAAczxC,EAASnE,EAAYrL,EACpE,CACF,CACA,QAAK6gD,IAGL7gD,IAAUA,EAAQ,IAAI,IACf,GAAa+G,EAAQkD,EAAOuF,EAASnE,EAAYk0C,EAAWv/C,GACrE,ECrDA,SAVA,SAASkhD,EAAY/2D,EAAO8f,EAAOuF,EAASnE,EAAYrL,GACtD,OAAI7V,IAAU8f,IAGD,MAAT9f,GAA0B,MAAT8f,IAAmB,GAAa9f,KAAW,GAAa8f,GACpE9f,GAAUA,GAAS8f,GAAUA,EAE/B,GAAgB9f,EAAO8f,EAAOuF,EAASnE,EAAY61C,EAAalhD,GACzE,ECeA,SANA,SAAqB7V,EAAO8f,EAAOoB,GAEjC,IAAIvb,GADJub,EAAkC,mBAAdA,EAA2BA,OAAa7R,GAClC6R,EAAWlhB,EAAO8f,QAASzQ,EACrD,YAAkBA,IAAX1J,EAAuB,GAAY3F,EAAO8f,OAAOzQ,EAAW6R,KAAgBvb,CACrF,ECLe,MAAMqxD,WAAyB7E,GA6B7C,WAAA5oD,CAAak4B,GACZzsB,MAAOysB,GAEPj6B,KAAKse,QAAU,CACdwhC,WAAW,EACX2P,eAAe,EACfC,SAAS,GAGV1vD,KAAK6xC,aAAe5X,EAAK4X,aACzB7xC,KAAK2vD,SAAW11B,EAAK21B,UAErB5vD,KAAK6vD,aAAe,IAAIj8C,IACxB5T,KAAK8vD,kBAAoB,IAAIvsD,OAAOisD,iBAAkBxvD,KAAK+vD,aAAa/oD,KAAMhH,MAC/E,CAKO,KAAAqtD,GACNrtD,KAAK+vD,aAAc/vD,KAAK8vD,kBAAkBE,cAC3C,CAKO,OAAAjqC,CAASgsB,GACf/xC,KAAK6vD,aAAahiD,IAAKkkC,GAElB/xC,KAAK00B,WACT10B,KAAK8vD,kBAAkB/pC,QAASgsB,EAAY/xC,KAAKse,QAEnD,CAKgB,aAAAgtC,CAAevZ,GAG9B,GAFA/xC,KAAK6vD,aAAal+C,OAAQogC,GAErB/xC,KAAK00B,UAAY,CAGrB10B,KAAK8vD,kBAAkBG,aAEvB,UAAYle,KAAc/xC,KAAK6vD,aAC9B7vD,KAAK8vD,kBAAkB/pC,QAASgsB,EAAY/xC,KAAKse,QAEnD,CACD,CAKgB,MAAAusC,GACfr9C,MAAMq9C,SAEN,UAAY9Y,KAAc/xC,KAAK6vD,aAC9B7vD,KAAK8vD,kBAAkB/pC,QAASgsB,EAAY/xC,KAAKse,QAEnD,CAKgB,OAAAwsC,GACft9C,MAAMs9C,UAEN9qD,KAAK8vD,kBAAkBG,YACxB,CAKgB,OAAAjqC,GACfxY,MAAMwY,UAENhmB,KAAK8vD,kBAAkBG,YACxB,CAOQ,YAAAF,CAAcG,GAErB,GAA6B,IAAxBA,EAAa53D,OACjB,OAGD,MAAMu5C,EAAe7xC,KAAK6xC,aAGpBse,EAAmB,IAAIv8C,IACvBw8C,EAA8B,IAAIx8C,IAIxC,UAAYy8C,KAAYH,EAAe,CACtC,MAAMz4C,EAAUo6B,EAAasQ,aAAckO,EAASzsD,QAE9C6T,IAKDA,EAAQvJ,GAAI,cAAiBuJ,EAAQvJ,GAAI,eAIvB,cAAlBmiD,EAASplD,MAAyBjL,KAAKswD,mBAAoBD,IAC/DD,EAA4BviD,IAAK4J,GAEnC,CAGA,UAAY44C,KAAYH,EAAe,CACtC,MAAMz4C,EAAUo6B,EAAasQ,aAAckO,EAASzsD,QAGpD,KAAK6T,IAAaA,EAAQvJ,GAAI,eAAiBuJ,EAAQvJ,GAAI,gBAIpC,kBAAlBmiD,EAASplD,KAA2B,CACxC,MAAMivB,EAAO2X,EAAagW,0BAA2BwI,EAASzsD,QAEzDs2B,IAASk2B,EAA4BxiD,IAAKssB,EAAKt7B,QACnDuxD,EAAiBtiD,IAAKqsB,IAKZA,GAAQqhB,GAAkB8U,EAASzsD,SAC7CwsD,EAA4BviD,IAC3BgkC,EAAasQ,aAAckO,EAASzsD,OAAOqD,YAG9C,CACD,CAKA,IAAIspD,GAAe,EAEnB,UAAY9yB,KAAY0yB,EACvBI,GAAe,EACfvwD,KAAK2vD,SAAStT,WAAY,OAAQ5e,GAGnC,UAAY8Z,KAAe6Y,EAA8B,CACxD,MAAMre,EAAaF,EAAayK,aAAc/E,GACxCiZ,EAAe7vD,MAAMrB,KAAMi4C,EAAY/b,eACvCi1B,EAAkB9vD,MAAMrB,KAAMuyC,EAAawV,kBAAmBtV,EAAY,CAAEyL,cAAc,KAI1F,GAAagT,EAAcC,EAAiB,MACjDF,GAAe,EACfvwD,KAAK2vD,SAAStT,WAAY,WAAY9E,GAExC,CAGKgZ,GASJvwD,KAAKi6B,KAAKy2B,aAMZ,CASQ,kBAAAJ,CAAoBD,GAC3B,IAAIM,EAAY,KAShB,OAN8B,OAAzBN,EAASp0B,aAAyD,IAAjCo0B,EAASO,aAAat4D,QAA8C,GAA9B+3D,EAASQ,WAAWv4D,SAC/Fq4D,EAAY3wD,KAAK6xC,aAAa8N,UAAW0Q,EAASQ,WAAY,GAAK,CAClErT,cAAc,KAITmT,GAAaA,EAAUziD,GAAI,UAAW,KAC9C,EAGD,SAAS,GAAW4iD,EAAkBC,GAErC,IAAKpwD,MAAMC,QAASkwD,GAKpB,OAAKA,IAAWC,MAIND,EAAO5iD,GAAI,WAAa6iD,EAAO7iD,GAAI,WACrC4iD,EAAOhuD,OAASiuD,EAAOjuD,IAKhC,CCpQe,MAAMkuD,WAAsB7F,GAqB1C,WAAAppD,CAAak4B,GACZzsB,MAAOysB,GAXR,KAAQg3B,kBAA4B,EAKpC,KAAgB7F,aAAe,CAAE,QAAS,QAQzCprD,KAAK0f,YAAa,EAClB,MAAMlc,EAAWxD,KAAKwD,SAEtBA,EAAS6L,GAA4B,SAAS,KAC7CrP,KAAKixD,kBAAmB,EAUxBjxD,KAAKkxD,iBAAmBn+B,YAAY,KACnC/yB,KAAKqtD,QACLpzB,EAAKM,QAAQ,QAAU,GACrB,GAAI,IAGR/2B,EAAS6L,GAA2B,QAAQ,CAAEgG,EAAKvS,KAClD,MAAMquD,EAAmB3tD,EAASyxB,UAAUkT,gBAElB,OAArBgpB,GAA6BA,IAAqBruD,EAAKc,SAC3DJ,EAASsvB,WAAY,EACrB9yB,KAAKixD,kBAAmB,EAIxBh3B,EAAKM,QAAQ,SACd,GAEF,CAKO,KAAA8yB,GACDrtD,KAAKixD,mBACTjxD,KAAKixD,kBAAmB,EACxBjxD,KAAKwD,SAASsvB,WAAY,EAE5B,CAKO,UAAAu4B,CAAYJ,GAClBjrD,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAC3B,CAKgB,OAAAjlC,GACVhmB,KAAKkxD,kBACTr+B,aAAc7yB,KAAKkxD,kBAGpB1jD,MAAMwY,SACP,ECzEc,MAAMorC,WAA0BzG,GAuD9C,WAAA5oD,CAAak4B,GACZzsB,MAAOysB,GAEPj6B,KAAKqxD,iBAAmBp3B,EAAKq3B,YAAa9B,IAC1CxvD,KAAKuxD,cAAgBt3B,EAAKq3B,YAAaN,IACvChxD,KAAKi1B,UAAYj1B,KAAKwD,SAASyxB,UAC/Bj1B,KAAK6xC,aAAe5X,EAAK4X,aAEzB7xC,KAAKwxD,WAAa,IAAI9jD,QAEtB1N,KAAKutD,kCAAoC,IAAUzqD,IAClD9C,KAAKwD,SAASgN,KAA4C,sBAAuB1N,EAAM,GACrF,KAEH9C,KAAKyxD,2BAA6BC,aAAa,IAAM1xD,KAAK2xD,sBAAsB,KAEhF3xD,KAAK4xD,+CAAiD,IAAU,IAAQ5xD,KAAKwD,SAAS44C,aAAc,GAAS,KAE7Gp8C,KAAK6xD,iBAAmB,CACzB,CAKgB,OAAA9rC,CAASgsB,GACxB,MAAMH,EAAcG,EAAW3yB,cASzB0yC,EAAyB,KACxB9xD,KAAKwD,SAAS44C,cAMpBp8C,KAAK+xD,uBAAwB,KAAMngB,GAEnC5xC,KAAKwD,SAAS44C,aAAc,EAG5Bp8C,KAAK4xD,+CAA+Cn+B,SAAQ,EAM7DzzB,KAAKwP,SAAUuiC,EAAY,eAzBM,KAChC/xC,KAAKwD,SAAS44C,aAAc,EAG5Bp8C,KAAK4xD,gDAAgD,GAqBc,CAAEhlD,SAAU,YAEhF5M,KAAKwP,SAAUuiC,EAAY,UAAW+f,EAAwB,CAAEllD,SAAU,UAAW8S,YAAY,IACjG1f,KAAKwP,SAAUuiC,EAAY,QAAS+f,EAAwB,CAAEllD,SAAU,UAAW8S,YAAY,IAG1F1f,KAAKwxD,WAAW5jD,IAAKgkC,KAM1B5xC,KAAKwP,SAAUoiC,EAAa,UAAWkgB,EAAwB,CAAEllD,SAAU,UAAW8S,YAAY,IAElG1f,KAAKwP,SAAUoiC,EAAa,mBAAmB,CAAEv8B,EAAK41C,KAahDjrD,KAAKwD,SAASm4B,cAAgB,EAAItyB,YAWvCrJ,KAAK+xD,uBAAwB9G,EAAUrZ,GAQvC5xC,KAAK4xD,iDAAgD,IAGtD5xD,KAAKwxD,WAAW3jD,IAAK+jC,GACtB,CAKgB,aAAA0Z,CAAevZ,GAC9B/xC,KAAK2P,cAAeoiC,EACrB,CAKgB,OAAA/rB,GACfxY,MAAMwY,UAENgsC,cAAehyD,KAAKyxD,4BACpBzxD,KAAKutD,kCAAkC95B,SACvCzzB,KAAK4xD,+CAA+Cn+B,QACrD,CAGQ,mBAAAw+B,GAKR,CAUQ,sBAAAF,CAAwB9G,EAAmBrZ,GAClD,IAAM5xC,KAAK00B,UACV,OAGD,MAAMud,EAAeL,EAAY1yB,YAAaizB,eAE9C,GAAKnyC,KAAK+qD,iCAAkC9Y,EAAa+P,YACxD,OAIDhiD,KAAKqxD,iBAAiBhE,QAEtB,MAAM7E,EAAmBxoD,KAAK6xC,aAAakQ,mBAAoB9P,GAM/D,GAAoC,GAA/BuW,EAAiBtb,YAWtB,GALAltC,KAAKi6B,KAAKi4B,iBAAkB,EAG5BlyD,KAAKuxD,cAAclE,SAEdrtD,KAAKi1B,UAAUhR,QAASukC,KAAsBxoD,KAAK6xC,aAAagQ,sBAAuB5P,GAO5F,KAAOjyC,KAAK6xD,iBAAmB,GAK9B7xD,KAAKiyD,2BAKN,GAAKjyD,KAAKi1B,UAAUoI,UAAWmrB,GAG9BxoD,KAAKi6B,KAAKy2B,kBACJ,CACN,MAAM5tD,EAAuC,CAC5C4qD,aAAc1tD,KAAKi1B,UACnBw4B,aAAcjF,EACdvW,gBAWDjyC,KAAKwD,SAASgN,KAAwC,kBAAmB1N,GAMzE9C,KAAKutD,kCAAmCzqD,EACzC,OArDC9C,KAAKi6B,KAAKi4B,iBAAkB,CAsD9B,CAKQ,kBAAAP,GACP3xD,KAAK6xD,iBAAmB,CACzB,ECjSc,MAAMM,WAA4BhH,GAShD,WAAAppD,CAAak4B,GACZzsB,MAAOysB,GANR,KAAgBmxB,aAAe,CAAE,mBAAoB,oBAAqB,kBAQzE,MAAM5nD,EAAWxD,KAAKwD,SAEtBA,EAAS6L,GAAuC,oBAAoB,KAOnE7L,EAASm4B,aAAc,CAAI,GACzB,CAAE/uB,SAAU,QAEfpJ,EAAS6L,GAAqC,kBAAkB,KAO/D7L,EAASm4B,aAAc,CAAK,GAC1B,CAAE/uB,SAAU,OAChB,CAKO,UAAAy+C,CAAYJ,GAKlBjrD,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAAU,CACnCnoD,KAAMmoD,EAASnoD,MAMjB,ECxDc,MAAMsvD,GAepB,WAAArwD,CAAaswD,EAAqC9tD,EAAoC,CAAC,GAKtFvE,KAAKsyD,OAAS/tD,EAAQguD,WAAaC,GAAUH,GAAuB,KAEpEryD,KAAKyyD,QAAUJ,CAChB,CAKA,SAAWK,GAKV,OAJM1yD,KAAKsyD,SACVtyD,KAAKsyD,OAASE,GAAUxyD,KAAKyyD,UAGvBzyD,KAAKsyD,MACb,CAKA,SAAW/3C,GACV,OAAOva,KAAKyyD,QAAQl4C,KACrB,CAWO,OAAAo4C,CAAS1nD,GACf,OAAOjL,KAAKyyD,QAAQE,QAAS1nD,EAC9B,CAOO,OAAA2nD,CAAS3nD,EAAcnI,GAC7B9C,KAAKyyD,QAAQG,QAAS3nD,EAAMnI,EAC7B,CAKA,iBAAW+vD,CAAer6D,GACzBwH,KAAKyyD,QAAQI,cAAgBr6D,CAC9B,CAEA,iBAAWq6D,GACV,OAAO7yD,KAAKyyD,QAAQI,aACrB,CAKA,cAAWC,CAAYt6D,GACtBwH,KAAKyyD,QAAQK,WAAat6D,CAC3B,CAEA,cAAWs6D,GACV,OAAO9yD,KAAKyyD,QAAQK,UACrB,CAKO,YAAAC,CAAcC,EAAgB94D,EAAWD,GAC/C+F,KAAKyyD,QAAQM,aAAcC,EAAO94D,EAAGD,EACtC,CAKA,cAAWg5D,GACV,MAAkC,QAA3BjzD,KAAKyyD,QAAQK,cAA4B9yD,KAAKyyD,QAAiBS,gBACvE,EAaD,SAASV,GAAUH,GAElB,MAAMK,EAAQ/xD,MAAMrB,KAAM+yD,EAAmBK,OAAS,IAChD9hC,EAAQjwB,MAAMrB,KAAM+yD,EAAmBzhC,OAAS,IAEtD,OAAK8hC,EAAMp6D,OACHo6D,EAID9hC,EACL7qB,QAAQ9F,GAAsB,SAAdA,EAAKkzD,OACrBn2D,KAAKiD,GAAQA,EAAKmzD,aACrB,CCtHe,MAAMC,WAAsBlI,GAA3C,kCAIC,KAAgBC,aAAe,cAKxB,UAAAC,CAAYJ,GAOlB,MAAMqI,EAAkBrI,EAASsI,kBAC3Bt5B,EAAOj6B,KAAKi6B,KACZu5B,EAAev5B,EAAKz2B,SAE1B,IAAIiwD,EAAoC,KACpC3wD,EAAsB,KACtB4wD,EAAiC,GA0BrC,GAxBKzI,EAASwI,eACbA,EAAe,IAAIrB,GAAcnH,EAASwI,eAGpB,OAAlBxI,EAASnoD,KACbA,EAAOmoD,EAASnoD,KAOL2wD,IACX3wD,EAAO2wD,EAAad,QAAS,eAWzBa,EAAav+B,UAAU8X,OAE3B2mB,EAAe/yD,MAAMrB,KAAMk0D,EAAav+B,UAAUmY,kBAQ5C,GAAKkmB,EAAgBh7D,OAC3Bo7D,EAAeJ,EAAgBt2D,KAAKmkD,IAKnC,MAAMwG,EAAY1tB,EAAK4X,aAAae,kBAAmBuO,EAASj8B,eAAgBi8B,EAASnX,aACnF4d,EAAU3tB,EAAK4X,aAAae,kBAAmBuO,EAASxG,aAAcwG,EAAS/W,WAErF,OAAKud,EACG1tB,EAAK4d,YAAa8P,EAAWC,GACzBA,EACJ3tB,EAAK4d,YAAa+P,QADnB,CAEP,IACG7hD,QAAU+e,KAAiCA,SAOhD,GAGU,EAAIzb,UAAY,CACzB,MAAM4oC,EAAiBgZ,EAASrnD,OAAwBwb,cAAcF,YAAaizB,eAEnFuhB,EAAe/yD,MAAMrB,KAAM26B,EAAK4X,aAAakQ,mBAAoB9P,GAAe7E,YAOjF,CAOA,GAAK,EAAI/jC,WAAmC,yBAAtB4hD,EAAS0I,WAAwC7wD,GAAQA,EAAKirB,SAAU,MAC7F/tB,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAAU,CACnC0I,UAAW,kBACXD,aAAc,CAAEz5B,EAAK4d,YAAa6b,EAAc,GAAI/yB,aAYtD,GAA2B,cAAtBsqB,EAAS0I,WAA6B7wD,GAAQA,EAAKysB,SAAU,MAAlE,CAGC,MAAM1Q,EAAQ/b,EAAK/F,MAAO,YAE1B,IAAI62D,EAAmBF,EAEvB,QAAUl5D,EAAI,EAAGA,EAAIqkB,EAAMvmB,OAAQkC,IAAM,CACxC,MAAMq5D,EAAWh1C,EAAOrkB,GAEP,IAAZq5D,IACJ7zD,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAAU,CACnCnoD,KAAM+wD,EACNJ,eACAC,aAAcE,EACdD,UAAW1I,EAAS0I,UACpBh4B,YAAasvB,EAAStvB,cAIvBi4B,EAAmB,CAAEJ,EAAav+B,UAAUoY,kBAGxC7yC,EAAI,EAAIqkB,EAAMvmB,SAClB0H,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAAU,CACnC0I,UAAW,kBACXD,aAAcE,IAIfA,EAAmB,CAAEJ,EAAav+B,UAAUoY,iBAE9C,CAOD,MAGArtC,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAAU,CACnCnoD,OACA2wD,eACAC,eACAC,UAAW1I,EAAS0I,UACpBh4B,YAAasvB,EAAStvB,aAMxB,ECtKc,MAAMm4B,WAA0BnJ,GAI9C,WAAA5oD,CAAak4B,GACZzsB,MAAOysB,GAEPj6B,KAAKwD,SAAS6L,GAA8B,WAAW,CAAEC,EAAOxM,KAC/D,GAAK9C,KAAK00B,azHiHmBpH,EyHjHUxqB,EAAKwqB,UzHkH5BjB,GAASG,YAC1Bc,GAAWjB,GAASC,WACpBgB,GAAWjB,GAASE,SACpBe,GAAWjB,GAASI,WyHrHqC,CACvD,MAAM/b,EAAY,IAAI2+B,GAAmBrvC,KAAKwD,SAAU,WAAYxD,KAAKwD,SAASyxB,UAAUoY,iBAE5FrtC,KAAKwD,SAASgN,KAAiCE,EAAW5N,GAErD4N,EAAUvE,KAAKF,QACnBqD,EAAMnD,MAER,CzHyGI,IAAyBmhB,CyHzG7B,GAEF,CAKgB,OAAAvH,GAAiB,CAKjB,aAAAulC,GAAuB,EC3BzB,MAAMyI,WAAoBpJ,GAIxC,WAAA5oD,CAAak4B,GACZzsB,MAAOysB,GAEP,MAAM5iB,EAAMrX,KAAKwD,SAEjB6T,EAAIhI,GAA8B,WAAW,CAAEgG,EAAKvS,KACnD,IACE9C,KAAK00B,WACN5xB,EAAKwqB,SAAWjB,GAASS,KACzBhqB,EAAK0qB,QAEL,OAGD,MAAMle,EAAQ,IAAI+/B,GAAmBh4B,EAAK,MAAOA,EAAI4d,UAAUoY,iBAE/Dh2B,EAAI7G,KAA4BlB,EAAOxM,GAElCwM,EAAMnD,KAAKF,QACfoJ,EAAIlJ,MACL,GAEF,CAKgB,OAAA4Z,GAAiB,CAKjB,aAAAulC,GAAuB,EC/BxC,SAJA,SAAmB9yD,GACjB,OAAO,GAAUA,EAAO,EAC1B,EC2De,MAAM,WAAa2a,KAmFjC,WAAApR,CAAagkC,GACZv4B,QApED,KAAgBwmD,SAAqC,IAAIziD,IA+BzD,KAAiB0iD,0BAA0E,IAAI5jC,QAK/F,KAAiB6jC,WAAiD,IAAI3iD,IAUtE,KAAQ4iD,gBAA0B,EAKlC,KAAQC,uBAAiC,EAKzC,KAAQC,oBAA8B,EAMtC,KAAQC,kCAA4C,EAQnDt0D,KAAKwD,SAAW,IAAI2b,GAAU4mB,GAC9B/lC,KAAK6xC,aAAe,IAAIuR,GAAcpjD,KAAKwD,UAE3CxD,KAAKyR,IAAK,yBAAyB,GACnCzR,KAAKyR,IAAK,mBAAmB,GAE7BzR,KAAK4vD,UAAY,IAAI/T,GAAU77C,KAAK6xC,aAAc7xC,KAAKwD,SAASyxB,WAChEj1B,KAAK4vD,UAAU5oD,KAAM,YAAa,cAAe,eAC/CzH,GAAIS,KAAKwD,SAAU,YAAa,cAAe,eAEjDxD,KAAKu0D,QAAU,IAAI,GAAgBv0D,KAAKwD,UAIxCxD,KAAKw0D,YAAahF,IAClBxvD,KAAKw0D,YAAaxD,IAClBhxD,KAAKw0D,YAAapD,IAClBpxD,KAAKw0D,YAAahJ,IAClBxrD,KAAKw0D,YAAalH,IAClBttD,KAAKw0D,YAAarC,IAClBnyD,KAAKw0D,YAAaV,IAClB9zD,KAAKw0D,YAAanB,IAClBrzD,KAAKw0D,YAAaT,IAGI/zD,KvC5ClBwD,SAAS6L,GAA+B,WAAYusC,GAAsB,CAAEhvC,SAAU,QuC6C1FolC,GAAyBhyC,MAGzBA,KAAKqP,GAAqB,UAAU,KACnCrP,KAAKy0D,UAGLz0D,KAAKwD,SAASgN,KAAsC,iBAGpDxQ,KAAKs0D,kCAAmC,CAAK,IAI9Ct0D,KAAKwP,SAA4CxP,KAAKwD,SAASyxB,UAAW,UAAU,KACnFj1B,KAAKs0D,kCAAmC,CAAI,IAI7Ct0D,KAAKwP,SAAiCxP,KAAKwD,SAAU,oBAAoB,KACxExD,KAAKs0D,kCAAmC,CAAI,IAKxC,EAAInrD,OACRnJ,KAAKwP,SAAiCxP,KAAKwD,SAAU,QAAQ,CAAE6R,EAAKvS,KACxC9C,KAAK6xC,aAAasQ,aAAcr/C,EAAKmoD,SAASyJ,gBAIxE10D,KAAK6xC,aAAa0W,oBACnB,GAGH,CAeO,aAAAoM,CAAejU,EAAsB1+C,EAAe,QAC1D,MAAM4yD,EAAW50D,KAAKwD,SAASstC,QAAS9uC,GAGxC4yD,EAAStsB,MAAQoY,EAAQiB,QAAQ/4C,cAEjC,MAAMisD,EAAmD,CAAC,EAS1D,UAAc7yD,KAAAA,EAAAA,MAAMxJ,KAAWmI,MAAMrB,KAAMohD,EAAQn7C,YAClDsvD,EAA0B7yD,GAASxJ,EAMrB,UAATwJ,EACJhC,KAAKu0D,QAAQ34B,SAAUpjC,EAAMuE,MAAO,KAAO63D,GAE3C50D,KAAKu0D,QAAQ9uD,aAAczD,EAAMxJ,EAAOo8D,GAI1C50D,KAAKi0D,0BAA0BxiD,IAAKivC,EAASmU,GAE7C,MAAMC,EAAiC,KACtC90D,KAAKu0D,QAAQ9uD,aAAc,oBAAsBmvD,EAASt/B,YAAa54B,WAAYk4D,GAE9EA,EAASt/B,WACbt1B,KAAKu0D,QAAQ34B,SAAU,eAAgBg5B,GAEvC50D,KAAKu0D,QAAQx5B,YAAa,eAAgB65B,EAC3C,EAIDE,IAEA90D,KAAKg0D,SAASviD,IAAKzP,EAAM0+C,GACzB1gD,KAAK6xC,aAAawM,aAAcqC,EAASkU,GACzC50D,KAAK4vD,UAAUvT,WAAY,WAAYuY,GACvC50D,KAAK4vD,UAAUvT,WAAY,aAAcuY,GACzC50D,KAAK4vD,UAAU9T,aAAajuC,IAAK6yC,EAAQthC,eAEzCw1C,EAASvlD,GAAI,mBAAmB,CAAEgG,EAAKlW,IAAUa,KAAK4vD,UAAUvT,WAAY,WAAYl9C,KACxFy1D,EAASvlD,GAAI,qBAAqB,CAAEgG,EAAKlW,IAAUa,KAAK4vD,UAAUvT,WAAY,aAAcl9C,KAC5Fy1D,EAASvlD,GAAI,eAAe,CAAEgG,EAAKlW,IAAUa,KAAK4vD,UAAUvT,WAAY,OAAQl9C,KAChFy1D,EAASvlD,GAAI,qBAAqB,IAAMrP,KAAKu6B,OAAQu6B,KAErDF,EAASvlD,GAAI,UAAU,KACtBrP,KAAKs0D,kCAAmC,CAAI,IAG7C,UAAYS,KAAY/0D,KAAKk0D,WAAWhpD,SACvC6pD,EAAShvC,QAAS26B,EAAS1+C,EAE7B,CAQO,aAAAgzD,CAAehzD,GACrB,MAAM0+C,EAAU1gD,KAAKg0D,SAAS5rD,IAAKpG,GAGnCrB,MAAMrB,KAAMohD,EAAQn7C,YAAa3H,SAAS,EAAIoE,KAAAA,KAAY0+C,EAAQh6C,gBAAiB1E,KAEnF,MAAM6yD,EAA2B70D,KAAKi0D,0BAA0B7rD,IAAKs4C,GAGrE,UAAYtqC,KAAay+C,EACxBnU,EAAQj7C,aAAc2Q,EAAWy+C,EAA0Bz+C,IAG5DpW,KAAKg0D,SAASriD,OAAQ3P,GACtBhC,KAAK6xC,aAAauM,iBAAkBsC,GAEpC,UAAYqU,KAAY/0D,KAAKk0D,WAAWhpD,SACvC6pD,EAASzJ,cAAe5K,EAE1B,CAQO,UAAAuU,CAAYjzD,EAAe,QACjC,OAAOhC,KAAKg0D,SAAS5rD,IAAKpG,EAC3B,CAeO,WAAAwyD,CAAaU,GACnB,IAAIH,EAAW/0D,KAAKk0D,WAAW9rD,IAAK8sD,GAEpC,GAAKH,EACJ,OAAOA,EAGRA,EAAW,IAAIG,EAAqBl1D,MAEpCA,KAAKk0D,WAAWziD,IAAKyjD,EAAqBH,GAE1C,UAAc/yD,EAAM+vC,KAAgB/xC,KAAKg0D,SACxCe,EAAShvC,QAASgsB,EAAY/vC,GAK/B,OAFA+yD,EAASlK,SAEFkK,CACR,CAWO,WAAAzD,CAA4C4D,GAClD,OAAOl1D,KAAKk0D,WAAW9rD,IAAK8sD,EAC7B,CAKO,gBAAAC,GACN,UAAYJ,KAAY/0D,KAAKk0D,WAAWhpD,SACvC6pD,EAASjK,SAEX,CAKO,eAAAsK,GACN,UAAYL,KAAY/0D,KAAKk0D,WAAWhpD,SACvC6pD,EAASlK,QAEX,CAqBO,oBAAAwK,EAA8D,WACpEtrC,EAAU,YACVC,EAAW,eACXC,EAAiB,GAAE,eACnBc,EAAiB,IAMd,CAAC,GACJ,MAAMjG,EAAQ9kB,KAAKwD,SAASyxB,UAAUoY,gBAEtC,IAAMvoB,EACL,OAID,MAAMwwC,EAAe,GAAW,CAAEvrC,aAAYC,cAAaC,iBAAgBc,mBAE5C,iBAAnBd,IACXA,EAAiB,CAChBxI,IAAKwI,EACLpI,OAAQoI,EACRlI,KAAMkI,EACNtI,MAAOsI,IAIT,MAAM1lB,EAAU,CACfX,OAAQ5D,KAAK6xC,aAAa2U,eAAgB1hC,GAC1CmF,iBACAc,iBACAhB,aACAC,eAGDhqB,KAAKwQ,KAAqC,uBAAwBjM,EAAS+wD,G7HzZtE,UACN,OACC1xD,EAAM,eACNqmB,EAAiB,EAAC,eAClBc,EAAiB,EAAC,WAClBhB,EAAU,YACVC,IAUD,MAAM0B,EAAeR,GAAWtnB,GAChC,IAAI+nB,EAA+BD,EAC/B6pC,EAAmC,KAKvC,IAHAtrC,EA+WD,SAAkCA,GACjC,MAA+B,iBAAnBA,EACJ,CACNxI,IAAKwI,EACLpI,OAAQoI,EACRlI,KAAMkI,EACNtI,MAAOsI,GAIFA,CACR,CA1XkBurC,CAAyBvrC,GAGlC0B,GAAgB,CACvB,IAAI8pC,EASHA,EAAwBlqC,GADpBI,GAAiBD,EACqB9nB,EAEA2xD,GAI3C1qC,GAA2B,CAC1BjsB,OAAQ62D,EACR3qC,QAAS,IAMDU,GAAyB5nB,EAAQ+nB,GAEzC5B,aACAgB,iBACAf,gBAKD,MAAMpC,EAAa4D,GAAyB5nB,EAAQ+nB,GAUpD,GARA7B,GAAwB,CACvBvmB,OAAQooB,EACRvI,KAAMwE,EACNqC,iBACAF,aACAC,gBAGI2B,EAAc/sB,QAAU+sB,GAY5B,GAPA4pC,EAAe5pC,EAAcE,aAC7BF,EAAgBA,EAAc/sB,QAMxB22D,EACL,YAGD5pC,EAAgB,IAElB,CACD,C6HsUE+pC,CAA4BnxD,EAC7B,CAMO,KAAA0oC,GACN,IAAMjtC,KAAKwD,SAASsvB,UAAY,CAC/B,MAAMuX,EAAWrqC,KAAKwD,SAASyxB,UAAUkT,gBAEpCkC,IACJrqC,KAAK6xC,aAAa5E,MAAO5C,GACzBrqC,KAAK0wD,cAQP,CACD,CAiCO,MAAAn2B,CAAiBhrB,GACvB,GAAKvP,KAAK21D,uBAAyB31D,KAAKo0D,sBAavC,MAAM,IAAI,EACT,0BACAp0D,MAIF,IAEC,GAAKA,KAAKm0D,eACT,OAAO5kD,EAAUvP,KAAKu0D,SAKvBv0D,KAAKm0D,gBAAiB,EACtB,MAAMyB,EAAiBrmD,EAAUvP,KAAKu0D,SActC,OAbAv0D,KAAKm0D,gBAAiB,GAKhBn0D,KAAKq0D,oBAAsBr0D,KAAKs0D,mCACrCt0D,KAAKo0D,uBAAwB,EAC7Bp0D,KAAKwD,SAASwtC,gBAAiBhxC,KAAKu0D,SACpCv0D,KAAKo0D,uBAAwB,EAE7Bp0D,KAAKwQ,KAAuB,WAGtBolD,CACR,CAAE,MAAQp0D,GAGT,EAAc2M,uBAAwB3M,EAAKxB,KAC5C,CACD,CAYO,WAAA0wD,GACN1wD,KAAKs0D,kCAAmC,EACxCt0D,KAAKsxD,YAAaN,IAAgB3D,QAClCrtD,KAAKu6B,QAAQ,QACd,CAKO,OAAAvU,GACN,UAAY+uC,KAAY/0D,KAAKk0D,WAAWhpD,SACvC6pD,EAAS/uC,UAGVhmB,KAAKwD,SAASwiB,UAEdhmB,KAAK2P,eACN,CAiBO,gBAAA+nC,CAAkB5M,EAAiClvB,GACzD,OAAO+sB,GAASC,UAAWkC,EAAgBlvB,EAC5C,CAOO,mBAAA+7B,CAAqB13C,GAC3B,OAAO0oC,GAASc,aAAcxpC,EAC/B,CAOO,oBAAA23C,CAAsB33C,GAC5B,OAAO0oC,GAASoB,cAAe9pC,EAChC,CAUO,WAAA43C,CAAanX,EAAiBC,GACpC,OAAO,IAAIoK,GAAOrK,EAAOC,EAC1B,CAKO,aAAAmX,CAAe73C,GACrB,OAAO8qC,GAAM0B,UAAWxsC,EACzB,CAQO,aAAA83C,CAAetgC,GACrB,OAAOszB,GAAMyB,UAAW/0B,EACzB,CAgGO,eAAAugC,IAAoB97C,GAC1B,OAAO,IAAIwwC,MAAcxwC,EAC1B,CASO,iBAAA25D,CAAmBC,GACzB91D,KAAKq0D,mBAAqByB,EAEb,GAARA,GAEJ91D,KAAKu6B,QAAQ,QAEf,CAMQ,OAAAk6B,GACPz0D,KAAK21D,uBAAwB,EAC7B31D,KAAKm1D,mBACLn1D,KAAK4vD,UAAUje,SACf3xC,KAAKo1D,kBACLp1D,KAAK21D,uBAAwB,CAC9B,ECpwBc,MAAe,GAoRtB,EAAAznD,GAON,MAAM,IAAI7V,MAAO,0BAClB,ECjQc,MAAe,WAAa,GAyB1C,WAAA0J,CAAaqjC,GACZ53B,QApBD,KAAgB5O,OAA4C,KAsB3DoB,KAAKulC,OAAS,GAAOH,EACtB,CAKA,YAAW5hC,GACV,OAAO,IACR,CAQA,SAAWoB,GACV,IAAIm3B,EAEJ,IAAM/7B,KAAKpB,OACV,OAAO,KAGR,GAAqD,QAA9Cm9B,EAAM/7B,KAAKpB,OAAOo9B,cAAeh8B,OACvC,MAAM,IAAI,EAAe,iCAAkCA,MAG5D,OAAO+7B,CACR,CASA,eAAWiO,GACV,IAAIjO,EAEJ,IAAM/7B,KAAKpB,OACV,OAAO,KAGR,GAA2D,QAApDm9B,EAAM/7B,KAAKpB,OAAOm3D,oBAAqB/1D,OAC7C,MAAM,IAAI,EAAe,iCAAkCA,MAG5D,OAAO+7B,CACR,CAQA,cAAW4B,GACV,OAAO,CACR,CAOA,aAAWyM,GACV,OAAMpqC,KAAKpB,OAIJoB,KAAKgqC,YAAehqC,KAAK29B,WAHxB,IAIT,CAKA,eAAW1B,GACV,MAAMr3B,EAAQ5E,KAAK4E,MAEnB,OAAmB,OAAVA,GAAkB5E,KAAKpB,OAAQi9B,SAAUj3B,EAAQ,IAAS,IACpE,CAKA,mBAAWgiB,GACV,MAAMhiB,EAAQ5E,KAAK4E,MAEnB,OAAmB,OAAVA,GAAkB5E,KAAKpB,OAAQi9B,SAAUj3B,EAAQ,IAAS,IACpE,CAMA,QAAWtO,GAEV,IAAIA,EAAgC0J,KAEpC,KAAQ1J,EAAKsI,QACZtI,EAAOA,EAAKsI,OAGb,OAAOtI,CACR,CAKO,UAAAilC,GAKN,OAAuB,OAAhBv7B,KAAKpB,QAA0BoB,KAAK1J,KAAKilC,YACjD,CAkBO,OAAAW,GACN,MAAMz8B,EAAO,GAEb,IAAIN,EAAgCa,KAEpC,KAAQb,EAAKP,QACZa,EAAKL,QAASD,EAAK6qC,aACnB7qC,EAAOA,EAAKP,OAGb,OAAOa,CACR,CAWO,YAAA08B,CAAc53B,EAA4D,CAAC,GACjF,MAAM63B,EAA4C,GAClD,IAAIx9B,EAAS2F,EAAQ83B,YAAcr8B,KAAOA,KAAKpB,OAE/C,KAAQA,GACPw9B,EAAW73B,EAAQ+3B,YAAc,OAAS,WAAa19B,GACvDA,EAASA,EAAOA,OAGjB,OAAOw9B,CACR,CAWO,iBAAAG,CAAmBp9B,EAAYoF,EAAqC,CAAC,GAC3E,MAAMi4B,EAAax8B,KAAKm8B,aAAc53B,GAChCk4B,EAAat9B,EAAKg9B,aAAc53B,GAEtC,IAAI/J,EAAI,EAER,KAAQgiC,EAAYhiC,IAAOiiC,EAAYjiC,IAAOgiC,EAAYhiC,IACzDA,IAGD,OAAa,IAANA,EAAU,KAAOgiC,EAAYhiC,EAAI,EACzC,CAQO,QAAAkiC,CAAUv9B,GAEhB,GAAKa,MAAQb,EACZ,OAAO,EAIR,GAAKa,KAAK1J,OAAS6I,EAAK7I,KACvB,OAAO,EAGR,MAAMqmC,EAAW38B,KAAKk8B,UAChBU,EAAWz9B,EAAK+8B,UAEhB/9B,EAASuY,EAAeimB,EAAUC,GAExC,OAASz+B,GACR,IAAK,SACJ,OAAO,EAER,IAAK,YACJ,OAAO,EAER,QACC,OAAOw+B,EAAUx+B,GAAqBy+B,EAAUz+B,GAEnD,CAQO,OAAA0+B,CAAS19B,GAEf,OAAKa,MAAQb,IAKRa,KAAK1J,OAAS6I,EAAK7I,OAKhB0J,KAAK08B,SAAUv9B,GACxB,CAQO,YAAA8mC,CAAclvC,GACpB,OAAOiJ,KAAKulC,OAAO33B,IAAK7W,EACzB,CAQO,YAAAskC,CAActkC,GACpB,OAAOiJ,KAAKulC,OAAOn9B,IAAKrR,EACzB,CAQO,aAAAivC,GACN,OAAOhmC,KAAKulC,OAAO9sB,SACpB,CAKO,gBAAA6lB,GACN,OAAOt+B,KAAKulC,OAAOtuC,MACpB,CAOO,MAAA+lC,GACN,MAAMC,EAAY,CAAC,EAYnB,OARKj9B,KAAKulC,OAAO1xB,OAChBopB,EAAK13B,WAAa5E,MAAMrB,KAAMU,KAAKulC,QAAStX,QAAQ,CAAE9vB,EAAQghD,KAC7DhhD,EAAQghD,EAAM,IAAQA,EAAM,GAErBhhD,IACL,CAAC,IAGE8+B,CACR,CAQO,MAAAM,CAAQy4B,GACd,OAAO,IAAMh2D,KAAK+B,YAAsB/B,KAAKulC,OAC9C,CAQO,OAAAlU,GACNrxB,KAAKpB,OAAQk+B,gBAAiB98B,KAAK4E,MACpC,CAUO,aAAAwiC,CAAerwC,EAAayB,GAClCwH,KAAKulC,OAAO9zB,IAAK1a,EAAKyB,EACvB,CASO,gBAAAy9D,CAAkB7wB,GACxBplC,KAAKulC,OAAS,GAAOH,EACtB,CAUO,gBAAAkC,CAAkBvwC,GACxB,OAAOiJ,KAAKulC,OAAO5zB,OAAQ5a,EAC5B,CAQO,gBAAAm/D,GACNl2D,KAAKulC,OAAO3zB,OACb,EAKD,GAAK/P,UAAUqM,GAAK,SAAUjD,GAC7B,MAAgB,SAATA,GAA4B,eAATA,CAC3B,ECjbe,MAAMkrD,GAYpB,WAAAp0D,CAAamlC,GARb,KAAQkvB,OAAsB,GASxBlvB,GACJlnC,KAAKm2C,aAAc,EAAGjP,EAExB,CAOA,CAASpmC,OAAOC,YACf,OAAOf,KAAKo2D,OAAQt1D,OAAOC,WAC5B,CAKA,UAAWzI,GACV,OAAO0H,KAAKo2D,OAAO99D,MACpB,CAKA,aAAW+9D,GACV,OAAOr2D,KAAKo2D,OAAOnoC,QAAQ,CAAEC,EAAK/uB,IAAU+uB,EAAM/uB,EAAKw+B,YAAY,EACpE,CAKO,OAAA24B,CAAS1xD,GACf,OAAO5E,KAAKo2D,OAAQxxD,IAAW,IAChC,CAKO,YAAA2xD,CAAcp3D,GACpB,MAAMyF,EAAQ5E,KAAKo2D,OAAOptD,QAAS7J,GAEnC,OAAiB,GAAVyF,EAAc,KAAOA,CAC7B,CAMO,kBAAA4xD,CAAoBr3D,GAC1B,MAAMyF,EAAQ5E,KAAKu2D,aAAcp3D,GAEjC,OAAiB,OAAVyF,EAAiB,KAAO5E,KAAKo2D,OAAOt0D,MAAO,EAAG8C,GAAQqpB,QAAQ,CAAEC,EAAK/uB,IAAU+uB,EAAM/uB,EAAKw+B,YAAY,EAC9G,CAQO,aAAA84B,CAAe7xD,GACrB,GAAKA,GAAS5E,KAAKo2D,OAAO99D,OACzB,OAAO0H,KAAKq2D,UAGb,MAAMl3D,EAAOa,KAAKo2D,OAAQxxD,GAE1B,IAAMzF,EAML,MAAM,IAAI,EAAe,qCAAsCa,MAGhE,OAAOA,KAAKw2D,mBAAoBr3D,EACjC,CAQO,aAAAu3D,CAAe96C,GACrB,IAAI+6C,EAAc,EAElB,UAAYx3D,KAAQa,KAAKo2D,OAAS,CACjC,GAAKx6C,GAAU+6C,GAAe/6C,EAAS+6C,EAAcx3D,EAAKw+B,WACzD,OAAO39B,KAAKu2D,aAAcp3D,GAG3Bw3D,GAAex3D,EAAKw+B,UACrB,CAEA,GAAKg5B,GAAe/6C,EAQnB,MAAM,IAAI,EAAe,sCACxB5b,KACA,CACC4b,SACAg7C,SAAU52D,OAKb,OAAOA,KAAK1H,MACb,CASO,YAAA69C,CAAcvxC,EAAesiC,GAEnC,UAAY/nC,KAAQ+nC,EACnB,KAAQ/nC,aAAgB,IAMvB,MAAM,IAAI,EAAe,sCAAuCa,MAIlEA,KAAKo2D,OCvIQ,SAAyBxyD,EAAkBV,EAAkBw9B,EAAe/7B,GAE1F,GAAK9L,KAAKC,IAAKoK,EAAO5K,OAAQsL,EAAOtL,QAzBf,IA0BrB,OAAOsL,EAAO9B,MAAO,EAAG4+B,GAAQvgC,OAAQ+C,GAAS/C,OAAQyD,EAAO9B,MAAO4+B,EAAQ/7B,EAAOf,EAAOtL,SACvF,CACN,MAAMu+D,EAAYl2D,MAAMrB,KAAMsE,GAG9B,OAFAizD,EAAUrvD,OAAQk5B,EAAO/7B,KAAUzB,GAE5B2zD,CACR,CACD,CD6HgBC,CAAmB92D,KAAKo2D,OAAQz1D,MAAMrB,KAAM4nC,GAAStiC,EAAO,EAC3E,CAUO,YAAAmyD,CAAcC,EAAoB7rD,EAAkB,GAC1D,OAAOnL,KAAKo2D,OAAO5uD,OAAQwvD,EAAY7rD,EACxC,CAQO,MAAA6xB,GACN,OAAOh9B,KAAKo2D,OAAOp5D,KAAKmC,GAAQA,EAAK69B,UACtC,EEtKc,MAAM,WAAa,GAkBjC,WAAAj7B,CAAae,EAAesiC,GAC3B53B,MAAO43B,GACPplC,KAAKo9B,MAAQt6B,GAAQ,EACtB,CAKA,cAAoB66B,GACnB,OAAO39B,KAAK8C,KAAKxK,MAClB,CAKA,QAAWwK,GACV,OAAO9C,KAAKo9B,KACb,CAOgB,MAAAJ,GACf,MAAMC,EAAYzvB,MAAMwvB,SAIxB,OAFAC,EAAKn6B,KAAO9C,KAAK8C,KAEVm6B,CACR,CAQgB,MAAAM,GACf,OAAO,IAAI,GAAMv9B,KAAK8C,KAAM9C,KAAKgmC,gBAClC,CAQA,eAAcixB,CAAUh6B,GACvB,OAAO,IAAI,GAAMA,EAAKn6B,KAAMm6B,EAAK13B,WAClC,EAiBD,GAAK1D,UAAUqM,GAAK,SAAUjD,GAC7B,MAAgB,UAATA,GAA6B,gBAATA,GAEjB,SAATA,GAA4B,eAATA,GAEV,SAATA,GAA4B,eAATA,CACrB,ECnEe,MAAM,WAAkB,GAyBtC,WAAAlJ,CAAa07B,EAAgBC,EAAsBplC,GAKlD,GAJAkV,QAEAxN,KAAKy9B,SAAWA,EAEXC,EAAe,GAAKA,EAAeD,EAASE,WAMhD,MAAM,IAAI,EAAe,qCAAsC39B,MAGhE,GAAK1H,EAAS,GAAKolC,EAAeplC,EAASmlC,EAASE,WAMnD,MAAM,IAAI,EAAe,+BAAgC39B,MAE1DA,KAAK8C,KAAO26B,EAAS36B,KAAKlG,UAAW8gC,EAAcA,EAAeplC,GAElE0H,KAAK09B,aAAeA,CACrB,CAOA,eAAWsM,GACV,OAAqC,OAA9BhqC,KAAKy9B,SAASuM,YAAuBhqC,KAAKy9B,SAASuM,YAAchqC,KAAK09B,aAAe,IAC7F,CAOA,cAAWC,GACV,OAAO39B,KAAK8C,KAAKxK,MAClB,CAOA,aAAW8xC,GACV,OAA4B,OAArBpqC,KAAKgqC,YAAuBhqC,KAAKgqC,YAAchqC,KAAK29B,WAAa,IACzE,CAUA,aAAWC,GACV,OAAO59B,KAAK29B,aAAe39B,KAAKy9B,SAASE,UAC1C,CAKA,UAAW/+B,GACV,OAAOoB,KAAKy9B,SAAS7+B,MACtB,CAKA,QAAWtI,GACV,OAAO0J,KAAKy9B,SAASnnC,IACtB,CAOO,OAAA4lC,GACN,MAAMz8B,EAAOO,KAAKy9B,SAASvB,UAM3B,OAJKz8B,EAAKnH,OAAS,IAClBmH,EAAMA,EAAKnH,OAAS,IAAO0H,KAAK09B,cAG1Bj+B,CACR,CAWO,YAAA08B,CAAc53B,EAGjB,CAAC,GACJ,MAAM63B,EAA2D,GACjE,IAAIx9B,EAAwD2F,EAAQ83B,YAAcr8B,KAAOA,KAAKpB,OAE9F,KAAQA,GACPw9B,EAAW73B,EAAQ+3B,YAAc,OAAS,WAAa19B,GACvDA,EAASA,EAAOA,OAGjB,OAAOw9B,CACR,CAQO,YAAA6J,CAAclvC,GACpB,OAAOiJ,KAAKy9B,SAASwI,aAAclvC,EACpC,CAQO,YAAAskC,CAActkC,GACpB,OAAOiJ,KAAKy9B,SAASpC,aAActkC,EACpC,CAQO,aAAAivC,GACN,OAAOhmC,KAAKy9B,SAASuI,eACtB,CAKO,gBAAA1H,GACN,OAAOt+B,KAAKy9B,SAASa,kBACtB,EAkBD,GAAUz8B,UAAUqM,GAAK,SAAUjD,GAClC,MAAgB,eAATA,GAAkC,qBAATA,GAEtB,cAATA,GAAiC,oBAATA,CAC1B,EC/Ne,MAAM,WAAgB,GAsBpC,WAAAlJ,CACCC,EACAojC,EACA9tB,GAEA9J,MAAO43B,GAlBR,KAAiBM,UAAsB,IAAIywB,GAoB1Cn2D,KAAKgC,KAAOA,EAEPsV,GACJtX,KAAK2lC,aAAc,EAAGruB,EAExB,CAKA,cAAWgkB,GACV,OAAOt7B,KAAK0lC,UAAUptC,MACvB,CAKA,aAAW+9D,GACV,OAAOr2D,KAAK0lC,UAAU2wB,SACvB,CAKA,WAAWl0B,GACV,OAA2B,IAApBniC,KAAKs7B,UACb,CAKO,QAAAO,CAAUj3B,GAChB,OAAO5E,KAAK0lC,UAAU4wB,QAAS1xD,EAChC,CAKO,WAAA42B,GACN,OAAOx7B,KAAK0lC,UAAW5kC,OAAOC,WAC/B,CAQO,aAAAi7B,CAAe78B,GACrB,OAAOa,KAAK0lC,UAAU6wB,aAAcp3D,EACrC,CAUO,mBAAA42D,CAAqB52D,GAC3B,OAAOa,KAAK0lC,UAAU8wB,mBAAoBr3D,EAC3C,CAkBO,aAAAu3D,CAAe96C,GACrB,OAAO5b,KAAK0lC,UAAUgxB,cAAe96C,EACtC,CAcO,aAAAs7C,CAAeC,GAErB,IAAIh4D,EAAaa,KAEjB,UAAY4E,KAASuyD,EACpBh4D,EAASA,EAAkB08B,SAAY18B,EAAkBu3D,cAAe9xD,IAGzE,OAAOzF,CACR,CASO,YAAAmnC,CAAc8wB,EAAoB7yD,EAAqC,CAAC,GAC9E,IAAI3F,EAAS2F,EAAQ83B,YAAcr8B,KAAOA,KAAKpB,OAE/C,KAAQA,GAAS,CAChB,GAAKA,EAAOoD,OAASo1D,EACpB,OAAOx4D,EAGRA,EAASA,EAAOA,MACjB,CAEA,OAAO,IACR,CAOgB,MAAAo+B,GACf,MAAMC,EAAYzvB,MAAMwvB,SAIxB,GAFAC,EAAKj7B,KAAOhC,KAAKgC,KAEZhC,KAAK0lC,UAAUptC,OAAS,EAAI,CAChC2kC,EAAK3lB,SAAW,GAEhB,UAAYnY,KAAQa,KAAK0lC,UACxBzI,EAAK3lB,SAAS1X,KAAMT,EAAK69B,SAE3B,CAEA,OAAOC,CACR,CAUgB,MAAAM,CAAQsJ,GAAO,GAC9B,MAAMvvB,EAAWuvB,EAAOlmC,MAAMrB,KAAMU,KAAK0lC,WAAY1oC,KAAKmC,GAAQA,EAAKo+B,QAAQ,UAAW,EAE1F,OAAO,IAAI,GAASv9B,KAAKgC,KAAMhC,KAAKgmC,gBAAiB1uB,EACtD,CASO,YAAA2vB,CAAcC,GACpBlnC,KAAK2lC,aAAc3lC,KAAKs7B,WAAY4L,EACrC,CAWO,YAAAvB,CAAc/gC,EAAegsB,GACnC,MAAMsW,EA0IR,SAAoBA,GAEnB,GAAqB,iBAATA,EACX,MAAO,CAAE,IAAI,GAAMA,IAGdvwB,EAAYuwB,KACjBA,EAAQ,CAAEA,IAIX,OAAOvmC,MAAMrB,KAAM4nC,GACjBlqC,KAAKmC,GACe,iBAARA,EACJ,IAAI,GAAMA,GAGbA,aAAgB,GACb,IAAI,GAAMA,EAAK2D,KAAM3D,EAAK6mC,iBAG3B7mC,GAEV,CAjKgB,CAAWyxB,GAEzB,UAAYzxB,KAAQ+nC,EAEE,OAAhB/nC,EAAKP,QACTO,EAAKkyB,UAGJlyB,EAAcP,OAASoB,KAG1BA,KAAK0lC,UAAUyQ,aAAcvxC,EAAOsiC,EACrC,CAYO,eAAApK,CAAiBl4B,EAAeuG,EAAkB,GACxD,MAAM+7B,EAAQlnC,KAAK0lC,UAAUqxB,aAAcnyD,EAAOuG,GAElD,UAAYhM,KAAQ+nC,EACjB/nC,EAAcP,OAAS,KAG1B,OAAOsoC,CACR,CASA,eAAc+vB,CAAUh6B,GACvB,IAAI3lB,EAEJ,GAAK2lB,EAAK3lB,SAAW,CACpBA,EAAW,GAEX,UAAYK,KAASslB,EAAK3lB,SACpBK,EAAM3V,KAEVsV,EAAS1X,KAAM,GAAQq3D,SAAUt/C,IAGjCL,EAAS1X,KAAM,GAAKq3D,SAAUt/C,GAGjC,CAEA,OAAO,IAAI,GAASslB,EAAKj7B,KAAMi7B,EAAK13B,WAAY+R,EACjD,EAkED,GAAQzV,UAAUqM,GAAK,SAAUjD,EAAcjJ,GAC9C,OAAMA,EAMCA,IAAShC,KAAKgC,OAAmB,YAATiJ,GAA+B,kBAATA,GALpC,YAATA,GAA+B,kBAATA,GAEnB,SAATA,GAA4B,eAATA,CAItB,ECxVe,MAAM,GAgEpB,WAAAlJ,CAAawC,GACZ,IAAMA,IAAcA,EAAQikC,aAAejkC,EAAQkkC,cAMlD,MAAM,IAAI,EACT,sCACA,MAIF,MAAMlkB,EAAYhgB,EAAQggB,WAAa,UAEvC,GAAkB,WAAbA,GAAuC,YAAbA,EAM9B,MAAM,IAAI,EAAe,sCAAuChgB,EAAS,CAAEggB,cAG5EvkB,KAAKukB,UAAYA,EACjBvkB,KAAKwoC,WAAajkC,EAAQikC,YAAc,KAEnCjkC,EAAQkkC,cACZzoC,KAAK0oC,UAAYnkC,EAAQkkC,cAAc1lB,QAEvC/iB,KAAK0oC,UAAY,GAASE,UAAW5oC,KAAKwoC,WAA+B,YAAlBxoC,KAAKukB,UAA0B,MAAQ,UAI/FvkB,KAAKylB,SAAS4xC,WAAa,SAE3Br3D,KAAK6oC,mBAAqBtkC,EAAQskC,iBAClC7oC,KAAK8oC,UAAYvkC,EAAQukC,QACzB9oC,KAAK+oC,mBAAqBxkC,EAAQwkC,iBAElC/oC,KAAKgpC,qBAAuBhpC,KAAKwoC,WAAaxoC,KAAKwoC,WAAW9H,MAAM9hC,OAAS,KAC7EoB,KAAKipC,mBAAqBjpC,KAAKwoC,WAAaxoC,KAAKwoC,WAAW7H,IAAI/hC,OAAS,KACzEoB,KAAKs3D,eAAiBt3D,KAAKylB,SAAS7mB,MACrC,CAOA,CAASkC,OAAOC,YACf,OAAOf,IACR,CAQA,YAAWylB,GACV,OAAOzlB,KAAK0oC,SACb,CAgBO,IAAAQ,CAAMA,GACZ,IAAI3nC,EAAM/I,EAAO4wC,EAAcmuB,EAE/B,GACCnuB,EAAeppC,KAAKylB,SACpB8xC,EAAoBv3D,KAAKs3D,iBAErB/1D,OAAM/I,SAAUwH,KAAKsB,eACfC,GAAQ2nC,EAAM1wC,IAEnB+I,IACLvB,KAAK0oC,UAAYU,EACjBppC,KAAKs3D,eAAiBC,EAExB,CAKO,IAAAj2D,GACN,MAAuB,WAAlBtB,KAAKukB,UACFvkB,KAAKqpC,QAELrpC,KAAKspC,WAEd,CAKQ,KAAAD,GACP,MAAME,EAAmBvpC,KAAKylB,SACxBA,EAAWzlB,KAAKylB,SAAS1C,QACzBnkB,EAASoB,KAAKs3D,eAGpB,GAAuB,OAAlB14D,EAAOA,QAAmB6mB,EAAS7J,SAAWhd,EAAOy3D,UACzD,MAAO,CAAE90D,MAAM,EAAM/I,WAAO,GAI7B,GAAKoG,IAAWoB,KAAKipC,oBAAsBxjB,EAAS7J,QAAU5b,KAAKwoC,WAAY7H,IAAI/kB,OAClF,MAAO,CAAEra,MAAM,EAAM/I,WAAO,GAK7B,MAAMg/D,EAAqBC,GAAuBhyC,EAAU7mB,GACtDO,EAAOq4D,GAAsBE,GAAsBjyC,EAAU7mB,EAAQ44D,GAE3E,GAAKr4D,aAAgB,GAAU,CAC9B,GAAMa,KAAK8oC,QAIJ,CAEN,GAAK9oC,KAAKwoC,YAAcxoC,KAAKwoC,WAAW7H,IAAIjE,SAAUjX,GACrD,MAAO,CAAElkB,MAAM,EAAM/I,WAAO,GAG7BitB,EAAS7J,QACV,MATG6J,EAAShmB,KAAwBG,KAAM,GACzCI,KAAKs3D,eAAiBn4D,EAYvB,OAFAa,KAAK0oC,UAAYjjB,EAEVkyC,GAAmB,eAAgBx4D,EAAMoqC,EAAkB9jB,EAAU,EAC7E,CAEA,GAAKtmB,aAAgB,GAAO,CAC3B,IAAIwqC,EAEJ,GAAK3pC,KAAK6oC,iBACTc,EAAkB,MACZ,CACN,IAAI/tB,EAASzc,EAAKirC,UAEbpqC,KAAKipC,oBAAsBrqC,GAAUoB,KAAKwoC,WAAY7H,IAAI/kB,OAASA,IACvEA,EAAS5b,KAAKwoC,WAAY7H,IAAI/kB,QAG/B+tB,EAAkB/tB,EAAS6J,EAAS7J,MACrC,CAEA,MAAMg8C,EAAmBnyC,EAAS7J,OAASzc,EAAK6qC,YAC1C/pC,EAAO,IAAI,GAAWd,EAAMy4D,EAAkBjuB,GAKpD,OAHAlkB,EAAS7J,QAAU+tB,EACnB3pC,KAAK0oC,UAAYjjB,EAEVkyC,GAAmB,OAAQ13D,EAAMspC,EAAkB9jB,EAAUkkB,EACrE,CAQA,OALElkB,EAAShmB,KAAwBT,MACnCymB,EAAS7J,SACT5b,KAAK0oC,UAAYjjB,EACjBzlB,KAAKs3D,eAAiB14D,EAAOA,OAExBoB,KAAK+oC,iBACF/oC,KAAKqpC,QAGNsuB,GAAmB,aAAc/4D,EAAmB2qC,EAAkB9jB,EAC9E,CAKQ,SAAA6jB,GACP,MAAMC,EAAmBvpC,KAAKylB,SACxBA,EAAWzlB,KAAKylB,SAAS1C,QACzBnkB,EAASoB,KAAKs3D,eAGpB,GAAuB,OAAlB14D,EAAOA,QAAuC,IAApB6mB,EAAS7J,OACvC,MAAO,CAAEra,MAAM,EAAM/I,WAAO,GAI7B,GAAKoG,GAAUoB,KAAKgpC,sBAAwBvjB,EAAS7J,QAAU5b,KAAKwoC,WAAY9H,MAAM9kB,OACrF,MAAO,CAAEra,MAAM,EAAM/I,WAAO,GAK7B,MAAMi9C,EAAiBhwB,EAAS7mB,OAC1B44D,EAAqBC,GAAuBhyC,EAAUgwB,GACtDt2C,EAAOq4D,GAAsBK,GAAuBpyC,EAAUgwB,EAAgB+hB,GAEpF,GAAKr4D,aAAgB,GAGpB,OAFAsmB,EAAS7J,SAEJ5b,KAAK8oC,SACT9oC,KAAK0oC,UAAYjjB,EAEVkyC,GAAmB,eAAgBx4D,EAAMoqC,EAAkB9jB,EAAU,KAG3EA,EAAShmB,KAAwBG,KAAMT,EAAKk3D,WAC9Cr2D,KAAK0oC,UAAYjjB,EACjBzlB,KAAKs3D,eAAiBn4D,EAEjBa,KAAK+oC,iBACF/oC,KAAKspC,YAGNquB,GAAmB,aAAcx4D,EAAMoqC,EAAkB9jB,IAGjE,GAAKtmB,aAAgB,GAAO,CAC3B,IAAIwqC,EAEJ,GAAK3pC,KAAK6oC,iBACTc,EAAkB,MACZ,CACN,IAAI/tB,EAASzc,EAAK6qC,YAEbhqC,KAAKgpC,sBAAwBpqC,GAAUoB,KAAKwoC,WAAY9H,MAAM9kB,OAASA,IAC3EA,EAAS5b,KAAKwoC,WAAY9H,MAAM9kB,QAGjC+tB,EAAkBlkB,EAAS7J,OAASA,CACrC,CAEA,MAAMg8C,EAAmBnyC,EAAS7J,OAASzc,EAAK6qC,YAC1C/pC,EAAO,IAAI,GAAWd,EAAMy4D,EAAmBjuB,EAAiBA,GAKtE,OAHAlkB,EAAS7J,QAAU+tB,EACnB3pC,KAAK0oC,UAAYjjB,EAEVkyC,GAAmB,OAAQ13D,EAAMspC,EAAkB9jB,EAAUkkB,EACrE,CAOA,OAJElkB,EAAShmB,KAAwBT,MACnCgB,KAAK0oC,UAAYjjB,EACjBzlB,KAAKs3D,eAAiB14D,EAAOA,OAEtB+4D,GAAmB,eAAgB/4D,EAAmB2qC,EAAkB9jB,EAAU,EAC1F,EAGD,SAASkyC,GACR1sD,EACAhL,EACAspC,EACAU,EACA3xC,GAEA,MAAO,CACNiJ,MAAM,EACN/I,MAAO,CACNyS,OACAhL,OACAspC,mBACAU,eACA3xC,UAGH,CC1Te,MAAM,WAAiB,GAmDrC,WAAAyJ,CACCzL,EACAmJ,EACA43D,EAAiC,UAIjC,GAFA7pD,SAEMlX,EAAK4X,GAAI,aAAgB5X,EAAK4X,GAAI,oBAQvC,MAAM,IAAI,EACT,8BACA5X,GAIF,KAAQmJ,aAAgBkB,QAA2B,IAAhBlB,EAAKnH,OAOvC,MAAM,IAAI,EACT,uCACAhC,EACA,CAAEmJ,SAKCnJ,EAAK4X,GAAI,eACbzO,EAAOA,EAAKqC,SAEZrC,EAAO,IAAKnJ,EAAK4lC,aAAcz8B,GAC/BnJ,EAAOA,EAAKA,MAGb0J,KAAK1J,KAAOA,EACZ0J,KAAKP,KAAOA,EACZO,KAAKq3D,WAAaA,CACnB,CAQA,UAAWz7C,GACV,OAAO5b,KAAKP,KAAMO,KAAKP,KAAKnH,OAAS,EACtC,CAEA,UAAWsjB,CAAQk8C,GAChB93D,KAAKP,KAAyBO,KAAKP,KAAKnH,OAAS,GAAMw/D,CAC1D,CAWA,UAAWl5D,GACV,IAAIA,EAAcoB,KAAK1J,KAEvB,QAAUkE,EAAI,EAAGA,EAAIwF,KAAKP,KAAKnH,OAAS,EAAGkC,IAG1C,GAFAoE,EAASA,EAAOi9B,SAAUj9B,EAAO83D,cAAe12D,KAAKP,KAAMjF,MAErDoE,EAgBL,MAAM,IAAI,EAAe,gCAAiCoB,KAAM,CAAEylB,SAAUzlB,OAI9E,GAAKpB,EAAOsP,GAAI,SACf,MAAM,IAAI,EAAe,gCAAiClO,KAAM,CAAEylB,SAAUzlB,OAG7E,OAAOpB,CACR,CAOA,SAAWgG,GACV,OAAO5E,KAAKpB,OAAO83D,cAAe12D,KAAK4b,OACxC,CAMA,YAAW6hB,GACV,OAAOg6B,GAAuBz3D,KAAMA,KAAKpB,OAC1C,CAKA,aAAWsrC,GAEV,MAAMtrC,EAASoB,KAAKpB,OAEpB,OAAO84D,GAAsB13D,KAAMpB,EAAQ64D,GAAuBz3D,KAAMpB,GACzE,CAKA,cAAWurC,GAEV,MAAMvrC,EAASoB,KAAKpB,OAEpB,OAAOi5D,GAAuB73D,KAAMpB,EAAQ64D,GAAuBz3D,KAAMpB,GAC1E,CAKA,aAAWkrC,GACV,OAAuB,IAAhB9pC,KAAK4b,MACb,CAKA,WAAW4tB,GACV,OAAOxpC,KAAK4b,QAAU5b,KAAKpB,OAAOy3D,SACnC,CAOO,WAAA1rB,CAAaD,GACnB,GAAK1qC,KAAK1J,MAAQo0C,EAAcp0C,KAC/B,MAAO,YAGR,MAAM6H,EAASuY,EAAe1W,KAAKP,KAAMirC,EAAcjrC,MAEvD,OAAStB,GACR,IAAK,OACJ,MAAO,OAER,IAAK,SACJ,MAAO,SAER,IAAK,YACJ,MAAO,QAER,QACC,OAAO6B,KAAKP,KAAMtB,GAAWusC,EAAcjrC,KAAMtB,GAAW,SAAW,QAE1E,CAyBO,uBAAAqsC,CACNtB,EACA3kC,EAA6B,CAAC,GAE9BA,EAAQkkC,cAAgBzoC,KAExB,MAAMyqC,EAAa,IAAI,GAAYlmC,GAGnC,OAFAkmC,EAAWvB,KAAMA,GAEVuB,EAAWhlB,QACnB,CAUO,aAAAsyC,GACN,OAAO/3D,KAAKP,KAAKqC,MAAO,GAAI,EAC7B,CAOO,YAAAq6B,GACN,MAAMv9B,EAASoB,KAAKpB,OAEpB,OAAKA,EAAOsP,GAAI,oBACR,CAAEtP,GAEFA,EAAOu9B,aAAc,CAAEE,aAAa,GAE7C,CAOO,YAAAiK,CAAc8wB,GACpB,MAAMx4D,EAASoB,KAAKpB,OAEpB,OAAKA,EAAOsP,GAAI,WACRtP,EAAO0nC,aAAc8wB,EAAY,CAAE/6B,aAAa,IAGjD,IACR,CAWO,aAAA27B,CAAevyC,GACrB,GAAKzlB,KAAK1J,MAAQmvB,EAASnvB,KAC1B,MAAO,GAIR,MAAMwT,EAAM4M,EAAe1W,KAAKP,KAAMgmB,EAAShmB,MAEzCw4D,EAAyB,iBAAPnuD,EAAoBjR,KAAKD,IAAKoH,KAAKP,KAAKnH,OAAQmtB,EAAShmB,KAAKnH,QAAWwR,EAEjG,OAAO9J,KAAKP,KAAKqC,MAAO,EAAGm2D,EAC5B,CAQO,iBAAA17B,CAAmB9W,GACzB,MAAM+W,EAAax8B,KAAKm8B,eAClBM,EAAahX,EAAS0W,eAE5B,IAAI3hC,EAAI,EAER,KAAQgiC,EAAYhiC,IAAOiiC,EAAYjiC,IAAOgiC,EAAYhiC,IACzDA,IAGD,OAAa,IAANA,EAAU,KAAOgiC,EAAYhiC,EAAI,EACzC,CAWO,YAAA8vC,CAAcne,GACpB,MAAMoe,EAAUvqC,KAAK+iB,QAEfnH,EAAS2uB,EAAQ3uB,OAASuQ,EAGhC,OAFAoe,EAAQ3uB,OAASA,EAAS,EAAI,EAAIA,EAE3B2uB,CACR,CAWO,OAAA1N,CAAS6N,GACf,MAA4C,SAArC1qC,KAAK2qC,YAAaD,EAC1B,CAyCO,QAAAhO,CAAUgO,GAChB,MAA4C,UAArC1qC,KAAK2qC,YAAaD,EAC1B,CAUO,OAAAzmB,CAASymB,GACf,MAA4C,QAArC1qC,KAAK2qC,YAAaD,EAC1B,CAUO,UAAAwtB,CAAYxtB,GAClB,GAAK1qC,KAAK1J,OAASo0C,EAAcp0C,KAChC,OAAO,EAGR,MAAM6hE,EAAct/D,KAAKD,IAAKoH,KAAKP,KAAKnH,OAAQoyC,EAAcjrC,KAAKnH,QAEnE,QAAU8/D,EAAQ,EAAGA,EAAQD,EAAaC,IAAU,CACnD,MAAM7+D,EAAOyG,KAAKP,KAAM24D,GAAU1tB,EAAcjrC,KAAM24D,GAGtD,GAAK7+D,GAAQ,GAAKA,EAAO,EACxB,OAAO,EACD,GAAc,IAATA,EAGX,OAAO8+D,GAAqB3tB,EAAe1qC,KAAMo4D,GAC3C,IAAe,IAAV7+D,EAGX,OAAO8+D,GAAqBr4D,KAAM0qC,EAAe0tB,EAInD,CAIA,OAAKp4D,KAAKP,KAAKnH,SAAWoyC,EAAcjrC,KAAKnH,SASnC0H,KAAKP,KAAKnH,OAASoyC,EAAcjrC,KAAKnH,OACxCggE,GAAiBt4D,KAAKP,KAAM04D,GAE5BG,GAAiB5tB,EAAcjrC,KAAM04D,GAE9C,CAUO,eAAAI,CAAiB9yC,GACvB,GAAKzlB,KAAK1J,OAASmvB,EAASnvB,KAC3B,OAAO,EAMR,MAAyD,QAAlDogB,EAHgB1W,KAAK+3D,gBACNtyC,EAASsyC,gBAGhC,CAeO,yBAAAS,CAA2BC,GACjC,IAAIt6D,EAEJ,OAASs6D,EAAUxtD,MAClB,IAAK,SACJ9M,EAAS6B,KAAK04D,iCAAkCD,GAChD,MACD,IAAK,OACL,IAAK,SACL,IAAK,WACJt6D,EAAS6B,KAAK24D,+BAAgCF,GAC9C,MACD,IAAK,QACJt6D,EAAS6B,KAAK44D,gCAAiCH,GAC/C,MACD,IAAK,QACJt6D,EAAS6B,KAAK64D,gCAAiCJ,GAC/C,MACD,QACCt6D,EAAS,GAASyqC,UAAW5oC,MAI/B,OAAO7B,CACR,CAOO,gCAAAu6D,CAAkCD,GACxC,OAAOz4D,KAAK84D,2BAA4BL,EAAUhzC,SAAUgzC,EAAUttD,QACvE,CAOO,8BAAAwtD,CAAgCF,GACtC,OAAOz4D,KAAK+4D,sBAAuBN,EAAUO,eAAgBP,EAAUpjB,eAAgBojB,EAAUttD,QAClG,CAOO,+BAAAytD,CAAiCH,GACvC,MAAMQ,EAAaR,EAAUQ,WAK7B,OAHoBA,EAAW1tB,iBAAkBvrC,OAC9Ci5D,EAAWv4B,MAAMzc,QAASjkB,OAA6B,UAAnBA,KAAKq3D,WAGpCr3D,KAAKk5D,aAAcT,EAAUU,cAAeV,EAAUW,oBAExDX,EAAUY,kBACPr5D,KAAK+4D,sBAAuBN,EAAUY,kBAAmBZ,EAAUngB,kBAAmB,GAEtFt4C,KAAK84D,2BAA4BL,EAAUngB,kBAAmB,EAGxE,CAOO,+BAAAugB,CAAiCJ,GACvC,MAAMQ,EAAaR,EAAUQ,WAG7B,IAAIl9B,EAeJ,OAjBoBk9B,EAAW1tB,iBAAkBvrC,OAAUi5D,EAAWv4B,MAAMzc,QAASjkB,OAKpF+7B,EAAM/7B,KAAKk5D,aAAcT,EAAUO,eAAgBP,EAAUpjB,gBAExDojB,EAAUO,eAAet8B,SAAU+7B,EAAUpjB,kBAEjDtZ,EAAMA,EAAIu9B,0BAA2Bb,EAAUc,iBAAkB,KAGlEx9B,EADW/7B,KAAKikB,QAASw0C,EAAUc,kBAC7B,GAAS3wB,UAAW6vB,EAAUc,kBAE9Bv5D,KAAK+4D,sBAAuBN,EAAUc,iBAAkBd,EAAUY,kBAAmB,GAGrFt9B,CACR,CAWO,yBAAAu9B,CAA2BE,EAA0BruD,GAC3D,MAAMsuD,EAAc,GAAS7wB,UAAW5oC,MAGxC,GAAKA,KAAK1J,MAAQkjE,EAAeljE,KAChC,OAAOmjE,EAGR,GAA8E,QAAzE/iD,EAAe8iD,EAAezB,gBAAiB/3D,KAAK+3D,kBAExD,GAAKyB,EAAe59C,OAAS5b,KAAK4b,OAAS,CAE1C,GAAK49C,EAAe59C,OAASzQ,EAAUnL,KAAK4b,OAE3C,OAAO,KAGP69C,EAAY79C,QAAUzQ,CAExB,OACM,GAA8E,UAAzEuL,EAAe8iD,EAAezB,gBAAiB/3D,KAAK+3D,iBAAgC,CAE/F,MAAMv9D,EAAIg/D,EAAe/5D,KAAKnH,OAAS,EAEvC,GAAKkhE,EAAe59C,QAAU5b,KAAKP,KAAMjF,GAAM,CAE9C,GAAKg/D,EAAe59C,OAASzQ,EAAUnL,KAAKP,KAAMjF,GAGjD,OAAO,KAGLi/D,EAAYh6D,KAAyBjF,IAAO2Q,CAEhD,CACD,CAEA,OAAOsuD,CACR,CAUO,0BAAAX,CAA4BY,EAA0BvuD,GAC5D,MAAMsuD,EAAc,GAAS7wB,UAAW5oC,MAGxC,GAAKA,KAAK1J,MAAQojE,EAAepjE,KAChC,OAAOmjE,EAGR,GAA8E,QAAzE/iD,EAAegjD,EAAe3B,gBAAiB/3D,KAAK+3D,kBAEnD2B,EAAe99C,OAAS5b,KAAK4b,QAAY89C,EAAe99C,QAAU5b,KAAK4b,QAA6B,cAAnB5b,KAAKq3D,cAG1FoC,EAAY79C,QAAUzQ,QAEjB,GAA8E,UAAzEuL,EAAegjD,EAAe3B,gBAAiB/3D,KAAK+3D,iBAAgC,CAE/F,MAAMv9D,EAAIk/D,EAAej6D,KAAKnH,OAAS,EAElCohE,EAAe99C,QAAU5b,KAAKP,KAAMjF,KAGtCi/D,EAAYh6D,KAAyBjF,IAAO2Q,EAEhD,CAEA,OAAOsuD,CACR,CAWO,qBAAAV,CAAuBC,EAA0B3jB,EAA0BlqC,GAIjF,GAFAkqC,EAAiBA,EAAeikB,0BAA2BN,EAAgB7tD,GAEtE6tD,EAAe/0C,QAASoxB,GAE5B,OAAO,GAASzM,UAAW5oC,MAI5B,MAAMy5D,EAAcz5D,KAAKs5D,0BAA2BN,EAAgB7tD,GAMpE,OAJgC,OAAhBsuD,GACbT,EAAe/0C,QAASjkB,OAA6B,UAAnBA,KAAKq3D,YACvC2B,EAAe1uB,aAAcn/B,GAAU8Y,QAASjkB,OAA6B,cAAnBA,KAAKq3D,WAK1Dr3D,KAAKk5D,aAAcF,EAAgB3jB,GAKnCokB,EAAYX,2BAA4BzjB,EAAgBlqC,EAEjE,CA+BO,YAAA+tD,CAAch2D,EAAkBU,GACtC,MAAMpJ,EAAI0I,EAAOzD,KAAKnH,OAAS,EAGzBqhE,EAAW,GAAS/wB,UAAWhlC,GAYrC,OAXA+1D,EAAStC,WAAar3D,KAAKq3D,WAK3BsC,EAAS/9C,OAAS+9C,EAAS/9C,OAAS5b,KAAKP,KAAMjF,GAAM0I,EAAO0Y,OAI1D+9C,EAAkBl6D,KAAO,IAAKk6D,EAASl6D,QAASO,KAAKP,KAAKqC,MAAOtH,EAAI,IAEhEm/D,CACR,CAKO,MAAA38B,GACN,MAAO,CACN1mC,KAAM0J,KAAK1J,KAAK0mC,SAChBv9B,KAAMkB,MAAMrB,KAAMU,KAAKP,MACvB43D,WAAYr3D,KAAKq3D,WAEnB,CAKO,KAAAt0C,GACN,OAAO,IAAM/iB,KAAK+B,YAAsB/B,KAAK1J,KAAM0J,KAAKP,KAAMO,KAAKq3D,WACpE,CAmBA,gBAAczuB,CACbkC,EACAlvB,EACAy7C,EAAiC,UAEjC,GAAKvsB,aAA0B,GAC9B,OAAO,IAAI,GAAUA,EAAex0C,KAAMw0C,EAAerrC,KAAMqrC,EAAeusB,YACxE,CACN,MAAMl4D,EAAO2rC,EAEb,GAAe,OAAVlvB,EACJA,EAAWzc,EAAck3D,cACnB,IAAe,UAAVz6C,EACX,OAAO5b,KAAK+pC,cAAe5qC,EAAMk4D,GAC3B,GAAe,SAAVz7C,EACX,OAAO5b,KAAKypC,aAActqC,EAAMk4D,GAC1B,GAAgB,IAAXz7C,IAAiBA,EAO5B,MAAM,IAAI,EAAe,yCAA0C,CAAE5b,KAAM8qC,GAC5E,CAEA,IAAM3rC,EAAK+O,GAAI,aAAgB/O,EAAK+O,GAAI,oBAMvC,MAAM,IAAI,EACT,kCACA,CAAElO,KAAM8qC,IAIV,MAAMrrC,EAAON,EAAK+8B,UAIlB,OAFAz8B,EAAKG,KAAMgc,GAEJ,IAAI5b,KAAMb,EAAK7I,KAAamJ,EAAM43D,EAC1C,CACD,CASA,mBAAc5tB,CAAcxpC,EAA+Bo3D,GAC1D,IAAMp3D,EAAKrB,OAOV,MAAM,IAAI,EACT,4BACA,CAAEoB,KAAMC,GACR,CAAE3J,KAAM2J,IAIV,OAAOD,KAAK4oC,UAAW3oC,EAAKrB,OAAQqB,EAAKmqC,UAAYitB,EACtD,CASA,oBAActtB,CAAe9pC,EAA+Bo3D,GAC3D,IAAMp3D,EAAKrB,OAOV,MAAM,IAAI,EACT,6BACAqB,EACA,CAAE3J,KAAM2J,IAIV,OAAOD,KAAK4oC,UAAW3oC,EAAKrB,OAAQqB,EAAK+pC,YAAcqtB,EACxD,CASA,eAAcJ,CAAUh6B,EAAW5lB,GAClC,GAAmB,eAAd4lB,EAAK3mC,KAAwB,CACjC,MAAMylC,EAAM,IAAI,GAAU1kB,EAAIuiD,UAAW38B,EAAKx9B,MAG9C,OAFAs8B,EAAIs7B,WAAap6B,EAAKo6B,WAEft7B,CACR,CAEA,IAAM1kB,EAAIy5B,QAAS7T,EAAK3mC,MAOvB,MAAM,IAAI,EACT,kCACA+gB,EACA,CAAE+d,SAAU6H,EAAK3mC,OAInB,OAAO,IAAI,GAAU+gB,EAAIy5B,QAAS7T,EAAK3mC,MAAS2mC,EAAKx9B,KAAMw9B,EAAKo6B,WACjE,EA4EM,SAASI,GAAuBhyC,EAAoBgwB,GAC1D,MAAMt2C,EAAOs2C,EAAe5Z,SAAU4Z,EAAeihB,cAAejxC,EAAS7J,SAE7E,OAAKzc,GAAQA,EAAK+O,GAAI,UAAa/O,EAAK6qC,YAAevkB,EAAS7J,OACxDzc,EAGD,IACR,CAwBO,SAASu4D,GACfjyC,EACAgwB,EACAhY,GAEA,OAAkB,OAAbA,EACG,KAGDgY,EAAe5Z,SAAU4Z,EAAeihB,cAAejxC,EAAS7J,QACxE,CAeO,SAASi8C,GACfpyC,EACAgwB,EACAhY,GAEA,OAAkB,OAAbA,EACG,KAGDgY,EAAe5Z,SAAU4Z,EAAeihB,cAAejxC,EAAS7J,QAAW,EACnF,CAeA,SAASy8C,GAAqBt2C,EAAgBJ,EAAiBy2C,GAC9D,OAAKA,EAAQ,IAAMr2C,EAAKtiB,KAAKnH,WAQvBggE,GAAiB32C,EAAMliB,KAAM24D,EAAQ,MA8C5C,SAA6Br8B,EAAeq8B,GAC3C,IAAIx5D,EAASm9B,EAAIn9B,OACbi7D,EAAM99B,EAAIt8B,KAAKnH,OAAS,EACxBuV,EAAM,EAEV,KAAQgsD,GAAOzB,GAAQ,CACtB,GAAKr8B,EAAIt8B,KAAMo6D,GAAQhsD,IAAQjP,EAAOy3D,UACrC,OAAO,EAMRxoD,EAAM,EACNgsD,IACAj7D,EAASA,EAAOA,MACjB,CAEA,OAAO,CACR,CAzDOk7D,CAAoB/3C,EAAMq2C,EAAQ,GAazC,CAOA,SAASE,GAAiB53D,EAA4Bm5D,GACrD,KAAQA,EAAMn5D,EAAIpI,QAAS,CAC1B,GAAoB,IAAfoI,EAAKm5D,GACT,OAAO,EAGRA,GACD,CAEA,OAAO,CACR,CAhMA,GAASh4D,UAAUqM,GAAK,SAAUjD,GACjC,MAAgB,aAATA,GAAgC,mBAATA,CAC/B,ECv9Be,MAAM,WAAc,GAiBlC,WAAAlJ,CAAa2+B,EAAiBC,GAC7BnzB,QAEAxN,KAAK0gC,MAAQ,GAASkI,UAAWlI,GACjC1gC,KAAK2gC,IAAMA,EAAM,GAASiI,UAAWjI,GAAQ,GAASiI,UAAWlI,GAIjE1gC,KAAK0gC,MAAM22B,WAAar3D,KAAKgrC,YAAc,SAAW,SACtDhrC,KAAK2gC,IAAI02B,WAAar3D,KAAKgrC,YAAc,SAAW,YACrD,CAcA,EAAUlqC,OAAOC,kBACT,IAAI,GAAY,CAAEynC,WAAYxoC,KAAM+oC,kBAAkB,GAC9D,CAMA,eAAWiC,GACV,OAAOhrC,KAAK0gC,MAAMzc,QAASjkB,KAAK2gC,IACjC,CAMA,UAAW7iB,GAIV,MAA0D,QAAnDpH,EAHiB1W,KAAK0gC,MAAMq3B,gBACb/3D,KAAK2gC,IAAIo3B,gBAGhC,CAKA,QAAWzhE,GACV,OAAO0J,KAAK0gC,MAAMpqC,IACnB,CASO,gBAAAi1C,CAAkB9lB,GACxB,OAAOA,EAASoX,QAAS78B,KAAK0gC,QAAWjb,EAASiX,SAAU18B,KAAK2gC,IAClE,CAWO,aAAA6K,CAAeF,EAAmBG,GAAiB,GACpDH,EAAWN,cACfS,GAAQ,GAGT,MAAMC,EAAgB1rC,KAAKurC,iBAAkBD,EAAW5K,QAAa+K,GAASzrC,KAAK0gC,MAAMzc,QAASqnB,EAAW5K,OACvGiL,EAAc3rC,KAAKurC,iBAAkBD,EAAW3K,MAAW8K,GAASzrC,KAAK2gC,IAAI1c,QAASqnB,EAAW3K,KAEvG,OAAO+K,GAAiBC,CACzB,CAKO,YAAAouB,CAAc95D,GACpB,MAAM87B,EAAM,GAASgO,cAAe9pC,GAEpC,OAAOD,KAAKurC,iBAAkBxP,IAAS/7B,KAAK0gC,MAAMzc,QAAS8X,EAC5D,CAQO,OAAA9X,CAASqnB,GACf,OAAOtrC,KAAK0gC,MAAMzc,QAASqnB,EAAW5K,QAAW1gC,KAAK2gC,IAAI1c,QAASqnB,EAAW3K,IAC/E,CAQO,cAAAmL,CAAgBR,GACtB,OAAOtrC,KAAK0gC,MAAMhE,SAAU4O,EAAW3K,MAAS3gC,KAAK2gC,IAAI9D,QAASyO,EAAW5K,MAC9E,CA6BO,aAAAkL,CAAeN,GACrB,MAAMO,EAAS,GAqBf,OAnBK7rC,KAAK8rC,eAAgBR,IAGpBtrC,KAAKurC,iBAAkBD,EAAW5K,QAGtCmL,EAAOjsC,KAAM,IAAI,GAAOI,KAAK0gC,MAAO4K,EAAW5K,QAG3C1gC,KAAKurC,iBAAkBD,EAAW3K,MAGtCkL,EAAOjsC,KAAM,IAAI,GAAO0rC,EAAW3K,IAAK3gC,KAAK2gC,OAI9CkL,EAAOjsC,KAAM,IAAI,GAAOI,KAAK0gC,MAAO1gC,KAAK2gC,MAGnCkL,CACR,CAuBO,eAAA3oB,CAAiBooB,GACvB,GAAKtrC,KAAK8rC,eAAgBR,GAAe,CAGxC,IAAIS,EAAmB/rC,KAAK0gC,MACxBsL,EAAiBhsC,KAAK2gC,IAc1B,OAZK3gC,KAAKurC,iBAAkBD,EAAW5K,SAGtCqL,EAAmBT,EAAW5K,OAG1B1gC,KAAKurC,iBAAkBD,EAAW3K,OAGtCqL,EAAiBV,EAAW3K,KAGtB,IAAI,GAAOoL,EAAkBC,EACrC,CAGA,OAAO,IACR,CAgCO,SAAAguB,CAAW1uB,EAAmBG,GAAiB,GACrD,IAAIwuB,EAAaj6D,KAAK8rC,eAAgBR,GAUtC,GARM2uB,IAEJA,EADIj6D,KAAK0gC,MAAMhE,SAAU4O,EAAW5K,OACvB+K,EAAQzrC,KAAK2gC,IAAIu3B,WAAY5sB,EAAW5K,OAAU1gC,KAAK2gC,IAAI1c,QAASqnB,EAAW5K,OAE/E+K,EAAQH,EAAW3K,IAAIu3B,WAAYl4D,KAAK0gC,OAAU4K,EAAW3K,IAAI1c,QAASjkB,KAAK0gC,SAIxFu5B,EACL,OAAO,KAGR,IAAIxxB,EAAgBzoC,KAAK0gC,MACrBgY,EAAc14C,KAAK2gC,IAUvB,OARK2K,EAAW5K,MAAMhE,SAAU+L,KAC/BA,EAAgB6C,EAAW5K,OAGvB4K,EAAW3K,IAAI9D,QAAS6b,KAC5BA,EAAcpN,EAAW3K,KAGnB,IAAI,GAAO8H,EAAeiQ,EAClC,CA6CO,oBAAAwhB,GACN,MAAMruB,EAAS,GACTosB,EAASj4D,KAAK0gC,MAAMs3B,cAAeh4D,KAAK2gC,KAAMroC,OAE9CyjC,EAAM,GAAS6M,UAAW5oC,KAAK0gC,OACrC,IAAIy5B,EAAYp+B,EAAIn9B,OAGpB,KAAQm9B,EAAIt8B,KAAKnH,OAAS2/D,EAAS,GAAI,CACtC,MAAM9sD,EAAUgvD,EAAU9D,UAAYt6B,EAAIngB,OAEzB,IAAZzQ,GACJ0gC,EAAOjsC,KAAM,IAAI,GAAOm8B,EAAKA,EAAIuO,aAAcn/B,KAG9C4wB,EAAat8B,KAAOs8B,EAAIt8B,KAAKqC,MAAO,GAAI,GAC1Ci6B,EAAIngB,SACJu+C,EAAYA,EAAUv7D,MACvB,CAGA,KAAQm9B,EAAIt8B,KAAKnH,QAAU0H,KAAK2gC,IAAIlhC,KAAKnH,QAAS,CACjD,MAAMsjB,EAAS5b,KAAK2gC,IAAIlhC,KAAMs8B,EAAIt8B,KAAKnH,OAAS,GAC1C6S,EAAUyQ,EAASmgB,EAAIngB,OAEZ,IAAZzQ,GACJ0gC,EAAOjsC,KAAM,IAAI,GAAOm8B,EAAKA,EAAIuO,aAAcn/B,KAGhD4wB,EAAIngB,OAASA,EACXmgB,EAAIt8B,KAAwBG,KAAM,EACrC,CAEA,OAAOisC,CACR,CAmBO,SAAAhB,CAAWtmC,EAA6B,CAAC,GAG/C,OAFAA,EAAQikC,WAAaxoC,KAEd,IAAI,GAAYuE,EACxB,CAeA,SAAQ2nC,CAAU3nC,EAA6B,CAAC,GAC/CA,EAAQikC,WAAaxoC,KACrBuE,EAAQwkC,kBAAmB,EAE3B,MAAM0B,EAAa,IAAI,GAAYlmC,GAEnC,UAAY/L,KAASiyC,QACdjyC,EAAMyH,IAEd,CAcA,aAAQksC,CAAc5nC,EAA6B,CAAC,GACnDA,EAAQikC,WAAaxoC,KAErB,MAAMyqC,EAAa,IAAI,GAAYlmC,SAE7BkmC,EAAWhlB,SAEjB,UAAYjtB,KAASiyC,QACdjyC,EAAMyxC,YAEd,CAYO,yBAAAuuB,CAA2BC,GACjC,OAASA,EAAUxtD,MAClB,IAAK,SACJ,OAAOjL,KAAK04D,iCAAkCD,GAC/C,IAAK,OACL,IAAK,SACL,IAAK,WACJ,OAAOz4D,KAAK24D,+BAAgCF,GAC7C,IAAK,QACJ,MAAO,CAAEz4D,KAAK44D,gCAAiCH,IAChD,IAAK,QACJ,MAAO,CAAEz4D,KAAK64D,gCAAiCJ,IAGjD,MAAO,CAAE,IAAI,GAAOz4D,KAAK0gC,MAAO1gC,KAAK2gC,KACtC,CASO,0BAAAy5B,CAA4BC,GAClC,MAAMxuB,EAAS,CAAE,IAAI,GAAO7rC,KAAK0gC,MAAO1gC,KAAK2gC,MAE7C,UAAY83B,KAAa4B,EACxB,QAAU7/D,EAAI,EAAGA,EAAIqxC,EAAOvzC,OAAQkC,IAAM,CACzC,MAAM2D,EAAS0tC,EAAQrxC,GAAIg+D,0BAA2BC,GAEtD5sB,EAAOrkC,OAAQhN,EAAG,KAAM2D,GACxB3D,GAAK2D,EAAO7F,OAAS,CACtB,CAOD,QAAUkC,EAAI,EAAGA,EAAIqxC,EAAOvzC,OAAQkC,IAAM,CACzC,MAAMsqB,EAAQ+mB,EAAQrxC,GAEtB,QAAU8/D,EAAI9/D,EAAI,EAAG8/D,EAAIzuB,EAAOvzC,OAAQgiE,IAAM,CAC7C,MAAMh5D,EAAOuqC,EAAQyuB,IAEhBx1C,EAAM0mB,cAAelqC,IAAUA,EAAKkqC,cAAe1mB,IAAWA,EAAMb,QAAS3iB,KACjFuqC,EAAOrkC,OAAQ8yD,EAAG,EAEpB,CACD,CAEA,OAAOzuB,CACR,CAMO,iBAAAtP,GACN,OAAOv8B,KAAK0gC,MAAMnE,kBAAmBv8B,KAAK2gC,IAC3C,CAOO,mBAAAsL,GACN,GAAKjsC,KAAKgrC,YACT,OAAO,KAGR,MAAMI,EAAiBprC,KAAK0gC,MAAMwJ,UAC5BmB,EAAgBrrC,KAAK2gC,IAAIwJ,WAE/B,OAAKiB,GAAkBA,EAAel9B,GAAI,YAAek9B,IAAmBC,EACpED,EAGD,IACR,CAOO,MAAApO,GACN,MAAO,CACN0D,MAAO1gC,KAAK0gC,MAAM1D,SAClB2D,IAAK3gC,KAAK2gC,IAAI3D,SAEhB,CAKO,KAAAja,GACN,OAAO,IAAM/iB,KAAK+B,YAAsB/B,KAAK0gC,MAAO1gC,KAAK2gC,IAC1D,CASO,gCAAA+3B,CAAkCD,EAA4B8B,GAAkB,GACtF,OAAOv6D,KAAK84D,2BAA4BL,EAAUhzC,SAAUgzC,EAAUttD,QAASovD,EAChF,CASO,8BAAA5B,CAAgCF,EAA0B8B,GAAkB,GAClF,MAAMvB,EAAiBP,EAAUO,eAC3B7tD,EAAUstD,EAAUttD,QACpBkqC,EAAiBojB,EAAUpjB,eAEjC,OAAOr1C,KAAK+4D,sBAAuBC,EAAgB3jB,EAAgBlqC,EAASovD,EAC7E,CASO,+BAAA3B,CAAiCH,GACvC,MAAM/3B,EAAQ1gC,KAAK0gC,MAAMk4B,gCAAiCH,GAC1D,IAAI93B,EAAM3gC,KAAK2gC,IAAIi4B,gCAAiCH,GAapD,OAXKz4D,KAAK2gC,IAAI1c,QAASw0C,EAAUngB,qBAChC3X,EAAM3gC,KAAK2gC,IAAI2J,aAAc,IAIzB5J,EAAMpqC,MAAQqqC,EAAIrqC,OAGtBqqC,EAAM3gC,KAAK2gC,IAAI2J,cAAe,IAGxB,IAAI,GAAO5J,EAAOC,EAC1B,CASO,+BAAAk4B,CAAiCJ,GAYvC,GAAKz4D,KAAK0gC,MAAMzc,QAASw0C,EAAUpjB,iBAAoBr1C,KAAK2gC,IAAI1c,QAASw0C,EAAUc,kBAClF,OAAO,IAAI,GAAOv5D,KAAK0gC,OAGxB,IAAIA,EAAQ1gC,KAAK0gC,MAAMm4B,gCAAiCJ,GACpD93B,EAAM3gC,KAAK2gC,IAAIk4B,gCAAiCJ,GASpD,OAPK/3B,EAAMpqC,MAAQqqC,EAAIrqC,OAItBqqC,EAAM3gC,KAAK2gC,IAAI2J,cAAe,IAG1B5J,EAAM7D,QAAS8D,IA2Bd83B,EAAUO,eAAet8B,SAAU+7B,EAAUpjB,iBAEjD3U,EAAQ,GAASkI,UAAWjI,GAC5BD,EAAM9kB,OAAS,IAET68C,EAAUc,iBAAiBt1C,QAASyc,KAEzCC,EAAM83B,EAAUc,kBAIjB74B,EAAQ+3B,EAAUpjB,gBAGZ,IAAI,GAAO3U,EAAOC,IAGnB,IAAI,GAAOD,EAAOC,EAC1B,CAkCO,0BAAAm4B,CAA4BY,EAA0BvuD,EAAiBovD,GAAkB,GAC/F,GAAKA,GAAUv6D,KAAKurC,iBAAkBmuB,GAKrC,MAAO,CACN,IAAI,GAAO15D,KAAK0gC,MAAOg5B,GACvB,IAAI,GACHA,EAAepvB,aAAcn/B,GAC7BnL,KAAK2gC,IAAIm4B,2BAA4BY,EAAgBvuD,KAGjD,CACN,MAAM2Z,EAAQ,IAAI,GAAO9kB,KAAK0gC,MAAO1gC,KAAK2gC,KAK1C,OAHE7b,EAAe4b,MAAQ5b,EAAM4b,MAAMo4B,2BAA4BY,EAAgBvuD,GAC/E2Z,EAAe6b,IAAM7b,EAAM6b,IAAIm4B,2BAA4BY,EAAgBvuD,GAEtE,CAAE2Z,EACV,CACD,CAaO,qBAAAi0C,CACNC,EACA3jB,EACAlqC,EACAovD,GAAkB,GAGlB,GAAKv6D,KAAKgrC,YAAc,CACvB,MAAMwvB,EAASx6D,KAAK0gC,MAAMq4B,sBAAuBC,EAAgB3jB,EAAgBlqC,GAEjF,MAAO,CAAE,IAAI,GAAOqvD,GACrB,CAcA,MAAMC,EAAY,GAAMluB,4BAA6BysB,EAAgB7tD,GAC/DuuD,EAAiBrkB,EAAeikB,0BAA2BN,EAAgB7tD,GAEjF,GAAKnL,KAAKurC,iBAAkB8J,KAAqBklB,IAC3CE,EAAUlvB,iBAAkBvrC,KAAK0gC,QAAW+5B,EAAUlvB,iBAAkBvrC,KAAK2gC,MAAQ,CACzF,MAAMD,EAAQ1gC,KAAK0gC,MAAMq4B,sBAAuBC,EAAgB3jB,EAAgBlqC,GAC1Ew1B,EAAM3gC,KAAK2gC,IAAIo4B,sBAAuBC,EAAgB3jB,EAAgBlqC,GAE5E,MAAO,CAAE,IAAI,GAAOu1B,EAAOC,GAC5B,CAID,IAAIxiC,EAEJ,MAAMu8D,EAAgB16D,KAAK4rC,cAAe6uB,GAC1C,IAAIE,EAAa,KAEjB,MAAMC,EAAS56D,KAAKkjB,gBAAiBu3C,GAsBrC,GApB6B,GAAxBC,EAAcpiE,OAElBqiE,EAAa,IAAI,GAChBD,EAAe,GAAIh6B,MAAM44B,0BAA2BN,EAAgB7tD,GACpEuvD,EAAe,GAAI/5B,IAAI24B,0BAA2BN,EAAgB7tD,IAEhC,GAAxBuvD,EAAcpiE,SAEzBqiE,EAAa,IAAI,GAChB36D,KAAK0gC,MACL1gC,KAAK2gC,IAAI24B,0BAA2BN,EAAgB7tD,KAKrDhN,EADIw8D,EACKA,EAAW7B,2BAA4BY,EAAiBvuD,EAAoB,OAAXyvD,GAAmBL,GAEpF,GAGLK,EAAS,CACb,MAAMC,EAAoB,IAAI,GAC7BD,EAAOl6B,MAAMw4B,aAAcuB,EAAU/5B,MAAOg5B,GAC5CkB,EAAOj6B,IAAIu4B,aAAcuB,EAAU/5B,MAAOg5B,IAGrB,GAAjBv7D,EAAO7F,OACX6F,EAAOqJ,OAAQ,EAAG,EAAGqzD,GAErB18D,EAAOyB,KAAMi7D,EAEf,CAEA,OAAO18D,CACR,CAcO,yBAAAm7D,CAA2BE,EAA0BruD,GAC3D,IAAI2vD,EAAW96D,KAAK0gC,MAAM44B,0BAA2BE,EAAgBruD,GACjE4vD,EAAS/6D,KAAK2gC,IAAI24B,0BAA2BE,EAAgBruD,GAEjE,OAAiB,MAAZ2vD,GAA8B,MAAVC,EACjB,MAGS,MAAZD,IACJA,EAAWtB,GAGG,MAAVuB,IACJA,EAASvB,GAGH,IAAI,GAAOsB,EAAUC,GAC7B,CAUA,kCAAcxuB,CAA6B9mB,EAAoB0G,GAC9D,MAAMuU,EAAQjb,EACRkb,EAAMlb,EAAS6kB,aAAcne,GAEnC,OAAOA,EAAQ,EAAI,IAAInsB,KAAM0gC,EAAOC,GAAQ,IAAI3gC,KAAM2gC,EAAKD,EAC5D,CASA,gBAAc8L,CAAW/0B,GACxB,OAAO,IAAIzX,KAAM,GAAS4oC,UAAWnxB,EAAS,GAAK,GAASmxB,UAAWnxB,EAASA,EAAQ4+C,WACzF,CAOA,gBAAc5pB,CAAWxsC,GACxB,OAAOD,KAAKusC,4BAA6B,GAASxC,cAAe9pC,GAAQA,EAAK09B,WAC/E,CAoBA,wBAAcq9B,CAAmBnvB,GAChC,GAAuB,IAAlBA,EAAOvzC,OAOX,MAAM,IAAI,EACT,uCACA,MAEK,GAAsB,GAAjBuzC,EAAOvzC,OAClB,OAAOuzC,EAAQ,GAAI9oB,QAMpB,MAAMk4C,EAAMpvB,EAAQ,GAGpBA,EAAO1qB,MAAM,CAAExlB,EAAGhD,IACVgD,EAAE+kC,MAAM7D,QAASlkC,EAAE+nC,OAAU,GAAK,IAI1C,MAAMw6B,EAAWrvB,EAAO7iC,QAASiyD,GAK3B98D,EAAS,IAAI6B,KAAMi7D,EAAIv6B,MAAOu6B,EAAIt6B,KAIxC,GAAKu6B,EAAW,EAEf,QAAU1gE,EAAI0gE,EAAW,EACnBrvB,EAAQrxC,GAAImmC,IAAI1c,QAAS9lB,EAAOuiC,OADJlmC,IAE9B2D,EAAgBuiC,MAAQ,GAASkI,UAAWiD,EAAQrxC,GAAIkmC,OAU7D,QAAUlmC,EAAI0gE,EAAW,EAAG1gE,EAAIqxC,EAAOvzC,QACjCuzC,EAAQrxC,GAAIkmC,MAAMzc,QAAS9lB,EAAOwiC,KADOnmC,IAE3C2D,EAAgBwiC,IAAM,GAASiI,UAAWiD,EAAQrxC,GAAImmC,KAO1D,OAAOxiC,CACR,CASA,eAAc84D,CAAUh6B,EAAW5lB,GAClC,OAAO,IAAIrX,KAAM,GAASi3D,SAAUh6B,EAAKyD,MAAOrpB,GAAO,GAAS4/C,SAAUh6B,EAAK0D,IAAKtpB,GACrF,EAaD,GAAMxV,UAAUqM,GAAK,SAAUjD,GAC9B,MAAgB,UAATA,GAA6B,gBAATA,CAC5B,EC99Be,MAAMkwD,WAAe/rD,KA8CnC,WAAArN,GACCyL,QA3CD,KAAQ4tD,oBAAsB,IAAI/qC,QAKlC,KAAQgrC,oBAAsB,IAAIhrC,QAMlC,KAAQirC,4BAA8B,IAAI/pD,IAQ1C,KAAQgqD,sBAAwB,IAAIhqD,IAOpC,KAAQiqD,sBAAwB,IAAIjqD,IAKpC,KAAQkqD,yBAA2B,IAAIlqD,IAMvC,KAAQmqD,oBAAsB,IAAI9nD,IASjC5T,KAAKqP,GAAoC,uBAAuB,CAAEgG,EAAKvS,KACtE,GAAKA,EAAK6vC,aACT,OAGD,MAAMgpB,EAAgB37D,KAAKo7D,oBAAoBhzD,IAAKtF,EAAK84D,cAAch9D,QAEvE,IAAM+8D,EASL,MAAM,IAAI,EAAe,+CAAgD37D,KAAM,CAAE47D,cAAe94D,EAAK84D,gBAGtG94D,EAAK6vC,aAAe3yC,KAAK67D,eAAgBF,EAAe74D,EAAK84D,cAAchgD,OAAQ,GACjF,CAAEhP,SAAU,QAGf5M,KAAKqP,GAAoC,uBAAuB,CAAEgG,EAAKvS,KACtE,GAAKA,EAAK84D,cACT,OAGD,MAAME,EAAY97D,KAAK+7D,uBAAwBj5D,EAAK6vC,cAC9CqpB,EAAch8D,KAAKq7D,oBAAoBjzD,IAAK0zD,GAC5CG,EAAcj8D,KAAKk8D,eAAgBp5D,EAAK6vC,aAAa/zC,OAAuBkE,EAAK6vC,aAAa/2B,OAAQkgD,GAE5Gh5D,EAAK84D,cAAgB,GAAchzB,UAAWozB,EAAcC,EAAa,GACvE,CAAErvD,SAAU,OAChB,CAWO,YAAAyxC,CACN8d,EACA5kB,GAEAv3C,KAAKo7D,oBAAoB3pD,IAAK0qD,EAAc5kB,GAC5Cv3C,KAAKq7D,oBAAoB5pD,IAAK8lC,EAAa4kB,EAC5C,CAgBO,iBAAAC,CACN7kB,EACAhzC,EAA+B,CAAC,GAEhC,MAAM43D,EAAen8D,KAAKq8D,eAAgB9kB,GAE1C,GAAKv3C,KAAKw7D,sBAAsB5tD,IAAK2pC,GACpC,UAAY+kB,KAAct8D,KAAKw7D,sBAAsBpzD,IAAKmvC,GACzDv3C,KAAK07D,oBAAoB7tD,IAAKyuD,GAI3B/3D,EAAQg4D,MACZv8D,KAAKy7D,yBAAyBhqD,IAAK8lC,EAAaA,EAAYjhD,OAE5D0J,KAAKq7D,oBAAoB1pD,OAAQ4lC,GAE5Bv3C,KAAKo7D,oBAAoBhzD,IAAK+zD,IAAkB5kB,GACpDv3C,KAAKo7D,oBAAoBzpD,OAAQwqD,GAGpC,CAaO,kBAAAK,CAAoBL,GAC1B,MAAM5kB,EAAcv3C,KAAKy8D,cAAeN,GAExCn8D,KAAKo7D,oBAAoBzpD,OAAQwqD,GAE5Bn8D,KAAKq7D,oBAAoBjzD,IAAKmvC,IAAiB4kB,GACnDn8D,KAAKq7D,oBAAoB1pD,OAAQ4lC,EAEnC,CASO,mBAAAmlB,CAAqBjlD,EAAsBzV,GACjD,MAAM26D,EAAW38D,KAAKu7D,sBAAsBnzD,IAAKpG,IAAU,IAAI4R,IAC/D+oD,EAAS9uD,IAAK4J,GAEd,MAAMkH,EAAQ3e,KAAKw7D,sBAAsBpzD,IAAKqP,IAAa,IAAI7D,IAC/D+K,EAAM9Q,IAAK7L,GAEXhC,KAAKu7D,sBAAsB9pD,IAAKzP,EAAM26D,GACtC38D,KAAKw7D,sBAAsB/pD,IAAKgG,EAASkH,EAC1C,CAQO,2BAAAi+C,CAA6BnlD,EAAsBzV,GACzD,MAAM66D,EAAiB78D,KAAKu7D,sBAAsBnzD,IAAKpG,GAElD66D,IACJA,EAAelrD,OAAQ8F,GAEK,GAAvBolD,EAAehpD,MACnB7T,KAAKu7D,sBAAsB5pD,OAAQ3P,IAIrC,MAAM86D,EAAiB98D,KAAKw7D,sBAAsBpzD,IAAKqP,GAElDqlD,IACJA,EAAenrD,OAAQ3P,GAEK,GAAvB86D,EAAejpD,MACnB7T,KAAKw7D,sBAAsB7pD,OAAQ8F,GAGtC,CAMO,uBAAAslD,GACN,MAAMC,EAAcr8D,MAAMrB,KAAMU,KAAK07D,qBAIrC,OAFA17D,KAAK07D,oBAAoB9pD,QAElBorD,CACR,CAOO,qBAAAC,GACN,UAAc1lB,EAAajhD,KAAU0J,KAAKy7D,yBAEpClkB,EAAYjhD,MAAQA,GACxB0J,KAAKo8D,kBAAmB7kB,GAI1Bv3C,KAAKy7D,yBAA2B,IAAIlqD,GACrC,CAKO,aAAA2rD,GACNl9D,KAAKo7D,oBAAsB,IAAI/qC,QAC/BrwB,KAAKq7D,oBAAsB,IAAIhrC,QAC/BrwB,KAAKu7D,sBAAwB,IAAIhqD,IACjCvR,KAAKw7D,sBAAwB,IAAIjqD,IACjCvR,KAAK07D,oBAAsB,IAAI9nD,IAC/B5T,KAAKy7D,yBAA2B,IAAIlqD,GACrC,CAsBO,cAAA8qD,CAAgB9kB,GACtB,OAAOv3C,KAAKq7D,oBAAoBjzD,IAAKmvC,EACtC,CAoBO,aAAAklB,CAAeN,GACrB,OAAOn8D,KAAKo7D,oBAAoBhzD,IAAK+zD,EACtC,CAQO,YAAAgB,CAAc1W,GACpB,OAAO,IAAI,GAAYzmD,KAAKo9D,gBAAiB3W,EAAU/lB,OAAS1gC,KAAKo9D,gBAAiB3W,EAAU9lB,KACjG,CAQO,WAAA08B,CAAaC,GACnB,OAAO,IAAI,GAAWt9D,KAAKu9D,eAAgBD,EAAW58B,OAAS1gC,KAAKu9D,eAAgBD,EAAW38B,KAChG,CASO,eAAAy8B,CAAiBzqB,GACvB,MAAM7vC,EAAsD,CAC3D6vC,eACA6qB,OAAQx9D,MAKT,OAFAA,KAAKwQ,KAAsC,sBAAuB1N,GAE3DA,EAAK84D,aACb,CAYO,cAAA2B,CACN3B,EACAr3D,EAAmC,CAAC,GAEpC,MAAMzB,EAAsD,CAC3D84D,gBACA4B,OAAQx9D,KACRy9D,UAAWl5D,EAAQk5D,WAKpB,OAFAz9D,KAAKwQ,KAAsC,sBAAuB1N,GAE3DA,EAAK6vC,YACb,CASO,oBAAA+qB,CAAsB17D,GAC5B,MAAM27D,EAAgB39D,KAAKu7D,sBAAsBnzD,IAAKpG,GAEtD,IAAM27D,EACL,OAAO,KAGR,MAAMhB,EAAW,IAAI/oD,IAErB,UAAY6D,KAAWkmD,EACtB,GAAKlmD,EAAQvJ,GAAI,oBAChB,UAAY6U,KAAStL,EAAQ65B,wBAC5BqrB,EAAS9uD,IAAKkV,QAGf45C,EAAS9uD,IAAK4J,GAIhB,OAAOklD,CACR,CAiCO,yBAAAiB,CACNC,EACAC,GAEA99D,KAAKs7D,4BAA4B7pD,IAAKosD,EAAiBC,EACxD,CAQO,sBAAA/B,CAAwBppB,GAC9B,IAAI/zC,EAAc+zC,EAAa/zC,OAE/B,MAASoB,KAAKq7D,oBAAoBztD,IAAKhP,IACtCA,EAASA,EAAOA,OAGjB,OAAOA,CACR,CAuBQ,cAAAs9D,CACPpV,EACAiX,EACAjC,GAEA,GAAKA,GAAahV,EAAa,CAK9B,OAH4B9mD,KAAKk8D,eAAgBpV,EAAWloD,OAAekoD,EAAWliD,MAAQk3D,GACvE97D,KAAKk8D,eAAgBpV,EAAYiX,EAAYjX,EAGrE,CAKA,GAAKA,EAAW54C,GAAI,SACnB,OAAO6vD,EAIR,IAAI9B,EAAc,EAElB,QAAUzhE,EAAI,EAAGA,EAAIujE,EAAYvjE,IAChCyhE,GAAej8D,KAAKg+D,eAAgBlX,EAAWjrB,SAAUrhC,IAG1D,OAAOyhE,CACR,CA0BO,cAAA+B,CAAgBzd,GACtB,GAAKvgD,KAAKs7D,4BAA4BlzD,IAAOm4C,EAAkBv+C,MAAS,CAGvE,OAFiBhC,KAAKs7D,4BAA4BlzD,IAAOm4C,EAAkBv+C,KAEpEuN,CAAUgxC,EAClB,CAAO,GAAKvgD,KAAKq7D,oBAAoBztD,IAAK2yC,GACzC,OAAO,EACD,GAAKA,EAASryC,GAAI,SACxB,OAAOqyC,EAASz9C,KAAKxK,OACf,GAAKioD,EAASryC,GAAI,aACxB,OAAO,EACD,CACN,IAAI9P,EAAM,EAEV,UAAYuZ,KAAW4oC,EAA0B/kB,cAChDp9B,GAAO4B,KAAKg+D,eAAgBrmD,GAG7B,OAAOvZ,CACR,CACD,CA4BO,cAAAy9D,CAAgB/U,EAA6CmX,GAEnE,IAAI1d,EAEA2d,EAAa,EAEbjC,EAAc,EACd8B,EAAa,EAGjB,GAAKjX,EAAW54C,GAAI,SACnB,OAAO,IAAI,GAAc44C,EAAYmX,GAMtC,KAAQhC,EAAcgC,GACrB1d,EAAauG,EAAmDjrB,SAAUkiC,GAC1EG,EAAal+D,KAAKg+D,eAAgBzd,GAClC0b,GAAeiC,EACfH,IAID,OAAK9B,GAAegC,EACZj+D,KAAKm+D,4BAA6B,IAAI,GAAcrX,EAAYiX,IAMhE/9D,KAAK67D,eAAgBtb,EAAW0d,GAAmBhC,EAAciC,GAE1E,CAeQ,2BAAAC,CAA6BxrB,GAGpC,MAAMxI,EAAawI,EAAaxI,WAC1BD,EAAYyI,EAAazI,UAE/B,OAAKC,aAAsB,GACnB,IAAI,GAAcA,EAAYA,EAAWrnC,KAAKxK,QAC1C4xC,aAAqB,GACzB,IAAI,GAAcA,EAAW,GAI9ByI,CACR,EC1jBc,MAAMyrB,GAArB,cAIC,KAAQC,YAAc,IAAI9sD,IAU1B,KAAQ+sD,mBAAqB,IAAI/sD,GAA6D,CAiBvF,GAAA1D,CACN5N,EACAgL,GAEAA,EAAOszD,GAA0BtzD,GAE5BhL,aAAgB,KACpBA,EAAOD,KAAKw+D,uBAAwBv+D,IAG/BD,KAAKq+D,YAAYzwD,IAAK3N,IAC3BD,KAAKq+D,YAAY5sD,IAAKxR,EAAM,IAAIsR,KAGjCvR,KAAKq+D,YAAYj2D,IAAKnI,GAAQwR,IAAKxG,GAAM,EAC1C,CAkBO,OAAAwzD,CACNx+D,EACAgL,GAQA,OANAA,EAAOszD,GAA0BtzD,GAE5BhL,aAAgB,KACpBA,EAAOD,KAAKw+D,uBAAwBv+D,MAGhCD,KAAKiC,KAAMhC,EAAMgL,KACrBjL,KAAKq+D,YAAYj2D,IAAKnI,GAAQwR,IAAKxG,GAAM,IAElC,EAIT,CAmBO,IAAAhJ,CACNhC,EACAgL,GAEAA,EAAOszD,GAA0BtzD,GAE5BhL,aAAgB,KACpBA,EAAOD,KAAKw+D,uBAAwBv+D,IAGrC,MAAMy+D,EAAkB1+D,KAAKq+D,YAAYj2D,IAAKnI,GAE9C,QAAyB,IAApBy+D,EACJ,OAAO,KAGR,MAAMlmE,EAAQkmE,EAAgBt2D,IAAK6C,GAEnC,YAAe,IAAVzS,EACG,KAGDA,CACR,CAkBO,MAAAmmE,CACN1+D,EACAgL,GAEAA,EAAOszD,GAA0BtzD,GAE5BhL,aAAgB,KACpBA,EAAOD,KAAKw+D,uBAAwBv+D,IAGrC,MAAMgC,EAAOjC,KAAKiC,KAAMhC,EAAMgL,GAE9B,OAAc,IAAThJ,GACJjC,KAAKq+D,YAAYj2D,IAAKnI,GAAQwR,IAAKxG,GAAM,IAElC,IACa,IAAThJ,GAIL,IACR,CAOO,iBAAA28D,CAAmBC,GACzB,MAAMjuC,EAAQ,GAEd,UAAc3wB,EAAM6+D,KAAiB9+D,KAAKq+D,YACzC,UAAc/uD,EAAOyvD,KAAgBD,EAAc,CAClD,MAAME,EAAc1vD,EAAMvS,MAAO,KAAO,GAEnCgiE,GAAcF,GAAcG,GAChCpuC,EAAMhxB,KAAM,CACX0P,QACArP,KAAMA,EAAK+B,MAAQ/B,EAAKg/D,aAG3B,CAGD,GAAKruC,EAAMt4B,OAiBV,MAAM,IAAI,EAAe,2CAA4C,KAAM,CAAEs4B,SAE/E,CAYO,sBAAA4tC,CAAwB30B,GAC9B,IAAIluB,EAAS,KAEb,MAAMujD,EAAWl/D,KAAKs+D,mBAAmBl2D,IAAKyhC,EAAUG,aAExD,GAAKk1B,EAAW,CACf,MAAMC,EAASD,EAAS92D,IAAKyhC,EAAUO,WAElC+0B,IACJxjD,EAASwjD,EAAO/2D,IAAKyhC,EAAUjrC,QAEjC,CAMA,OAJM+c,IACLA,EAAS3b,KAAKo/D,uBAAwBv1B,IAGhCluB,CACR,CAUQ,sBAAAyjD,CAAwBv1B,GAC/B,MAAMnJ,EAAQmJ,EAAUG,YAClBrJ,EAAMkJ,EAAUO,UAChBxrC,EAASirC,EAAUjrC,OAEnB+c,EAAS7a,OAAQ,cAAgB+oC,EAAU/mC,MACjD,IAAIo8D,EACAC,EAkBJ,OAhBAD,EAAWl/D,KAAKs+D,mBAAmBl2D,IAAKs4B,GAElCw+B,IACLA,EAAW,IAAI3tD,IACfvR,KAAKs+D,mBAAmB7sD,IAAKivB,EAAOw+B,IAGrCC,EAASD,EAAS92D,IAAKu4B,GAEjBw+B,IACLA,EAAS,IAAI5tD,IACb2tD,EAASztD,IAAKkvB,EAAKw+B,IAGpBA,EAAO1tD,IAAK7S,EAAQ+c,GAEbA,CACR,EAWD,SAAS4iD,GAA0BtzD,GAClC,MAAM4T,EAAQ5T,EAAKlO,MAAO,KAG1B,MAAmB,UAAd8hB,EAAO,GACJA,EAAO,GAII,aAAdA,EAAO,IAAoC,gBAAdA,EAAO,GACjC5T,EAGD4T,EAAMvmB,OAAS,EAAIumB,EAAO,GAAM,IAAMA,EAAO,GAAMA,EAAO,EAClE,C,+aClRe,MAAMwgD,WAA2BjwD,KAqB/C,WAAArN,CAAau9D,GACZ9xD,QAEAxN,KAAKu/D,eAAiB,IAAEC,WAAYx/D,MAASs/D,GAC7Ct/D,KAAKy/D,gBAAkB,IAAIpvC,OAC5B,CAgBO,cAAAqvC,CACNC,EACAC,EACAplC,GAEA,MAAM8kC,EAAgBt/D,KAAK6/D,qBAAsBrlC,EAAQmlC,EAAOG,qBAGhE,UAAYvlC,KAAUolC,EAAOI,qBAC5B//D,KAAKggE,qBAAsBzlC,EAAOv4B,KAAMu4B,EAAOzV,MAAOw6C,GAIvD,MAAMW,EAAUjgE,KAAKkgE,eAAgBP,EAAOQ,cAG5C,UAAYznD,KAASunD,EACA,WAAfvnD,EAAMzN,KACVjL,KAAKogE,eAAgB,GAAM7zB,4BAA6B7zB,EAAM+M,SAAU/M,EAAMpgB,QAAUgnE,GAC9D,aAAf5mD,EAAMzN,KACjBjL,KAAKqgE,iBAAkB,GAAM9zB,4BAA6B7zB,EAAM+M,SAAU/M,EAAMpgB,QAAUgnE,GAChE,WAAf5mD,EAAMzN,KACjBjL,KAAKsgE,eAAgB5nD,EAAM+M,SAAU/M,EAAMpgB,OAAQogB,EAAM1W,KAAMs9D,GAG/Dt/D,KAAKugE,kBAAmB7nD,EAAMoM,MAAOpM,EAAM+rC,aAAc/rC,EAAM8nD,kBAAmB9nD,EAAM+nD,kBAAmBnB,GAI7G,UAAYhD,KAAcgD,EAAc9B,OAAOT,0BAA4B,CAC1E,MAAM2D,EAAcd,EAAQx3D,IAAKk0D,GAAcqE,WAE/C3gE,KAAKggE,qBAAsB1D,EAAYoE,EAAapB,GACpDt/D,KAAK4gE,kBAAmBtE,EAAYoE,EAAapB,EAClD,CAGA,UAAY/kC,KAAUolC,EAAOkB,kBAC5B7gE,KAAK4gE,kBAAmBrmC,EAAOv4B,KAAMu4B,EAAOzV,MAAOw6C,GAIpDA,EAAc9B,OAAOP,wBAGrBqC,EAAcwB,WAAWlC,kBAAmB,SAC7C,CAaO,OAAA1nE,CACN4tB,EACA86C,EACAplC,EACAj2B,EAA8C,CAAC,GAE/C,MAAM+6D,EAAgBt/D,KAAK6/D,qBAAsBrlC,OAAQ,EAAWj2B,GAEpEvE,KAAKogE,eAAgBt7C,EAAOw6C,GAE5B,UAAct9D,EAAM8iB,KAAW86C,EAC9B5/D,KAAK4gE,kBAAmB5+D,EAAM8iB,EAAOw6C,GAItCA,EAAcwB,WAAWlC,kBAAmB,SAC7C,CAcO,gBAAAmC,CACN9rC,EACA2qC,EACAplC,GAEA,MAAM8kC,EAAgBt/D,KAAK6/D,qBAAsBrlC,GAGjDx6B,KAAKwQ,KAAmC,iBAAkB,CAAEykB,aAAaqqC,GAGzE,MAAM0B,EAAY/rC,EAAUE,mBAAoB7+B,KAEhD,IAAMgpE,EAAc9B,OAAOf,cAAeuE,GACzC,OAID,MAAMC,EAAqBtgE,MAAMrB,KAAMsgE,EAAQsB,qBAAsBjsC,EAAUE,qBAM/E,GAJAn1B,KAAKmhE,4BAA6B7B,EAAcwB,WAAY7rC,EAAWgsC,GAEvEjhE,KAAKwQ,KAA8B,YAAa,CAAEykB,aAAaqqC,GAEzDrqC,EAAU+V,YAAhB,CAIA,UAAYo2B,KAAUH,EAErB,GAAK3B,EAAcwB,WAAW7+D,KAAMgzB,EAAW,aAAemsC,EAAOp/D,MAAS,CAC7E,MAAM0+D,EAAcU,EAAOT,WAE3B,IAAMU,GAA+BpsC,EAAUE,mBAAqBisC,EAAQ9B,EAAc9B,QACzF,SAGD,MAAM16D,EAAO,CACZ7C,KAAMg1B,EACNqnC,WAAY8E,EAAOp/D,KACnB0+D,eAGD1gE,KAAKwQ,KAA8B,aAAc4wD,EAAOp/D,OAASc,EAAMw8D,EACxE,CAGD,UAAYvoE,KAAOk+B,EAAUqJ,mBAE5B,GAAKghC,EAAcwB,WAAW7+D,KAAMgzB,EAAW,aAAel+B,GAAQ,CACrE,MAAM+L,EAAO,CACZ7C,KAAMg1B,EACNnQ,MAAOmQ,EAAUoY,gBACjBoX,aAAc1tD,EACdypE,kBAAmB,KACnBC,kBAAmBxrC,EAAUoG,aAActkC,IAG5CiJ,KAAKwQ,KAA8B,aAAczZ,UAAc+L,EAAMw8D,EACtE,CAjCD,CAmCD,CAeQ,cAAAc,CACPt7C,EACAw6C,EACA/6D,EAA6C,CAAC,GAExCA,EAAQ+8D,qBAEbthE,KAAKuhE,yBAA0BjC,EAAcwB,WAAYngE,MAAMrB,KAAMwlB,IAItE,UAAYhiB,KAAQnC,MAAMrB,KAAMwlB,EAAM+lB,UAAW,CAAE/B,SAAS,KAAW9rC,IAAKwkE,IAC3ExhE,KAAKyhE,aAAc,SAAU3+D,EAAMw8D,EAErC,CAUQ,cAAAgB,CACP76C,EACAntB,EACA0J,EACAs9D,GAEAt/D,KAAKwQ,KAA2B,UAAWxO,IAAS,CAAEyjB,WAAUntB,UAAUgnE,EAC3E,CAcQ,iBAAAiB,CACPz7C,EACA/tB,EACAyc,EACAC,EACA6rD,GAGAt/D,KAAK0hE,wBAAyBpC,EAAcwB,WAAYh8C,EAAO,aAAc/tB,KAG7E,UAAYyB,KAASssB,EAAQ,CAC5B,MAAMhiB,EAAO,CACZ7C,KAAMzH,EAAMyH,KACZ6kB,MAAO,GAAMynB,4BAA6B/zC,EAAM+wC,iBAAkB/wC,EAAMF,QACxEmsD,aAAc1tD,EACdypE,kBAAmBhtD,EACnBitD,kBAAmBhtD,GAGpBzT,KAAKyhE,aAAc,aAAc1qE,IAAQ+L,EAAMw8D,EAChD,CACD,CAcQ,gBAAAe,CAAkBv7C,EAAcw6C,GAEvC,MAAMqC,EAAehhE,MAAMrB,KAAMwlB,EAAM+lB,UAAW,CAAE/B,SAAS,KAG7D9oC,KAAKuhE,yBAA0BjC,EAAcwB,WAAYa,GAGzD,UAAY7+D,KAAQ6+D,EAAa3kE,IAAKwkE,IACrCxhE,KAAKyhE,aAAc,SAAU,SAAK3+D,GAAL,CAAW8+D,cAAc,IAAQtC,EAEhE,CAWQ,iBAAAsB,CACPtE,EACAoE,EACApB,GAGA,GAAkC,cAA7BoB,EAAYpqE,KAAK8+B,SACrB,OAID,MAAM5jB,EAAY,aAAc8qD,IAahC,GARAgD,EAAcwB,WAAWjzD,IAAK6yD,EAAalvD,GAE3CxR,KAAKwQ,KAA8BgB,EAAW,CAAE8qD,aAAYoE,eAAepB,GAMrEA,EAAcwB,WAAWrC,QAASiC,EAAalvD,GAArD,CAOAxR,KAAK0hE,wBAAyBpC,EAAcwB,WAAYJ,EAAalvD,GAErE,UAAYvR,KAAQygE,EAAYx0B,WAAa,CAE5C,IAAMozB,EAAcwB,WAAW7+D,KAAMhC,EAAMuR,GAC1C,SAGD,MAAM1O,EAAO,CAAE7C,OAAM6kB,MAAO,GAAM2nB,UAAWxsC,GAAQq8D,aAAYoE,eAEjE1gE,KAAKwQ,KAA8BgB,EAAW1O,EAAMw8D,EACrD,CAhBA,CAiBD,CAUQ,oBAAAU,CAAsB1D,EAAoBoE,EAAoBpB,GAEnC,cAA7BoB,EAAYpqE,KAAK8+B,UAItBp1B,KAAKwQ,KAAiC,gBAAiB8rD,IAAe,CAAEA,aAAYoE,eAAepB,EACpG,CAWQ,cAAAY,CAAgBD,GACvB,MAAMn9D,EAA2D,CAAEm9D,WAInE,OAFAjgE,KAAKwQ,KAAkC,gBAAiB1N,GAEjDA,EAAKm9D,OACb,CAUQ,wBAAAsB,CACPT,EACAa,GAEA,UAAYnpE,KAASmpE,EAAe,CACnC,MAAM1hE,EAAOzH,EAAMyH,KAGnB,GAA2C,OAAtC6gE,EAAW7+D,KAAMhC,EAAM,UAAsB,CACjD6gE,EAAWjzD,IAAK5N,EAAM,UAEtB,UAAYlJ,KAAOkJ,EAAKq+B,mBACvBwiC,EAAWjzD,IAAK5N,EAAM,aAAelJ,EAEvC,CACD,CAEA,OAAO+pE,CACR,CAUQ,uBAAAY,CACPZ,EACAh8C,EACA7Z,GAEA,UAAYhL,KAAQ6kB,EAAMonB,WACzB40B,EAAWjzD,IAAK5N,EAAMgL,GAGvB,OAAO61D,CACR,CAUQ,2BAAAK,CACPL,EACA7rC,EACA2qC,GAEAkB,EAAWjzD,IAAKonB,EAAW,aAE3B,UAAYmsC,KAAUxB,EACrBkB,EAAWjzD,IAAKonB,EAAW,aAAemsC,EAAOp/D,MAGlD,UAAYjL,KAAOk+B,EAAUqJ,mBAC5BwiC,EAAWjzD,IAAKonB,EAAW,aAAel+B,GAG3C,OAAO+pE,CACR,CAWQ,YAAAW,CACPx2D,EACAnI,EACAw8D,GAEA,MAAM9tD,EAqTR,SAA6CvG,EAAanI,GACzD,MAAMd,EAAOc,EAAK7C,KAAKiO,GAAI,WAAcpL,EAAK7C,KAAK+B,KAAO,QAE1D,MAAO,GAAIiJ,KAAUjJ,GACtB,CAzToB6/D,CAAc52D,EAAMnI,GAChCy8B,EAAUz8B,EAAK7C,KAAKiO,GAAI,cAAiBoxD,EAAcwB,WAAWtC,uBAAwB17D,EAAK7C,MAAS6C,EAAK7C,KAE7G6hE,EAA2B9hE,KAAKy/D,gBAAgBr3D,IAAKk3D,GACrDyC,EAAqBD,EAAyB15D,IAAKm3B,GAEzD,GAAMwiC,EAEC,IAAMA,EAAmBn0D,IAAK4D,GAGpC,OAFAuwD,EAAmBl0D,IAAK2D,EAGzB,MALCswD,EAAyBrwD,IAAK8tB,EAAS,IAAI3rB,IAAK,CAAEpC,KAOnDxR,KAAKwQ,KAA4BgB,EAAW1O,EAAMw8D,EACnD,CAQQ,yBAAA0C,CACP/hE,EACAq/D,GAEA,MAAMx8D,EAAgC,CACrC7C,OACA6kB,MAAO,GAAM2nB,UAAWxsC,IAGzB,UAAYlJ,KAAO+L,EAAK7C,KAAKq+B,mBAC5Bx7B,EAAK2hD,aAAe1tD,EACpB+L,EAAK09D,kBAAoB,KACzB19D,EAAK29D,kBAAoB39D,EAAK7C,KAAKo7B,aAActkC,GAEjDiJ,KAAKyhE,aAAc,aAAc1qE,IAAQ+L,EAAMw8D,EAEjD,CAYQ,oBAAAO,CACPrlC,EACAynC,EAA4B,IAAIruD,IAChCrP,EAA8C,CAAC,GAE/C,MAAM+6D,EAAuC,SACzCt/D,KAAKu/D,gBADoC,CAE5CuB,WAAY,IAAI,GAChBtmC,SACAj2B,UACA29D,YAAajiE,GAAQD,KAAKogE,eAAgB,GAAM3zB,UAAWxsC,GAAQq/D,GACnE6C,gBAAiB1qD,GAAWzX,KAAKogE,eAAgB,GAAM5zB,UAAW/0B,GAAW6nD,EAAe,CAAEgC,qBAAqB,IACnHc,kBAAmBniE,GAAQD,KAAKgiE,0BAA2B/hE,EAAMq/D,GACjE+C,aAAc9qB,IAAgB0qB,EAAer0D,IAAK0xD,EAAc9B,OAAOnB,eAAgB9kB,MAKxF,OAFAv3C,KAAKy/D,gBAAgBhuD,IAAK6tD,EAAe,IAAI/tD,KAEtC+tD,CACR,EA0ND,SAAS+B,GACRzF,EACAwF,EACA5D,GAEA,MAAM14C,EAAQs8C,EAAOT,WACfvkC,EAAYz7B,MAAMrB,KAAMs8D,EAAcz/B,gBAC5CC,EAAUjQ,QACViQ,EAAU/wB,UAUV,OAR4B+wB,EAA8B/H,MAAM5c,IAC/D,GAAKqN,EAAMi1C,aAActiD,GAAY,CAGpC,QAFoB+lD,EAAOf,cAAehlD,GAErB+uB,kBAAmB,eACzC,IAIF,CAQA,SAASg7B,GAAwBhpE,GAIhC,MAAO,CACNyH,KAJYzH,EAAMyH,KAKlB6kB,MAJiB,GAAMynB,4BAA6B/zC,EAAM+wC,iBAAkB/wC,EAAMF,QAMpF,CC33Be,MAAM,WAAkB8W,EAAc,KAkEpD,WAAArN,IACI7F,GAWHsR,QA1ED,KAAQo/B,oBAA8B,EAKtC,KAAUrH,OAA+B,IAAIh0B,IAG7C,KAAOo7B,QAAwB,GAoEzBzwC,EAAK5D,QACT0H,KAAKoiC,SAAUlmC,EAEjB,CAiBA,UAAWw/B,GACV,GAAK17B,KAAK2sC,QAAQr0C,OAAS,EAAI,CAC9B,MAAMwsB,EAAQ9kB,KAAK2sC,QAAS3sC,KAAK2sC,QAAQr0C,OAAS,GAElD,OAAO0H,KAAK4sC,mBAAqB9nB,EAAM6b,IAAM7b,EAAM4b,KACpD,CAEA,OAAO,IACR,CAUA,SAAWuM,GACV,GAAKjtC,KAAK2sC,QAAQr0C,OAAS,EAAI,CAC9B,MAAMwsB,EAAQ9kB,KAAK2sC,QAAS3sC,KAAK2sC,QAAQr0C,OAAS,GAElD,OAAO0H,KAAK4sC,mBAAqB9nB,EAAM4b,MAAQ5b,EAAM6b,GACtD,CAEA,OAAO,IACR,CAMA,eAAWqK,GAGV,OAAgB,IAFDhrC,KAAK2sC,QAAQr0C,QAGpB0H,KAAK2sC,QAAS,GAAI3B,WAI3B,CAKA,cAAWkC,GACV,OAAOltC,KAAK2sC,QAAQr0C,MACrB,CAKA,cAAW60C,GACV,OAAQntC,KAAKgrC,aAAehrC,KAAK4sC,kBAClC,CASO,OAAA3oB,CAASypB,GACf,GAAK1tC,KAAKktC,YAAcQ,EAAeR,WACtC,OAAO,EACD,GAAyB,IAApBltC,KAAKktC,WAChB,OAAO,EAGR,IAAMltC,KAAK07B,OAAQzX,QAASypB,EAAehS,UAAc17B,KAAKitC,MAAOhpB,QAASypB,EAAeT,OAC5F,OAAO,EAGR,UAAYU,KAAa3tC,KAAK2sC,QAAU,CACvC,IAAIiB,GAAQ,EAEZ,UAAYtC,KAAcoC,EAAef,QACxC,GAAKgB,EAAU1pB,QAASqnB,GAAe,CACtCsC,GAAQ,EACR,KACD,CAGD,IAAMA,EACL,OAAO,CAET,CAEA,OAAO,CACR,CAKA,UAAQR,GACP,UAAYtoB,KAAS9kB,KAAK2sC,cACnB,IAAI,GAAO7nB,EAAM4b,MAAO5b,EAAM6b,IAEtC,CAUO,aAAA0M,GACN,IAAI5c,EAAQ,KAEZ,UAAY3L,KAAS9kB,KAAK2sC,QACnBlc,IAAS3L,EAAM4b,MAAMhE,SAAUjM,EAAMiQ,SAC1CjQ,EAAQ3L,GAIV,OAAO2L,EAAQ,IAAI,GAAOA,EAAMiQ,MAAOjQ,EAAMkQ,KAAQ,IACtD,CAUO,YAAA2M,GACN,IAAI5c,EAAO,KAEX,UAAY5L,KAAS9kB,KAAK2sC,QACnBjc,IAAQ5L,EAAM6b,IAAI9D,QAASnM,EAAKiQ,OACrCjQ,EAAO5L,GAIT,OAAO4L,EAAO,IAAI,GAAOA,EAAKgQ,MAAOhQ,EAAKiQ,KAAQ,IACnD,CASO,gBAAAxL,GACN,MAAM1E,EAAQzwB,KAAKqtC,gBAEnB,OAAO5c,EAAQA,EAAMiQ,MAAM3d,QAAU,IACtC,CASO,eAAAyqB,GACN,MAAMC,EAAYztC,KAAKstC,eAEvB,OAAOG,EAAYA,EAAU9M,IAAI5d,QAAU,IAC5C,CAwDO,KAAAqf,IACHlmC,GASH,IAAM+xC,EAAYC,EAAe3pC,GAAYrI,EAO7C,GAL6B,iBAAjBgyC,IACX3pC,EAAU2pC,EACVA,OAAgB,GAGG,OAAfD,EACJjuC,KAAKmuC,WAAY,SACX,GAAKF,aAAsB,GACjCjuC,KAAKmuC,WAAYF,EAAWb,YAAaa,EAAWd,iBAC9C,GAAKc,GAAwD,mBAAjCA,EAAoBb,UAGtDptC,KAAKmuC,WAAcF,EAAkCb,YAAea,EAAkCd,iBAChG,GAAKc,aAAsB,GACjCjuC,KAAKmuC,WAAY,CAAEF,KAAgB1pC,KAAaA,EAAQiqC,eAClD,GAAKP,aAAsB,GACjCjuC,KAAKmuC,WAAY,CAAE,IAAI,GAAOF,UACxB,GAAKA,aAAsB,GAAO,CACxC,MAAMO,IAAajqC,KAAaA,EAAQiqC,SACxC,IAAI1pB,EAEJ,GAAsB,MAAjBopB,EACJppB,EAAQ,GAAM0nB,UAAWyB,QACnB,GAAsB,MAAjBC,EACXppB,EAAQ,GAAM2nB,UAAWwB,OACnB,SAAuB,IAAlBC,EAQX,MAAM,IAAI,EAAe,kDAAmD,CAAEluC,KAAMiuC,IAPpFnpB,EAAQ,IAAI,GAAO,GAAS8jB,UAAWqF,EAAYC,GAQpD,CAEAluC,KAAKmuC,WAAY,CAAErpB,GAAS0pB,EAC7B,KAAO,KAAK73B,EAAYs3B,GAgBvB,MAAM,IAAI,EAAe,uCAAwC,CAAEjuC,KAAMiuC,IAdzEjuC,KAAKmuC,WAAYF,EAAY1pC,KAAaA,EAAQiqC,SAenD,CACD,CAYU,UAAAL,CAAYS,EAA4BC,GAA0B,GAC3E,MAAMhD,EAASlrC,MAAMrB,KAAMsvC,GAGrB0zB,EAAcz2B,EAAOxX,MAAM8iB,IAChC,KAAQA,aAAoB,IAY3B,MAAM,IAAI,EACT,uCACA,CAAEn3C,KAAM4uC,IAIV,OAAO5uC,KAAK2sC,QAAQp2B,OAAOgsD,IAClBA,EAASt+C,QAASkzB,IACxB,KAICtL,EAAOvzC,SAAW0H,KAAK2sC,QAAQr0C,QAAWgqE,KAI/CtiE,KAAKwiE,kBAAmB32B,GAExB7rC,KAAK4sC,qBAAuBiC,EAE5B7uC,KAAKwQ,KAAiC,eAAgB,CAAEiyD,cAAc,IACvE,CAWO,QAAAh0B,CAAU3D,EAAiClvB,GACjD,GAAqB,OAAhB5b,KAAK07B,OAMT,MAAM,IAAI,EAAe,qCAAsC,CAAE17B,KAAM8qC,IAGxE,MAAM4D,EAAW,GAAS9F,UAAWkC,EAAgBlvB,GAErD,GAA4C,QAAvC8yB,EAAS/D,YAAa3qC,KAAKitC,OAC/B,OAGD,MAAMvR,EAAS17B,KAAK07B,OAEf17B,KAAK2sC,QAAQr0C,QACjB0H,KAAK0iE,YAGiC,UAAlCh0B,EAAS/D,YAAajP,IAC1B17B,KAAK8uC,WAAY,IAAI,GAAOJ,EAAUhT,IACtC17B,KAAK4sC,oBAAqB,IAE1B5sC,KAAK8uC,WAAY,IAAI,GAAOpT,EAAQgT,IACpC1uC,KAAK4sC,oBAAqB,GAG3B5sC,KAAKwQ,KAAiC,eAAgB,CAAEiyD,cAAc,GACvE,CAQO,YAAApnC,CAActkC,GACpB,OAAOiJ,KAAKulC,OAAOn9B,IAAKrR,EACzB,CAQO,aAAAivC,GACN,OAAOhmC,KAAKulC,OAAO9sB,SACpB,CAKO,gBAAA6lB,GACN,OAAOt+B,KAAKulC,OAAOtuC,MACpB,CAQO,YAAAgvC,CAAclvC,GACpB,OAAOiJ,KAAKulC,OAAO33B,IAAK7W,EACzB,CAWO,eAAA2P,CAAiB3P,GAClBiJ,KAAKimC,aAAclvC,KACvBiJ,KAAKulC,OAAO5zB,OAAQ5a,GAEpBiJ,KAAKwQ,KAAqC,mBAAoB,CAAE6tB,cAAe,CAAEtnC,GAAO0rE,cAAc,IAExG,CAYO,YAAAh9D,CAAc1O,EAAayB,GAC5BwH,KAAKq7B,aAActkC,KAAUyB,IACjCwH,KAAKulC,OAAO9zB,IAAK1a,EAAKyB,GAEtBwH,KAAKwQ,KAAqC,mBAAoB,CAAE6tB,cAAe,CAAEtnC,GAAO0rE,cAAc,IAExG,CAOO,kBAAAz0B,GACN,OAAyB,IAApBhuC,KAAKktC,WACF,KAGDltC,KAAKqtC,gBAAiBpB,qBAC9B,CAiEA,kBAAQ02B,GACP,MAAMC,EAAU,IAAIl1D,QAEpB,UAAYoX,KAAS9kB,KAAKotC,YAAc,CAEvC,MAAMy1B,EAAaC,GAAgBh+C,EAAM4b,MAAOkiC,GAE3CG,GAAsBF,EAAY/9C,WAChC+9C,GAGP,UAAYrqE,KAASssB,EAAM+lB,YAAc,CACxC,MAAMm4B,EAAQxqE,EAAMyH,KAED,cAAdzH,EAAMyS,MAAwBg4D,GAAqBD,EAAcJ,EAAS99C,WACxEk+C,EAER,CAEA,MAAME,EAAWJ,GAAgBh+C,EAAM6b,IAAKiiC,GAEvCO,GAAoBD,EAAUp+C,WAC5Bo+C,EAER,CACD,CAUO,qBAAAE,CAAuB3rD,EAAmBzX,KAAK07B,OAAQplC,MAC7D,MAAM+sE,EAAqB,GAASz6B,UAAWnxB,EAAS,GAClD6rD,EAAmB,GAAS16B,UAAWnxB,EAAS,OAEtD,OAAO4rD,EAAmBnL,WAAYl4D,KAAKm1B,qBAC1CmuC,EAAiBpL,WAAYl4D,KAAKwtC,kBACpC,CAMU,UAAAsB,CAAYhqB,GACrB9kB,KAAKujE,YAAaz+C,GAClB9kB,KAAK2sC,QAAQ/sC,KAAM,IAAI,GAAOklB,EAAM4b,MAAO5b,EAAM6b,KAClD,CAKU,WAAA4iC,CAAaz+C,GACtB,QAAUtqB,EAAI,EAAGA,EAAIwF,KAAK2sC,QAAQr0C,OAAQkC,IACzC,GAAKsqB,EAAMgnB,eAAgB9rC,KAAK2sC,QAASnyC,IAQxC,MAAM,IAAI,EACT,mCACA,CAAEwF,KAAM8kB,GACR,CAAEkqB,WAAYlqB,EAAOmqB,kBAAmBjvC,KAAK2sC,QAASnyC,IAI1D,CAMU,iBAAAgoE,CAAmB32B,GAC5B7rC,KAAKwjE,mBAEL,UAAY1+C,KAAS+mB,EACpB7rC,KAAK8uC,WAAYhqB,EAEnB,CAMU,gBAAA0+C,GACT,KAAQxjE,KAAK2sC,QAAQr0C,OAAS,GAC7B0H,KAAK0iE,WAEP,CAKU,SAAAA,GACT1iE,KAAK2sC,QAAQ3tC,KACd,EAgED,SAASykE,GAAkBhsD,EAAkCmrD,GAC5D,OAAKA,EAAQh1D,IAAK6J,KAIlBmrD,EAAQ/0D,IAAK4J,GAENA,EAAQnhB,KAAKkN,SAAUpL,MAAMsrE,OAAOC,QAASlsD,MAAeA,EAAQ7Y,OAC5E,CAKA,SAASqkE,GAAqBxrD,EAAkBmrD,EAA2C99C,GAC1F,OAAO2+C,GAAkBhsD,EAASmrD,IAAagB,GAAmBnsD,EAASqN,EAC5E,CAOA,SAASg+C,GAAgBr9C,EAAoBm9C,GAC5C,MACMc,EADUj+C,EAAS7mB,OACFtI,KAAKkN,SAAUpL,MAAMsrE,OAEtCtnC,EAAY3W,EAAS7mB,OAAOu9B,aAAc,CAAEG,aAAa,EAAMD,aAAa,IAElF,IAAIwnC,GAAiB,EAErB,MAAMb,EAAQ5mC,EAAU7K,MAAQ9Z,IAE1BosD,IAILA,EAAiBH,EAAOI,QAASrsD,IAEzBosD,GAAkBJ,GAAkBhsD,EAASmrD,MAOtD,OAFAxmC,EAAUx+B,SAAS6Z,GAAWmrD,EAAQ/0D,IAAK4J,KAEpCurD,CACR,CAKA,SAASY,GAAmBZ,EAAal+C,GACxC,MAAMi/C,EA+EP,SAA4B5kE,GAC3B,MAAMukE,EAASvkE,EAAK7I,KAAKkN,SAAUpL,MAAMsrE,OAEzC,IAAI9kE,EAASO,EAAKP,OAElB,KAAQA,GAAS,CAChB,GAAK8kE,EAAOC,QAAS/kE,GACpB,OAAOA,EAGRA,EAASA,EAAOA,MACjB,CACD,CA3FqBolE,CAAmBhB,GAEvC,IAAMe,EACL,OAAO,EAMR,OAFwBj/C,EAAM0mB,cAAe,GAAMiB,UAAWs3B,IAAe,EAG9E,CAkBA,SAAShB,GAAsBF,EAAiC/9C,GAC/D,QAAM+9C,OAID/9C,EAAMkmB,cAAe63B,EAAW1gC,WAIhCrd,EAAM4b,MAAMw3B,WAAY,GAAStvB,UAAWi6B,EAAYA,EAAWxM,aAIjEuN,GAAmBf,EAAY/9C,GACvC,CAkBA,SAASq+C,GAAoBD,EAA+Bp+C,GAC3D,QAAMo+C,OAIDp+C,EAAMkmB,cAAek4B,EAAS/gC,WAI9Brd,EAAM6b,IAAIu3B,WAAY,GAAStvB,UAAWs6B,EAAU,KAIlDU,GAAmBV,EAAUp+C,GACrC,CAzLA,GAAUjjB,UAAUqM,GAAK,SAAUjD,GAClC,MAAgB,cAATA,GAAiC,oBAATA,CAChC,EChuBe,MAAM,WAAkBmE,EAAc,KAMpD,WAAArN,CAAa2+B,EAAiBC,GAC7BnzB,MAAOkzB,EAAOC,GAEdsjC,GAAiB5iE,KAAMrB,KACxB,CAOO,MAAA8gB,GACN9gB,KAAK2P,eACN,CAKO,OAAAu0D,GACN,OAAO,IAAI,GAAOlkE,KAAK0gC,MAAO1gC,KAAK2gC,IACpC,CAKA,gBAAcwjC,CAAWr/C,GACxB,OAAO,IAAI,GAAWA,EAAM4b,MAAO5b,EAAM6b,IAC1C,EA4ED,SAASsjC,KACRjkE,KAAKwP,SACJxP,KAAK1J,KAAKkN,SAAUpL,MACpB,kBACA,CAAEkX,EAAOpT,KACR,MAAMu8D,EAAYv8D,EAAM,GAElBu8D,EAAU2L,qBAIhBvsD,GAAUxW,KAAMrB,KAAMy4D,EAAW,GAElC,CAAE7rD,SAAU,OAEd,CAKA,SAASiL,GAA4B4gD,GAEpC,MAAM5sB,EAAS7rC,KAAKw4D,0BAA2BC,GACzCt6D,EAAS,GAAM68D,kBAAmBnvB,GAElCw4B,GAAqBlmE,EAAO8lB,QAASjkB,MACrCskE,EAgCP,SAA0Cx/C,EAAc2zC,GACvD,OAASA,EAAUxtD,MAClB,IAAK,SACJ,OAAO6Z,EAAMymB,iBAAkBktB,EAAUhzC,UAC1C,IAAK,OACL,IAAK,SACL,IAAK,WACL,IAAK,QACJ,OAAOX,EAAMymB,iBAAkBktB,EAAUO,iBACxCl0C,EAAM4b,MAAMzc,QAASw0C,EAAUO,iBAC/Bl0C,EAAMymB,iBAAkBktB,EAAUpjB,gBACpC,IAAK,QACJ,OAAOvwB,EAAMymB,iBAAkBktB,EAAUU,gBAAmBr0C,EAAMymB,iBAAkBktB,EAAUngB,mBAGhG,OAAO,CACR,CAhDwBisB,CAAiCvkE,KAAMy4D,GAE9D,IAAIc,EAAmB,KAEvB,GAAK8K,EAAoB,CAGK,cAAxBlmE,EAAO7H,KAAK8+B,WAGfmkC,EADsB,UAAlBd,EAAUxtD,KACOwtD,EAA6BO,eAG7BP,EAA8Bc,kBAIrD,MAAMgJ,EAAWviE,KAAKkkE,UAEpBlkE,KAAc0gC,MAAQviC,EAAOuiC,MAC7B1gC,KAAc2gC,IAAMxiC,EAAOwiC,IAE7B3gC,KAAKwQ,KAAiC,eAAgB+xD,EAAU,CAAEhJ,oBACnE,MAAY+K,GAEXtkE,KAAKwQ,KAAmC,iBAAkBxQ,KAAKkkE,UAAW,CAAE3K,oBAE9E,CAvGA,GAAU13D,UAAUqM,GAAK,SAAUjD,GAClC,MAAgB,cAATA,GAAiC,oBAATA,GAEtB,SAARA,GAA4B,gBAATA,CACrB,ECzDA,MAAMu5D,GAAc,aA0BL,MAAM,WAA0Bp1D,EAAc,KAW5D,WAAArN,CAAasV,GACZ7J,QAEAxN,KAAKkvC,WAAa,IAAIu1B,GAAeptD,GAErCrX,KAAKkvC,WAAW99B,SAAU,gBAAiB7R,GAAIS,MAC/CA,KAAKkvC,WAAW99B,SAAU,oBAAqB7R,GAAIS,MACnDA,KAAKkvC,WAAW99B,SAAU,iBAAkB7R,GAAIS,KACjD,CAMA,eAAWgrC,GACV,OAAOhrC,KAAKkvC,WAAWlE,WACxB,CAYA,UAAWtP,GACV,OAAO17B,KAAKkvC,WAAWxT,MACxB,CASA,SAAWuR,GACV,OAAOjtC,KAAKkvC,WAAWjC,KACxB,CAKA,cAAWC,GACV,OAAOltC,KAAKkvC,WAAWhC,UACxB,CAMA,eAAWw3B,GACV,OAAO1kE,KAAKkvC,WAAWw1B,WACxB,CASA,cAAWv3B,GACV,OAAOntC,KAAKkvC,WAAW/B,UACxB,CAOA,uBAAWw3B,GACV,OAAO3kE,KAAKkvC,WAAWy1B,mBACxB,CAQA,WAAW/E,GACV,OAAO5/D,KAAKkvC,WAAW0wB,OACxB,CAOA,WAAWjzB,GACV,OAAO3sC,KAAKkvC,WAAWvC,OACxB,CAKO,SAAAS,GACN,OAAOptC,KAAKkvC,WAAW9B,WACxB,CASO,gBAAAjY,GACN,OAAOn1B,KAAKkvC,WAAW/Z,kBACxB,CASO,eAAAqY,GACN,OAAOxtC,KAAKkvC,WAAW1B,iBACxB,CAUO,aAAAH,GACN,OAAOrtC,KAAKkvC,WAAW7B,eACxB,CAUO,YAAAC,GACN,OAAOttC,KAAKkvC,WAAW5B,cACxB,CAuDO,iBAAAq1B,GACN,OAAO3iE,KAAKkvC,WAAWyzB,mBACxB,CAOO,kBAAA30B,GACN,OAAOhuC,KAAKkvC,WAAWlB,oBACxB,CAUO,qBAAAo1B,CAAuB3rD,GAC7B,OAAOzX,KAAKkvC,WAAWk0B,sBAAuB3rD,EAC/C,CAKO,OAAAuO,GACNhmB,KAAKkvC,WAAWlpB,SACjB,CAKO,gBAAAsY,GACN,OAAOt+B,KAAKkvC,WAAW5Q,kBACxB,CAQO,aAAA0H,GACN,OAAOhmC,KAAKkvC,WAAWlJ,eACxB,CAQO,YAAA3K,CAActkC,GACpB,OAAOiJ,KAAKkvC,WAAW7T,aAActkC,EACtC,CAQO,YAAAkvC,CAAclvC,GACpB,OAAOiJ,KAAKkvC,WAAWjJ,aAAclvC,EACtC,CAKO,OAAAg+B,GACN/0B,KAAKkvC,WAAW01B,gBAChB5kE,KAAKkvC,WAAW21B,mBAAmB,EACpC,CAUO,cAAAC,CAAgBC,GACtB/kE,KAAKkvC,WAAW41B,eAAgBC,EACjC,CAcO,SAAA31B,CAAWtE,EAAiClvB,GAClD5b,KAAKkvC,WAAWT,SAAU3D,EAAgBlvB,EAC3C,CAUO,MAAAuzB,IAAWjzC,GACjB8D,KAAKkvC,WAAW9M,SAAUlmC,EAC3B,CAWO,aAAAkrC,CAAerwC,EAAayB,GAClCwH,KAAKkvC,WAAWzpC,aAAc1O,EAAKyB,EACpC,CAYO,gBAAA8uC,CAAkBvwC,GACxBiJ,KAAKkvC,WAAWxoC,gBAAiB3P,EAClC,CAOO,oBAAAiuE,GACN,OAAOhlE,KAAKkvC,WAAW+1B,qBACxB,CAgBO,gBAAAC,GACN,OAAOllE,KAAKkvC,WAAWi2B,iBACxB,CAaO,eAAAC,CAAiBxsD,GACvB5Y,KAAKkvC,WAAWm2B,eAAgBzsD,EACjC,CASA,4BAAc0sD,CAAuBvuE,GACpC,OAAOytE,GAAcztE,CACtB,CAOA,2BAAcwuE,CAAsBxuE,GACnC,OAAOA,EAAI6tD,WAAY4f,GACxB,EAKD,GAAkB3iE,UAAUqM,GAAK,SAAUjD,GAC1C,MAAgB,cAATA,GACE,mBAARA,GACQ,qBAARA,GACQ,2BAARA,CACF,EAiFA,MAAMw5D,WAAsB,GA6D3B,WAAA1iE,CAAasV,GACZ7J,QAzDD,KAAOoyD,QAA8B,IAAI,GAAY,CAAEzvC,WAAY,SA0BnE,KAAQq1C,mBAAoD,IAAIj0D,IAKhE,KAAQk0D,0BAA6C,KAKrD,KAAQC,kBAA4B,EAQpC,KAAQC,2BAA0C,IAAI/xD,IAKtD,KAAQgyD,iBAAgC,IAAIhyD,IAU3C5T,KAAK6lE,OAASxuD,EAAIjf,MAClB4H,KAAK8lE,UAAYzuD,EAGjBrX,KAAKwP,SAAoCxP,KAAK6lE,OAAQ,kBAAkB,CAAExwD,EAAKnZ,KAC9E,MAAMu8D,EAAYv8D,EAAM,GAElBu8D,EAAU2L,qBAAyC,UAAlB3L,EAAUxtD,MAAsC,UAAlBwtD,EAAUxtD,MAAsC,QAAlBwtD,EAAUxtD,OAKjF,GAAvBjL,KAAK2sC,QAAQr0C,QAAe0H,KAAKylE,2BACrCzlE,KAAK+lE,uBAAwB/lE,KAAKylE,2BAInCzlE,KAAKylE,0BAA4B,KAE5BzlE,KAAK0lE,mBACT1lE,KAAK0lE,kBAAmB,EACxB1lE,KAAKwQ,KAAyC,eAAgB,CAAEiyD,cAAc,KAC/E,GACE,CAAE71D,SAAU,WAGf5M,KAAKqP,GAAuC,gBAAgB,KAC3DrP,KAAKgmE,yBAA0BhmE,KAAKotC,YAAa,IAKlDptC,KAAKwP,SAAuCxP,KAAK6lE,OAAOjG,QAAS,UAAU,CAAEvqD,EAAK+rD,EAAQmB,EAAUprB,KACnGn3C,KAAKimE,cAAe7E,EAAQjqB,EAAU,IAIvCn3C,KAAKwP,SAA+BxP,KAAK8lE,UAAW,UAAU,CAAEzwD,EAAK6wD,MA2kBvE,SAAyC9tE,EAAc8tE,GACtD,MAAMvG,EAASvnE,EAAMoL,SAASm8D,OAE9B,UAAYjnD,KAASinD,EAAOQ,aAAe,CAC1C,GAAmB,UAAdznD,EAAMzN,KACV,SAGD,MAAMk7D,EAAeztD,EAAM+M,SAAS7mB,OACZ8Z,EAAMpgB,SAAW6tE,EAAa9P,WAGrDj+D,EAAMguE,cAAeF,GAAO1rC,IAC3B,MAAM6rC,EAAmB1lE,MAAMrB,KAAM6mE,EAAa7nC,oBAChDv4B,QAAQhP,GAAOA,EAAI6tD,WAAY4f,MAEjC,UAAYztE,KAAOsvE,EAClB7rC,EAAO9zB,gBAAiB3P,EAAKovE,EAC9B,GAGH,CACD,CAhmBGG,CAAgCtmE,KAAK6lE,OAAQK,EAAO,GAEtD,CAEA,eAAoBl7B,GAGnB,OAAkB,IAFHhrC,KAAK2sC,QAAQr0C,OAEN0H,KAAK8lE,UAAUS,mBAAmBv7B,YAAcx9B,MAAMw9B,WAC7E,CAEA,UAAoBtP,GACnB,OAAOluB,MAAMkuB,QAAU17B,KAAK8lE,UAAUS,mBAAmB7lC,KAC1D,CAEA,SAAoBuM,GACnB,OAAOz/B,MAAMy/B,OAASjtC,KAAK8lE,UAAUS,mBAAmB5lC,GACzD,CAEA,cAAoBuM,GACnB,OAAOltC,KAAK2sC,QAAQr0C,OAAS0H,KAAK2sC,QAAQr0C,OAAS,CACpD,CAMA,eAAWosE,GACV,OAAO1kE,KAAK2sC,QAAQr0C,OAAS,CAC9B,CAMA,uBAAWqsE,GACV,QAAS3kE,KAAK2lE,2BAA2B9xD,IAC1C,CAKO,OAAAmS,GACN,QAAUxrB,EAAI,EAAGA,EAAIwF,KAAK2sC,QAAQr0C,OAAQkC,IACzCwF,KAAK2sC,QAASnyC,GAAIsmB,SAGnB9gB,KAAK2P,eACN,CAEA,UAAiBy9B,GACXptC,KAAK2sC,QAAQr0C,aACVkV,MAAM4/B,kBAEPptC,KAAK8lE,UAAUS,kBAEvB,CAEgB,aAAAl5B,GACf,OAAO7/B,MAAM6/B,iBAAmBrtC,KAAK8lE,UAAUS,kBAChD,CAEgB,YAAAj5B,GACf,OAAO9/B,MAAM8/B,gBAAkBttC,KAAK8lE,UAAUS,kBAC/C,CAEgB,KAAAnkC,IAAUlmC,GACzBsR,MAAM40B,SAAUlmC,GAChB8D,KAAK6kE,mBAAmB,GACxB7kE,KAAK4kE,eACN,CAEgB,QAAAn2B,CAAU3D,EAAiClvB,GAC1DpO,MAAMihC,SAAU3D,EAAgBlvB,GAChC5b,KAAK6kE,mBAAmB,GACxB7kE,KAAK4kE,eACN,CAEgB,YAAAn/D,CAAc1O,EAAayB,GAC1C,GAAKwH,KAAKonC,cAAerwC,EAAKyB,GAAU,CAEvC,MAAM6lC,EAAgB,CAAEtnC,GACxBiJ,KAAKwQ,KAA6C,mBAAoB,CAAE6tB,gBAAeokC,cAAc,GACtG,CACD,CAEgB,eAAA/7D,CAAiB3P,GAChC,GAAKiJ,KAAKsnC,iBAAkBvwC,GAAQ,CAEnC,MAAMsnC,EAAgB,CAAEtnC,GACxBiJ,KAAKwQ,KAA6C,mBAAoB,CAAE6tB,gBAAeokC,cAAc,GACtG,CACD,CAEO,eAAA0C,GACN,MAAMqB,EAAc,IAUpB,OANAxmE,KAAK2lE,2BAA2B93D,IAAK24D,GAES,IAAzCxmE,KAAK2lE,2BAA2B9xD,MACpC7T,KAAK6kE,mBAAmB,GAGlB2B,CACR,CAEO,cAAAnB,CAAgBzsD,GACtB,IAAM5Y,KAAK2lE,2BAA2B/3D,IAAKgL,GAS1C,MAAM,IAAI,EACT,2CACA5Y,KACA,CAAE4Y,IAAAA,IAIJ5Y,KAAK2lE,2BAA2Bh0D,OAAQiH,GAGlC5Y,KAAK2kE,qBACV3kE,KAAK6kE,mBAAmB,EAE1B,CAEO,cAAAC,CAAgBC,GACtB/kE,KAAK4lE,iBAAiB/3D,IAAKk3D,GAC3B/kE,KAAK4kE,eACN,CAEmB,iBAAApC,CAAmB32B,GACrC7rC,KAAKgmE,yBAA0Bn6B,GAE/Br+B,MAAMg1D,kBAAmB32B,EAC1B,CAEmB,SAAA62B,GAClB1iE,KAAK2sC,QAAQ3tC,MAAO8hB,QACrB,CAEmB,UAAAguB,CAAYhqB,GAC9B,MAAM2hD,EAAYzmE,KAAK0mE,cAAe5hD,GAGjC2hD,GACJzmE,KAAK2sC,QAAQ/sC,KAAM6mE,EAErB,CAEQ,wBAAAT,CAA0Bn6B,GACjC,UAAY/mB,KAAS+mB,EACpB,IAAM7rC,KAAK8lE,UAAUa,wBAAyB7hD,GAQ7C,MAAM,IAAI,EACT,oCACA9kB,KACA,CAAE8kB,SAIN,CAOQ,aAAA4hD,CAAe5hD,GAGtB,GAFA9kB,KAAKujE,YAAaz+C,GAEbA,EAAMxuB,MAAQ0J,KAAK8lE,UAAUlM,UAGjC,OAGD,MAAM6M,EAAY,GAAUtC,UAAWr/C,GAgBvC,OAZA2hD,EAAUp3D,GAAI,gBAAgB,CAAEgG,EAAKktD,EAAUz/D,KAG9C,GAFA9C,KAAK0lE,kBAAmB,EAEnBe,EAAUnwE,MAAQ0J,KAAK8lE,UAAUlM,UAAY,CACjD55D,KAAKylE,0BAA4B3iE,EAAKy2D,iBAEtC,MAAM30D,EAAQ5E,KAAK2sC,QAAQ3jC,QAASy9D,GACpCzmE,KAAK2sC,QAAQnlC,OAAQ5C,EAAO,GAC5B6hE,EAAU3lD,QACX,KAGM2lD,CACR,CAEO,aAAA7B,GACN,IAAM5kE,KAAK4lE,iBAAiB/xD,KAC3B,OAGD,MAAM+rD,EAAU,GAChB,IAAIgH,GAAU,EAEd,UAAYxF,KAAUphE,KAAK6lE,OAAOjG,QAAU,CAC3C,MAAMiH,EAAczF,EAAOp/D,KAAKjF,MAAO,IAAK,GAAK,GAEjD,IAAMiD,KAAK4lE,iBAAiBh4D,IAAKi5D,GAChC,SAGD,MAAMnG,EAAcU,EAAOT,WAE3B,UAAYmG,KAAkB9mE,KAAKotC,YAC7BszB,EAAYl1B,cAAes7B,GAAiBA,EAAe97B,cAC/D40B,EAAQhgE,KAAMwhE,EAGjB,CAEA,MAAM2F,EAAapmE,MAAMrB,KAAMU,KAAK4/D,SAEpC,UAAYwB,KAAUxB,EACf5/D,KAAK4/D,QAAQhyD,IAAKwzD,KACvBphE,KAAK4/D,QAAQ/xD,IAAKuzD,GAElBwF,GAAU,GAIZ,UAAYxF,KAAUzgE,MAAMrB,KAAMU,KAAK4/D,SAChCA,EAAQrwC,SAAU6xC,KACvBphE,KAAK4/D,QAAQ35D,OAAQm7D,GAErBwF,GAAU,GAIPA,GACJ5mE,KAAKwQ,KAA0C,gBAAiB,CAAEu2D,aAAYtE,cAAc,GAE9F,CAEQ,aAAAwD,CAAe7E,EAAgBV,GACtC,MAAMmG,EAAczF,EAAOp/D,KAAKjF,MAAO,IAAK,GAAK,GAEjD,IAAMiD,KAAK4lE,iBAAiBh4D,IAAKi5D,GAChC,OAGD,IAAID,GAAU,EAEd,MAAMG,EAAapmE,MAAMrB,KAAMU,KAAK4/D,SAC9BoH,EAAYhnE,KAAK4/D,QAAQhyD,IAAKwzD,GAEpC,GAAMV,EAKC,CACN,IAAIuG,GAAY,EAEhB,UAAYH,KAAkB9mE,KAAKotC,YAClC,GAAKszB,EAAYl1B,cAAes7B,GAAiBA,EAAe97B,aAAgB,CAC/Ei8B,GAAY,EAEZ,KACD,CAGIA,IAAcD,GAClBhnE,KAAK4/D,QAAQ/xD,IAAKuzD,GAElBwF,GAAU,IACEK,GAAaD,IACzBhnE,KAAK4/D,QAAQ35D,OAAQm7D,GAErBwF,GAAU,EAEZ,MAxBMI,IACJhnE,KAAK4/D,QAAQ35D,OAAQm7D,GACrBwF,GAAU,GAwBPA,GACJ5mE,KAAKwQ,KAA0C,gBAAiB,CAAEu2D,aAAYtE,cAAc,GAE9F,CAKO,iBAAAoC,CAAmBqC,GACzB,MAAMC,EAAgB,GAAOnnE,KAAKonE,6BAC5BC,EAAgB,GAAOrnE,KAAKgmC,iBAElC,GAAKkhC,EAEJlnE,KAAKwlE,mBAAqB,IAAIj0D,IAC9BvR,KAAKulC,OAAS,IAAIh0B,SAGlB,UAAcxa,EAAK6V,KAAc5M,KAAKwlE,mBACpB,OAAZ54D,IACJ5M,KAAKulC,OAAO5zB,OAAQ5a,GACpBiJ,KAAKwlE,mBAAmB7zD,OAAQ5a,IAKnCiJ,KAAKi2D,iBAAkBkR,GAGvB,MAAMP,EAAU,GAIhB,UAAcU,EAAQ7zD,KAAczT,KAAKgmC,gBAClCqhC,EAAcz5D,IAAK05D,IAAYD,EAAcj/D,IAAKk/D,KAAa7zD,GACpEmzD,EAAQhnE,KAAM0nE,GAKhB,UAAcC,KAAYF,EACnBrnE,KAAKimC,aAAcshC,IACxBX,EAAQhnE,KAAM2nE,GAKXX,EAAQtuE,OAAS,GACrB0H,KAAKwQ,KAA6C,mBAAoB,CAAE6tB,cAAeuoC,EAASnE,cAAc,GAEhH,CAMQ,aAAAr7B,CAAerwC,EAAayB,EAAgBiqE,GAAwB,GAC3E,MAAM71D,EAAW61D,EAAe,SAAW,MAE3C,GAAiB,OAAZ71D,GAA2D,UAAtC5M,KAAKwlE,mBAAmBp9D,IAAKrR,GAEtD,OAAO,EAMR,OAHiByW,MAAM6tB,aAActkC,KAGnByB,IAIlBwH,KAAKulC,OAAO9zB,IAAK1a,EAAKyB,GAGtBwH,KAAKwlE,mBAAmB/zD,IAAK1a,EAAK6V,IAE3B,EACR,CASQ,gBAAA06B,CAAkBvwC,EAAa0rE,GAAwB,GAC9D,MAAM71D,EAAW61D,EAAe,SAAW,MAE3C,OAAiB,OAAZ71D,GAA2D,UAAtC5M,KAAKwlE,mBAAmBp9D,IAAKrR,MAMvDiJ,KAAKwlE,mBAAmB/zD,IAAK1a,EAAK6V,KAG5BY,MAAMy4B,aAAclvC,KAI1BiJ,KAAKulC,OAAO5zB,OAAQ5a,IAEb,GACR,CAMQ,gBAAAk/D,CAAkB7wB,GACzB,MAAMwhC,EAAU,IAAIhzD,IAEpB,UAAc2zD,EAAQ/zD,KAAcxT,KAAKgmC,gBAEnCZ,EAAMh9B,IAAKm/D,KAAa/zD,GAK7BxT,KAAKsnC,iBAAkBigC,GAAQ,GAGhC,UAAcxwE,EAAKyB,KAAW4sC,EAAQ,CAEpBplC,KAAKonC,cAAerwC,EAAKyB,GAAO,IAGhDouE,EAAQ/4D,IAAK9W,EAEf,CAEA,OAAO6vE,CACR,CAKA,oBAAQ3B,GACP,MAAMxmB,EAAkBz+C,KAAKm1B,mBAAoBv2B,OAEjD,GAAKoB,KAAKgrC,aAAeyT,EAAgBtc,QACxC,UAAYprC,KAAO0nD,EAAgBngB,mBAClC,GAAKvnC,EAAI6tD,WAAY4f,IAAgB,CACpC,MAAMgD,EAAUzwE,EAAIkb,OAAQuyD,SAEtB,CAAEgD,EAAS/oB,EAAgBpjB,aAActkC,GAChD,CAGH,CAOQ,yBAAAqwE,GACP,MAAM3hD,EAAWzlB,KAAKm1B,mBAChBuuC,EAAS1jE,KAAK6lE,OAAOnC,OAE3B,GAA+B,cAA1Bj+C,EAASnvB,KAAK8+B,SAClB,OAAO,KAGR,IAAIgQ,EAAQ,KAEZ,GAAMplC,KAAKgrC,YAkBJ,CAGN,MAAMb,EAAa1kB,EAASgY,SAAWhY,EAASgY,SAAWhY,EAAS0kB,WAC9DD,EAAYzkB,EAASgY,SAAWhY,EAASgY,SAAWhY,EAASykB,UAenE,GAZMlqC,KAAK2kE,sBAEVv/B,EAAQqiC,GAAmBt9B,EAAYu5B,IAIlCt+B,IACLA,EAAQqiC,GAAmBv9B,EAAWw5B,KAKjC1jE,KAAK2kE,sBAAwBv/B,EAAQ,CAC1C,IAAIjmC,EAAOgrC,EAEX,KAAQhrC,IAASimC,GAChBjmC,EAAOA,EAAKynB,gBACZwe,EAAQqiC,GAAmBtoE,EAAMukE,EAEnC,CAGA,IAAMt+B,EAAQ,CACb,IAAIjmC,EAAO+qC,EAEX,KAAQ/qC,IAASimC,GAChBjmC,EAAOA,EAAK88B,YACZmJ,EAAQqiC,GAAmBtoE,EAAMukE,EAEnC,CAGMt+B,IACLA,EAAQplC,KAAKilE,sBAEf,KA5DyB,CAExB,MAAMngD,EAAQ9kB,KAAKqtC,gBAGnB,UAAY70C,KAASssB,EAAQ,CAE5B,GAAKtsB,EAAMyH,KAAKiO,GAAI,YAAew1D,EAAOgE,SAAUlvE,EAAMyH,MAAS,CAElEmlC,EAAQqiC,GAAmBjvE,EAAMyH,KAAMyjE,GACvC,KACD,CAEA,GAAmB,QAAdlrE,EAAMyS,KAAiB,CAC3Bm6B,EAAQ5sC,EAAMyH,KAAK+lC,gBACnB,KACD,CACD,CACD,CA4CA,OAAOZ,CACR,CAMQ,sBAAA2gC,CAAwBxM,GAE/B,MAAMuN,EAAiB9mE,KAAK6lE,OAAOnC,OAAOiE,yBAA0BpO,GAG/DuN,GAEJ9mE,KAAK8uC,WAAYg4B,EAGnB,EASD,SAASW,GAAmBtoE,EAAmBukE,GAC9C,IAAMvkE,EACL,OAAO,KAGR,GAAKA,aAAgB,IAAaA,aAAgB,GACjD,OAAOA,EAAK6mC,gBAGb,IAAM09B,EAAOkE,SAAUzoE,GACtB,OAAO,KAIR,IAAMukE,EAAOgE,SAAUvoE,GACtB,MAAO,GAGR,MAAMoG,EAAuC,GAG7C,UAAcxO,EAAKyB,KAAW2G,EAAK6mC,gBAEjC09B,EAAOmE,eAAgB,QAAS9wE,KACwB,IAAxD2sE,EAAOoE,uBAAwB/wE,GAAMgxE,gBAErCxiE,EAAW3F,KAAM,CAAE7I,EAAKyB,IAI1B,OAAO+M,CACR,CC1tCe,MAAMyiE,GAMpB,WAAAjmE,CAAakmE,GACZjoE,KAAKkoE,aAAeD,CACrB,CAUO,GAAAp6D,CAAKs6D,GACX,UAAY3I,KAAcx/D,KAAKkoE,aAC9BC,EAAkB3I,GAGnB,OAAOx/D,IACR,EC8Bc,MAAMooE,WAAwBJ,GAiJrC,gBAAAK,CAAkBr/C,GASxB,OAAOhpB,KAAK6N,IAmsDd,SAAmCmb,GASlC,MAAM5wB,EAAQkwE,GAA6Bt/C,EAAO5wB,OAC5C6hC,EAAOsuC,GAA0Bv/C,EAAOiR,KAAM,aAI/C7hC,EAAMmN,WAAWjN,SACrBF,EAAMkf,UAAW,GAGlB,OAASkoD,IACRA,EAAWnwD,GACV,UAAWjX,EAAM4J,OA/uBb,SAAwBwmE,EAAwCC,EAA6BC,IACnG,MAAO,CACNrzD,EACAvS,EACAw8D,KAEA,IAAMmJ,EAAU3lE,EAAK7C,KAAMq/D,EAAcwB,WAAY,CAAE6H,WAAW,IACjE,OAGD,MAAMpxB,EAAcixB,EAAgB1lE,EAAK7C,KAAMq/D,EAAex8D,GAE9D,IAAMy0C,EACL,OAIDkxB,EAAU3lE,EAAK7C,KAAMq/D,EAAcwB,YAEnC,MAAMnuB,EAAe2sB,EAAc9B,OAAOD,eAAgBz6D,EAAKgiB,MAAM4b,OAErE4+B,EAAc9B,OAAOnf,aAAcv7C,EAAK7C,KAAMs3C,GAC9C+nB,EAAc9kC,OAAO90B,OAAQitC,EAAc4E,GAG3C+nB,EAAc8C,kBAAmBt/D,EAAK7C,MAGtC2oE,GAAwBrxB,EAAaz0C,EAAK7C,KAAKu7B,cAAe8jC,EAAe,CAAEsC,aAAc9+D,EAAK8+D,cAAgB,CAEpH,CAktBGiH,CAAe5uC,EAAM6uC,GAAgB1wE,IACrC,CAAEwU,SAAUoc,EAAO+/C,mBAAqB,YAGpC3wE,EAAMkf,UAAYlf,EAAMmN,WAAWjN,SACvCknE,EAAWnwD,GAAgC,gBAAiB25D,GAAqB5wE,GAAS,CAAEwU,SAAU,OACvG,CAEF,CAhuDmBq8D,CAA0BjgD,GAC5C,CAqJO,kBAAAkgD,CAAoBlgD,GAQ1B,OAAOhpB,KAAK6N,IA4kDd,SACCmb,GASA,MAAM5wB,EAAQkwE,GAA6Bt/C,EAAO5wB,OAC5C6hC,EAAOsuC,GAA0Bv/C,EAAOiR,KAAM,aAMpD,OAFA7hC,EAAMkf,UAAW,EAERkoD,IACR,GAAKA,EAAWD,eAAemE,OAAOyF,WAAY/wE,EAAM4J,KAAM,SA2C7D,MAAM,IAAI,EAAe,kDAAmDw9D,EAAY,CAAE7a,YAAavsD,EAAM4J,OAjxBzG,IAA0BwmE,EAA0CC,EAoxBzEjJ,EAAWnwD,GACV,UAAWjX,EAAM4J,QArxBawmE,EAsxBbvuC,EAtxBuDwuC,EAsxBjDK,GAAgB1wE,GArxBlC,CACNid,EACAvS,EACAw8D,KAEA,IAAMmJ,EAAU3lE,EAAK7C,KAAMq/D,EAAcwB,WAAY,CAAE6H,WAAW,IACjE,OAGD,MAAMS,EAAW,IAAI73D,IAErB+tD,EAAc9kC,OAAO2d,qBA8yCvB,SAA4B1gC,EAAuB2xD,EAA8C9J,GAChG,MAAO,CAAE9kC,EAAwB0d,KAChC,MAAMmxB,EAAO7uC,EAAOuZ,uBAAwB,SAE5C,IAAIz8B,EAAoC,KAExC,GAAsB,aAAjB4gC,EACJ5gC,EAAW3W,MAAMrB,KAAMmY,EAAQ+jB,mBACzB,IAA4B,mBAAhB0c,EAQlB,MAAM,IAAI,EAAe,+BAAgConB,EAAcE,WAAY,CAAEtnB,iBAPrF5gC,EAAW3W,MAAMrB,KAAMmY,EAAQ+jB,eAAgBz1B,QAAQ0R,GAAWygC,EAAczgC,IAQjF,CAIA,OAFA2xD,EAAS33D,IAAK43D,EAAM/xD,GAEb+xD,CAAI,CAEb,CAr0C6CC,CAAmBxmE,EAAK7C,KAAMmpE,EAAU9J,IAGnF,MAAM/nB,EAAcixB,EAAgB1lE,EAAK7C,KAAMq/D,EAAex8D,GAI9D,GAFAw8D,EAAc9kC,OAAO6d,qBAEfd,EACL,QAk0CH,SACC9/B,EACA2xD,EACA9J,GAEA,MAAMiK,EAAkB5oE,MAAMrB,KAAM8pE,EAASl+D,UAAWs+D,OAClDC,EAAwB,IAAI71D,IAAK21D,GAEvC,GAAKE,EAAsB51D,MAAQ01D,EAAgBjxE,OAQlD,MAAM,IAAI,EAAe,iCAAkCgnE,EAAcE,WAAY,CAAE/nD,YAGxF,GAAKgyD,EAAsB51D,MAAQ4D,EAAQ6jB,WAS1C,MAAM,IAAI,EAAe,oCAAqCgkC,EAAcE,WAAY,CAAE/nD,WAE5F,CA51CEiyD,CAAuB5mE,EAAK7C,KAAMmpE,EAAU9J,GAG5CmJ,EAAU3lE,EAAK7C,KAAMq/D,EAAcwB,YAEnC,MAAMnuB,EAAe2sB,EAAc9B,OAAOD,eAAgBz6D,EAAKgiB,MAAM4b,OAErE4+B,EAAc9B,OAAOnf,aAAcv7C,EAAK7C,KAAMs3C,GAC9C+nB,EAAc9kC,OAAO90B,OAAQitC,EAAc4E,GAG3C+nB,EAAc8C,kBAAmBt/D,EAAK7C,MAs1CxC,SACCs3C,EACA6xB,EACA9J,EACA/6D,GAGA+6D,EAAc9B,OAAOnuD,GAAoC,sBAAuBs6D,EAAuB,CAAE/8D,SAAU,YAEnH,IAAIg9D,EAAkC,KAClCC,EAA4C,KAGhD,KAAQD,EAAaC,KAAsBT,EAC1CR,GAAwBrxB,EAAasyB,EAAkBvK,EAAe/6D,GAEtE+6D,EAAc9kC,OAAO8a,KACpBgqB,EAAc9kC,OAAOud,cAAe6xB,GACpCtK,EAAc9kC,OAAOod,qBAAsBgyB,IAE5CtK,EAAc9kC,OAAOv0B,OAAQ2jE,GAK9B,SAASD,EAAuBt0D,EAAcvS,GAM7C,MAAM2U,EAAU3U,EAAK84D,cAAc1xB,UAG7BtlC,EAAQilE,EAAkB7gE,QAASyO,GAEpC7S,EAAQ,IAIb9B,EAAK6vC,aAAe7vC,EAAK06D,OAAO3B,eAAgB+N,EAAchlE,GAC/D,CAlBA06D,EAAc9B,OAAOpxD,IAAK,sBAAuBu9D,EAmBlD,CA73CEG,CAAWvyB,EAAa6xB,EAAU9J,EAAe,CAAEsC,aAAc9+D,EAAK8+D,cAAgB,GAivBrF,CAAEh1D,SAAUoc,EAAO+/C,mBAAqB,WAGzCvJ,EAAWnwD,GAAgC,gBAAiB25D,GAAqB5wE,GAAS,CAAEwU,SAAU,OAAS,CAEjH,CAppDmBm9D,CAA4B/gD,GAC9C,CAwFO,kBAAAghD,CACNhhD,GAiBA,OAAOhpB,KAAK6N,IA0jDd,SAAqCmb,GASpCA,EAAS,GAAWA,GAEpB,IAAI5wB,EAAQ4wB,EAAO5wB,MAEE,iBAATA,IACXA,EAAQ,CAAErB,IAAKqB,IAGhB,IAAIoZ,EAAY,aAAcpZ,EAAMrB,MAE/BqB,EAAM4J,OACVwP,GAAa,IAAMpZ,EAAM4J,MAG1B,GAAK5J,EAAM8S,OACV,UAAY++D,KAAc7xE,EAAM8S,OAC7B8d,EAAOiR,KAAegwC,GAAe1B,GAA4Bv/C,EAAOiR,KAAegwC,GAAc,kBAGxGjhD,EAAOiR,KAAOsuC,GAA0Bv/C,EAAOiR,KAAa,aAG7D,MAAMuuC,EAAiB0B,GAA0DlhD,GAEjF,OAASw2C,IACRA,EAAWnwD,GACVmC,EA98BI,SAAeg3D,GACrB,MAAO,CACNnzD,EACAvS,EAOAw8D,KAEA,IAAMA,EAAcwB,WAAW7+D,KAAMa,EAAK7C,KAAMoV,EAAIrT,MACnD,OAKD,MAAMmoE,EAAiB3B,EAAgB1lE,EAAK09D,kBAAmBlB,EAAex8D,GAGxEsnE,EAAiB5B,EAAgB1lE,EAAK29D,kBAAmBnB,EAAex8D,GAE9E,IAAMqnE,IAAmBC,EACxB,OAGD9K,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MAEjD,MAAMqoE,EAAa/K,EAAc9kC,OAC3Bwc,EAAgBqzB,EAAW7mE,SAASyxB,UAE1C,GAAKnyB,EAAK7C,gBAAgB,IAAkB6C,EAAK7C,gBAAgB,GAEhEoqE,EAAWvzB,KAAME,EAAc3J,gBAAkB+8B,OAC3C,CAEN,IAAI3jB,EAAY6Y,EAAc9B,OAAOH,YAAav6D,EAAKgiB,OAGvB,OAA3BhiB,EAAK09D,mBAA8B2J,IACvC1jB,EAAY4jB,EAAWnzB,OAAQuP,EAAW0jB,IAGX,OAA3BrnE,EAAK29D,mBAA8B2J,GACvCC,EAAWvzB,KAAM2P,EAAW2jB,EAE9B,EAEF,CA85BGtzB,CAAM0xB,GACN,CAAE57D,SAAUoc,EAAO+/C,mBAAqB,UACxC,CAEH,CAlmDmBuB,CAA4BthD,GAC9C,CAmFO,oBAAAuhD,CACNvhD,GAiBA,OAAOhpB,KAAK6N,IA+gDd,SAAuCmb,GAStCA,EAAS,GAAWA,GAEpB,IAAI5wB,EAAQ4wB,EAAO5wB,MAEE,iBAATA,IACXA,EAAQ,CAAErB,IAAKqB,IAGhB,IAAIoZ,EAAY,aAAcpZ,EAAMrB,MAE/BqB,EAAM4J,OACVwP,GAAa,IAAMpZ,EAAM4J,MAG1B,GAAK5J,EAAM8S,OACV,UAAY++D,KAAc7xE,EAAM8S,OAC7B8d,EAAOiR,KAAegwC,GAAeO,GAA8BxhD,EAAOiR,KAAegwC,SAG5FjhD,EAAOiR,KAAOuwC,GAA4BxhD,EAAOiR,MAGlD,MAAMuuC,EAAiB0B,GAAmDlhD,GAE1E,OAASw2C,IAniBV,IAA0BiL,EAoiBxBjL,EAAWnwD,GACVmC,GAriBuBi5D,EAsiBNjC,EAriBZ,CACNnzD,EACAvS,EAOAw8D,KAEA,IAAMA,EAAcwB,WAAW7+D,KAAMa,EAAK7C,KAAMoV,EAAIrT,MACnD,OAGD,MAAM0oE,EAAeD,EAAkB3nE,EAAK09D,kBAAmBlB,EAAex8D,GACxEm2C,EAAewxB,EAAkB3nE,EAAK29D,kBAAmBnB,EAAex8D,GAE9E,IAAM4nE,IAAiBzxB,EACtB,OAGDqmB,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MAEjD,MAAMu1C,EAAc+nB,EAAc9B,OAAOf,cAAe35D,EAAK7C,MACvDoqE,EAAa/K,EAAc9kC,OAIjC,IAAM+c,EAyCL,MAAM,IAAI,EAAe,4CAA6C+nB,EAAcE,WAAY18D,GAIjG,GAAgC,OAA3BA,EAAK09D,mBAA8BkK,EACvC,GAAyB,SAApBA,EAAa3zE,IAAiB,CAClC,MAAM2nC,EAAU,GAASgsC,EAAalyE,OAEtC,UAAY2tC,KAAazH,EACxB2rC,EAAWtvC,YAAaoL,EAAWoR,EAErC,MAAO,GAAyB,SAApBmzB,EAAa3zE,IAAiB,CACzC,MAAME,EAAOD,OAAOC,KAAMyzE,EAAalyE,OAEvC,UAAYzB,KAAOE,EAClBozE,EAAW11B,YAAa59C,EAAKwgD,EAE/B,MACC8yB,EAAW3jE,gBAAiBgkE,EAAa3zE,IAAKwgD,GAKhD,GAAgC,OAA3Bz0C,EAAK29D,mBAA8BxnB,EACvC,GAAyB,SAApBA,EAAaliD,IAAiB,CAClC,MAAM2nC,EAAU,GAASua,EAAazgD,OAEtC,UAAY2tC,KAAazH,EACxB2rC,EAAWzuC,SAAUuK,EAAWoR,EAElC,MAAO,GAAyB,SAApB0B,EAAaliD,IAAiB,CACzC,MAAME,EAAOD,OAAOC,KAAMgiD,EAAazgD,OAEvC,UAAYzB,KAAOE,EAClBozE,EAAW31B,SAAU39C,EAAOkiD,EAAazgD,MAAmCzB,GAAOwgD,EAErF,MACC8yB,EAAW5kE,aAAcwzC,EAAaliD,IAAKkiD,EAAazgD,MAAiB++C,EAE3E,GAybC,CAAE3qC,SAAUoc,EAAO+/C,mBAAqB,UACxC,CAEH,CAvjDmB4B,CAA8B3hD,GAChD,CAqEO,eAAA4hD,CAAiB5hD,GAKvB,OAAOhpB,KAAK6N,IAy/Cd,SAAkCmb,GAKjC,MAAMiR,EAAOsuC,GAA0Bv/C,EAAOiR,KAAM,MAEpD,OAASulC,IA52BH,IAA0BgJ,EA62B/BhJ,EAAWnwD,GACV,aAAc2Z,EAAO5wB,SA92BSowE,EA+2BbvuC,EA92BZ,CACN5kB,EACAvS,EAKAw8D,KAIAx8D,EAAK+nE,WAAY,EACjB,MAAMC,EAAmBtC,EAAgB1lE,EAAMw8D,GAE/Cx8D,EAAK+nE,WAAY,EACjB,MAAME,EAAiBvC,EAAgB1lE,EAAMw8D,GAE7C,IAAMwL,IAAqBC,EAC1B,OAGD,MAAMrK,EAAc59D,EAAK49D,YAKzB,GAAKA,EAAY11B,cAAgBs0B,EAAcwB,WAAWrC,QAASiC,EAAarrD,EAAIrT,MACnF,OAID,UAAYxJ,KAASkoE,EACpB,IAAMpB,EAAcwB,WAAWrC,QAASjmE,EAAMyH,KAAMoV,EAAIrT,MACvD,OAIF,MAAMw7D,EAAS8B,EAAc9B,OACvB6M,EAAa/K,EAAc9kC,OAGjC6vC,EAAW3kE,OAAQ83D,EAAOD,eAAgBmD,EAAYhgC,OAASoqC,GAC/DxL,EAAc9B,OAAOd,oBAAqBoO,EAAkBhoE,EAAKw5D,YAG3DoE,EAAY11B,cACjBq/B,EAAW3kE,OAAQ83D,EAAOD,eAAgBmD,EAAY//B,KAAOoqC,GAC7DzL,EAAc9B,OAAOd,oBAAqBqO,EAAgBjoE,EAAKw5D,aAGhEjnD,EAAIlJ,MAAM,GA6zBT,CAAES,SAAUoc,EAAO+/C,mBAAqB,WAEzCvJ,EAAWnwD,GACV,gBAAiB2Z,EAAO5wB,SAnzBnB,CACNid,EACAvS,EACAw8D,KAEA,MAAM3C,EAAW2C,EAAc9B,OAAOE,qBAAsB56D,EAAKw5D,YAEjE,GAAMK,EAAN,CAIA,UAAYllD,KAAWklD,EACtB2C,EAAc9B,OAAOZ,4BAA6BnlD,EAAS3U,EAAKw5D,YAChEgD,EAAc9kC,OAAO5oB,MAAO0tD,EAAc9kC,OAAOsd,cAAergC,GAAWA,GAG5E6nD,EAAc9kC,OAAOgd,yBAA0B10C,EAAKw5D,YAEpDjnD,EAAIlJ,MATJ,CASU,GAmyBT,CAAES,SAAUoc,EAAO+/C,mBAAqB,UACxC,CAEH,CA5gDmBiC,CAAyBhiD,GAC3C,CA4DO,iBAAAiiD,CAAmBjiD,GAKzB,OAAOhpB,KAAK6N,IA+/Cd,SAAoCmb,GAKnC,OAASw2C,IAngBV,IAAwB0L,EAogBtB1L,EAAWnwD,GACV,aAAc2Z,EAAO5wB,SArgBA8yE,EAsgBNliD,EAAOiR,KArgBjB,CACN5kB,EACAvS,EAMAw8D,KAEA,IAAMx8D,EAAK7C,KACV,OAGD,KAAQ6C,EAAK7C,gBAAgB,IAAkB6C,EAAK7C,gBAAgB,IAA6B6C,EAAK7C,KAAKiO,GAAI,eAC9G,OAGD,MAAMi9D,EAAaC,GAAmBF,EAAqBpoE,EAAMw8D,GAEjE,IAAM6L,EACL,OAGD,IAAM7L,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAGD,MAAMqoE,EAAa/K,EAAc9kC,OAC3B+c,EAAc8zB,GAA0ChB,EAAYc,GACpEn0B,EAAgBqzB,EAAW7mE,SAASyxB,UAE1C,GAAKnyB,EAAK7C,gBAAgB,IAAkB6C,EAAK7C,gBAAgB,GAChEoqE,EAAWvzB,KAAME,EAAc3J,gBAAkBkK,OAC3C,CACN,MAAMkP,EAAY6Y,EAAc9B,OAAOH,YAAav6D,EAAKgiB,OACnDwmD,EAAiBjB,EAAWvzB,KAAM2P,EAAWlP,GAEnD,UAAY9/B,KAAW6zD,EAAep/B,WACrC,GAAKz0B,EAAQvJ,GAAI,qBAAwBuJ,EAAQ4lB,UAAWka,GAAgB,CAC3E+nB,EAAc9B,OAAOd,oBAAqBjlD,EAAS3U,EAAKw5D,YAIxD,KACD,CAEF,IAudC,CAAE1vD,SAAUoc,EAAO+/C,mBAAqB,WAEzCvJ,EAAWnwD,GACV,aAAc2Z,EAAO5wB,QArcxB,SAA2B8yE,GAC1B,MAAO,CACN71D,EACAvS,EAKAw8D,KAEA,IAAMx8D,EAAK7C,KACV,OAGD,KAAQ6C,EAAK7C,gBAAgB,IAC5B,OAGD,MAAMkrE,EAAaC,GAAmBF,EAAqBpoE,EAAMw8D,GAEjE,IAAM6L,EACL,OAGD,IAAM7L,EAAcwB,WAAW7+D,KAAMa,EAAK7C,KAAMoV,EAAIrT,MACnD,OAGD,MAAMu1C,EAAc+nB,EAAc9B,OAAOf,cAAe35D,EAAK7C,MAE7D,GAAKs3C,GAAeA,EAAY/Q,kBAAmB,gBAAmB,CAErE84B,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MAGjD,UAAYxJ,KAAS,GAAWg0C,UAAW1pC,EAAK7C,MAC/Cq/D,EAAcwB,WAAWrC,QAASjmE,EAAMyH,KAAMoV,EAAIrT,MAGtBu1C,EAAY/Q,kBAAmB,eAE5D+kC,CAAsBh0B,EAAa4zB,EAAY7L,EAAc9kC,QAE7D8kC,EAAc9B,OAAOd,oBAAqBnlB,EAAaz0C,EAAKw5D,WAC7D,EAEF,CAwZGkP,CAAkBxiD,EAAOiR,MACzB,CAAErtB,SAAUoc,EAAO+/C,mBAAqB,WAEzCvJ,EAAWnwD,GACV,gBAAiB2Z,EAAO5wB,QArY3B,SAA0B8yE,GACzB,MAAO,CACN71D,EACAvS,EAIAw8D,KAGA,GAAKx8D,EAAK49D,YAAY11B,YACrB,OAGD,MAAMmgC,EAAaC,GAAmBF,EAAqBpoE,EAAMw8D,GAEjE,IAAM6L,EACL,OAID,MAAMM,EAAuBJ,GAA0C/L,EAAc9kC,OAAQ2wC,GAGvFxO,EAAW2C,EAAc9B,OAAOE,qBAAsB56D,EAAKw5D,YAEjE,GAAMK,EAAN,CAIA,UAAYllD,KAAWklD,EAGtB,GAFA2C,EAAc9B,OAAOZ,4BAA6BnlD,EAAS3U,EAAKw5D,YAE3D7kD,EAAQvJ,GAAI,oBAChBoxD,EAAc9kC,OAAO0c,OAAQooB,EAAc9kC,OAAOsd,cAAergC,GAAWg0D,OACtE,CAE0Bh0D,EAAQ+uB,kBAAmB,kBAE3DklC,CAAyBj0D,EAAS0zD,EAAWtrE,GAAKy/D,EAAc9kC,OACjE,CAGD8kC,EAAc9kC,OAAOgd,yBAA0B10C,EAAKw5D,YAEpDjnD,EAAIlJ,MAjBJ,CAiBU,CAEZ,CAuVGw/D,CAAiB3iD,EAAOiR,MACxB,CAAErtB,SAAUoc,EAAO+/C,mBAAqB,UACxC,CAEH,CArhDmB6C,CAA2B5iD,GAC7C,CA0HO,YAAA6iD,CAAc7iD,GAKpB,OAAOhpB,KAAK6N,IAm1Cd,SAA+Bmb,GAK9BA,EAAS,GAAWA,GAEpB,MAAMqxB,EAAQrxB,EAAO5wB,MACrB,IAAI6hC,EAAOjR,EAAOiR,KAGZA,IACLA,EAAOqiC,IAAA,CACNjiB,QACAr4C,KAAMs6D,EAAWrqD,OAAQ+W,EAAO5wB,MAAME,OAAS,MAIjD,OAASknE,IAnzBV,IAA2BsM,EAozBzBtM,EAAWnwD,GACV,aAAcgrC,KArzBUyxB,EAszBN7xC,EArzBb,CACN5kB,EACAvS,EAIAw8D,KAEA,MAAMyM,EAAiBD,EAAahpE,EAAKw5D,WAAYgD,GAErD,IAAMyM,EACL,OAGD,MAAMrL,EAAc59D,EAAK49D,YAEnBpB,EAAcwB,WAAWrC,QAASiC,EAAarrD,EAAIrT,QAKzDgqE,GAAsBtL,GAAa,EAAOpB,EAAex8D,EAAMipE,GAC/DC,GAAsBtL,GAAa,EAAMpB,EAAex8D,EAAMipE,GAE9D12D,EAAIlJ,OAAM,GA8xBT,CAAES,SAAUoc,EAAO+/C,mBAAqB,WAEzCvJ,EAAWnwD,GACV,gBAAiBgrC,IAhsBpB,SAA2ByxB,GAC1B,MAAO,CACNz2D,EACAvS,EACAw8D,KAEA,MAAM2M,EAAWH,EAAahpE,EAAKw5D,WAAYgD,GAE/C,IAAM2M,EACL,OAGD,MAAMtP,EAAW2C,EAAc9B,OAAOE,qBAAsB56D,EAAKw5D,YAEjE,GAAMK,EAAN,CAIA,UAAYllD,KAAWklD,EACtB2C,EAAc9B,OAAOZ,4BAA6BnlD,EAAS3U,EAAKw5D,YAE3D7kD,EAAQvJ,GAAI,qBAChBg+D,EAA2B,QAASD,EAAS5xB,qBAAuB5iC,GACpEy0D,EAA2B,QAASD,EAAS5xB,oBAAsB5iC,GACnEy0D,EAA2B,QAASD,EAAS5xB,mBAAqB5iC,GAClEy0D,EAA2B,QAASD,EAAS5xB,kBAAoB5iC,IAEjE6nD,EAAc9kC,OAAO5oB,MAAO0tD,EAAc9kC,OAAOsd,cAAergC,GAAWA,GAI7E6nD,EAAc9kC,OAAOgd,yBAA0B10C,EAAKw5D,YAEpDjnD,EAAIlJ,MAjBJ,CAmBA,SAAS+/D,EAA2BtlC,EAAuBnvB,GAC1D,GAAKA,EAAQwuB,aAAcW,GAAkB,CAC5C,MAAMo2B,EAAc,IAAIppD,IAAK6D,EAAQ4jB,aAAcuL,GAAiB7pC,MAAO,MAE3EigE,EAAYrrD,OAAQs6D,EAAUjqE,MAEL,GAApBg7D,EAAYnpD,KAChByrD,EAAc9kC,OAAO9zB,gBAAiBkgC,EAAenvB,GAErD6nD,EAAc9kC,OAAO/0B,aAAcmhC,EAAejmC,MAAMrB,KAAM09D,GAAc9/D,KAAM,KAAOua,EAE3F,CACD,EAEF,CAgpBG00D,CAAkBlyC,GAClB,CAAErtB,SAAUoc,EAAO+/C,mBAAqB,UACxC,CAEH,CAj3CmBqD,CAAsBpjD,GACxC,EA6FM,SAASqiD,GAA0C7wC,EAAwB2wC,GACjF,MAAM5zB,EAAc/c,EAAOoZ,uBAAwB,OAAQu3B,EAAW5lE,YAYtE,OAVK4lE,EAAWzsC,SACf6Y,EAAYhQ,UAAW4jC,EAAWzsC,SAGC,iBAAxBysC,EAAWv+D,WACpB2qC,EAAqBpG,UAAYg6B,EAAWv+D,UAG7C2qC,EAAqBnG,IAAM+5B,EAAWtrE,GAEjC03C,CACR,CAseA,SAASy0B,GACRlnD,EACAunD,EACA/M,EACAx8D,EACAipE,GAEA,MAAMnQ,EAAgByQ,EAAUvnD,EAAM4b,MAAQ5b,EAAM6b,IAC9C2rC,EAAe1Q,EAAc1xB,WAAa0xB,EAAc1xB,UAAUh8B,GAAI,WAAc0tD,EAAc1xB,UAAY,KAC9GqiC,EAAgB3Q,EAAczxB,YAAcyxB,EAAczxB,WAAWj8B,GAAI,WAAc0tD,EAAczxB,WAAa,KAExH,GAAKmiC,GAAgBC,EAAgB,CACpC,IAAIpQ,EACAz/B,EAGC2vC,GAAWC,IAAiBD,IAAYE,GAG5CpQ,EAAemQ,EACf5vC,GAAW,IAIXy/B,EAAeoQ,EACf7vC,GAAW,GAGZ,MAAM6a,EAAc+nB,EAAc9B,OAAOf,cAAeN,GAIxD,GAAK5kB,EAGJ,YAYH,SACCA,EACA80B,EACA3vC,EACA4iC,EACAx8D,EACAipE,GAEA,MAAMnlC,EAAgB,QAASmlC,EAAe1xB,SAAWgyB,EAAU,QAAU,SAAW3vC,EAAW,SAAW,UAExGsgC,EAAczlB,EAAYtR,aAAcW,GAAkB2Q,EAAYlc,aAAcuL,GAAiB7pC,MAAO,KAAQ,GAG1HigE,EAAY59D,QAAS2sE,EAAe/pE,MAEpCs9D,EAAc9kC,OAAO/0B,aAAcmhC,EAAeo2B,EAAY9/D,KAAM,KAAOq6C,GAC3E+nB,EAAc9B,OAAOd,oBAAqBnlB,EAAaz0C,EAAKw5D,WAC7D,CA/BGkQ,CAAyBj1B,EAAa80B,EAAS3vC,EAAU4iC,EAAex8D,EAAMipE,EAIhF,EAgCD,SACCtmD,EACA4mD,EACA/M,EACAx8D,EACAipE,GAEA,MAAMlO,EAAkB,GAAIkO,EAAe1xB,SAAWgyB,EAAU,QAAU,QAEpEjnC,EAAQ2mC,EAAe/pE,KAAO,CAAE,KAAQ+pE,EAAe/pE,MAAS,KAChEu1C,EAAc+nB,EAAc9kC,OAAO6Z,gBAAiBwpB,EAAiBz4B,GAE3Ek6B,EAAc9kC,OAAO90B,OAAQ+f,EAAU8xB,GACvC+nB,EAAc9B,OAAOd,oBAAqBnlB,EAAaz0C,EAAKw5D,WAC7D,CA1CCmQ,CAFqBnN,EAAc9B,OAAOD,eAAgB3B,GAErByQ,EAAS/M,EAAex8D,EAAMipE,EACpE,CAgyBA,SAASzD,GAA6BlwE,GAmBrC,MAdqB,iBAATA,IACXA,EAAQ,CAAE4J,KAAM5J,IAIXA,EAAMmN,WAEC5E,MAAMC,QAASxI,EAAMmN,cACjCnN,EAAMmN,WAAa,CAAEnN,EAAMmN,aAF3BnN,EAAMmN,WAAa,GAMpBnN,EAAMkf,WAAalf,EAAMkf,SAElBlf,CACR,CAgBA,SAASmwE,GACRtuC,EACAyyC,GAEA,MAAoB,mBAARzyC,EAEJA,EAGC,CAAE0yC,EAAoBrN,IAOhC,SACCsN,EACAtN,EACAoN,GAEqC,iBAAzBE,IAEXA,EAAwB,CAAE5qE,KAAM4qE,IAGjC,IAAIn1D,EACJ,MAAM4yD,EAAa/K,EAAc9kC,OAC3Bj1B,EAAavO,OAAO+wB,OAAQ,CAAC,EAAG6kD,EAAsBrnE,YAE5D,GAAwB,aAAnBmnE,EACJj1D,EAAU4yD,EAAWt2B,uBAAwB64B,EAAsB5qE,KAAMuD,QACnE,GAAwB,aAAnBmnE,EAAiC,CAC5C,MAAMnoE,EAAU,CACfqI,SAAUggE,EAAsBhgE,UAAY,GAAqB4kC,kBAGlE/5B,EAAU4yD,EAAWz2B,uBAAwBg5B,EAAsB5qE,KAAMuD,EAAYhB,EACtF,MAECkT,EAAU4yD,EAAWh2B,gBAAiBu4B,EAAsB5qE,KAAMuD,GAGnE,GAAKqnE,EAAsB/tC,OAAS,CACnC,MAAM5nC,EAAOD,OAAOC,KAAM21E,EAAsB/tC,QAEhD,UAAY9nC,KAAOE,EAClBozE,EAAW31B,SAAU39C,EAAK61E,EAAsB/tC,OAAQ9nC,GAAO0gB,EAEjE,CAEA,GAAKm1D,EAAsBluC,QAAU,CACpC,MAAMA,EAAUkuC,EAAsBluC,QAEtC,GAAuB,iBAAXA,EACX2rC,EAAWzuC,SAAU8C,EAASjnB,QAE9B,UAAY0uB,KAAazH,EACxB2rC,EAAWzuC,SAAUuK,EAAW1uB,EAGnC,CAEA,OAAOA,CACR,CAtDEo1D,CAAiC5yC,EAAMqlC,EAAeoN,EACxD,CAuDA,SAASxC,GAA+FlhD,GACvG,OAAKA,EAAO5wB,MAAM8S,OACR,CAAE4hE,EAA0BxN,EAAsCx8D,KAC1E,MAAMm3B,EAAOjR,EAAOiR,KAAM6yC,GAE1B,OAAK7yC,EACGA,EAAM6yC,EAAqBxN,EAAex8D,GAG3C,IAAI,EAGLkmB,EAAOiR,IAEhB,CAQA,SAASuwC,GAA4BvwC,GACpC,MAAoB,iBAARA,EACJ6yC,IAAA,CAA2B/1E,IAAKkjC,EAAMzhC,MAAOs0E,IAC1B,iBAAR7yC,EAEbA,EAAKzhC,MACF,IAAMyhC,EAIN6yC,IAAA,CAA2B/1E,IAAKkjC,EAAKljC,IAAKyB,MAAOs0E,IAIlD7yC,CAET,CAKA,SAASmxC,GACRF,EACApoE,EAIAw8D,GAGA,MAAM6L,EAA2C,mBAAvBD,EACzBA,EAAqBpoE,EAAMw8D,GAC3B4L,EAED,OAAMC,GAKAA,EAAWv+D,WAChBu+D,EAAWv+D,SAAW,IAIjBu+D,EAAWtrE,KAChBsrE,EAAWtrE,GAAKiD,EAAKw5D,YAGf6O,GAbC,IAcT,CAuCA,SAASnC,GAAqB5wE,GAC7B,MAAM20E,EA9BP,SAAsC30E,GACrC,MAAO,CAAE+G,EAAiBo7B,KACzB,IAAMp7B,EAAK+O,GAAI,UAAW9V,EAAM4J,MAC/B,OAAO,EAGR,GAAoB,aAAfu4B,EAAOtvB,MACX,GAAK7S,EAAMmN,WAAWgqB,SAAUgL,EAAOkqB,cACtC,OAAO,OAIR,GAAKrsD,EAAMkf,SACV,OAAO,EAIT,OAAO,CAAK,CAEd,CAWuB01D,CAA6B50E,GAEnD,MAAO,CACNid,EACAvS,KAEA,MAAMmqE,EAAqD,GAErDnqE,EAAKoqE,sBACVpqE,EAAKoqE,oBAAsB,IAAIt5D,KAGhC,UAAY2mB,KAAUz3B,EAAKm9D,QAAU,CAGpC,MAAM9gE,EAAsB,aAAfo7B,EAAOtvB,KAAsBsvB,EAAOzV,MAAM4b,MAAMwJ,UAAY3P,EAAO9U,SAAS7mB,OAEzF,GAAMO,GAAS4tE,EAAe5tE,EAAMo7B,IAOpC,IAAMz3B,EAAKoqE,oBAAoBt/D,IAAKzO,GAAS,CAC5C2D,EAAKoqE,oBAAoBr/D,IAAK1O,GAE9B,MAAMsmB,EAAW,GAAcskB,cAAe5qC,GAC9C,IAAIguE,EAAcF,EAAe30E,OAKjC,QAAUkC,EAAIyyE,EAAe30E,OAAS,EAAGkC,GAAK,EAAGA,IAAM,CACtD,MAAM+/B,EAAS0yC,EAAgBzyE,GAEzB4yE,GADgC,aAAf7yC,EAAOtvB,KAAsBsvB,EAAOzV,MAAM4b,MAAQnG,EAAO9U,UACxCklB,YAAallB,GAErD,GAAyB,UAApB2nD,GAA+C,UAAf7yC,EAAOtvB,MAAwC,QAApBmiE,EAC/D,MAGDD,EAAc3yE,CACf,CAEAyyE,EAAezlE,OAAQ2lE,EAAa,EAAG,CACtCliE,KAAM,SACNjJ,KAAQ7C,EAAuB6C,KAC/ByjB,WACAntB,OAAQ,GACC,CACT2S,KAAM,WACNjJ,KAAQ7C,EAAuB6C,KAC/ByjB,WACAntB,OAAQ,GAEV,OAtCC20E,EAAertE,KAAM26B,EAuCvB,CAEAz3B,EAAKm9D,QAAUgN,CAAc,CAE/B,CAUA,SAASnE,GAAgB1wE,GACxB,MAAO,CAAE+G,EAAM2hE,EAAYv8D,EAAU,CAAC,KACrC,MAAM8M,EAAS,CAAE,UAGjB,UAAYu1B,KAAiBxuC,EAAMmN,WAC7BpG,EAAK8mC,aAAcW,IACvBv1B,EAAOzR,KAAM,aAAcgnC,KAI7B,QAAMv1B,EAAOkF,OAAOjH,GAASwxD,EAAW7+D,KAAM9C,EAAMmQ,OAI9C/K,EAAQokE,WACbt3D,EAAOzT,SAAS0R,GAASwxD,EAAWrC,QAASt/D,EAAMmQ,MAG7C,EAAI,CAEb,CAsHA,SAASs5D,GACRrxB,EACA81B,EACA/N,EACA/6D,GAGA,UAAY+oE,KAAkBD,EAEvBE,GAAch2B,EAAYjhD,KAAMg3E,EAAgBhO,EAAe/6D,IAEpE+6D,EAAc4C,YAAaoL,EAG9B,CAOA,SAASC,GACR3Y,EACA4Y,EACAlO,EACA/6D,GAEA,MAAM,OAAEi2B,EAAM,OAAEgjC,GAAW8B,EAG3B,IAAM/6D,EAAQq9D,aACb,OAAO,EAGR,MAAM6L,EAAgBjQ,EAAOf,cAAe+Q,GAG5C,SAAMC,GAAiBA,EAAcn3E,MAAQs+D,OAKvC0K,EAAc+C,aAAcoL,KAKlCjzC,EAAO8a,KACN9a,EAAOsd,cAAe21B,GACtBjQ,EAAOD,eAAgB,GAAcxzB,cAAeyjC,MAG9C,GACR,CASA,SAAS9E,GACRzoE,EACA6gE,GACA,UAAE6H,GAAuC,CAAC,GAE1C,OAAKA,EACG7H,EAAW7+D,KAAMhC,EAAM,UAEvB6gE,EAAWrC,QAASx+D,EAAM,SAEnC,CC9sFO,SAASytE,GAAyBlzC,GACxC,MAAM,OAAEkpC,EAAM,SAAElgE,GAAag3B,EAAOpiC,MAEpC,UAAY9B,KAAQkN,EAASmqE,WAC5B,GAAKr3E,EAAK6rC,UAAYuhC,EAAOyF,WAAY7yE,EAAM,UAEzCotE,EAAOyF,WAAY7yE,EAAM,aAM7B,OALAkkC,EAAOquC,cAAe,YAAavyE,IAK5B,EAKV,OAAO,CACR,CAUO,SAASs3E,GACfnoD,EACAooD,EACAnK,GAEA,MAAMn2D,EAAUm2D,EAAOoK,cAAeroD,GAGtC,QAAMi+C,EAAOyF,WAAY57D,EAAS,gBAK5Bm2D,EAAOyF,WAAY57D,EAAQ3N,KAAM,aAAeiuE,EAKvD,CAUO,SAASE,GAAiBtoD,EAAoB+U,GACpD,MAAMwzC,EAAYxzC,EAAOl1B,cAAe,aAIxC,OAFAk1B,EAAO90B,OAAQsoE,EAAWvoD,GAEnB+U,EAAOkd,iBAAkBs2B,EAAW,EAC5C,C,8RC9Ce,MAAMC,WAAsBjG,GAsDnC,gBAAAK,CAAkBr/C,GAKxB,OAAOhpB,KAAK6N,IAAKqgE,GAAwBllD,GAC1C,CAuFO,kBAAAmlD,CAAoBnlD,GAQ1B,OAAOhpB,KAAK6N,IAscd,SAAmCmb,GAQlCA,EAAS,GAAWA,GAEpBolD,GAA+BplD,GAE/B,MAAMqlD,EAAYC,GAA6BtlD,GAAe,GAExD27B,EAAc4pB,GAA8BvlD,EAAOiR,MACnDzoB,EAAYmzC,EAAc,WAAYA,IAA0B,UAEtE,OAAS6a,IACRA,EAAWnwD,GAAwBmC,EAAW68D,EAAW,CAAEzhE,SAAUoc,EAAO+/C,mBAAqB,OAAS,CAE5G,CA1dmByF,CAA0BxlD,GAC5C,CA8HO,oBAAAuhD,CAAsBvhD,GAkB5B,OAAOhpB,KAAK6N,IA4Vd,SAAqCmb,GAepCA,EAAS,GAAWA,GAEpB,IAAIylD,EAAyB,MAEF,iBAAfzlD,EAAOiR,MAAoBjR,EAAOiR,KAAKljC,OAClD03E,EA4QF,SAA+CzlD,GACnB,iBAAfA,EAAOiR,OAClBjR,EAAOiR,KAAO,CAAEljC,IAAKiyB,EAAOiR,OAG7B,MAAMljC,EAAciyB,EAAOiR,KAAKljC,IAChC,IAAI23E,EAEJ,GAAY,SAAP33E,GAAyB,SAAPA,EAAiB,CAGvC23E,EAAa,CACZ,CAHsB,SAAP33E,EAAiB,UAAY,UAG/BiyB,EAAOiR,KAAKzhC,MAE3B,KAAO,CAGNk2E,EAAa,CACZnpE,WAAY,CACX,CAAExO,QAJsC,IAArBiyB,EAAOiR,KAAKzhC,MAAuB,UAAYwwB,EAAOiR,KAAKzhC,OAOjF,CAEKwwB,EAAOiR,KAAKj4B,OAChB0sE,EAAW1sE,KAAOgnB,EAAOiR,KAAKj4B,MAK/B,OAFAgnB,EAAOiR,KAAOy0C,EAEP33E,CACR,CA3SY43E,CAAsC3lD,IAGjDolD,GAA+BplD,EAAQylD,GAEvC,MAAMJ,EAAYC,GAA6BtlD,GAAe,GAE9D,OAASw2C,IACRA,EAAWnwD,GAAwB,UAAWg/D,EAAW,CAAEzhE,SAAUoc,EAAO+/C,mBAAqB,OAAS,CAE5G,CA1XmB6F,CAA4B5lD,GAC9C,CAqDO,eAAA6lD,CAAiB7lD,GAKvB,OAAOhpB,KAAK6N,IA6Ud,SAAgCmb,GAK/B,MAAM5wB,EA6bP,SAA8CA,GAC7C,MAAO,CAAEm/C,EAAa+nB,KACrB,MAAMhD,EAA6B,iBAATlkE,EAAoBA,EAAQA,EAAOm/C,EAAa+nB,GAE1E,OAAOA,EAAc9kC,OAAOl1B,cAAe,UAAW,CAAE,YAAag3D,GAAc,CAErF,CAncewS,CAAqC9lD,EAAO5wB,OAE1D,OAAO81E,I,2HAAwB,IAAKllD,G,EAAL,CAAa5wB,S,qBAC7C,CArVmB22E,CAAuB/lD,GACzC,CAyEO,YAAAgmD,CAAchmD,GAKpB,OAAOhpB,KAAK6N,IA+Qd,SAA6Bmb,GAK5BA,EAAS,GAAWA,GAGdA,EAAO5wB,QACZ4wB,EAAO5wB,MAAQ4J,GACPA,EAAOgnB,EAAOiR,KAAO,IAAMj4B,EAAOgnB,EAAOiR,MAIlD,MAAMg1C,EAAmB,CACxBh1C,KAAMjR,EAAOiR,KACb7hC,MAAO4wB,EAAO5wB,OAGT82E,EAAiBC,GAA2BC,GAA6BH,EAAkB,UAC3FI,EAAeF,GAA2BC,GAA6BH,EAAkB,QAE/F,OAASzP,IACRA,EAAWnwD,GACV,WAAY2Z,EAAOiR,aACnBi1C,EACA,CAAEtiE,SAAUoc,EAAO+/C,mBAAqB,WAEzCvJ,EAAWnwD,GACV,WAAY2Z,EAAOiR,WACnBo1C,EACA,CAAEziE,SAAUoc,EAAO+/C,mBAAqB,WAezC,MAAMuG,EAAe,EAAWtiE,IAC1BuiE,EAAc,EAAWziE,QACzB0iE,EAAiB,EAAWpnE,IAAK4gB,EAAO+/C,mBAAsBwG,EAEpE/P,EAAWnwD,GACV,UAgBH,SAAkC2Z,GAIjC,MAAO,CACN3T,EACAvS,EACAw8D,KAEA,MAAMmQ,EAAW,QAASzmD,EAAOiR,OAqCjC,SAASy1C,EAAmBjqD,EAAyBkqD,GACpD,UAAYC,KAAkBD,EAAkB,CAC/C,MAAMrT,EAAatzC,EAAO5wB,MAAOw3E,EAAgBtQ,GAC3C7nD,EAAU6nD,EAAc9kC,OAAOl1B,cAAe,UAAW,CAAE,YAAag3D,IAE9EgD,EAAc9kC,OAAO90B,OAAQ+R,EAASgO,GAEjC3iB,EAAK+sE,YAAY5rD,QAASwB,GAC9B3iB,EAAK+sE,YAAc/sE,EAAK+sE,YAAYvlC,aAAc,GAElDxnC,EAAK+sE,YAAc/sE,EAAK+sE,YAAY/W,2BAA4BrzC,EAAU,GAG3E3iB,EAAKw6D,WAAax6D,EAAKw6D,WAAYxE,2BAA4BrzC,EAAU,GAAK,EAC/E,CACD,EA/CE65C,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAEvqE,WAAYkqE,EAAW,gBACvEnQ,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAEvqE,WAAYkqE,EAAW,kBACvEnQ,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAEvqE,WAAYkqE,EAAW,iBACvEnQ,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAEvqE,WAAYkqE,EAAW,qBASnE3sE,EAAKw6D,YACVtmE,OAAO+wB,OAAQjlB,EAAMw8D,EAAc6C,gBAAiBr/D,EAAKgtE,SAAUhtE,EAAK+sE,cAGpEvQ,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAEvqE,WAAYkqE,EAAW,gBAC9EC,EAAmB5sE,EAAKw6D,WAAY38B,IAAK79B,EAAKgtE,SAASz0C,aAAco0C,EAAW,cAAgB1yE,MAAO,MAGnGuiE,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAEvqE,WAAYkqE,EAAW,kBAC9EC,EAAmB5sE,EAAKw6D,WAAY38B,IAAK79B,EAAKgtE,SAASz0C,aAAco0C,EAAW,gBAAkB1yE,MAAO,MAGrGuiE,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAEvqE,WAAYkqE,EAAW,iBAC9EC,EAAmB5sE,EAAKw6D,WAAY58B,MAAO59B,EAAKgtE,SAASz0C,aAAco0C,EAAW,eAAiB1yE,MAAO,MAGtGuiE,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAEvqE,WAAYkqE,EAAW,mBAC9EC,EAAmB5sE,EAAKw6D,WAAY58B,MAAO59B,EAAKgtE,SAASz0C,aAAco0C,EAAW,iBAAmB1yE,MAAO,MAkB7G,CAEF,CA9EGgzE,CAAyBd,GACzB,CAAEriE,SAAU0iE,EAAeE,GAC3B,CAEH,CAvUmBQ,CAAoBhnD,GACtC,EA4ID,SAASklD,GAAwBllD,GAOhC,MAAMqlD,EAAYc,GAFlBnmD,EAAS,GAAWA,IAId27B,EAAc4pB,GAA8BvlD,EAAOiR,MACnDzoB,EAAYmzC,EAAc,WAAYA,IAA0B,UAEtE,OAAS6a,IACRA,EAAWnwD,GAAwBmC,EAAW68D,EAAW,CAAEzhE,SAAUoc,EAAO+/C,mBAAqB,UAAY,CAE/G,CA8PA,SAASwF,GAA8B0B,GACtC,MAA0B,iBAAdA,EACJA,EAGkB,iBAAdA,GAAoD,iBAAnBA,EAAWjuE,KAChDiuE,EAAWjuE,KAGZ,IACR,CAQA,SAASmtE,GAA2BnmD,GAInC,MAAMud,EAAU,IAAI,GAASvd,EAAOiR,MAEpC,MAAO,CACN5kB,EACAvS,EACAw8D,KAEA,MAAM4Q,EAAgB3pC,EAAQ1pC,MAAOiG,EAAKgtE,UAE1C,IAAMI,EACL,OAGD,MAAMrzE,EAAQqzE,EAAcrzE,MAK5B,GAFAA,EAAMmF,MAAO,GAEPs9D,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAUjzE,GACnD,OAGD,MAAMs/D,EAwBR,SACC/jE,EACAqkB,EACA6iD,GAEA,OAAKlnE,aAAiBkQ,SACdlQ,EAAOqkB,EAAO6iD,GAEdA,EAAc9kC,OAAOl1B,cAAelN,EAE7C,CAlCuB+3E,CAAiBnnD,EAAO5wB,MAAO0K,EAAKgtE,SAAUxQ,GAE7DnD,GAIAmD,EAAc8Q,WAAYjU,EAAcr5D,EAAK+sE,eAInDvQ,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAUjzE,GACjDyiE,EAAc6C,gBAAiBr/D,EAAKgtE,SAAU3T,GAC9CmD,EAAc+Q,uBAAwBlU,EAAcr5D,GAAM,CAE5D,CAwEA,SAASsrE,GAA+BplD,EAAasnD,EAAwC,MAC5F,MAAMC,EAA+C,OAA3BD,GACzB,CAAE/4B,GAA8BA,EAAYlc,aAAci1C,IAErDv5E,EAA6B,iBAAhBiyB,EAAO5wB,MAAoB4wB,EAAO5wB,MAAQ4wB,EAAO5wB,MAAMrB,IACpEyB,EAA+B,iBAAhBwwB,EAAO5wB,YAAkD,IAAtB4wB,EAAO5wB,MAAMI,MAAuB+3E,EAAoBvnD,EAAO5wB,MAAMI,MAE7HwwB,EAAO5wB,MAAQ,CAAErB,MAAKyB,QACvB,CAUA,SAAS81E,GACRtlD,EAOA8f,GAEA,MAAMvC,EAAU,IAAI,GAASvd,EAAOiR,MAEpC,MAAO,CACN5kB,EACAvS,EACAw8D,KAKA,IAAMx8D,EAAKw6D,YAAcx0B,EACxB,OAGD,MAAMjsC,EAAQ0pC,EAAQ1pC,MAAOiG,EAAKgtE,UAGlC,IAAMjzE,EACL,OAYD,IA6CF,SAAgCozE,EAAiBH,GAEhD,MAAMU,EAAoC,mBAAdP,EAA2BA,EAAYH,GAAaG,EAEhF,GAA4B,iBAAhBO,IAA6BjC,GAA8BiC,GACtE,OAAO,EAGR,OAAQA,EAAa9xC,UAAY8xC,EAAajrE,aAAeirE,EAAa3xC,MAC3E,CA/DO4xC,CAAuBznD,EAAOiR,KAAMn3B,EAAKgtE,iBAKtCjzE,EAAMA,MAAMmF,KAJnBnF,EAAMA,MAAMmF,MAAO,GAQds9D,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAUjzE,EAAMA,OACzD,OAGD,MAAM6zE,EAAW1nD,EAAO5wB,MAAMrB,IACxBkzE,EAAmD,mBAAtBjhD,EAAO5wB,MAAMI,MAC/CwwB,EAAO5wB,MAAMI,MAAOsK,EAAKgtE,SAAUxQ,GAAkBt2C,EAAO5wB,MAAMI,MAGnE,GAAoB,OAAfyxE,EACJ,OAKKnnE,EAAKw6D,YAEVtmE,OAAO+wB,OAAQjlB,EAAMw8D,EAAc6C,gBAAiBr/D,EAAKgtE,SAAUhtE,EAAK+sE,cAIzE,MAAMc,EAgDR,SACCrT,EACAsT,EAIA9nC,EACAw2B,GAEA,IAAInhE,GAAS,EAGb,UAAYgB,KAAQwB,MAAMrB,KAAMg+D,EAAWpxB,SAAU,CAAEpD,aAEhDw2B,EAAcoE,OAAOmE,eAAgB1oE,EAAMyxE,EAAe75E,OAMhEoH,GAAS,EAGJgB,EAAK8mC,aAAc2qC,EAAe75E,MAIvCuoE,EAAc9kC,OAAO/0B,aAAcmrE,EAAe75E,IAAK65E,EAAep4E,MAAO2G,IAG9E,OAAOhB,CACR,CA/E0B0yE,CAAgB/tE,EAAKw6D,WAAa,CAAEvmE,IAAK25E,EAAUl4E,MAAOyxE,GAAcnhC,EAASw2B,GAIpGqR,IAMCrR,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAE9tE,MAAM,MAC1DnF,EAAMA,MAAMmF,MAAO,GAGpBs9D,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAUjzE,EAAMA,OACxD,CAEF,CAgFA,SAASuyE,GACRpmD,EAIA/d,GASA,MAAO,CAENgvB,KAAM,GAAIjR,EAAOiR,QAAUhvB,IAC3B7S,MAVsD,CAAEm/C,EAAa+nB,KACrE,MAAM/U,EAAWhT,EAAYlc,aAAc,QACrCihC,EAAatzC,EAAO5wB,MAAOmyD,EAAU+U,GAE3C,OAAOA,EAAc9kC,OAAOl1B,cAAe,UAAW,CAAE,YAAag3D,GAAc,EAQrF,CCpnCO,SAASwU,GAA0B14E,GACzCA,EAAMoL,SAASk3B,mBAAmBF,GAMnC,SAA6BA,EAAgBpiC,GAC5C,MAAM68B,EAAY78B,EAAMoL,SAASyxB,UAC3ByuC,EAAStrE,EAAMsrE,OAEf73B,EAAS,GAEf,IAAIoF,GAAW,EAEf,UAAYqsB,KAAcroC,EAAUmY,YAAc,CAGjD,MAAM2jC,EAAiBC,GAAgB1T,EAAYoG,GAS9CqN,IAAmBA,EAAe9sD,QAASq5C,IAC/CzxB,EAAOjsC,KAAMmxE,GACb9/B,GAAW,GAEXpF,EAAOjsC,KAAM09D,EAEf,CAGKrsB,GACJzW,EAAOgZ,aA0KF,SAAkC3H,GACxC,MAAMolC,EAAgB,IAAKplC,GACrBqlC,EAAuB,IAAIt9D,IACjC,IAAIu9D,EAAoB,EAExB,KAAQA,EAAoBF,EAAc34E,QAAS,CAClD,MAAM84E,EAAeH,EAAeE,GAC9BE,EAAiBJ,EAAcnvE,MAAO,EAAGqvE,GAE/C,UAAcG,EAAoBC,KAAmBF,EAAe54D,UACnE,IAAKy4D,EAAqBtjE,IAAK0jE,GAI/B,GAAKF,EAAantD,QAASstD,GAC1BL,EAAqBrjE,IAAKyjE,QACpB,GAAKF,EAAatlC,eAAgBylC,GAAkB,CAC1DL,EAAqBrjE,IAAKyjE,GAC1BJ,EAAqBrjE,IAAKsjE,GAE1B,MAAMK,EAAcJ,EAAapX,UAAWuX,GAC5CN,EAAcrxE,KAAM4xE,EACrB,CAGDL,GACD,CAIA,OAF8BF,EAAclrE,QAAQ,CAAEuG,EAAG1H,KAAYssE,EAAqBtjE,IAAKhJ,IAGhG,CAzMuB6sE,CAAyB5lC,GAAU,CAAE2C,SAAUvZ,EAAUkY,aAG/E,OAAO,CACR,CAxC6CukC,CAAoBl3C,EAAQpiC,IACzE,CAgDO,SAAS44E,GAAgBlsD,EAAc4+C,GAC7C,OAAK5+C,EAAMkmB,YAeZ,SAAkClmB,EAAc4+C,GAC/C,MAAMiO,EAAmB7sD,EAAM4b,MAEzBkxC,EAAwBlO,EAAOiE,yBAA0BgK,GAM/D,IAAMC,EAAwB,CAC7B,MAAMC,EAAiBF,EAAiBx1C,eAAe9wB,UAAUkmB,MAAQtxB,GAA2ByjE,EAAOgE,SAAUznE,KAErH,OAAK4xE,EACG,GAAMplC,UAAWolC,GAGlB,IACR,CAEA,IAAMD,EAAsB5mC,YAC3B,OAAO4mC,EAGR,MAAME,EAAgBF,EAAsBlxC,MAG5C,GAAKixC,EAAiB1tD,QAAS6tD,GAC9B,OAAO,KAGR,OAAO,IAAI,GAAOA,EACnB,CA7CSC,CAAyBjtD,EAAO4+C,GAqDzC,SAAoC5+C,EAAc4+C,GACjD,MAAM,MAAEhjC,EAAK,IAAEC,GAAQ7b,EAEjBktD,EAAuBtO,EAAOyF,WAAYzoC,EAAO,SACjDuxC,EAAqBvO,EAAOyF,WAAYxoC,EAAK,SAE7CuxC,EAAoBxO,EAAOyO,gBAAiBzxC,GAC5C0xC,EAAkB1O,EAAOyO,gBAAiBxxC,GAGhD,GAAKuxC,IAAsBE,EAAkB,CAI5C,GAAKJ,GAAwBC,EAC5B,OAAO,KAQR,GAiEF,SAA2CvxC,EAAiBC,EAAe+iC,GAC1E,MAAM2O,EAAmB3xC,EAAMwJ,YAAcw5B,EAAOI,QAASpjC,EAAMwJ,YAAiBw5B,EAAOyF,WAAYzoC,EAAO,SACxG4xC,EAAiB3xC,EAAIwJ,aAAeu5B,EAAOI,QAASnjC,EAAIwJ,aAAkBu5B,EAAOyF,WAAYxoC,EAAK,SAGxG,OAAO0xC,GAAkBC,CAC1B,CAvEOC,CAAkC7xC,EAAOC,EAAK+iC,GAAW,CAC7D,MACM8O,EAD0B9xC,EAAMwJ,WAAaw5B,EAAO+O,aAAc/xC,EAAMwJ,WACjC,KAAOw5B,EAAOiE,yBAA0BjnC,EAAO,WAGtFgyC,EADuB/xC,EAAIwJ,YAAcu5B,EAAO+O,aAAc9xC,EAAIwJ,YAChC,KAAOu5B,EAAOiE,yBAA0BhnC,EAAK,YAG/EoZ,EAAay4B,EAAaA,EAAW9xC,MAAQA,EAC7CsZ,EAAW04B,EAAWA,EAAS/xC,IAAMA,EAE3C,OAAO,IAAI,GAAOoZ,EAAYC,EAC/B,CACD,CAEA,MAAM24B,EAAiBT,IAAsBA,EAAkBhkE,GAAI,eAC7D0kE,EAAeR,IAAoBA,EAAgBlkE,GAAI,eAI7D,GAAKykE,GAAkBC,EAAe,CACrC,MAAMC,EAAqBnyC,EAAMwJ,WAAavJ,EAAIwJ,YAAgBzJ,EAAMwJ,UAAUtrC,SAAW+hC,EAAIwJ,WAAWvrC,OAEtGk0E,EAAcH,KAAqBE,IAAqBJ,GAAc/xC,EAAMwJ,UAAWw5B,IACvFqP,EAAYH,KAAmBC,IAAqBJ,GAAc9xC,EAAIwJ,WAAYu5B,IAIxF,IAAI8O,EAAa9xC,EACbgyC,EAAW/xC,EAUf,OARKmyC,IACJN,EAAa,GAASzoC,cAAeipC,GAA4Bd,EAAmBxO,KAGhFqP,IACJL,EAAW,GAASjpC,aAAcupC,GAA4BZ,EAAiB1O,KAGzE,IAAI,GAAO8O,EAAYE,EAC/B,CAGA,OAAO,IACR,CArHQO,CAA2BnuD,EAAO4+C,EAC1C,CAyHA,SAASsP,GAA4BE,EAAoBxP,GACxD,IAAIyP,EAAcD,EACdt0E,EAAkCu0E,EAGtC,KAAQzP,EAAOI,QAASllE,IAAYA,EAAOA,QAC1Cu0E,EAAcv0E,EACdA,EAASA,EAAOA,OAGjB,OAAOu0E,CACR,CAuDA,SAASV,GAActzE,EAAYukE,GAClC,OAAOvkE,GAAQukE,EAAO+O,aAActzE,EACrC,CCnQe,MAAMi0E,WAA0BjgE,KA2B9C,WAAApR,CAAa3J,EAAc2tC,GAC1Bv4B,QAEAxN,KAAK5H,MAAQA,EACb4H,KAAKi6B,KAAO,IAAI,GAAM8L,GACtB/lC,KAAKw9D,OAAS,IAAIrC,GAElBn7D,KAAKqzE,mBAAqB,IAAIhU,GAAoB,CACjD7B,OAAQx9D,KAAKw9D,OACbkG,OAAQtrE,EAAMsrE,SAGf,MAAMrsD,EAAMrX,KAAK5H,MAAMoL,SACjByxB,EAAY5d,EAAI4d,UAChB2qC,EAAU5/D,KAAK5H,MAAMwnE,QA+J7B,IAA0BpC,EAAgBkG,EAAgBzpC,EAxJxDj6B,KAAKwP,SAA8BxP,KAAK5H,MAAO,kBAAkB,KAChE4H,KAAKi6B,KAAK47B,mBAAmB,EAAM,GACjC,CAAEjpD,SAAU,YAEf5M,KAAKwP,SAA6BxP,KAAK5H,MAAO,iBAAiB,KAC9D4H,KAAKi6B,KAAK47B,mBAAmB,EAAO,GAClC,CAAEjpD,SAAU,WAKf5M,KAAKwP,SAA+B6H,EAAK,UAAU,KAClDrX,KAAKi6B,KAAKM,QAAQC,IACjBx6B,KAAKqzE,mBAAmB3T,eAAgBroD,EAAIsoD,OAAQC,EAASplC,GAC7Dx6B,KAAKqzE,mBAAmBtS,iBAAkB9rC,EAAW2qC,EAASplC,EAAQ,GACpE,GACD,CAAE5tB,SAAU,QAGf5M,KAAKwP,SAA4CxP,KAAKi6B,KAAKz2B,SAAU,kBFychE,SAAiCpL,EAAcolE,GACrD,MAAO,CACNnoD,EACAvS,KAEA,MAAMk0C,EAAgBl0C,EAAK2qD,aAErB5hB,EAA4B,GAElC,UAAY4a,KAAazP,EAAc5J,YACtCvB,EAAOjsC,KAAM49D,EAAOL,aAAc1W,IAGnC,MAAM6sB,EAAiBl7E,EAAM4/C,gBAAiBnM,EAAQ,CAAE2C,SAAUwI,EAAc7J,aAE1EmmC,EAAervD,QAAS7rB,EAAMoL,SAASyxB,YAC5C78B,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAc8/B,EAAgB,GAEvC,CAEF,CE7dGC,CAAwBvzE,KAAK5H,MAAO4H,KAAKw9D,SAI1Cx9D,KAAKwP,SAAkCxP,KAAKi6B,KAAKz2B,SAAU,eAgInCg6D,EA/HNx9D,KAAKw9D,OA+HiBkG,EA/HT1jE,KAAK5H,MAAMsrE,OA+HczpC,EA/HNj6B,KAAKi6B,KAgIjD,CAAE5kB,EAAKvS,KAGb,IAAKm3B,EAAKz2B,SAASm4B,aAAgB,EAAItyB,UAIvC,QAAU7O,EAAI,EAAGA,EAAIsI,EAAK4wD,aAAap7D,OAAQkC,IAAM,CACpD,MAAMisD,EAAY3jD,EAAK4wD,aAAcl5D,GAC/B8iE,EAAaE,EAAOL,aAAc1W,GAClCsqB,EAAiBC,GAAgB1T,EAAYoG,GAE7CqN,IAAkBA,EAAe9sD,QAASq5C,KAIhDx6D,EAAK4wD,aAAcl5D,GAAMgjE,EAAOH,YAAa0T,GAC9C,IAhJC,CAAEnkE,SAAU,SAIb5M,KAAKqzE,mBAAmBhkE,GAAqD,gBJmuBvE,CACNgG,EACAvS,EACAw8D,KAEA,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAGD,MAAMqoE,EAAa/K,EAAc9kC,OAC3BmY,EAAe2sB,EAAc9B,OAAOD,eAAgBz6D,EAAKgiB,MAAM4b,OAC/Dke,EAAWyrB,EAAW12B,WAAY7wC,EAAK7C,KAAK6C,MAElDunE,EAAW3kE,OAAQitC,EAAciM,EAAU,GIhvBgE,CAAEhyC,SAAU,WACvH5M,KAAKqzE,mBAAmBhkE,GAAyB,UJyvB3C,CACNgG,EACAvS,EACAw8D,KAEAA,EAAc8C,kBAAmBt/D,EAAK7C,MAIhC6C,EAAK8+D,eAAgB9+D,EAAK7C,KAAKiO,GAAI,YAAgBpL,EAAK7C,KAAKkiC,SAClEm9B,EAAc6C,gBAAiBr/D,EAAK7C,KACrC,GIpwB0F,CAAE2M,SAAU,WACtG5M,KAAKqzE,mBAAmBhkE,GAAyB,UJixB3C,CACNgG,EACAvS,EACAw8D,KAGA,MAAM3X,EAAY2X,EAAc9B,OAAOD,eAAgBz6D,EAAK2iB,UAEtD+tD,EAAW1wE,EAAK2iB,SAAS6kB,aAAcxnC,EAAKxK,QAC5CsvD,EAAU0X,EAAc9B,OAAOD,eAAgBiW,EAAU,CAAE/V,WAAW,IAEtEhX,EAAY6Y,EAAc9kC,OAAOqd,YAAa8P,EAAWC,GAGzD52B,EAAUsuC,EAAc9kC,OAAOv0B,OAAQwgD,EAAUtb,cAIvD,UAAYxzB,KAAS2nD,EAAc9kC,OAAOud,cAAe/mB,GAAUkb,WAClEozB,EAAc9B,OAAOpB,kBAAmBzkD,EAAsB,CAAE4kD,OAAO,GACxE,GIryBqE,CAAE3vD,SAAU,QAGjF5M,KAAKqzE,mBAAmBhkE,GAAiC,kBJg7BnD,CACNgG,EACAvS,EACAw8D,KAEA,MAAM+K,EAAa/K,EAAc9kC,OAC3Bwc,EAAgBqzB,EAAW7mE,SAASyxB,UAE1C,UAAYnQ,KAASkyB,EAAc5J,YAE7BtoB,EAAMkmB,aAEHlmB,EAAM6b,IAAI/hC,OAAqB28B,cACrC+jC,EAAc9kC,OAAO+a,gBAAiBzwB,EAAM4b,OAK/C2pC,EAAW72B,aAAc,KAAM,IIj8B/BxzC,KAAKqzE,mBAAmBhkE,GAA4B,aJs0B9C,CACNgG,EACAvS,EACAw8D,KAEA,MAAMrqC,EAAYnyB,EAAKmyB,UAEvB,GAAKA,EAAU+V,YACd,OAGD,IAAMs0B,EAAcwB,WAAWrC,QAASxpC,EAAW,aAClD,OAGD,MAAMwyB,EAA+B,GAErC,UAAY3iC,KAASmQ,EAAUmY,YAC9Bqa,EAAW7nD,KAAM0/D,EAAc9B,OAAOH,YAAav4C,IAGpDw6C,EAAc9kC,OAAOgZ,aAAciU,EAAY,CAAEjZ,SAAUvZ,EAAUkY,YAAc,GI31BO,CAAEvgC,SAAU,QACtG5M,KAAKqzE,mBAAmBhkE,GAA4B,aJy3B9C,CACNgG,EACAvS,EACAw8D,KAEA,MAAMrqC,EAAYnyB,EAAKmyB,UAEvB,IAAMA,EAAU+V,YACf,OAGD,IAAMs0B,EAAcwB,WAAWrC,QAASxpC,EAAW,aAClD,OAGD,MAAMo1C,EAAa/K,EAAc9kC,OAC3BohC,EAAgB3mC,EAAUE,mBAC1Bwd,EAAe2sB,EAAc9B,OAAOD,eAAgB3B,GACpD6X,EAAiBpJ,EAAWv1B,gBAAiBnC,GAEnD03B,EAAW72B,aAAcigC,EAAgB,GI74BqD,CAAE7mE,SAAU,QAK1G5M,KAAKi6B,KAAKz2B,SAASqtC,MAAM38B,OAAQlU,KAAK5H,MAAMoL,SAASqtC,OAAQ/e,OAAOx7B,IAEnE,GAAsB,cAAjBA,EAAK8+B,SACT,OAAO,KAGR,MAAMw/B,EAAW,IAAIvsB,GAAqBroC,KAAKi6B,KAAKz2B,SAAUlN,EAAK0L,MAKnE,OAHA4yD,EAASx/B,SAAW9+B,EAAK8+B,SACzBp1B,KAAKw9D,OAAOnf,aAAc/nD,EAAMs+D,GAEzBA,CAAQ,GAYjB,CAMO,OAAA5uC,GACNhmB,KAAKi6B,KAAKjU,UACVhmB,KAAK2P,eACN,CAqCO,eAAA+jE,CAAiBC,GACvB,MAAMrX,EAAoC,iBAAhBqX,EAA2BA,EAAeA,EAAa3xE,KAC3E4xE,EAAgB5zE,KAAK5H,MAAMwnE,QAAQx3D,IAAKk0D,GAE9C,IAAMsX,EAOL,MAAM,IAAI,EAAe,qDAAsD5zE,KAAM,CAAEs8D,eAGxFt8D,KAAK5H,MAAMmiC,QAAQ,KAClBv6B,KAAK5H,MAAMwnE,QAAQiU,SAAUD,EAAe,GAE9C,CAYO,aAAAE,CAAe7zE,GACrBD,KAAK5H,MAAMmiC,QAAQ,KAClBv6B,KAAK5H,MAAMoL,SAASm8D,OAAOoU,aAAc9zE,EAAM,GAEjD,EC5Mc,MAAM+zE,GAArB,cAOC,KAAQjwC,aAAe,IAAIxyB,GAA+D,CAqDnF,GAAA1D,CACN4J,EACAqnD,GAEA,IAAImV,EAGCx8D,EAAQvJ,GAAI,UAAauJ,EAAQvJ,GAAI,oBACzClO,KAAK+jC,aAAatyB,IAAKgG,GAAS,IAM3BzX,KAAK+jC,aAAan2B,IAAK6J,GAI5Bw8D,EAAqBj0E,KAAK+jC,aAAa37B,IAAKqP,IAH5Cw8D,EAAqB,IAAIC,GAAwBz8D,GACjDzX,KAAK+jC,aAAatyB,IAAKgG,EAASw8D,IAKjCA,EAAmBpmE,IAAKixD,GACzB,CAkCO,IAAA78D,CAAMwV,EAAkCqnD,GAC9C,MAAMmV,EAAqBj0E,KAAK+jC,aAAa37B,IAAKqP,GAElD,YAA4B,IAAvBw8D,EACG,KAIHx8D,EAAQvJ,GAAI,UAAauJ,EAAQvJ,GAAI,oBAClC+lE,EAICA,EAA+ChyE,KAAM68D,EAC/D,CAiCO,OAAAL,CAAShnD,EAAkCqnD,GACjD,QAAK9+D,KAAKiC,KAAMwV,EAASqnD,KACnBrnD,EAAQvJ,GAAI,UAAauJ,EAAQvJ,GAAI,oBAEzClO,KAAK+jC,aAAatyB,IAAKgG,GAAS,GAG9BzX,KAAK+jC,aAAa37B,IAAKqP,GAAsCgnD,QAASK,IAGlE,EAIT,CAiCO,MAAAH,CAAQlnD,EAAeqnD,GAC7B,MAAMmV,EAAqBj0E,KAAK+jC,aAAa37B,IAAKqP,QAEtB,IAAvBw8D,IACCx8D,EAAQvJ,GAAI,UAAauJ,EAAQvJ,GAAI,oBAEzClO,KAAK+jC,aAAatyB,IAAKgG,GAAS,GAG9Bw8D,EAA+CtV,OAAQG,GAG5D,CAMA,6BAAcqV,CAAwB18D,GACrC,MAAMqnD,EAAc,CACnBrnD,UACAzV,MAAM,EACNuD,WAAY,GACZm5B,QAAS,GACTG,OAAQ,IAGHt5B,EAAakS,EAAQ6mB,mBAE3B,UAAYloB,KAAa7Q,EAEN,SAAb6Q,GAAqC,SAAbA,GAI7B0oD,EAAYv5D,WAAW3F,KAAMwW,GAG9B,MAAMsoB,EAAUjnB,EAAQknB,gBAExB,UAAYwH,KAAazH,EACxBogC,EAAYpgC,QAAQ9+B,KAAMumC,GAG3B,MAAMtH,EAASpnB,EAAQqnB,gBAEvB,UAAYz5B,KAASw5B,EACpBigC,EAAYjgC,OAAOj/B,KAAMyF,GAG1B,OAAOy5D,CACR,CAWA,iBAAcsV,CAAY90E,EAA+B+0E,GAKxD,GAJMA,IACLA,EAAW,IAAIL,IAGX10E,EAAK4O,GAAI,SAGb,OAFAmmE,EAASxmE,IAAKvO,GAEP+0E,EAIH/0E,EAAK4O,GAAI,YACbmmE,EAASxmE,IAAKvO,EAAM00E,GAAeG,uBAAwB70E,IAGvDA,EAAK4O,GAAI,qBACbmmE,EAASxmE,IAAKvO,GAGf,UAAYqY,KAAWrY,EAAqCk8B,cAC3D64C,EAAWL,GAAeI,WAAYz8D,EAAO08D,GAG9C,OAAOA,CACR,EA0BD,MAAMC,GAAmB,CAAE,aAAc,UAAW,UAQ7C,MAAMJ,GAkBZ,WAAAnyE,CAAazC,GACZU,KAAKyX,QAAUnY,EACfU,KAAKu0E,gBAAkB,KACvBv0E,KAAK+jC,aAAe,CACnBx+B,WAAY,IAAIgM,IAChBstB,OAAQ,IAAIttB,IACZmtB,QAAS,IAAIntB,IAEf,CA2BO,GAAA1D,CAAKixD,GACNA,EAAY98D,OAChBhC,KAAKu0E,iBAAkB,GAGxB,UAAYtpE,KAAQqpE,GACdrpE,KAAQ6zD,GACZ9+D,KAAK43B,KAAM3sB,EAAM6zD,EAAa7zD,GAGjC,CA0BO,IAAAhJ,CAAM68D,GAEZ,GAAKA,EAAY98D,OAAShC,KAAKu0E,gBAC9B,OAAOv0E,KAAKu0E,gBAGb,UAAYtpE,KAAQqpE,GACnB,GAAKrpE,KAAQ6zD,EAAc,CAC1B,MAAMtmE,EAAQwH,KAAKw0E,MAAOvpE,EAAM6zD,EAAa7zD,IAE7C,IAAe,IAAVzS,EACJ,OAAOA,CAET,CAID,OAAO,CACR,CAwBO,OAAAimE,CAASK,GACVA,EAAY98D,OAChBhC,KAAKu0E,iBAAkB,GAGxB,UAAYtpE,KAAQqpE,GACdrpE,KAAQ6zD,GACZ9+D,KAAKy0E,SAAUxpE,EAAM6zD,EAAa7zD,GAGrC,CAuBO,MAAA0zD,CAAQG,GACTA,EAAY98D,OAChBhC,KAAKu0E,iBAAkB,GAGxB,UAAYtpE,KAAQqpE,GACdrpE,KAAQ6zD,GACZ9+D,KAAK00E,QAASzpE,EAAM6zD,EAAa7zD,GAGpC,CAWQ,IAAA2sB,CAAM3sB,EAAsBhL,GACnC,MAAM2wB,EAAQ,GAAS3wB,GAASA,EAAO,CAAEA,GACnC6+D,EAAc9+D,KAAK+jC,aAAc94B,GAEvC,UAAYjJ,KAAQ4uB,EAAQ,CAC3B,GAAc,eAAT3lB,IAAoC,UAATjJ,GAA6B,UAATA,GAenD,MAAM,IAAI,EAAe,mCAAoChC,MAK9D,GAFA8+D,EAAYrtD,IAAKzP,GAAM,GAET,WAATiJ,EACJ,UAAY+5B,KAAYhlC,KAAKyX,QAAQjU,SAASuiC,gBAAgBxB,iBAAkBviC,GAC/E88D,EAAYrtD,IAAKuzB,GAAU,EAG9B,CACD,CAUQ,KAAAwvC,CAAOvpE,EAAsBhL,GACpC,MAAM2wB,EAAQ,GAAS3wB,GAASA,EAAO,CAAEA,GACnC6+D,EAAc9+D,KAAK+jC,aAAc94B,GAEvC,UAAYjJ,KAAQ4uB,EACnB,GAAc,eAAT3lB,GAAoC,UAATjJ,GAA6B,UAATA,EAS7C,CACN,MAAMxJ,EAAQsmE,EAAY12D,IAAKpG,GAE/B,QAAe,IAAVxJ,EACJ,OAAO,KAGR,IAAMA,EACL,OAAO,CAET,KAnByE,CACxE,MAAMm8E,EAAyB,SAAR3yE,EAAkB,UAAY,SAG/CxJ,EAAQwH,KAAKw0E,MAAOG,EAAgB,IAAK30E,KAAK+jC,aAAc4wC,GAAiB19E,SAEnF,IAAe,IAAVuB,EACJ,OAAOA,CAET,CAaD,OAAO,CACR,CAQQ,QAAAi8E,CAAUxpE,EAAsBhL,GACvC,MAAM2wB,EAAQ,GAAS3wB,GAASA,EAAO,CAAEA,GACnC6+D,EAAc9+D,KAAK+jC,aAAc94B,GAEvC,UAAYjJ,KAAQ4uB,EACnB,GAAc,eAAT3lB,GAAoC,UAATjJ,GAA6B,UAATA,GAQnD,GAFA88D,EAAYrtD,IAAKzP,GAAM,GAEV,UAARiJ,EACJ,UAAY2pE,KAAa50E,KAAKyX,QAAQjU,SAASuiC,gBAAgBxB,iBAAkBviC,GAChF88D,EAAYrtD,IAAKmjE,GAAW,OAV0C,CACxE,MAAMD,EAAyB,SAAR3yE,EAAkB,UAAY,SAGrDhC,KAAKy0E,SAAUE,EAAgB,IAAK30E,KAAK+jC,aAAc4wC,GAAiB19E,QACzE,CAUF,CAQQ,OAAAy9E,CAASzpE,EAAsBhL,GACtC,MAAM2wB,EAAQ,GAAS3wB,GAASA,EAAO,CAAEA,GACnC6+D,EAAc9+D,KAAK+jC,aAAc94B,GAEvC,UAAYjJ,KAAQ4uB,EACnB,GAAc,eAAT3lB,GAAoC,UAATjJ,GAA6B,UAATA,EAK7C,EAGS,IAFD88D,EAAY12D,IAAKpG,IAG9B88D,EAAYrtD,IAAKzP,GAAM,EAEzB,KAXyE,CACxE,MAAM2yE,EAAyB,SAAR3yE,EAAkB,UAAY,SAGrDhC,KAAK00E,QAASC,EAAgB,IAAK30E,KAAK+jC,aAAc4wC,GAAiB19E,QACxE,CAQF,ECtoBc,MAAM49E,WAAe1hE,KAYnC,WAAApR,GACCyL,QAZD,KAAiBsnE,mBAAkE,CAAC,EAKpF,KAAiBC,qBAA4D,CAAC,EAS7E/0E,KAAKiV,SAAU,cACfjV,KAAKiV,SAAU,kBAEfjV,KAAKqP,GAAI,kBAAkB,CAAEgG,EAAKnZ,KACjCA,EAAM,GAAM,IAAI84E,GAAe94E,EAAM,GAAK,GACxC,CAAE0Q,SAAU,YAEf5M,KAAKqP,GAAI,cAAc,CAAEgG,EAAKnZ,KAC7BA,EAAM,GAAM,IAAI84E,GAAe94E,EAAM,IACrCA,EAAM,GAAM8D,KAAKi1E,cAAe/4E,EAAM,GAAK,GACzC,CAAE0Q,SAAU,WAChB,CAWO,QAAAsoE,CAAUC,EAAkBjtE,GAClC,GAAKlI,KAAK80E,mBAAoBK,GAoB7B,MAAM,IAAI,EACT,oCACAn1E,KACA,CACCm1E,aAKHn1E,KAAK80E,mBAAoBK,GAAa,CACrCn+E,OAAO+wB,OAAQ,CAAC,EAAG7f,IAGpBlI,KAAKo1E,aACN,CAyBO,MAAAliC,CAAQiiC,EAAkBjtE,GAChC,IAAMlI,KAAK80E,mBAAoBK,GAU9B,MAAM,IAAI,EAAe,oCAAqCn1E,KAAM,CACnEm1E,aAIFn1E,KAAK80E,mBAAoBK,GAAWv1E,KAAM5I,OAAO+wB,OAAQ,CAAC,EAAG7f,IAE7DlI,KAAKo1E,aACN,CAUO,cAAAC,GAKN,OAJMr1E,KAAKs1E,sBACVt1E,KAAKu1E,WAGCv1E,KAAKs1E,oBACb,CAUO,aAAAL,CAAeh1E,GACrB,IAAIk1E,EAYJ,OATCA,EADmB,iBAARl1E,EACAA,EACA,OAAQA,IAAUA,EAAKiO,GAAI,UAAajO,EAAKiO,GAAI,eACjD,QAIEjO,EAAc+B,KAGrBhC,KAAKq1E,iBAAkBF,EAC/B,CAWO,YAAAK,CAAcv1E,GACpB,QAASD,KAAKi1E,cAAeh1E,EAC9B,CAiBO,OAAA0jE,CAAS1jE,GACf,MAAMw1E,EAAMz1E,KAAKi1E,cAAeh1E,GAEhC,SAAWw1E,IAAOA,EAAI9R,QACvB,CAqBO,OAAAG,CAAS7jE,GACf,MAAMw1E,EAAMz1E,KAAKi1E,cAAeh1E,GAEhC,QAAMw1E,MAIKA,EAAI3R,UAAW2R,EAAI/N,SAC/B,CAqBO,QAAAA,CAAUznE,GAChB,MAAMw1E,EAAMz1E,KAAKi1E,cAAeh1E,GAEhC,QAAMw1E,MAMKA,EAAI/N,UAAc+N,EAAI3R,SAAW2R,EAAIhD,cAAgBgD,EAAIC,UACrE,CAiBO,QAAA9N,CAAU3nE,GAChB,MAAMw1E,EAAMz1E,KAAKi1E,cAAeh1E,GAEhC,SAAWw1E,IAAOA,EAAI7N,SACvB,CAmBO,YAAA6K,CAAcxyE,GACpB,MAAMw1E,EAAMz1E,KAAKi1E,cAAeh1E,GAEhC,QAAMw1E,MAIKA,EAAIhD,eAAgBgD,EAAI/N,SACpC,CAmBO,SAAAgO,CAAWz1E,GACjB,MAAMw1E,EAAMz1E,KAAKi1E,cAAeh1E,GAEhC,QAAMw1E,MAIKA,EAAIC,YAAaD,EAAI/N,SACjC,CAuBO,UAAAyB,CAAY57D,EAAkCkoE,GAEpD,QAAMA,GAICz1E,KAAK21E,mBAAoBF,EAAYloE,EAC7C,CAkBO,cAAAs6D,CAAgBt6D,EAAkCq5B,GACxD,MAAM6uC,EAAMz1E,KAAKi1E,cAAiB1nE,EAAiBmjB,MAEnD,QAAM+kD,GAICA,EAAIG,gBAAgBrmD,SAAUqX,EACtC,CAgBO,UAAAivC,CAAYC,EAA2CC,GAC7D,GAAKD,aAAiC,GAAW,CAChD,MAAM3rC,EAAa2rC,EAAsB3rC,WACnCD,EAAY4rC,EAAsB5rC,UAExC,KAAQC,aAAsB,IAM7B,MAAM,IAAI,EACT,uCACAnqC,MAIF,KAAQkqC,aAAqB,IAM5B,MAAM,IAAI,EACT,sCACAlqC,MAIF,OAAOA,KAAK61E,WAAY1rC,EAAYD,EACrC,CAEA,UAAYvyB,KAASo+D,EAAev6C,cACnC,IAAMx7B,KAAKmpE,WAAY2M,EAAuBn+D,GAC7C,OAAO,EAIT,OAAO,CACR,CA6CO,aAAAq+D,CAAezmE,GACrBvP,KAAKqP,GAA2B,cAAc,CAAEgG,GAAOic,EAAK2kD,MAI3D,IAAMA,EACL,OAGD,MAAMC,EAAW3mE,EAAU+hB,EAAK2kD,GAER,kBAAZC,IACX7gE,EAAIlJ,OACJkJ,EAAIlE,OAAS+kE,EACd,GACE,CAAEtpE,SAAU,QAChB,CA6CO,iBAAAupE,CAAmB5mE,GACzBvP,KAAKqP,GAA+B,kBAAkB,CAAEgG,GAAOic,EAAKsV,MACnE,MAAMsvC,EAAW3mE,EAAU+hB,EAAKsV,GAER,kBAAZsvC,IACX7gE,EAAIlJ,OACJkJ,EAAIlE,OAAS+kE,EACd,GACE,CAAEtpE,SAAU,QAChB,CA8CO,sBAAAwpE,CAAwBxvC,EAAuBtzB,GACrDtT,KAAK+0E,qBAAsBnuC,GAAkB5vC,OAAO+wB,OAAQ/nB,KAAK8nE,uBAAwBlhC,GAAiBtzB,EAC3G,CAOO,sBAAAw0D,CAAwBlhC,GAC9B,OAAO5mC,KAAK+0E,qBAAsBnuC,IAAmB,CAAC,CACvD,CASO,eAAAurC,CAAiBkE,GACvB,IAAI5+D,EAEJ,GAAK4+D,aAAsC,GAC1C5+D,EAAU4+D,EAA2Bz3E,WAC/B,CAMN6Y,GALe4+D,aAAsC,GACpD,CAAEA,GACF11E,MAAMrB,KAAM+2E,EAA2BjpC,cAItCnf,QAAwB,CAAExW,EAASqN,KACnC,MAAMwxD,EAAsBxxD,EAAMyX,oBAElC,OAAM9kB,EAICA,EAAQ8kB,kBAAmB+5C,EAAgC,CAAEj6C,aAAa,IAHzEi6C,CAGiF,GACvF,KACL,CAEA,MAASt2E,KAAK8jE,QAASrsD,IACjBA,EAAQ7Y,QACZ6Y,EAAUA,EAAQ7Y,OAMpB,OAAO6Y,CACR,CAYO,yBAAA8+D,CAA2BthD,EAA0C7e,GAC3E,GAAK6e,EAAU+V,YAAc,CAC5B,MACMz9B,EAAU,IADM0nB,EAAUE,mBAEdgH,eACjB,IAAI,GAAM,GAAIlH,EAAU+Q,kBAIzB,OAAOhmC,KAAK6nE,eAAgBt6D,EAAS6I,EACtC,CAAO,CACN,MAAMy1B,EAAS5W,EAAUmY,YAGzB,UAAYtoB,KAAS+mB,EACpB,UAAYrzC,KAASssB,EACpB,GAAK9kB,KAAK6nE,eAAgBrvE,EAAMyH,KAAMmW,GAErC,OAAO,CAIX,CAGA,OAAO,CACR,CASA,eAAQogE,CAAgB3qC,EAAyBz1B,GAChDy1B,EAw0CF,UAAsCA,GACrC,UAAY/mB,KAAS+mB,QACb/mB,EAAMo1C,sBAEf,CA50CWuc,CAA4B5qC,GAErC,UAAY/mB,KAAS+mB,QACb7rC,KAAK02E,wBAAyB5xD,EAAO1O,EAE9C,CAsBO,wBAAAuxD,CAA0BliD,EAAoBlB,EAA6C,QACjG,GAA+B,cAA1BkB,EAASnvB,KAAK8+B,SAGlB,OAAO,KAIR,GAAKp1B,KAAKmpE,WAAY1jD,EAAU,SAC/B,OAAO,IAAI,GAAOA,GAGnB,IAAIkxD,EAAgBC,EAGpB,MAAMC,EAAiBpxD,EAAS0W,eAAmC9wB,UAAUkmB,MAAMtxB,GAAQD,KAAK8jE,QAAS7jE,MACxGwlB,EAASnvB,KAEQ,QAAbiuB,GAAoC,YAAbA,IAC3BoyD,EAAiB,IAAI,GAAY,CAChCnuC,WAAY,GAAMgE,UAAWqqC,GAC7BpuC,cAAehjB,EACflB,UAAW,cAIK,QAAbA,GAAoC,WAAbA,IAC3BqyD,EAAgB,IAAI,GAAY,CAC/BpuC,WAAY,GAAMgE,UAAWqqC,GAC7BpuC,cAAehjB,KAIjB,UAAY3iB,KAquCd,UAA0B0rC,EAAkCsoC,GAC3D,IAAIv1E,GAAO,EAEX,MAASA,GAAO,CAGf,GAFAA,GAAO,EAEFitC,EAAW,CACf,MAAMuoC,EAAOvoC,EAASltC,OAEhBy1E,EAAKx1E,OACVA,GAAO,OACD,CACLm1C,OAAQlI,EACRh2C,MAAOu+E,EAAKv+E,OAGf,CAEA,GAAKs+E,EAAU,CACd,MAAMC,EAAOD,EAAQx1E,OAEfy1E,EAAKx1E,OACVA,GAAO,OACD,CACLm1C,OAAQogC,EACRt+E,MAAOu+E,EAAKv+E,OAGf,CACD,CACD,CAnwCsBw+E,CAAgBL,EAAgBC,GAAkB,CACrE,MAAM3rE,EAASnI,EAAK4zC,QAAUigC,EAAiB,aAAe,eACxDn+E,EAAQsK,EAAKtK,MAEnB,GAAKA,EAAMyS,MAAQA,GAAQjL,KAAK0nE,SAAUlvE,EAAMyH,MAC/C,OAAO,GAAMwsC,UAAWj0C,EAAMyH,MAG/B,GAAKD,KAAKmpE,WAAY3wE,EAAMyxC,aAAc,SACzC,OAAO,IAAI,GAAOzxC,EAAMyxC,aAE1B,CAEA,OAAO,IACR,CAYO,iBAAAgtC,CAAmBxxD,EAAoBtmB,GAC7C,IAAIP,EAAS6mB,EAAS7mB,OAEtB,KAAQA,GAAS,CAChB,GAAKoB,KAAKmpE,WAAYvqE,EAAQO,GAC7B,OAAOP,EAIR,GAAKoB,KAAK8jE,QAASllE,GAClB,OAAO,KAGRA,EAASA,EAAOA,MACjB,CAEA,OAAO,IACR,CASO,oBAAAs4E,CACN/3E,EACAoG,EACAi1B,GAEA,MAAMpiC,EAAQoiC,EAAOpiC,MAErB,UAAcwuC,EAAe8d,KAAoB1tD,OAAOyhB,QAASlT,GAC3DnN,EAAMsrE,OAAOmE,eAAgB1oE,EAAMynC,IACvCpM,EAAO/0B,aAAcmhC,EAAe8d,EAAgBvlD,EAGvD,CAOO,0BAAAg4E,CAA4BjwC,EAAuB1M,GACzD,UAAYr7B,KAAQ+nC,EAEnB,GAAK/nC,EAAK+O,GAAI,SACbkpE,GAAmCp3E,KAAMb,EAAMq7B,OAM3C,CACJ,MACM68C,EADc,GAAM7qC,UAAWrtC,GACAgtC,eAErC,UAAY1mB,KAAY4xD,EAAmB,CAG1CD,GAAmCp3E,KAFtBylB,EAAS0kB,YAAc1kB,EAAS7mB,OAES47B,EACvD,CACD,CAEF,CAYO,yBAAA88C,CAA2Bn4E,EAAY4U,EAAsByC,GACnE,MAAMjR,EAAsC,CAAC,EAE7C,UAAcqhC,EAAe8d,KAAoBvlD,EAAK6mC,gBAAkB,CACvE,MAAMuxC,EAAsBv3E,KAAK8nE,uBAAwBlhC,QAEZ,IAAxC2wC,EAAqBxjE,UAIH,IAAlByC,GAA+BA,IAAkB+gE,EAAqBxjE,KAC1ExO,EAAYqhC,GAAkB8d,GAEhC,CAEA,OAAOn/C,CACR,CAKO,aAAAuoE,CAAevgE,GACrB,OAAO,IAAIynE,GAAeznE,EAC3B,CAEQ,WAAA6nE,GACPp1E,KAAKs1E,qBAAuB,IAC7B,CAEQ,QAAAC,GACP,MAAMiC,EAA4E,CAAC,EAC7EC,EAAcz3E,KAAK80E,mBACnB4C,EAAY1gF,OAAOC,KAAMwgF,GAE/B,UAAYtC,KAAYuC,EACvBF,EAAqBrC,GAAawC,GAAqBF,EAAatC,GAAYA,GAGjF,UAAYA,KAAYuC,EACvBE,GAAsBJ,EAAqBrC,GAG5C,UAAYA,KAAYuC,EACvBG,GAAuBL,EAAqBrC,GAG7C,UAAYA,KAAYuC,EACvBI,GAAmBN,EAAqBrC,GAGzC,UAAYA,KAAYuC,EACvBK,GAA0BP,EAAqBrC,GAC/C6C,GAA8BR,EAAqBrC,GAGpD,UAAYA,KAAYuC,EACvBO,GAAgBT,EAAqBrC,GACrC+C,GAAoBV,EAAqBrC,GACzCgD,GAAwBX,EAAqBrC,GAG9Cn1E,KAAKs1E,qBAAuBkC,CAC7B,CAEQ,kBAAA7B,CACPF,EACAloE,EACA6qE,EAA2B7qE,EAAQjV,OAAS,GAE5C,MAAM+/E,EAAc9qE,EAAQ+qE,QAASF,GAErC,GAAK3C,EAAI8C,QAAQhpD,SAAU8oD,EAAYr2E,MAAS,CAC/C,GAAyB,GAApBo2E,EACJ,OAAO,EACD,CACN,MAAMI,EAAax4E,KAAKi1E,cAAeoD,GAEvC,OAAOr4E,KAAK21E,mBAAoB6C,EAAajrE,EAAS6qE,EAAmB,EAC1E,CACD,CACC,OAAO,CAET,CAYA,wBAAS1B,CAAyB5xD,EAAc1O,GAC/C,IAAIsqB,EAAQ5b,EAAM4b,MACdC,EAAM7b,EAAM4b,MAEhB,UAAYzgC,KAAQ6kB,EAAMonB,SAAU,CAAEpD,SAAS,IACzC7oC,EAAKiO,GAAI,mBACNlO,KAAK02E,wBAAyB,GAAMlqC,UAAWvsC,GAAQmW,IAGzDpW,KAAK6nE,eAAgB5nE,EAAMmW,KAC1BsqB,EAAMzc,QAAS0c,WACd,IAAI,GAAOD,EAAOC,IAGzBD,EAAQ,GAAS+I,aAAcxpC,IAGhC0gC,EAAM,GAAS8I,aAAcxpC,GAGxBygC,EAAMzc,QAAS0c,WACd,IAAI,GAAOD,EAAOC,GAE1B,EAsfM,MAAMq0C,GAMZ,WAAAjzE,CAAawL,GACZ,GAAKA,aAAmBynE,GACvB,OAAOznE,EAGR,IAAIqjB,EAGHA,EADsB,iBAAXrjB,EACH,CAAEA,GACE5M,MAAMC,QAAS2M,GAKnBA,EAFAA,EAAQ4uB,aAAc,CAAEE,aAAa,IAK9Cr8B,KAAKgwB,OAASY,EAAM5zB,IAAKy7E,GAC1B,CAKA,UAAWngF,GACV,OAAO0H,KAAKgwB,OAAO13B,MACpB,CAKA,QAAWo4B,GACV,OAAO1wB,KAAKgwB,OAAQhwB,KAAKgwB,OAAO13B,OAAS,EAC1C,CAOA,CAASwI,OAAOC,YACf,OAAOf,KAAKgwB,OAAQlvB,OAAOC,WAC5B,CA4BO,IAAAnB,CAAMK,GACZ,MAAMqxB,EAAM,IAAI0jD,GAAe,CAAE/0E,IAIjC,OAFAqxB,EAAItB,OAAS,IAAKhwB,KAAKgwB,UAAWsB,EAAItB,QAE/BsB,CACR,CAKO,OAAAgnD,CAAS1zE,GACf,OAAO5E,KAAKgwB,OAAQprB,EACrB,CAKA,SAAQ8zE,SACA14E,KAAKgwB,OAAOhzB,KAAKiD,GAAQA,EAAK+B,MACtC,CAcO,QAAA+rB,CAAU4qD,GAChB,OAAOh4E,MAAMrB,KAAMU,KAAK04E,YAAax7E,KAAM,KAAM6wB,SAAU4qD,EAC5D,CAcO,UAAA/zB,CAAY+zB,GAClB,OAAOh4E,MAAMrB,KAAMU,KAAK04E,YAAax7E,KAAM,KAAM0nD,WAAY+zB,EAC9D,EAmJD,SAAShB,GAAqBiB,EAA8CzD,GAC3E,MAAM0D,EAAW,CAChB72E,KAAMmzE,EAENoD,QAAS,GACTO,eAAgB,GAChBC,WAAY,GAEZnD,gBAAiB,GACjBoD,kBAAmB,GAEnBC,cAAe,GAEfC,iBAAkB,IAkBnB,OAwID,SAAoBN,EAA8CC,GACjE,UAAYM,KAAkBP,EAAkB,CAC/C,MAAMQ,EAAYpiF,OAAOC,KAAMkiF,GAAiBpzE,QAAQ/D,GAAQA,EAAK4iD,WAAY,QAEjF,UAAY5iD,KAAQo3E,EACnBP,EAAU72E,KAAWm3E,EAAgBn3E,EAEvC,CACD,CA/JCq3E,CAAWT,EAAiBC,GAE5BS,GAAcV,EAAiBC,EAAU,WACzCS,GAAcV,EAAiBC,EAAU,kBACzCS,GAAcV,EAAiBC,EAAU,cAEzCS,GAAcV,EAAiBC,EAAU,mBACzCS,GAAcV,EAAiBC,EAAU,qBAEzCS,GAAcV,EAAiBC,EAAU,iBAEzCS,GAAcV,EAAiBC,EAAU,oBA4K1C,SAA6BD,EAA8CC,GAC1E,UAAYM,KAAkBP,EAAkB,CAC/C,MAAMW,EAAcJ,EAAeK,eAE9BD,IACJV,EAASC,eAAgBl5E,KAAM25E,GAC/BV,EAASE,WAAYn5E,KAAM25E,GAC3BV,EAASG,kBAAmBp5E,KAAM25E,GAClCV,EAASK,iBAAkBt5E,KAAM25E,GAEnC,CACD,CArLCE,CAAoBb,EAAiBC,GAE9BA,CACR,CAEA,SAASjB,GACRJ,EACArC,GAEA,MAAMl1E,EAAOu3E,EAAqBrC,GAElC,UAAYuE,KAAqBz5E,EAAKg5E,cAAgB,CACrD,MAAMU,EAAkBnC,EAAqBkC,GAGvCC,GAINA,EAAgBpB,QAAQ34E,KAAMu1E,EAC/B,CAIAl1E,EAAKg5E,cAAc3gF,OAAS,CAC7B,CAEA,SAASu/E,GACRL,EACArC,GAEA,UAAYyE,KAA0BpC,EAAqBrC,GAAW2D,eAErE,GAAKtB,EAAqBoC,GAA2B,CAC5BC,GAAoBrC,EAAqBoC,GAEjDh8E,SAASk8E,IACxBA,EAAYvB,QAAQ34E,KAAMu1E,EAAU,GAEtC,QAGMqC,EAAqBrC,GAAW2D,cACxC,CAEA,SAAShB,GACRN,EACArC,GAEA,UAAY4E,KAAsBvC,EAAqBrC,GAAW4D,WAAc,CAC/E,MAAMQ,EAAc/B,EAAqBuC,GAGzC,GAAKR,EAAc,CAClB,MAAMS,EAAYT,EAAYhB,QAE9Bf,EAAqBrC,GAAWoD,QAAQ34E,QAASo6E,EAClD,CACD,QAEOxC,EAAqBrC,GAAW4D,UACxC,CAEA,SAAShB,GACRP,EACArC,GAEA,UAAY8E,KAAwBzC,EAAqBrC,GAAW6D,kBAAqB,CACxF,MAAMO,EAAc/B,EAAqByC,GAEzC,GAAKV,EAAc,CAClB,MAAMW,EAAoBX,EAAY3D,gBAEtC4B,EAAqBrC,GAAWS,gBAAgBh2E,QAASs6E,EAC1D,CACD,QAEO1C,EAAqBrC,GAAW6D,iBACxC,CAEA,SAAShB,GACRR,EACArC,GAEA,MAAMl1E,EAAOu3E,EAAqBrC,GAElC,UAAYgF,KAA2Bl6E,EAAKi5E,iBAAoB,CAC/D,MAAMK,EAAc/B,EAAqB2C,GAEzC,GAAKZ,EAAc,CAClB,MAAMH,EAAYpiF,OAAOC,KAAMsiF,GAAcxzE,QAAQ/D,GAAQA,EAAK4iD,WAAY,QAE9E,UAAY5iD,KAAQo3E,EACXp3E,KAAQ/B,IACfA,EAAM+B,GAASu3E,EAAav3E,GAG/B,CACD,QAEO/B,EAAKi5E,gBACb,CAIA,SAASjB,GACRT,EACArC,GAEA,MAAM0D,EAAWrB,EAAqBrC,GAChCiF,EAAgBvB,EAASN,QAAQxyE,QAAQs0E,GAAe7C,EAAqB6C,KAEnFxB,EAASN,QAAU53E,MAAMrB,KAAM,IAAIsU,IAAKwmE,GACzC,CAGA,SAASlC,GACRV,EACArC,GAEA,MAAM0D,EAAWrB,EAAqBrC,GAEtC,UAAYmF,KAAyBzB,EAASN,QAAU,CAC7Bf,EAAqB8C,GAE7BrB,cAAcr5E,KAAMu1E,EACvC,CACD,CAEA,SAASgD,GACRX,EACArC,GAEA,MAAM0D,EAAWrB,EAAqBrC,GAEtC0D,EAASjD,gBAAkBj1E,MAAMrB,KAAM,IAAIsU,IAAKilE,EAASjD,iBAC1D,CAYA,SAAS0D,GACRV,EACAC,EACA9kE,GAQA,UAAYolE,KAAkBP,EAAkB,CAC/C,MAAMpgF,EAAQ2gF,EAAgBplE,GAET,iBAATvb,EACXqgF,EAAU9kE,GAAgBnU,KAAMpH,GACrBmI,MAAMC,QAASpI,IAC1BqgF,EAAU9kE,GAAgBnU,QAASpH,EAErC,CACD,CAeA,SAASqhF,GAAoBrC,EAA2ErC,GACvG,MAAM0D,EAAWrB,EAAqBrC,GAEtC,OAGmBtwE,EAHD2yE,EAIXxgF,OAAOC,KAAM4N,GAAM7H,KAAKjG,GAAO8N,EAAK9N,MAJHgP,QAAQ0vE,GAAOA,EAAI8C,QAAQhpD,SAAUspD,EAAS72E,QAGvF,IAAoB6C,CAFpB,CAMA,SAAS4zE,GAAgB8B,GACxB,MAAuB,iBAAXA,GAAuBA,EAAQrsE,GAAI,oBACvC,CACNlM,KAAwB,iBAAXu4E,EAAsBA,EAAU,oBAE7C,iBAAEj8C,GAAoB,EAEtB,YAAAjD,GAAgB,GAGV,CAENr5B,KAAMu4E,EAAQrsE,GAAI,WAAcqsE,EAAQv4E,KAAO,QAE/C,iBAAEs8B,SACMi8C,EAAQj8C,kBAChB,EAEAjD,aAActkC,GACNwjF,EAAQl/C,aAActkC,GAIjC,CAwDA,SAASqgF,GAAmC1T,EAAgBvkE,EAAYq7B,GACvE,UAAYpkB,KAAajX,EAAKm/B,mBACvBolC,EAAOmE,eAAgB1oE,EAAMiX,IAClCokB,EAAO9zB,gBAAiB0P,EAAWjX,EAGtC,C,8RCv7De,MAAMq7E,WAAyBprE,KAyC7C,WAAArN,CAAau9D,G,MACZ9xD,QA/BD,KAAQitE,YAAc,IAAIlpE,IAO1B,KAAQmpE,eAAiB,IAAInpE,IAO7B,KAAQopE,aAAqC,KAO7C,KAAQC,qBAAuB,IAAIhnE,IAYlC5T,KAAKs/D,e,2HAAgB,IACjBA,G,QADiB,CAEpBwB,WAAY,KACZtmC,OAAQ,KACRqgD,MAAO,KACP3Y,YAAa,CAAE4N,EAAUD,IAAiB7vE,KAAK86E,aAAchL,EAAUD,GACvE1N,gBAAiB,CAAE5qB,EAAawjC,IAAuB/6E,KAAKg7E,iBAAkBzjC,EAAawjC,GAC3F3K,WAAY,CAAE5C,EAAW/nD,IAAczlB,KAAKi7E,YAAazN,EAAW/nD,GACpE4qD,uBAAwB,CAAElU,EAAcr5D,IAAU9C,KAAKk7E,wBAAyB/e,EAAcr5D,GAE9Fq4E,qBAAsB,CAAE3N,EAAWqC,IAAiB7vE,KAAKo7E,sBAAuB5N,EAAWqC,GAC3FwL,cAAelf,GAAgBn8D,KAAKs7E,eAAgBnf,GACpDof,iBAAkBpf,GAAgBn8D,KAAKw7E,kBAAmBrf,MAE5D,CAeO,OAAAjlE,CACNqgD,EACA/c,EACAjtB,EAAmC,CAAE,UAErCvN,KAAKwQ,KAA8B,cAAe+mC,GAIlDv3C,KAAK26E,aA8aP,SACCc,EACAjhD,GAEA,IAAI/U,EAEJ,UAAYxlB,KAAQ,IAAI+0E,GAAeyG,GAAsB,CAC5D,MAAMl2E,EAAsC,CAAC,EAE7C,UAAYxO,KAAOkJ,EAAKq+B,mBACvB/4B,EAAYxO,GAAQkJ,EAAKo7B,aAActkC,GAGxC,MAAMgI,EAAUy7B,EAAOl1B,cAAerF,EAAK+B,KAAMuD,GAE5CkgB,GACJ+U,EAAO90B,OAAQ3G,EAAS0mB,GAGzBA,EAAW,GAAcmjB,UAAW7pC,EAAS,EAC9C,CAEA,OAAO0mB,CACR,CArcsBi2D,CAAmBnuE,EAASitB,GAIhDx6B,KAAKs/D,cAAc9kC,OAASA,EAG5Bx6B,KAAKs/D,cAAcwB,WAAakT,GAAeI,WAAY78B,GAG3Dv3C,KAAKs/D,cAAcub,MAAQ,CAAC,EAG5B,MAAM,WAAEvd,GAAet9D,KAAK86E,aAAcvjC,EAAav3C,KAAK26E,cAGtDgB,EAAmBnhD,EAAOkZ,yBAGhC,GAAK4pB,EAAa,CAEjBt9D,KAAK47E,uBAGL,UAAY37E,KAAQU,MAAMrB,KAAMU,KAAK26E,aAAa/7E,OAAO48B,eACxDhB,EAAOqrB,OAAQ5lD,EAAM07E,GAIpBA,EAA0B/b,QA2W/B,SAA0Cic,EAAkCrhD,GAC3E,MAAMshD,EAAiB,IAAIloE,IACrBgsD,EAAU,IAAIruD,IAGduT,EAAQ,GAAW0nB,UAAWqvC,GAAY3vC,WAGhD,UAAYjsC,KAAQ6kB,EAEd7kB,EAAKiO,GAAI,UAAW,YACxB4tE,EAAejuE,IAAK5N,GAKtB,UAAY87E,KAAiBD,EAAiB,CAC7C,MAAMxf,EAAayf,EAAc1gD,aAAc,aACzC2gD,EAAkBxhD,EAAOod,qBAAsBmkC,GAG/Cnc,EAAQhyD,IAAK0uD,GAIhBsD,EAAQx3D,IAAKk0D,GAAsB37B,IAAMq7C,EAAgBj5D,QAH3D68C,EAAQnuD,IAAK6qD,EAAY,IAAI,GAAY0f,EAAgBj5D,UAO1DyX,EAAOv0B,OAAQ81E,EAChB,CAEA,OAAOnc,CACR,CA5YyCqc,CAAiCN,EAAkBnhD,EAC1F,CAeA,OAZAx6B,KAAK26E,aAAe,KAGpB36E,KAAKy6E,YAAY7oE,QACjB5R,KAAK06E,eAAe9oE,QACpB5R,KAAK46E,qBAAqBhpE,QAGxB5R,KAAKs/D,cAAuB9kC,OAAS,KACvCx6B,KAAKs/D,cAAcub,MAAQ,KAGpBc,CACR,CAKQ,YAAAb,CAAchL,EAA2CD,GAIhE,MAAM/sE,EAA6B,CAAEgtE,WAAUD,cAAavS,WAAY,MAuBxE,GArBKwS,EAAS5hE,GAAI,WACjBlO,KAAKwQ,KACJ,WAAYs/D,EAAS9tE,OACrBc,EACA9C,KAAKs/D,eAEKwQ,EAAS5hE,GAAI,SACxBlO,KAAKwQ,KACJ,OACA1N,EACA9C,KAAKs/D,eAGNt/D,KAAKwQ,KACJ,mBACA1N,EACA9C,KAAKs/D,eAKFx8D,EAAKw6D,cAAiBx6D,EAAKw6D,sBAAsB,IAQrD,MAAM,IAAI,EAAe,8CAA+Ct9D,MAGzE,MAAO,CAAEs9D,WAAYx6D,EAAKw6D,WAAYuS,YAAa/sE,EAAK+sE,YACzD,CAKQ,gBAAAmL,CACPlL,EACAoM,GAKA,IAAIC,EAAkBD,EAAqBhuE,GAAI,YAC9CguE,EAAuB,GAActzC,UAAWszC,EAAsB,GAEvE,MAAM5e,EAAa,IAAI,GAAY6e,GAEnC,UAAYj+B,KAAav9C,MAAMrB,KAAMwwE,EAASt0C,eAAkB,CAC/D,MAAMr9B,EAAS6B,KAAK86E,aAAc58B,EAAWi+B,GAExCh+E,EAAOm/D,sBAAsB,KAC/BA,EAAoB38B,IAAMxiC,EAAOm/D,WAAW38B,IAC9Cw7C,EAAkBh+E,EAAO0xE,YAE3B,CAEA,MAAO,CAAEvS,aAAYuS,YAAasM,EACnC,CAKQ,WAAAlB,CACPzN,EACA/nD,GAKA,MAAM22D,EAAcp8E,KAAKo7E,sBAAuB5N,EAAW/nD,GAG3D,QAAM22D,IAKNp8E,KAAKs/D,cAAc9kC,OAAQ90B,OAAQ8nE,EAAW4O,EAAY32D,WAEnD,EACR,CAKQ,uBAAAy1D,CAAyB/e,EAA4Br5D,GAC5D,MAAM+b,EAAQ7e,KAAKs7E,eAAgBnf,GAE7B3hC,EAASx6B,KAAKs/D,cAAc9kC,OAG5B13B,EAAKw6D,aACVx6D,EAAKw6D,WAAa9iC,EAAOqd,YACxBrd,EAAOod,qBAAsBukB,GAC7B3hC,EAAOmd,oBAAqB94B,EAAOA,EAAMvmB,OAAS,MAIpD,MAAM+jF,EAAoBr8E,KAAK06E,eAAetyE,IAAK+zD,GASlDr5D,EAAK+sE,YANDwM,EAMe7hD,EAAOkd,iBAAkB2kC,EAAmB,GAI5Cv5E,EAAKw6D,WAAW38B,GAErC,CAKQ,qBAAAy6C,CAAuBj8E,EAAiB0wE,GAI/C,MAAM,OAAEnM,EAAM,OAAElpC,GAAWx6B,KAAKs/D,cAGhC,IAAIgd,EAAgB5Y,EAAOuT,kBAAmBpH,EAAa1wE,GAE3D,GAAKm9E,EAAgB,CAEpB,GAAKA,IAAkBzM,EAAYjxE,OAClC,MAAO,CAAE6mB,SAAUoqD,GAIf7vE,KAAK26E,aAAc/7E,OAAOu9B,eAAe5M,SAAU+sD,KACvDA,EAAgB,KAElB,CAEA,IAAMA,EAEL,OAAM1O,GAAiBiC,EAAa1wE,EAAMukE,GAInC,CACNj+C,SAAUsoD,GAAiB8B,EAAar1C,IAJjC,KAST,MAAM4hD,EAAcp8E,KAAKs/D,cAAc9kC,OAAQz9B,MAAO8yE,EAAayM,GAgB7DjuE,EAA6B,GAEnC,UAAYkuE,KAAmBH,EAAYt3D,MAAM+lB,YAChD,GAA6B,cAAxB0xC,EAAgBtxE,KACpBoD,EAAMzO,KAAM28E,EAAgBt8E,UACtB,CAEN,MAAMu8E,EAAenuE,EAAMrP,MACrBy9E,EAAYF,EAAgBt8E,KAElCD,KAAK08E,mBAAoBF,EAAeC,EACzC,CAGD,MAAME,EAAeP,EAAYt3D,MAAM6b,IAAI/hC,OAG3C,OAFAoB,KAAK06E,eAAejpE,IAAKtS,EAAMw9E,GAExB,CACNl3D,SAAU22D,EAAY32D,SACtBk3D,eAEF,CAOQ,kBAAAD,CAAoBF,EAA4BC,GACjDz8E,KAAKy6E,YAAY7sE,IAAK4uE,IAC3Bx8E,KAAKy6E,YAAYhpE,IAAK+qE,EAAc,CAAEA,IAGvC,MAAMz8E,EAAOC,KAAKy6E,YAAYryE,IAAKo0E,GAEnCx8E,KAAKy6E,YAAYhpE,IAAKgrE,EAAW18E,GACjCA,EAAKH,KAAM68E,EACZ,CAKQ,cAAAnB,CAAgB7jE,GACvB,IAAIoH,EAQJ,OAHCA,EAHK7e,KAAKy6E,YAAY7sE,IAAK6J,GAGnBzX,KAAKy6E,YAAYryE,IAAKqP,GAFtB,CAAEA,GAKJoH,CACR,CAKQ,iBAAA28D,CAAmB/jE,GAC1BzX,KAAK46E,qBAAqB/sE,IAAK4J,EAChC,CAQQ,oBAAAmkE,GACP,IAAIgB,GAAa,EAEjB,UAAYnlE,KAAWzX,KAAKy6E,YAAYxjF,OAClCwgB,EAAQ0qB,UAAYniC,KAAK46E,qBAAqBhtE,IAAK6J,KACvDzX,KAAKs/D,cAAc9kC,OAAQv0B,OAAQwR,GACnCzX,KAAKy6E,YAAY9oE,OAAQ8F,GAEzBmlE,GAAa,GAIVA,GACJ58E,KAAK47E,sBAEP,ECjfc,MAAMiB,GAIb,OAAAC,CAAS73B,GACf,MACMnE,EADMt9C,SAASygD,eAAeC,mBAAoB,IAClC5+C,cAAe,OAIrC,OAFAw7C,EAAUn7C,YAAas/C,GAEhBnE,EAAUr6B,SAClB,ECJc,MAAMs2D,GAuBpB,WAAAh7E,CAAayB,GAPb,KAAO4lD,cAAwB,EAQ9BppD,KAAKg9E,UAAY,IAAIj4B,UACrB/kD,KAAK6xC,aAAe,IAAIuR,GAAc5/C,EAAU,CAAE8/C,cAAe,SACjEtjD,KAAKi9E,WAAa,IAAIJ,EACvB,CAQO,MAAAK,CAAQ34B,GAEd,MAAMD,EAActkD,KAAK6xC,aAAakN,UAAWwF,GAGjD,OAAOvkD,KAAKi9E,WAAWH,QAASx4B,EACjC,CAQO,MAAA64B,CAAQr6E,GAEd,MAAMwhD,EAActkD,KAAKo9E,OAAQt6E,GAGjC,OAAO9C,KAAK6xC,aAAa8N,UAAW2E,EAAa,CAAE8E,aAAcppD,KAAKopD,cACvE,CAWO,yBAAAF,CAA2BrrB,GACjC79B,KAAK6xC,aAAaqX,0BAA2BrrB,EAC9C,CAaO,aAAAw/C,CAAepyE,GACrBjL,KAAK6xC,aAAawR,gBAA0B,UAARp4C,EAAmB,aAAe,MACvE,CAMU,MAAAmyE,CAAQt6E,GAIXA,EAAKjG,MAAO,4CACjBiG,EAAO,SAAUA,YAGlB,MAAMU,EAAWxD,KAAKg9E,UAAUh4B,gBAAiBliD,EAAM,aACjDmiD,EAAWzhD,EAASkwC,yBACpBwR,EAAiB1hD,EAASgiB,KAAKlf,WAErC,KAAQ4+C,EAAe5sD,OAAS,GAC/B2sD,EAASt/C,YAAau/C,EAAgB,IAGvC,OAAOD,CACR,ECjEc,MAAMq4B,WAAuBluE,KAwD3C,WAAArN,CAAa3J,EAAc2tC,GAC1Bv4B,QAEAxN,KAAK5H,MAAQA,EACb4H,KAAKw9D,OAAS,IAAIrC,GAElBn7D,KAAKqzE,mBAAqB,IAAIhU,GAAoB,CACjD7B,OAAQx9D,KAAKw9D,OACbkG,OAAQtrE,EAAMsrE,SAEf1jE,KAAKqzE,mBAAmBhkE,GAAqD,gBVsuBvE,CACNgG,EACAvS,EACAw8D,KAEA,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAGD,MAAMqoE,EAAa/K,EAAc9kC,OAC3BmY,EAAe2sB,EAAc9B,OAAOD,eAAgBz6D,EAAKgiB,MAAM4b,OAC/Dke,EAAWyrB,EAAW12B,WAAY7wC,EAAK7C,KAAK6C,MAElDunE,EAAW3kE,OAAQitC,EAAciM,EAAU,GUnvBgE,CAAEhyC,SAAU,WACvH5M,KAAKqzE,mBAAmBhkE,GAAyB,UV4vB3C,CACNgG,EACAvS,EACAw8D,KAEAA,EAAc8C,kBAAmBt/D,EAAK7C,MAIhC6C,EAAK8+D,eAAgB9+D,EAAK7C,KAAKiO,GAAI,YAAgBpL,EAAK7C,KAAKkiC,SAClEm9B,EAAc6C,gBAAiBr/D,EAAK7C,KACrC,GUvwB0F,CAAE2M,SAAU,WAEtG5M,KAAKu9E,iBAAmB,IAAI/C,GAAkB,CAC7C9W,OAAQtrE,EAAMsrE,SAGf1jE,KAAKwzD,aAAe,IAAI,GAAcztB,GACtC/lC,KAAK+lC,gBAAkBA,EACvB/lC,KAAKw9E,cAAgB,IAAIT,GAAmB/8E,KAAKwzD,cACjDxzD,KAAKy9E,UAAYz9E,KAAKw9E,cACtBx9E,KAAK09E,YAAc,IAAI,GAAoB19E,KAAKwzD,cAOhDxzD,KAAKu9E,iBAAiBluE,GAAqB,QR+WrC,CACNgG,EACAvS,GACE4gE,SAAQ5C,aAAYtmC,aAEtB,IAAI/U,EAAW3iB,EAAK+sE,YAGpB,IAAM/O,EAAW7+D,KAAMa,EAAKgtE,UAC3B,OAGD,IAAMpM,EAAOyF,WAAY1jD,EAAU,SAAY,CAC9C,IAAMmoD,GAAiBnoD,EAAU,QAASi+C,GACzC,OAID,GAAyC,GAApC5gE,EAAKgtE,SAAShtE,KAAK+qB,OAAOv1B,OAC9B,OAID,MAAM6xC,EAAa1kB,EAAS0kB,WAE5B1kB,EAAWsoD,GAAiBtoD,EAAU+U,GAEjC2P,GAAcA,EAAWj8B,GAAI,UAAW,aAE5CssB,EAAO8a,KAAM9a,EAAOsd,cAAe3N,GAAc1kB,GACjDA,EAAW+U,EAAOmd,oBAAqBxN,GAEzC,CAEA22B,EAAWrC,QAAS37D,EAAKgtE,UAEzB,MAAM51C,EAAOM,EAAOmZ,WAAY7wC,EAAKgtE,SAAShtE,MAE9C03B,EAAO90B,OAAQw0B,EAAMzU,GAErB3iB,EAAKw6D,WAAa9iC,EAAOqd,YACxBpyB,EACAA,EAAS6kB,aAAcpQ,EAAKyD,aAE7B76B,EAAK+sE,YAAc/sE,EAAKw6D,WAAW38B,GAAG,GQ3Z4B,CAAE/zB,SAAU,WAC9E5M,KAAKu9E,iBAAiBluE,GAAwB,WRyVxC,CACNgG,EACAvS,EACAw8D,KAGA,IAAMx8D,EAAKw6D,YAAcgC,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAE9tE,MAAM,IAAW,CAC5F,MAAM,WAAEs7D,EAAU,YAAEuS,GAAgBvQ,EAAc6C,gBAAiBr/D,EAAKgtE,SAAUhtE,EAAK+sE,aAEvF/sE,EAAKw6D,WAAaA,EAClBx6D,EAAK+sE,YAAcA,CACpB,IQpWmF,CAAEjjE,SAAU,WAC/F5M,KAAKu9E,iBAAiBluE,GAAiC,oBRwVjD,CACNgG,EACAvS,EACAw8D,KAGA,IAAMx8D,EAAKw6D,YAAcgC,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAE9tE,MAAM,IAAW,CAC5F,MAAM,WAAEs7D,EAAU,YAAEuS,GAAgBvQ,EAAc6C,gBAAiBr/D,EAAKgtE,SAAUhtE,EAAK+sE,aAEvF/sE,EAAKw6D,WAAaA,EAClBx6D,EAAK+sE,YAAcA,CACpB,IQnWqG,CAAEjjE,SAAU,WAEjHuG,IAAkBtR,UAAUoT,SAAS5T,KAAMrB,KAAM,QACjDmT,IAAkBtR,UAAUoT,SAAS5T,KAAMrB,KAAM,OACjDmT,IAAkBtR,UAAUoT,SAAS5T,KAAMrB,KAAM,OACjDmT,IAAkBtR,UAAUoT,SAAS5T,KAAMrB,KAAM,UACjDmT,IAAkBtR,UAAUoT,SAAS5T,KAAMrB,KAAM,WAIjDA,KAAKqP,GAA6B,QAAQ,KACzCrP,KAAKwQ,KAAgC,QAAS,GAC5C,CAAE5D,SAAU,WAIf5M,KAAKqP,GAA8B,SAAS,KAC3CrP,KAAK5H,MAAMguE,cAAe,CAAEuX,YAAY,GAASjQ,GAAyB,GACxE,CAAE9gE,SAAU,UAChB,CAkBO,GAAAxE,CACN7D,EAII,CAAC,GAEL,MAAM,SAAE6wB,EAAW,YAAQvH,EAAO,SAAYtpB,EAE9C,IAAMvE,KAAK49E,oBAAqB,CAAExoD,IAejC,MAAM,IAAI,EAAe,uCAAwCp1B,MAGlE,MAAM1J,EAAO0J,KAAK5H,MAAMoL,SAASstC,QAAS1b,GAc1C,OAZM9+B,EAAKilC,cASV,EAAY,mCAAoCv7B,MAGnC,UAAT6tB,GAAqB7tB,KAAK5H,MAAMylF,WAAYvnF,EAAM,CAAEwnF,mBAAmB,IAIrE99E,KAAK6C,UAAWvM,EAAMiO,GAHrB,EAIT,CAWO,SAAA1B,CACNk7E,EACAx5E,EAAmC,CAAC,GAGpC,MAAMy5E,EAAuBh+E,KAAKm9E,OAAQY,EAAwBx5E,GAGlE,OAAOvE,KAAKy9E,UAAUP,OAAQc,EAC/B,CAcO,MAAAb,CACNY,EACAx5E,EAAmC,CAAC,GAEpC,MAAMivD,EAAexzD,KAAKwzD,aACpB6W,EAAarqE,KAAK09E,YAGxB19E,KAAKw9D,OAAON,gBAGZ,MAAMI,EAAa,GAAW9wB,UAAWuxC,GACnCC,EAAuB,IAAI,GAAsBxqB,GAEvDxzD,KAAKw9D,OAAOnf,aAAc0/B,EAAwBC,GAOlD,MAAMpe,EAAUme,EAAuB7vE,GAAI,oBAC1C6vE,EAAuBne,QA0V1B,SAAuCnoD,GACtC,MAAMtZ,EAAwC,GACxCkZ,EAAMI,EAAQnhB,KAAKkN,SAEzB,IAAM6T,EACL,OAAO,IAAI9F,IAGZ,MAAM0sE,EAAe,GAAWzxC,UAAW/0B,GAE3C,UAAY2pD,KAAU/pD,EAAIjf,MAAMwnE,QAAU,CACzC,MAAMc,EAAcU,EAAOT,WAErBud,EAAoBxd,EAAY11B,YAChCmzC,EAA4Bzd,EAAYhgC,MAAMzc,QAASg6D,EAAav9C,QAAWggC,EAAY//B,IAAI1c,QAASg6D,EAAat9C,KAE3H,GAAKu9C,GAAqBC,EACzBhgF,EAAOyB,KAAM,CAAEwhE,EAAOp/D,KAAM0+D,QACtB,CACN,MAAM0d,EAAqBH,EAAa/6D,gBAAiBw9C,GAEpD0d,GACJjgF,EAAOyB,KAAM,CAAEwhE,EAAOp/D,KAAMo8E,GAE9B,CACD,CA0CA,OA5BAjgF,EAAOgjB,MAAM,EAAIk9D,EAAI9xE,IAAQ+xE,EAAI7xE,MAChC,GAAwC,UAAnCF,EAAGo0B,IAAIgK,YAAal+B,EAAGi0B,OAE3B,OAAO,EACD,GAAwC,WAAnCn0B,EAAGm0B,MAAMiK,YAAal+B,EAAGk0B,KAEpC,OAAQ,EAIR,OAASp0B,EAAGm0B,MAAMiK,YAAal+B,EAAGi0B,QACjC,IAAK,SACJ,OAAO,EACR,IAAK,QACJ,OAAQ,EACT,QACC,OAASn0B,EAAGo0B,IAAIgK,YAAal+B,EAAGk0B,MAC/B,IAAK,SACJ,OAAO,EACR,IAAK,QACJ,OAAQ,EACT,QACC,OAAO29C,EAAGC,cAAeF,IAG9B,IAGM,IAAI9sE,IAAKpT,EACjB,CA7ZGqgF,CAA8BT,GAI/B,OAFA/9E,KAAKqzE,mBAAmBn8E,QAASomE,EAAYsC,EAASyK,EAAY9lE,GAE3Dy5E,CACR,CA2BO,IAAAtnD,CAAM5zB,GACZ,GAAK9C,KAAK5H,MAAMoL,SAASmL,QAQxB,MAAM,IAAI,EAAe,yCAA0C3O,MAGpE,IAAIy+E,EAAsC,CAAC,EAQ3C,GANqB,iBAAT37E,EACX27E,EAAYC,KAAO57E,EAEnB27E,EAAc37E,GAGT9C,KAAK49E,oBAAqB5mF,OAAOC,KAAMwnF,IAe5C,MAAM,IAAI,EAAe,wCAAyCz+E,MAWnE,OARAA,KAAK5H,MAAMguE,cAAe,CAAEuX,YAAY,IAASnjD,IAChD,UAAYpF,KAAYp+B,OAAOC,KAAMwnF,GAAgB,CACpD,MAAMzd,EAAYhhE,KAAK5H,MAAMoL,SAASstC,QAAS1b,GAE/CoF,EAAO90B,OAAQ1F,KAAK2+E,MAAOF,EAAarpD,GAAY4rC,GAAaA,EAAW,EAC7E,KAGMvoC,QAAQlc,SAChB,CAsCO,GAAA9K,CAAK3O,EAAuCyB,EAAqC,CAAC,GACxF,IAAIq6E,EAAkC,CAAC,EAQvC,GANqB,iBAAT97E,EACX87E,EAAQF,KAAO57E,EAEf87E,EAAU97E,GAGL9C,KAAK49E,oBAAqB5mF,OAAOC,KAAM2nF,IAe5C,MAAM,IAAI,EAAe,uCAAwC5+E,MAGlEA,KAAK5H,MAAMguE,cAAe7hE,EAAQs6E,WAAa,CAAC,GAAGrkD,IAClDA,EAAOgZ,aAAc,MACrBhZ,EAAOskD,yBAA0B9+E,KAAK5H,MAAMoL,SAASyxB,UAAUqJ,oBAE/D,UAAYlJ,KAAYp+B,OAAOC,KAAM2nF,GAAY,CAEhD,MAAM5d,EAAYhhE,KAAK5H,MAAMoL,SAASstC,QAAS1b,GAE/CoF,EAAOv0B,OAAQu0B,EAAOud,cAAeipB,IACrCxmC,EAAO90B,OAAQ1F,KAAK2+E,MAAOC,EAASxpD,GAAY4rC,GAAaA,EAAW,EACzE,IAEF,CAYO,KAAA2d,CAAO77E,EAAcyK,EAAmC,SAE9D,MAAMywE,EAAuBh+E,KAAKy9E,UAAUN,OAAQr6E,GAGpD,OAAO9C,KAAKjC,QAASigF,EAAsBzwE,EAC5C,CAgBO,OAAAxP,CACNghF,EACAxxE,EAAmC,SAEnC,OAAOvN,KAAK5H,MAAMmiC,QAAQC,GAClBx6B,KAAKu9E,iBAAiBrmF,QAAS6nF,EAAuBvkD,EAAQjtB,IAEvE,CAYO,sBAAAyxE,CAAwBzvE,GAC9BA,EAAUvP,KAAK+lC,gBAChB,CAYO,yBAAAmjB,CAA2BrrB,GAE5B79B,KAAKy9E,WAAaz9E,KAAKy9E,YAAcz9E,KAAKw9E,eAC9Cx9E,KAAKy9E,UAAUv0B,0BAA2BrrB,GAG3C79B,KAAKw9E,cAAct0B,0BAA2BrrB,EAC/C,CAKO,OAAA7X,GACNhmB,KAAK2P,eACN,CAQQ,mBAAAiuE,CAAqBqB,GAC5B,UAAY7pD,KAAY6pD,EACvB,IAAMj/E,KAAK5H,MAAMoL,SAASstC,QAAS1b,GAClC,OAAO,EAIT,OAAO,CACR,ECvdc,MAAM8pD,GAYpB,WAAAn9E,CACCo9E,EACAC,GAVD,KAAiBC,SAAW,IAAI9tE,IAa/BvR,KAAKs/E,UAAY,GAASH,GAC1Bn/E,KAAKu/E,yBAA0B,CAAEv9E,KAAM,WAAYimE,YAAajoE,KAAKs/E,UAAWE,YAAY,IAE5Fx/E,KAAKy/E,QAAU,GAASL,GACxBp/E,KAAKu/E,yBAA0B,CAAEv9E,KAAM,SAAUimE,YAAajoE,KAAKy/E,QAASD,YAAY,GACzF,CA8BO,QAAAE,CACNC,EACAngB,GAEA,MAAMggB,EAAax/E,KAAKs/E,UAAU/vD,SAAUiwC,GAG5C,IAFiBx/D,KAAKy/E,QAAQlwD,SAAUiwC,KAErBggB,EAMlB,MAAM,IAAI,EACT,iDACAx/E,MAIFA,KAAKu/E,yBAA0B,CAAEv9E,KAAM29E,EAAO1X,YAAa,CAAEzI,GAAcggB,cAC5E,CAwEO,IAAK/nC,GACX,IAAMz3C,KAAKq/E,SAASzxE,IAAK6pC,GAMxB,MAAM,IAAI,EAAe,+BAAgCz3C,MAG1D,OAAOA,KAAKq/E,SAASj3E,IAAKqvC,EAC3B,CA0EO,gBAAA4wB,CAAkBngE,GAOxBlI,KAAK4/E,IAAK,YAAavX,iBAAkBngE,GAGzC,UAAY,MAAE9P,EAAK,KAAE6hC,KAAU4lD,GAA0B33E,GACxDlI,KAAK4/E,IAAK,UACRvX,iBAAkB,CAClBjwE,QACA6hC,OACA8uC,kBAAmB7gE,EAAW6gE,mBAGlC,CAiKO,kBAAAiB,CACN9hE,GAoBAlI,KAAK4/E,IAAK,YAAa5V,mBAAoB9hE,GAG3C,UAAY,MAAE9P,EAAK,KAAE6hC,KAAU4lD,GAA0B33E,GACxDlI,KAAK4/E,IAAK,UACRzR,mBAAoB,CACpBl0C,OACA7hC,QACA2wE,kBAAmB7gE,EAAW6gE,mBAGlC,CAiHO,oBAAAwB,CACNriE,GAmBAlI,KAAK4/E,IAAK,YAAarV,qBAAsBriE,GAG7C,UAAY,MAAE9P,EAAK,KAAE6hC,KAAU4lD,GAA0B33E,GACxDlI,KAAK4/E,IAAK,UACRrV,qBAAsB,CACtBtwC,OACA7hC,SAGJ,CAOQ,wBAAAmnF,EACP,KAAEv9E,EAAI,YAAEimE,EAAW,WAAEuX,IAMrB,GAAKx/E,KAAKq/E,SAASzxE,IAAK5L,GAMvB,MAAM,IAAI,EAAe,0BAA2BhC,MAGrD,MAAM8/E,EAAUN,EACf,IAAIpX,GAAiBH,GACrB,IAAIgG,GAAehG,GAEpBjoE,KAAKq/E,SAAS5tE,IAAKzP,EAAM89E,EAC1B,EAOD,SAAUD,GAA0B33E,GACnC,GAAKA,EAAW9P,MAAM8S,OACrB,UAAY1S,KAAS0P,EAAW9P,MAAM8S,OAAS,CAC9C,MAAM9S,EAAQ,CAAErB,IAAKmR,EAAW9P,MAAMrB,IAAKyB,SACrCyhC,EAAO/xB,EAAW+xB,KAAMzhC,GACxBunF,EAAa73E,EAAW63E,WAAa73E,EAAW63E,WAAYvnF,QAAU,QAErEwnF,GAAsB5nF,EAAO6hC,EAAM8lD,EAC3C,YAEOC,GAAsB93E,EAAW9P,MAAO8P,EAAW+xB,KAAM/xB,EAAW63E,WAE7E,CAEA,SAAUC,GAAsB5nF,EAAgB6hC,EAAe8lD,GAG9D,QAFM,CAAE3nF,QAAO6hC,QAEV8lD,EACJ,UAAYE,KAAkB,GAASF,QAChC,CAAE3nF,QAAO6hC,KAAMgmD,EAGxB,CCvrBe,MAAeC,GA+B7B,WAAAn+E,CAAao+E,GACZngF,KAAKmgF,YAAcA,EACnBngF,KAAKokE,oBAA2C,OAArBpkE,KAAKmgF,YAChCngF,KAAKkmE,MAAQ,IACd,CA0CO,SAAAka,GACP,CAOO,MAAApjD,GAGN,MAAMC,EAAYjmC,OAAO+wB,OAAQ,CAAC,EAAG/nB,MAUrC,OARAi9B,EAAKojD,YAAgBrgF,KAAK+B,YAAqBokC,iBAGxClJ,EAAKipC,aAGLjpC,EAAKmnC,oBAELnnC,CACR,CAKA,oBAAkBkJ,GACjB,MAAO,WACR,CAQA,eAAc8wB,CAAUh6B,EAAWz5B,GAClC,OAAO,IAAMxD,KAAei9B,EAAKkjD,YAClC,ECxGM,SAAS30E,GAASia,EAAoByhB,GAC5C,MAAMo5C,EAAkBC,GAAiBr5C,GAGnCtrB,EAAS0kE,EAAgBryD,QAAQ,CAAEC,EAAK/uB,IAAU+uB,EAAM/uB,EAAKw+B,YAAY,GACzE/+B,EAAS6mB,EAAS7mB,OAGxB4hF,GAAsB/6D,GACtB,MAAM7gB,EAAQ6gB,EAAS7gB,MAUvB,OANAhG,EAAO+mC,aAAc/gC,EAAO07E,GAG5BG,GAAoB7hF,EAAQgG,EAAQ07E,EAAgBhoF,QACpDmoF,GAAoB7hF,EAAQgG,GAErB,IAAI,GAAO6gB,EAAUA,EAAS6kB,aAAc1uB,GACpD,CAQO,SAASyV,GAAoBvM,GACnC,IAAMA,EAAMhH,OAMX,MAAM,IAAI,EACT,wCACA9d,MAIF,MAAMpB,EAASkmB,EAAM4b,MAAM9hC,OAG3B4hF,GAAsB17D,EAAM4b,OAC5B8/C,GAAsB17D,EAAM6b,KAG5B,MAAM3P,EAAUpyB,EAAOk+B,gBAAiBhY,EAAM4b,MAAM97B,MAAOkgB,EAAM6b,IAAI/7B,MAAQkgB,EAAM4b,MAAM97B,OAMzF,OAFA67E,GAAoB7hF,EAAQkmB,EAAM4b,MAAM97B,OAEjCosB,CACR,CAUO,SAAS0vD,GAAkBtrC,EAAoBC,GACrD,IAAMD,EAAYt3B,OAMjB,MAAM,IAAI,EACT,sCACA9d,MAIF,MAAMknC,EAAQ7V,GAAS+jB,GAMvB,OAAO5pC,GAFP6pC,EAAiBA,EAAeikB,0BAA2BlkB,EAAY1U,MAAO0U,EAAYzU,IAAI/kB,OAASw5B,EAAY1U,MAAM9kB,QAEzFsrB,EACjC,CA4CO,SAASq5C,GAAiBr5C,GAChC,MAAMwnC,EAA0B,IAEhC,SAASx3E,EAASgwC,GACjB,GAAqB,iBAATA,EACXwnC,EAAW9uE,KAAM,IAAI,GAAMsnC,SACrB,GAAKA,aAAiB,GAC5BwnC,EAAW9uE,KAAM,IAAI,GAAMsnC,EAAMpkC,KAAMokC,EAAMlB,uBACvC,GAAKkB,aAAiB,GAC5BwnC,EAAW9uE,KAAMsnC,QACX,GAAKvwB,EAAYuwB,GACvB,UAAY/nC,KAAQ+nC,EACnBhwC,EAASiI,EAIZ,CAEAjI,CAASgwC,GAGT,QAAU1sC,EAAI,EAAGA,EAAIk0E,EAAWp2E,OAAQkC,IAAM,CAC7C,MAAM2E,EAAOuvE,EAAYl0E,GACnBq7C,EAAO64B,EAAYl0E,EAAI,GAExB2E,aAAgB,IAAQ02C,aAAgB,IAAQ8qC,GAAqBxhF,EAAM02C,KAE/E64B,EAAWlnE,OAAQhN,EAAI,EAAG,EAAG,IAAI,GAAMq7C,EAAK/yC,KAAO3D,EAAK2D,KAAM+yC,EAAK7P,kBACnExrC,IAEF,CAEA,OAAOk0E,CACR,CAWA,SAAS+R,GAAoBhpE,EAAqC7S,GACjE,MAAMulC,EAAa1yB,EAAQokB,SAAUj3B,EAAQ,GACvCslC,EAAYzyB,EAAQokB,SAAUj3B,GAGpC,GAAKulC,GAAcD,GAAaC,EAAWj8B,GAAI,UAAag8B,EAAUh8B,GAAI,UAAayyE,GAAqBx2C,EAAYD,GAAc,CAErI,MAAM02C,EAAa,IAAI,GAAMz2C,EAAWrnC,KAAOonC,EAAUpnC,KAAMqnC,EAAWnE,iBAG1EvuB,EAAQqlB,gBAAiBl4B,EAAQ,EAAG,GAGpC6S,EAAQkuB,aAAc/gC,EAAQ,EAAGg8E,EAClC,CACD,CAQA,SAASJ,GAAsB/6D,GAC9B,MAAMgY,EAAWhY,EAASgY,SACpBhmB,EAAUgO,EAAS7mB,OAEzB,GAAK6+B,EAAW,CACf,MAAMojD,EAAap7D,EAAS7J,OAAS6hB,EAASuM,YACxCplC,EAAQ64B,EAAS74B,MAEvB6S,EAAQqlB,gBAAiBl4B,EAAO,GAEhC,MAAMk8E,EAAY,IAAI,GAAMrjD,EAAS36B,KAAKmP,OAAQ,EAAG4uE,GAAcpjD,EAASuI,iBACtE+6C,EAAa,IAAI,GAAMtjD,EAAS36B,KAAKmP,OAAQ4uE,GAAcpjD,EAASuI,iBAE1EvuB,EAAQkuB,aAAc/gC,EAAO,CAAEk8E,EAAWC,GAC3C,CACD,CASA,SAASJ,GAAqBK,EAAaC,GAC1C,MAAMC,EAAYF,EAAMh7C,gBAClBm7C,EAAYF,EAAMj7C,gBAExB,UAAYmZ,KAAQ+hC,EAAY,CAC/B,GAAK/hC,EAAM,KAAQ8hC,EAAM5lD,aAAc8jB,EAAM,IAC5C,OAAO,EAGRgiC,EAAU7/E,MACX,CAEA,OAAO6/E,EAAU7/E,OAAOC,IACzB,CC9Oe,MAAM6/E,WAAsBlB,GA0B1C,WAAAn+E,CAAai3D,EAA0B7tD,EAAiBkqC,EAA0B8qC,GACjF3yE,MAAO2yE,GAEPngF,KAAKg5D,eAAiBA,EAAej2C,QAErC/iB,KAAKg5D,eAAe3B,WAAa,SAEjCr3D,KAAKmL,QAAUA,EAEfnL,KAAKq1C,eAAiBA,EAAetyB,QACrC/iB,KAAKq1C,eAAegiB,WAAa,QAClC,CAKA,QAAWpsD,GACV,MAA0C,cAArCjL,KAAKq1C,eAAe/+C,KAAK8+B,SACtB,SACyC,cAArCp1B,KAAKg5D,eAAe1iE,KAAK8+B,SAC7B,WAGD,MACR,CAKA,sBAAWisD,GACV,MAAO,CACN,GAAM90C,4BAA6BvsC,KAAKg5D,eAAgBh5D,KAAKmL,SAC7D,GAAMohC,4BAA6BvsC,KAAKq1C,eAAgB,GAE1D,CAKO,KAAAtyB,GACN,OAAO,IAAIq+D,GAAephF,KAAKg5D,eAAgBh5D,KAAKmL,QAASnL,KAAKq1C,eAAgBr1C,KAAKmgF,YACxF,CAgBO,kBAAAmB,GACN,OAAOthF,KAAKq1C,eAAeikB,0BAA2Bt5D,KAAKg5D,eAAgBh5D,KAAKmL,QACjF,CAKO,WAAAo2E,GACN,MAAMC,EAAoBxhF,KAAKg5D,eAAeF,2BAA4B94D,KAAKq1C,eAAgBr1C,KAAKmL,SAEpG,OAAO,IAAIi2E,GAAephF,KAAKshF,qBAAsBthF,KAAKmL,QAASq2E,EAAmBxhF,KAAKmgF,YAAe,EAC3G,CAMgB,SAAAC,GACf,MAAMqB,EAAgBzhF,KAAKg5D,eAAep6D,OACpC8iF,EAAgB1hF,KAAKq1C,eAAez2C,OACpC+iF,EAAe3hF,KAAKg5D,eAAep9C,OACnCgmE,EAAe5hF,KAAKq1C,eAAez5B,OAKzC,GAAK+lE,EAAe3hF,KAAKmL,QAAUs2E,EAAcprB,UAMhD,MAAM,IAAI,EACT,oCAAqCr2D,MAEhC,GAAKyhF,IAAkBC,GAAiBC,EAAeC,GAAgBA,EAAeD,EAAe3hF,KAAKmL,QAMhH,MAAM,IAAI,EACT,mCAAoCnL,MAE/B,GAAKA,KAAKg5D,eAAe1iE,MAAQ0J,KAAKq1C,eAAe/+C,MACuC,UAA7FogB,EAAe1W,KAAKg5D,eAAejB,gBAAiB/3D,KAAKq1C,eAAe0iB,iBAAgC,CAC5G,MAAMv9D,EAAIwF,KAAKg5D,eAAev5D,KAAKnH,OAAS,EAE5C,GAAK0H,KAAKq1C,eAAe51C,KAAMjF,IAAOmnF,GAAgB3hF,KAAKq1C,eAAe51C,KAAMjF,GAAMmnF,EAAe3hF,KAAKmL,QAMzG,MAAM,IAAI,EACT,kCAAmCnL,KAGtC,CAEF,CAMO,QAAA6hF,GACNnB,GAAO,GAAMn0C,4BAA6BvsC,KAAKg5D,eAAgBh5D,KAAKmL,SAAWnL,KAAKq1C,eACrF,CAKgB,MAAArY,GACf,MAAMC,EAAYzvB,MAAMwvB,SAKxB,OAHAC,EAAK+7B,eAAiBh5D,KAAKg5D,eAAeh8B,SAC1CC,EAAKoY,eAAiBr1C,KAAKq1C,eAAerY,SAEnCC,CACR,CAKA,oBAA2BkJ,GAC1B,MAAO,eACR,CAQA,eAAuB8wB,CAAUh6B,EAAWz5B,GAC3C,MAAMw1D,EAAiB,GAAS/B,SAAUh6B,EAAK+7B,eAAgBx1D,GACzD6xC,EAAiB,GAAS4hB,SAAUh6B,EAAKoY,eAAgB7xC,GAE/D,OAAO,IAAIxD,KAAMg5D,EAAgB/7B,EAAK9xB,QAASkqC,EAAgBpY,EAAKkjD,YACrE,ECvLc,MAAM2B,WAAwB5B,GA8B5C,WAAAn+E,CAAa0jB,EAAoByhB,EAAgBi5C,GAChD3yE,MAAO2yE,GAEPngF,KAAKylB,SAAWA,EAAS1C,QACzB/iB,KAAKylB,SAAS4xC,WAAa,SAC3Br3D,KAAKknC,MAAQ,IAAIivB,GAAUoqB,GAAiBr5C,IAC5ClnC,KAAK+hF,yBAA0B,CAChC,CAKA,QAAW92E,GACV,MAAO,QACR,CAKA,WAAWE,GACV,OAAOnL,KAAKknC,MAAMmvB,SACnB,CAKA,sBAAWgrB,GACV,OAAOrhF,KAAKylB,SAAS1C,OACtB,CAKO,KAAAA,GACN,MAAMmkB,EAAQ,IAAIivB,GAAU,IAAKn2D,KAAKknC,OAAQlqC,KAAKmC,GAAQA,EAAKo+B,QAAQ,MAClE73B,EAAS,IAAIo8E,GAAiB9hF,KAAKylB,SAAUyhB,EAAOlnC,KAAKmgF,aAI/D,OAFAz6E,EAAOq8E,wBAA0B/hF,KAAK+hF,wBAE/Br8E,CACR,CAKO,WAAA67E,GACN,MAAM3nB,EAAY55D,KAAKylB,SAASnvB,KAAKkN,SAAUo2D,UACzCooB,EAAa,IAAI,GAAUpoB,EAAW,CAAE,IAE9C,OAAO,IAAIwnB,GAAephF,KAAKylB,SAAUzlB,KAAKknC,MAAMmvB,UAAW2rB,EAAYhiF,KAAKmgF,YAAe,EAChG,CAMgB,SAAAC,GACf,MAAMsB,EAAgB1hF,KAAKylB,SAAS7mB,OAEpC,IAAM8iF,GAAiBA,EAAcrrB,UAAYr2D,KAAKylB,SAAS7J,OAM9D,MAAM,IAAI,EACT,oCACA5b,KAGH,CAMO,QAAA6hF,GAKN,MAAMI,EAAgBjiF,KAAKknC,MAC3BlnC,KAAKknC,MAAQ,IAAIivB,GAAU,IAAK8rB,GAAgBjlF,KAAKmC,GAAQA,EAAKo+B,QAAQ,MAE1E/xB,GAASxL,KAAKylB,SAAUw8D,EACzB,CAKgB,MAAAjlD,GACf,MAAMC,EAAYzvB,MAAMwvB,SAKxB,OAHAC,EAAKxX,SAAWzlB,KAAKylB,SAASuX,SAC9BC,EAAKiK,MAAQlnC,KAAKknC,MAAMlK,SAEjBC,CACR,CAKA,oBAA2BkJ,GAC1B,MAAO,iBACR,CAQA,eAAuB8wB,CAAUh6B,EAAWz5B,GAC3C,MAAM8T,EAAW,GAEjB,UAAYK,KAASslB,EAAKiK,MACpBvvB,EAAM3V,KAEVsV,EAAS1X,KAAM,GAAQq3D,SAAUt/C,IAGjCL,EAAS1X,KAAM,GAAKq3D,SAAUt/C,IAIhC,MAAMjS,EAAS,IAAIo8E,GAAiB,GAAS7qB,SAAUh6B,EAAKxX,SAAUjiB,GAAY8T,EAAU2lB,EAAKkjD,aAGjG,OAFAz6E,EAAOq8E,wBAA0B9kD,EAAK8kD,wBAE/Br8E,CACR,EC/Jc,MAAMw8E,WAAuBhC,GAmC3C,WAAAn+E,CACCo3D,EACAhuD,EACAmtC,EACA+gB,EACA8mB,GAEA3yE,MAAO2yE,GAEPngF,KAAKm5D,cAAgBA,EAAcp2C,QAGnC/iB,KAAKm5D,cAAc9B,WAAa,SAEhCr3D,KAAKmL,QAAUA,EACfnL,KAAKs4C,kBAAoBA,EACzBt4C,KAAKq5D,kBAAoBA,EAAoBA,EAAkBt2C,QAAU,KAEpE/iB,KAAKq5D,oBACTr5D,KAAKq5D,kBAAkBhC,WAAa,SAEtC,CAKA,QAAWpsD,GACV,MAAO,OACR,CAOA,sBAAWmuD,GACV,MAAM35D,EAAOO,KAAKs4C,kBAAkB74C,KAAKqC,QAGzC,OAFArC,EAAKG,KAAM,GAEJ,IAAI,GAAUI,KAAKs4C,kBAAkBhiD,KAAMmJ,EACnD,CAMA,cAAWw5D,GACV,MAAMt4B,EAAM3gC,KAAKm5D,cAAc7uB,aAAcllB,OAAOC,mBAEpD,OAAO,IAAI,GAAOrlB,KAAKm5D,cAAex4B,EACvC,CAKA,sBAAW0gD,GAEV,MAAMx1C,EAAS,CACd,GAAMU,4BAA6BvsC,KAAKm5D,cAAe,GACvD,GAAM5sB,4BAA6BvsC,KAAKs4C,kBAAmB,IAO5D,OAJKt4C,KAAKq5D,mBACTxtB,EAAOjsC,KAAM,GAAM2sC,4BAA6BvsC,KAAKq5D,kBAAmB,IAGlExtB,CACR,CAOO,KAAA9oB,GACN,OAAO,IAAIm/D,GAAgBliF,KAAKm5D,cAAen5D,KAAKmL,QAASnL,KAAKs4C,kBAAmBt4C,KAAKq5D,kBAAmBr5D,KAAKmgF,YACnH,CAKO,WAAAoB,GACN,MAAM3nB,EAAY55D,KAAKm5D,cAAc7iE,KAAKkN,SAAUo2D,UAC9CP,EAAoB,IAAI,GAAUO,EAAW,CAAE,IAErD,OAAO,IAAIuoB,GAAgBniF,KAAKo5D,mBAAoBp5D,KAAKmL,QAASnL,KAAKm5D,cAAeE,EAAmBr5D,KAAKmgF,YAAe,EAC9H,CAMgB,SAAAC,GACf,MAAM3oE,EAAUzX,KAAKm5D,cAAcv6D,OAC7Bgd,EAAS5b,KAAKm5D,cAAcv9C,OAGlC,IAAMnE,GAAWA,EAAQ4+C,UAAYz6C,EAMpC,MAAM,IAAI,EAAe,mCAAoC5b,MACvD,IAAMyX,EAAQ7Y,OAMpB,MAAM,IAAI,EAAe,gCAAiCoB,MACpD,GAAKA,KAAKmL,SAAWsM,EAAQ4+C,UAAYr2D,KAAKm5D,cAAcv9C,OAMlE,MAAM,IAAI,EAAe,mCAAoC5b,MACvD,GAAKA,KAAKq5D,oBAAsBr5D,KAAKq5D,kBAAkBnvB,UAM7D,MAAM,IAAI,EAAe,6CAA8ClqC,KAEzE,CAMO,QAAA6hF,GACN,MAAMO,EAAepiF,KAAKm5D,cAAcv6D,OAExC,GAAKoB,KAAKq5D,kBACTqnB,GAAO,GAAMn0C,4BAA6BvsC,KAAKq5D,kBAAmB,GAAKr5D,KAAKs4C,uBACtE,CACN,MAAMnD,EAAeitC,EAAsB7kD,SAE3C/xB,GAASxL,KAAKs4C,kBAAmBnD,EAClC,CAOAurC,GALoB,IAAI,GACvB,GAAS93C,UAAWw5C,EAAcpiF,KAAKm5D,cAAcv9C,QACrD,GAASgtB,UAAWw5C,EAAcA,EAAa/rB,YAG5Br2D,KAAKo5D,mBAC1B,CAKgB,MAAAp8B,GACf,MAAMC,EAAYzvB,MAAMwvB,SASxB,OAPAC,EAAKk8B,cAAgBn5D,KAAKm5D,cAAcn8B,SACxCC,EAAKqb,kBAAoBt4C,KAAKs4C,kBAAkBtb,SAE3Ch9B,KAAKq5D,oBACTp8B,EAAKo8B,kBAAoBr5D,KAAKq5D,kBAAkBr8B,UAG1CC,CACR,CAKA,oBAA2BkJ,GAC1B,MAAO,gBACR,CAMA,2BAAck8C,CAAsBlpB,GACnC,MAAM15D,EAAO05D,EAAc15D,KAAKqC,MAAO,GAAI,GAG3C,OAFArC,EAAMA,EAAKnH,OAAS,KAEb,IAAI,GAAU6gE,EAAc7iE,KAAMmJ,EAAM,aAChD,CAQA,eAAuBw3D,CAAUh6B,EAAWz5B,GAC3C,MAAM21D,EAAgB,GAASlC,SAAUh6B,EAAKk8B,cAAe31D,GACvD80C,EAAoB,GAAS2e,SAAUh6B,EAAKqb,kBAAmB90C,GAC/D61D,EAAoBp8B,EAAKo8B,kBAAoB,GAASpC,SAAUh6B,EAAKo8B,kBAAmB71D,GAAa,KAE3G,OAAO,IAAIxD,KAAMm5D,EAAel8B,EAAK9xB,QAASmtC,EAAmB+gB,EAAmBp8B,EAAKkjD,YAC1F,ECpOc,MAAMgC,WAAuBjC,GAgC3C,WAAAn+E,CACCi3D,EACA7tD,EACAkqC,EACAgkB,EACA8mB,GAEA3yE,MAAO2yE,GAEPngF,KAAKg5D,eAAiBA,EAAej2C,QAErC/iB,KAAKg5D,eAAe3B,WAAa,aAEjCr3D,KAAKmL,QAAUA,EACfnL,KAAKq1C,eAAiBA,EAAetyB,QAIrC/iB,KAAKq1C,eAAegiB,WAAa,SACjCr3D,KAAKq5D,kBAAoBA,EAAkBt2C,OAC5C,CAKA,QAAW9X,GACV,MAAO,OACR,CAKA,oBAAWsuD,GACV,OAAO,IAAI,GAAUv5D,KAAKg5D,eAAe1iE,KAAM0J,KAAKg5D,eAAev5D,KAAKqC,MAAO,GAAI,GACpF,CAMA,cAAWm3D,GACV,MAAMt4B,EAAM3gC,KAAKg5D,eAAe1uB,aAAcllB,OAAOC,mBAErD,OAAO,IAAI,GAAOrlB,KAAKg5D,eAAgBr4B,EACxC,CAKA,sBAAW0gD,GACV,MAAMiB,EAAgBtiF,KAAKg5D,eAAep6D,OAE1C,MAAO,CACN,GAAM6tC,UAAW61C,GAGjB,GAAM/1C,4BAA6BvsC,KAAKq1C,eAAgB,GACxD,GAAM9I,4BAA6BvsC,KAAKq5D,kBAAmB,GAE7D,CAKO,KAAAt2C,GACN,OAAO,IAAIo/D,GAAgBniF,KAAKg5D,eAAgBh5D,KAAKmL,QAASnL,KAAKq1C,eAAgBr1C,KAAKq5D,kBAAmBr5D,KAAKmgF,YACjH,CAKO,WAAAoB,GAIN,MAAMlsC,EAAiBr1C,KAAKq1C,eAAewjB,gCAAiC74D,MAEtEP,EAAOO,KAAKg5D,eAAev5D,KAAKqC,MAAO,GAAI,GAC3Cw2C,EAAoB,IAAI,GAAUt4C,KAAKg5D,eAAe1iE,KAAMmJ,GAAOo5D,gCAAiC74D,MAE1G,OAAO,IAAIkiF,GAAgB7sC,EAAgBr1C,KAAKmL,QAASmtC,EAAmBt4C,KAAKq5D,kBAAmBr5D,KAAKmgF,YAAe,EACzH,CAMgB,SAAAC,GACf,MAAMqB,EAAgBzhF,KAAKg5D,eAAep6D,OACpC8iF,EAAgB1hF,KAAKq1C,eAAez2C,OAG1C,IAAM6iF,EAAc7iF,OAMnB,MAAM,IAAI,EAAe,0CAA2CoB,MAC9D,IAAM0hF,EAAc9iF,OAM1B,MAAM,IAAI,EAAe,0CAA2CoB,MAC9D,GAAKA,KAAKmL,SAAWs2E,EAAcprB,UAMzC,MAAM,IAAI,EAAe,mCAAoCr2D,KAE/D,CAMO,QAAA6hF,GACN,MAAMS,EAAgBtiF,KAAKg5D,eAAep6D,OAG1C8hF,GAFoB,GAAMl0C,UAAW81C,GAEjBtiF,KAAKq1C,gBACzBqrC,GAAO,GAAMj0C,UAAW61C,GAAiBtiF,KAAKq5D,kBAC/C,CAKgB,MAAAr8B,GACf,MAAMC,EAAYzvB,MAAMwvB,SAMxB,OAJAC,EAAK+7B,eAAiB/7B,EAAK+7B,eAAeh8B,SAC1CC,EAAKoY,eAAiBpY,EAAKoY,eAAerY,SAC1CC,EAAKo8B,kBAAoBp8B,EAAKo8B,kBAAkBr8B,SAEzCC,CACR,CAKA,oBAA2BkJ,GAC1B,MAAO,gBACR,CAQA,eAAuB8wB,CAAUh6B,EAAWz5B,GAC3C,MAAMw1D,EAAiB,GAAS/B,SAAUh6B,EAAK+7B,eAAgBx1D,GACzD6xC,EAAiB,GAAS4hB,SAAUh6B,EAAKoY,eAAgB7xC,GACzD61D,EAAoB,GAASpC,SAAUh6B,EAAKo8B,kBAAmB71D,GAErE,OAAO,IAAIxD,KAAMg5D,EAAgB/7B,EAAK9xB,QAASkqC,EAAgBgkB,EAAmBp8B,EAAKkjD,YACxF,EC7Mc,MAAMoC,WAAwBrC,GA6C5C,WAAAn+E,CACCC,EACAugE,EACAprB,EACAyoB,EACA5qC,EACAmrD,GAEA3yE,MAAO2yE,GAEPngF,KAAKgC,KAAOA,EACZhC,KAAKuiE,SAAWA,EAAWA,EAASx/C,QAAU,KAC9C/iB,KAAKm3C,SAAWA,EAAWA,EAASp0B,QAAU,KAC9C/iB,KAAKg1B,YAAcA,EAEnBh1B,KAAKwiF,SAAW5iB,CACjB,CAKA,QAAW30D,GACV,MAAO,QACR,CAKA,sBAAWo2E,GACV,MAAMx1C,EAAS,GAcf,OAZK7rC,KAAKuiE,UACT12B,EAAOjsC,KAAMI,KAAKuiE,SAASx/C,SAGvB/iB,KAAKm3C,WACJn3C,KAAKuiE,SACT12B,EAAOjsC,QAASI,KAAKm3C,SAASvL,cAAe5rC,KAAKuiE,WAElD12B,EAAOjsC,KAAMI,KAAKm3C,SAASp0B,UAItB8oB,CACR,CAKO,KAAA9oB,GACN,OAAO,IAAIw/D,GAAiBviF,KAAKgC,KAAMhC,KAAKuiE,SAAUviE,KAAKm3C,SAAUn3C,KAAKwiF,SAAUxiF,KAAKg1B,YAAah1B,KAAKmgF,YAC5G,CAKO,WAAAoB,GACN,OAAO,IAAIgB,GAAiBviF,KAAKgC,KAAMhC,KAAKm3C,SAAUn3C,KAAKuiE,SAAUviE,KAAKwiF,SAAUxiF,KAAKg1B,YAAah1B,KAAKmgF,YAAe,EAC3H,CAMO,QAAA0B,GACD7hF,KAAKm3C,SACTn3C,KAAKwiF,SAASC,KAAMziF,KAAKgC,KAAMhC,KAAKm3C,UAAU,EAAMn3C,KAAKg1B,aAEzDh1B,KAAKwiF,SAASnxD,QAASrxB,KAAKgC,KAE9B,CAMgB,MAAAg7B,GACf,MAAMC,EAAYzvB,MAAMwvB,SAYxB,OAVKh9B,KAAKuiE,WACTtlC,EAAKslC,SAAWviE,KAAKuiE,SAASvlC,UAG1Bh9B,KAAKm3C,WACTla,EAAKka,SAAWn3C,KAAKm3C,SAASna,iBAGxBC,EAAKulD,SAELvlD,CACR,CAKA,oBAA2BkJ,GAC1B,MAAO,iBACR,CAQA,eAAuB8wB,CAAUh6B,EAAWz5B,GAC3C,OAAO,IAAI++E,GACVtlD,EAAKj7B,KACLi7B,EAAKslC,SAAW,GAAMtL,SAAUh6B,EAAKslC,SAAU/+D,GAAa,KAC5Dy5B,EAAKka,SAAW,GAAM8f,SAAUh6B,EAAKka,SAAU3zC,GAAa,KAC5DA,EAASpL,MAAMwnE,QACf3iC,EAAKjI,YACLiI,EAAKkjD,YAEP,EC7ID,SAJA,SAAiB3nF,EAAO8f,GACtB,OAAO,GAAY9f,EAAO8f,EAC5B,ECRe,MAAMoqE,WAA2BxC,GAgD/C,WAAAn+E,CAAa+iB,EAAc/tB,EAAayc,EAAmBC,EAAmB0sE,GAC7E3yE,MAAO2yE,GAEPngF,KAAK8kB,MAAQA,EAAM/B,QACnB/iB,KAAKjJ,IAAMA,EACXiJ,KAAKwT,cAAwB,IAAbA,EAAyB,KAAOA,EAChDxT,KAAKyT,cAAwB,IAAbA,EAAyB,KAAOA,CACjD,CAKA,QAAWxI,GACV,OAAuB,OAAlBjL,KAAKwT,SACF,eACsB,OAAlBxT,KAAKyT,SACT,kBAEA,iBAET,CAKA,sBAAW4tE,GACV,OAAOrhF,KAAK8kB,MAAM/B,OACnB,CAKO,KAAAA,GACN,OAAO,IAAI2/D,GAAoB1iF,KAAK8kB,MAAO9kB,KAAKjJ,IAAKiJ,KAAKwT,SAAUxT,KAAKyT,SAAUzT,KAAKmgF,YACzF,CAKO,WAAAoB,GACN,OAAO,IAAImB,GAAoB1iF,KAAK8kB,MAAO9kB,KAAKjJ,IAAKiJ,KAAKyT,SAAUzT,KAAKwT,SAAUxT,KAAKmgF,YAAe,EACxG,CAKgB,MAAAnjD,GACf,MAAMC,EAAYzvB,MAAMwvB,SAIxB,OAFAC,EAAKnY,MAAQ9kB,KAAK8kB,MAAMkY,SAEjBC,CACR,CAMgB,SAAAmjD,GACf,IAAMpgF,KAAK8kB,MAAMhH,OAMhB,MAAM,IAAI,EAAe,qCAAsC9d,MAGhE,UAAYC,KAAQD,KAAK8kB,MAAMonB,SAAU,CAAEpD,SAAS,IAAW,CAC9D,GAAuB,OAAlB9oC,KAAKwT,WAAsB,GAASvT,EAAKo7B,aAAcr7B,KAAKjJ,KAAOiJ,KAAKwT,UAS5E,MAAM,IAAI,EACT,sCACAxT,KACA,CAAEC,OAAMlJ,IAAKiJ,KAAKjJ,IAAKyB,MAAOwH,KAAKwT,WAIrC,GAAuB,OAAlBxT,KAAKwT,UAAuC,OAAlBxT,KAAKyT,UAAqBxT,EAAKgmC,aAAcjmC,KAAKjJ,KAQhF,MAAM,IAAI,EACT,uCACAiJ,KACA,CAAEb,KAAMc,EAAMlJ,IAAKiJ,KAAKjJ,KAG3B,CACD,CAMO,QAAA8qF,GAEA,GAAS7hF,KAAKwT,SAAUxT,KAAKyT,WPvD9B,SAAwBqR,EAAc/tB,EAAayB,GAEzDgoF,GAAsB17D,EAAM4b,OAC5B8/C,GAAsB17D,EAAM6b,KAG5B,UAAY1gC,KAAQ6kB,EAAMonB,SAAU,CAAEpD,SAAS,IAAW,CAIzD,MAAM3pC,EAAOc,EAAKiO,GAAI,cAAiBjO,EAAKw9B,SAAWx9B,EAExC,OAAVzH,EACJ2G,EAAKioC,cAAerwC,EAAKyB,GAEzB2G,EAAKmoC,iBAAkBvwC,GAIxB0pF,GAAoBthF,EAAKP,OAASO,EAAKyF,MACxC,CAGA67E,GAAoB37D,EAAM6b,IAAI/hC,OAAQkmB,EAAM6b,IAAI/7B,MACjD,COiCGwiC,CAAepnC,KAAK8kB,MAAO9kB,KAAKjJ,IAAKiJ,KAAKyT,SAE5C,CAKA,oBAA2B0yB,GAC1B,MAAO,oBACR,CAQA,eAAuB8wB,CAAUh6B,EAAWz5B,GAC3C,OAAO,IAAIk/E,GAAoB,GAAMzrB,SAAUh6B,EAAKnY,MAAOthB,GAAYy5B,EAAKlmC,IAAKkmC,EAAKzpB,SAAUypB,EAAKxpB,SAAUwpB,EAAKkjD,YACrH,ECrLc,MAAMwC,WAAoBzC,GACxC,QAAWj1E,GACV,MAAO,MACR,CAKA,sBAAWo2E,GACV,OAAO,IACR,CAKO,KAAAt+D,GACN,OAAO,IAAI4/D,GAAa3iF,KAAKmgF,YAC9B,CAKO,WAAAoB,GACN,OAAO,IAAIoB,GAAa3iF,KAAKmgF,YAAe,EAC7C,CAGO,QAAA0B,GACP,CAKA,oBAA2B17C,GAC1B,MAAO,aACR,EChCc,MAAMy8C,WAAwB1C,GAyB5C,WAAAn+E,CAAa0jB,EAAoBo9D,EAAiBvrC,EAAiB6oC,GAClE3yE,MAAO2yE,GAEPngF,KAAKylB,SAAWA,EAEhBzlB,KAAKylB,SAAS4xC,WAAa,SAE3Br3D,KAAK6iF,QAAUA,EACf7iF,KAAKs3C,QAAUA,CAChB,CAKA,QAAWrsC,GACV,MAAO,QACR,CAKA,sBAAWo2E,GACV,OAAOrhF,KAAKylB,SAASykB,SACtB,CAOO,KAAAnnB,GACN,OAAO,IAAI6/D,GAAiB5iF,KAAKylB,SAAS1C,QAAS/iB,KAAK6iF,QAAS7iF,KAAKs3C,QAASt3C,KAAKmgF,YACrF,CAKO,WAAAoB,GACN,OAAO,IAAIqB,GAAiB5iF,KAAKylB,SAAS1C,QAAS/iB,KAAKs3C,QAASt3C,KAAK6iF,QAAS7iF,KAAKmgF,YAAe,EACpG,CAMgB,SAAAC,GACf,MAAM3oE,EAAUzX,KAAKylB,SAASykB,UAE9B,KAAQzyB,aAAmB,IAM1B,MAAM,IAAI,EACT,kCACAzX,MAEK,GAAKyX,EAAQzV,OAAShC,KAAK6iF,QAMjC,MAAM,IAAI,EACT,8BACA7iF,KAGH,CAMO,QAAA6hF,GACU7hF,KAAKylB,SAASykB,UAEXloC,KAAOhC,KAAKs3C,OAChC,CAKgB,MAAAta,GACf,MAAMC,EAAYzvB,MAAMwvB,SAIxB,OAFAC,EAAKxX,SAAWzlB,KAAKylB,SAASuX,SAEvBC,CACR,CAKA,oBAA2BkJ,GAC1B,MAAO,iBACR,CAQA,eAAuB8wB,CAAUh6B,EAAWz5B,GAC3C,OAAO,IAAIo/E,GAAiB,GAAS3rB,SAAUh6B,EAAKxX,SAAUjiB,GAAYy5B,EAAK4lD,QAAS5lD,EAAKqa,QAASra,EAAKkjD,YAC5G,EChIc,MAAM2C,WAA+B5C,GAoCnD,WAAAn+E,CACCzL,EACAS,EACAyc,EACAC,EACA0sE,GAEA3yE,MAAO2yE,GAEPngF,KAAK1J,KAAOA,EACZ0J,KAAKjJ,IAAMA,EACXiJ,KAAKwT,cAAwB,IAAbA,EAAyB,KAAOA,EAChDxT,KAAKyT,cAAwB,IAAbA,EAAyB,KAAOA,CACjD,CAKA,QAAWxI,GACV,OAAuB,OAAlBjL,KAAKwT,SACF,mBACsB,OAAlBxT,KAAKyT,SACT,sBAEA,qBAET,CAKA,sBAAW4tE,GACV,OAAOrhF,KAAK1J,IACb,CAOO,KAAAysB,GACN,OAAO,IAAI+/D,GAAwB9iF,KAAK1J,KAAM0J,KAAKjJ,IAAKiJ,KAAKwT,SAAUxT,KAAKyT,SAAUzT,KAAKmgF,YAC5F,CAKO,WAAAoB,GACN,OAAO,IAAIuB,GAAwB9iF,KAAK1J,KAAM0J,KAAKjJ,IAAKiJ,KAAKyT,SAAUzT,KAAKwT,SAAUxT,KAAKmgF,YAAe,EAC3G,CAMgB,SAAAC,GACf,GAAKpgF,KAAK1J,MAAQ0J,KAAK1J,KAAKA,MAAQ0J,KAAK1J,KAAK4X,GAAI,oBASjD,MAAM,IAAI,EACT,qCACAlO,KACA,CAAE1J,KAAM0J,KAAK1J,KAAMS,IAAKiJ,KAAKjJ,MAI/B,GAAuB,OAAlBiJ,KAAKwT,UAAqBxT,KAAK1J,KAAK+kC,aAAcr7B,KAAKjJ,OAAUiJ,KAAKwT,SAS1E,MAAM,IAAI,EACT,0CACAxT,KACA,CAAE1J,KAAM0J,KAAK1J,KAAMS,IAAKiJ,KAAKjJ,MAI/B,GAAuB,OAAlBiJ,KAAKwT,UAAuC,OAAlBxT,KAAKyT,UAAqBzT,KAAK1J,KAAK2vC,aAAcjmC,KAAKjJ,KAQrF,MAAM,IAAI,EACT,2CACAiJ,KACA,CAAE1J,KAAM0J,KAAK1J,KAAMS,IAAKiJ,KAAKjJ,KAGhC,CAMgB,QAAA8qF,GACQ,OAAlB7hF,KAAKyT,SACTzT,KAAK1J,KAAK8wC,cAAepnC,KAAKjJ,IAAKiJ,KAAKyT,UAExCzT,KAAK1J,KAAKgxC,iBAAkBtnC,KAAKjJ,IAEnC,CAKgB,MAAAimC,GACf,MAAMC,EAAYzvB,MAAMwvB,SAIxB,OAFAC,EAAK3mC,KAAO0J,KAAK1J,KAAK0mC,SAEfC,CACR,CAKA,oBAA2BkJ,GAC1B,MAAO,wBACR,CAQA,eAAuB8wB,CAAUh6B,EAAWz5B,GAC3C,IAAMA,EAASstC,QAAS7T,EAAK3mC,MAO5B,MAAM,IAAI,EAAe,2CAA4C0J,KAAM,CAAEo1B,SAAU6H,EAAK3mC,OAG7F,OAAO,IAAIwsF,GAAwBt/E,EAASstC,QAAS7T,EAAK3mC,MAAS2mC,EAAKlmC,IAAKkmC,EAAKzpB,SAAUypB,EAAKxpB,SAAUwpB,EAAKkjD,YACjH,ECrMc,MAAM4C,WAAsB7C,GA8B1C,WAAAn+E,CACCqzB,EACAuvB,EACAq+B,EACAx/E,EACA28E,GAaA,GAXA3yE,MAAO2yE,GAEPngF,KAAKo1B,SAAWA,EAChBp1B,KAAK2kD,YAAcA,EACnB3kD,KAAKgjF,MAAQA,EACbhjF,KAAK8lE,UAAYtiE,GAMXxD,KAAK8lE,UAAUh1B,QAAS9wC,KAAKo1B,UAAa,CAClCp1B,KAAK8lE,UAAUmd,WAAYjjF,KAAK2kD,YAAa3kD,KAAKo1B,UAE1D8tD,aAAc,CACpB,CACD,CAKA,QAAoBj4E,GACnB,OAAOjL,KAAKgjF,MAAQ,UAAY,YACjC,CAKA,sBAAW3B,GACV,OAAOrhF,KAAK8lE,UAAUh1B,QAAS9wC,KAAKo1B,SACrC,CAKgB,KAAArS,GACf,OAAO,IAAIggE,GAAe/iF,KAAKo1B,SAAUp1B,KAAK2kD,YAAa3kD,KAAKgjF,MAAOhjF,KAAK8lE,UAAW9lE,KAAKmgF,YAC7F,CAKgB,WAAAoB,GACf,OAAO,IAAIwB,GAAe/iF,KAAKo1B,SAAUp1B,KAAK2kD,aAAc3kD,KAAKgjF,MAAOhjF,KAAK8lE,UAAW9lE,KAAKmgF,YAAe,EAC7G,CAKgB,QAAA0B,GACf7hF,KAAK8lE,UAAUh1B,QAAS9wC,KAAKo1B,UAAY8tD,YAAcljF,KAAKgjF,KAC7D,CAKgB,MAAAhmD,GACf,MAAMC,EAAYzvB,MAAMwvB,SAIxB,cAFOC,EAAK6oC,UAEL7oC,CACR,CAKA,oBAA2BkJ,GAC1B,MAAO,eACR,CAQA,eAAuB8wB,CAAUh6B,EAAWz5B,GAC3C,OAAO,IAAIu/E,GAAe9lD,EAAK7H,SAAU6H,EAAK0nB,YAAa1nB,EAAK+lD,MAAOx/E,EAAUy5B,EAAKkjD,YACvF,EC/GD,MAAM9lB,GAIF,CAAC,EAELA,GAAYqoB,GAAmBv8C,WAAcu8C,GAC7CroB,GAAYynB,GAAgB37C,WAAc27C,GAC1CznB,GAAYkoB,GAAgBp8C,WAAco8C,GAC1CloB,GAAY+mB,GAAcj7C,WAAci7C,GACxC/mB,GAAYsoB,GAAYx8C,WAAcw8C,GACtCtoB,GAAY6lB,GAAU/5C,WAAc+5C,GACpC7lB,GAAYuoB,GAAgBz8C,WAAcy8C,GAC1CvoB,GAAYyoB,GAAuB38C,WAAc28C,GACjDzoB,GAAY0oB,GAAc58C,WAAc48C,GACxC1oB,GAAY6nB,GAAe/7C,WAAc+7C,GACzC7nB,GAAY8nB,GAAeh8C,WAAcg8C,GAK1B,MAAegB,GAO7B,eAAclsB,CAAUh6B,EAAWz5B,GAClC,OAAO62D,GAAYp9B,EAAKojD,aAAcppB,SAAUh6B,EAAMz5B,EACvD,EC3BD,MAAM4/E,GAAkB,IAAI7xE,IAqB5B,SAAS8xE,GAIRC,EACAC,EACAC,GAEA,IAAIC,EAASL,GAAgBh7E,IAAKk7E,GAE5BG,IACLA,EAAS,IAAIlyE,IACb6xE,GAAgB3xE,IAAK6xE,EAAYG,IAGlCA,EAAOhyE,IAAK8xE,EAAYC,EACzB,CAwBA,SAASE,GAAwB/nF,GAChC,MAAO,CAAEA,EACV,CAUO,SAAS,GAAWA,EAAchD,EAAc4U,EAAiC,CAAC,GACxF,MAAMi2E,EA1BP,SAA4BF,EAAsBC,GACjD,MAAME,EAASL,GAAgBh7E,IAAKk7E,GAEpC,OAAKG,GAAUA,EAAO71E,IAAK21E,GACnBE,EAAOr7E,IAAKm7E,GAGbG,EACR,CAkBgCC,CAAmBhoF,EAAEoG,YAAapJ,EAAEoJ,aAGnE,IAGC,OAAOyhF,EAFP7nF,EAAIA,EAAEonB,QAE4BpqB,EAAG4U,EACtC,CAAE,MAAQrJ,GAUT,MAAMA,CACP,CAED,CAoCO,SAAS0/E,GACfC,EACAC,EACAv/E,GASAs/E,EAAcA,EAAY/hF,QAC1BgiF,EAAcA,EAAYhiF,QAE1B,MAAMiiF,EAAiB,IAAIC,GAAgBz/E,EAAQf,SAAUe,EAAQ0/E,aAAc1/E,EAAQ2/E,iBAC3FH,EAAeI,sBAAuBN,GACtCE,EAAeI,sBAAuBL,GAEtC,MAAMM,EAAqBL,EAAeK,mBAG1C,GAA2B,GAAtBP,EAAYvrF,QAAqC,GAAtBwrF,EAAYxrF,OAC3C,MAAO,CAAEurF,cAAaC,cAAaM,sBAqIpC,MAAMC,EAAqB,IAAIh0D,QAG/B,UAAYi0D,KAAMT,EACjBQ,EAAmB5yE,IAAK6yE,EAAI,GAI7B,MAAMxhF,EAAO,CACZyhF,iBAAkBV,EAAaA,EAAYvrF,OAAS,GAAI6nF,YAAe,EACvEqE,iBAAkBV,EAAaA,EAAYxrF,OAAS,GAAI6nF,YAAe,EACvEsE,yBAA0BZ,EAAYvrF,OACtCosF,yBAA0BZ,EAAYxrF,QAIvC,IAAIkC,EAAI,EAGR,KAAQA,EAAIqpF,EAAYvrF,QAAS,CAEhC,MAAMqsF,EAAMd,EAAarpF,GAGnBoqF,EAASP,EAAmBj8E,IAAKu8E,GAGvC,GAAKC,GAAUd,EAAYxrF,OAAS,CACnCkC,IACA,QACD,CAEA,MAAMqqF,EAAMf,EAAac,GAGnBE,EAAU,GAAWH,EAAKE,EAAKd,EAAegB,WAAYJ,EAAKE,GAAK,IACpEG,EAAU,GAAWH,EAAKF,EAAKZ,EAAegB,WAAYF,EAAKF,GAAK,IAI1EZ,EAAekB,eAAgBN,EAAKE,GAEpCd,EAAeI,sBAAuBW,EAASH,GAC/CZ,EAAeI,sBAAuBa,EAASH,GAM/C,UAAYK,KAAUJ,EAMrBT,EAAmB5yE,IAAKyzE,EAAQN,EAASI,EAAQ1sF,QAIlDurF,EAAYr8E,OAAQhN,EAAG,KAAMsqF,GAC7BhB,EAAYt8E,OAAQo9E,EAAQ,KAAMI,EACnC,CAEA,GAAKzgF,EAAQ4gF,aAAe,CAE3B,MAAMC,EAAyBvB,EAAYvrF,OAASwK,EAAK2hF,yBACnDY,EAAyBvB,EAAYxrF,OAASwK,EAAK4hF,yBAMzDS,GAActB,EAAawB,EAAyBD,GACpDD,GAAcrB,EAAasB,EAAyBC,EACrD,CAMA,OAHAC,GAAoBzB,EAAa/gF,EAAK0hF,kBACtCc,GAAoBxB,EAAahhF,EAAKyhF,kBAE/B,CAAEV,cAAaC,cAAaM,qBACpC,CA4BA,MAAMJ,GAiBL,WAAAjiF,CAAayB,EAAoBygF,EAAmCC,GAAkB,GAMrFlkF,KAAKokF,mBAAqB,IAAI7yE,IAG9BvR,KAAKulF,SAAW/hF,EAASgiF,QAGzBxlF,KAAKylF,cAAgBxB,EAErBjkF,KAAK0lF,mBAAqBxB,EAK1BlkF,KAAK2lF,WAAa,IAAIp0E,GACvB,CAmBO,qBAAA4yE,CAAuB9pB,EAA8BurB,EAA6B,MACxF,MAAMC,EAAoBD,EAAW5lF,KAAKokF,mBAAmBh8E,IAAKw9E,GAAa,KAE/E,UAAYntB,KAAa4B,EACxBr6D,KAAKokF,mBAAmB3yE,IAAKgnD,EAAWotB,GAAqBptB,EAE/D,CAOO,cAAAwsB,CAAgBN,EAAgBE,GAQtC,GAAKF,aAAevD,GACdyD,aAAe1C,GACdwC,EAAItvC,eAAepxB,QAAS4gE,EAAI7rB,iBAAoB6rB,EAAI5rB,WAAW1tB,iBAAkBo5C,EAAItvC,gBAC7Fr1C,KAAK8lF,aAAcnB,EAAKE,EAAK,kBAClBF,EAAItvC,eAAepxB,QAAS4gE,EAAItrB,kBAC3Cv5D,KAAK8lF,aAAcnB,EAAKE,EAAK,iBAClBF,EAAItvC,eAAexY,QAASgoD,EAAI7rB,iBAC3Ch5D,KAAK8lF,aAAcnB,EAAKE,EAAK,mBAEnBA,aAAezD,KACrBuD,EAAItvC,eAAepxB,QAAS4gE,EAAI7rB,iBAAoB2rB,EAAItvC,eAAe3Y,SAAUmoD,EAAI7rB,gBACzFh5D,KAAK8lF,aAAcnB,EAAKE,EAAK,gBAE7B7kF,KAAK8lF,aAAcnB,EAAKE,EAAK,qBAGzB,GAAKF,aAAezC,IAC1B,GAAK2C,aAAe1C,GACdwC,EAAIxrB,cAAcz8B,SAAUmoD,EAAI7rB,iBACpCh5D,KAAK8lF,aAAcnB,EAAKE,EAAK,oBAExB,GAAKA,aAAezD,GAC1B,GAAKuD,EAAIxrB,cAAcl1C,QAAS4gE,EAAI7rB,iBAAoB2rB,EAAIxrB,cAAcz8B,SAAUmoD,EAAI7rB,gBACvFh5D,KAAK8lF,aAAcnB,EAAKE,EAAK,mBACvB,CACN,MAAM//D,EAAQ,GAAMynB,4BAA6Bs4C,EAAI7rB,eAAgB6rB,EAAI15E,SAEzE,GAAKw5E,EAAIxrB,cAAcZ,gBAAiBssB,EAAI7rB,iBAAoBl0C,EAAMymB,iBAAkBo5C,EAAIxrB,eAAkB,CAC7G,MAAMhuD,EAAU2Z,EAAM6b,IAAI/kB,OAAS+oE,EAAIxrB,cAAcv9C,OAC/CA,EAAS+oE,EAAIxrB,cAAcv9C,OAASkJ,EAAM4b,MAAM9kB,OAEtD5b,KAAK8lF,aAAcnB,EAAKE,EAAK,CAAE15E,UAASyQ,UACzC,CACD,OAEK,GAAK+oE,aAAexC,GACrB0C,aAAe1C,IACbwC,EAAItvC,eAAepxB,QAAS4gE,EAAI7rB,iBACrCh5D,KAAK8lF,aAAcnB,EAAKE,EAAK,uBAGzBF,EAAI3rB,eAAe/0C,QAAS4gE,EAAIxvC,iBACpCr1C,KAAK8lF,aAAcnB,EAAKE,EAAK,uBAGzBF,EAAI3rB,eAAe/0C,QAAS4gE,EAAI7rB,iBACpCh5D,KAAK8lF,aAAcnB,EAAKE,EAAK,qBAEnBA,aAAe3C,IACrByC,EAAI3rB,eAAe/0C,QAAS4gE,EAAI1rB,gBACpCn5D,KAAK8lF,aAAcnB,EAAKE,EAAK,sBAGzB,GAAKF,aAAepC,GAAkB,CAC5C,MAAM7hB,EAAcikB,EAAIxtC,SAExB,IAAMupB,EACL,OAGD,GAAKmkB,aAAezD,GAAgB,CACnC,MAAMnoB,EAAa,GAAM1sB,4BAA6Bs4C,EAAI7rB,eAAgB6rB,EAAI15E,SAExE46E,EAAe9sB,EAAW1tB,iBAAkBm1B,EAAYhgC,QAC7Du4B,EAAWv4B,MAAMzc,QAASy8C,EAAYhgC,OAEjCslD,EAAgB/sB,EAAW1tB,iBAAkBm1B,EAAY//B,MAC9Ds4B,EAAWt4B,IAAI1c,QAASy8C,EAAY//B,MAE9BolD,IAAgBC,GAAoB/sB,EAAWztB,cAAek1B,IACpE1gE,KAAK8lF,aAAcnB,EAAKE,EAAK,CAC5BoB,KAAMF,EAAe,OAAS,QAC9BtmF,KAAMsmF,EAAerlB,EAAYhgC,MAAMjhC,KAAKqC,QAAU4+D,EAAY//B,IAAIlhC,KAAKqC,SAG9E,MAAO,GAAK+iF,aAAe1C,GAAiB,CAC3C,MAAM+D,EAAmBxlB,EAAYhgC,MAAMzc,QAAS4gE,EAAIxvC,gBAClD8wC,EAA8BzlB,EAAYhgC,MAAMzc,QAAS4gE,EAAItrB,kBAC7D6sB,EAA4B1lB,EAAY//B,IAAI1c,QAAS4gE,EAAItrB,kBACzD8sB,EAAoB3lB,EAAY//B,IAAI1c,QAAS4gE,EAAI7rB,iBAElDktB,GAAoBC,GAA+BC,GAA6BC,IACpFrmF,KAAK8lF,aAAcnB,EAAKE,EAAK,CAC5BqB,mBACAC,8BACAC,4BACAC,qBAGH,CACD,CACD,CAKO,UAAAtB,CAAYJ,EAAgBE,EAAgByB,GAClD,MAAO,CACNA,YACAC,WAAYvmF,KAAKwmF,WAAY7B,GAC7B8B,WAAYzmF,KAAKwmF,WAAY3B,GAC7B6B,WAAY1mF,KAAKylF,cAAgBzlF,KAAK2mF,aAAchC,EAAKE,GAAQ,KACjE+B,WAAY5mF,KAAKylF,cAAgBzlF,KAAK2mF,aAAc9B,EAAKF,GAAQ,KACjET,gBAAiBlkF,KAAK0lF,iBAExB,CAOO,UAAAc,CAAYlC,GAIlB,MAAMuC,EAAa7mF,KAAKokF,mBAAmBh8E,IAAKk8E,GAGhD,OAASuC,EAAoBC,WAAa9mF,KAAKulF,SAASwB,kBAAmBF,EAC5E,CAwBO,YAAAF,CAAchC,EAAgBE,GAEpC,MAAMmC,EAAQhnF,KAAKokF,mBAAmBh8E,IAAKy8E,GACrCoC,EAAUjnF,KAAKulF,SAAS2B,mBAAoBF,GAGlD,IAAMC,EACL,OAAO,KAGR,MAAME,EAAQnnF,KAAKokF,mBAAmBh8E,IAAKu8E,GACrCyC,EAAapnF,KAAK2lF,WAAWv9E,IAAK++E,GAGxC,OAAKC,GACGA,EAAWh/E,IAAK6+E,IAGjB,IACR,CAKQ,YAAAnB,CAAcnB,EAAgBE,EAAgBwC,GAErD,MAAMF,EAAQnnF,KAAKokF,mBAAmBh8E,IAAKu8E,GACrCqC,EAAQhnF,KAAKokF,mBAAmBh8E,IAAKy8E,GAE3C,IAAIuC,EAAapnF,KAAK2lF,WAAWv9E,IAAK++E,GAEhCC,IACLA,EAAa,IAAI71E,IACjBvR,KAAK2lF,WAAWl0E,IAAK01E,EAAOC,IAG7BA,EAAW31E,IAAKu1E,EAAOK,EACxB,EAgDD,SAAS/B,GAAoBjrB,EAAsC8lB,GAClE,UAAY1nB,KAAa4B,EACxB5B,EAAU0nB,YAAcA,GAE1B,CAKA,SAASgF,GAAc9qB,EAA8BlvD,GACpD,QAAU3Q,EAAI,EAAGA,EAAI2Q,EAAS3Q,IAC7B6/D,EAAWz6D,KAAM,IAAI+iF,GAAa,GAEpC,CA2HA,SAAS2E,GAAsCC,EAAkCxwF,EAAa0c,GAC7F,MAGM+zE,EAHQD,EAAgBrgD,MAGJovB,QAAS,GAAKj7B,aAActkC,GAEtD,GAAKywF,GAAe/zE,EACnB,OAAO,KAGR,MAAMqR,EAAQ,IAAI,GAAOyiE,EAAgB9hE,SAAU8hE,EAAgB9hE,SAAS6kB,aAAci9C,EAAgBp8E,UAE1G,OAAO,IAAIu3E,GAAoB59D,EAAO/tB,EAAKywF,EAAa/zE,EAAU,EACnE,CAm7CA,SAASg0E,GAA2B9rF,EAAkBhD,GACrD,OAAqF,OAA9EgD,EAAE05C,eAAeikB,0BAA2B3gE,EAAEqgE,eAAgBrgE,EAAEwS,QACxE,CAYA,SAASu8E,GAA+B77C,EAAsBwJ,GAU7D,MAAMglB,EAAa,GAGnB,QAAU7/D,EAAI,EAAGA,EAAIqxC,EAAOvzC,OAAQkC,IAAM,CAEzC,MAAMsqB,EAAQ+mB,EAAQrxC,GAChB8pF,EAAK,IAAIlD,GACdt8D,EAAM4b,MACN5b,EAAM6b,IAAI/kB,OAASkJ,EAAM4b,MAAM9kB,OAC/By5B,EACA,GAGDglB,EAAWz6D,KAAM0kF,GAGjB,QAAUhqB,EAAI9/D,EAAI,EAAG8/D,EAAIzuB,EAAOvzC,OAAQgiE,IAOvCzuB,EAAQyuB,GAAMzuB,EAAQyuB,GAAIvB,sBAAuBurB,EAAGtrB,eAAgBsrB,EAAGjvC,eAAgBivC,EAAGn5E,SAAW,GAGtGkqC,EAAiBA,EAAe0jB,sBAAuBurB,EAAGtrB,eAAgBsrB,EAAGjvC,eAAgBivC,EAAGn5E,QACjG,CAEA,OAAOkvD,CACR,CA7mDAgpB,GAAmBX,GAAoBA,IAAoB,CAAE/mF,EAAGhD,EAAG4U,KAQlE,GAAK5R,EAAE5E,MAAQ4B,EAAE5B,KAAO4E,EAAEmpB,MAAM4b,MAAM63B,gBAAiB5/D,EAAEmsB,MAAM4b,OAAU,CAExE,MAAM25B,EAAa1+D,EAAEmpB,MAAM8mB,cAAejzC,EAAEmsB,OAAQ9nB,KAAK8nB,GACjD,IAAI49D,GAAoB59D,EAAOnpB,EAAE5E,IAAK4E,EAAE6X,SAAU7X,EAAE8X,SAAU,KAIhEmnD,EAASj/D,EAAEmpB,MAAM5B,gBAAiBvqB,EAAEmsB,OAW1C,OATK81C,GAICrtD,EAAQ+4E,WACZjsB,EAAWz6D,KAAM,IAAI8iF,GAAoB9nB,EAAQjiE,EAAE5B,IAAK4B,EAAE8a,SAAU9X,EAAE8X,SAAU,IAIxD,GAArB4mD,EAAW/hE,OACR,CAAE,IAAIqqF,GAAa,IAGpBtoB,CACR,CAEC,MAAO,CAAE1+D,EACV,IAGD0nF,GAAmBX,GAAoBZ,IAAiB,CAAEnmF,EAAGhD,KAO5D,GAAKgD,EAAEmpB,MAAM4b,MAAM63B,gBAAiB5/D,EAAE8sB,WAAc9pB,EAAEmpB,MAAMymB,iBAAkB5yC,EAAE8sB,UAAa,CAG5F,MACMtnB,EADQxC,EAAEmpB,MAAMg0C,2BAA4BngE,EAAE8sB,SAAU9sB,EAAEwS,SAAUxS,EAAEopF,yBACvD/kF,KAAKvE,GAClB,IAAIiqF,GAAoBjqF,EAAGkD,EAAE5E,IAAK4E,EAAE6X,SAAU7X,EAAE8X,SAAU9X,EAAEwkF,eAGpE,GAAKxnF,EAAEopF,wBAA0B,CA4ChC,MAAMuC,EAAKgD,GAAsC3uF,EAAGgD,EAAE5E,IAAK4E,EAAE6X,UAExD8wE,GACJnmF,EAAOiB,QAASklF,EAElB,CAGA,OAAOnmF,CACR,CAKA,OAFAxC,EAAEmpB,MAAQnpB,EAAEmpB,MAAMg0C,2BAA4BngE,EAAE8sB,SAAU9sB,EAAEwS,SAAS,GAAS,GAEvE,CAAExP,EAAG,IAwBb0nF,GAAmBX,GAAoBP,IAAgB,CAAExmF,EAAGhD,KAC3D,MAAMkzC,EAAS,GAOVlwC,EAAEmpB,MAAM4b,MAAM63B,gBAAiB5/D,EAAE4gE,oBAChC59D,EAAEmpB,MAAMymB,iBAAkB5yC,EAAE4gE,mBAAsB59D,EAAEmpB,MAAM4b,MAAMzc,QAAStrB,EAAE4gE,oBAC/E1tB,EAAOjsC,KAAM,GAAM2sC,4BAA6B5zC,EAAE0gE,kBAAmB,IAIvE,MAAMv0C,EAAQnpB,EAAEmpB,MAAM+zC,gCAAiClgE,GAQvD,OALMmsB,EAAMkmB,aACXa,EAAOjsC,KAAMklB,GAIP+mB,EAAO7uC,KAAK8nB,GACX,IAAI49D,GAAoB59D,EAAOnpB,EAAE5E,IAAK4E,EAAE6X,SAAU7X,EAAE8X,SAAU9X,EAAEwkF,cACrE,IAGJkD,GAAmBX,GAAoBtB,IAAe,CAAEzlF,EAAGhD,KAC1D,MAAMkzC,EAiBP,SAAqC/mB,EAAc6iE,GAClD,MAAMltB,EAAY,GAAMluB,4BAA6Bo7C,EAAO3uB,eAAgB2uB,EAAOx8E,SAInF,IAAIyvD,EAAS,KACTD,EAA2B,GAG1BF,EAAUjvB,cAAe1mB,GAAO,GAEpC81C,EAAS91C,EACEA,EAAM4b,MAAM63B,gBAAiBkC,EAAU/5B,QAGlDi6B,EAAa71C,EAAM8mB,cAAe6uB,GAClCG,EAAS91C,EAAM5B,gBAAiBu3C,IAOhCE,EAAa,CAAE71C,GAGhB,MAAM3mB,EAAuB,GAI7B,QAAU5E,KAAQohE,EAAa,CAG9BphE,EAAOA,EAAK+/D,0BAA2BquB,EAAO3uB,eAAgB2uB,EAAOx8E,SAGrE,MAAMkqC,EAAiBsyC,EAAOrG,qBAGxB/mB,EAAShhE,EAAKmnC,MAAM63B,gBAAiBljB,GAGrCuyC,EAAQruF,EAAKu/D,2BAA4BzjB,EAAgBsyC,EAAOx8E,QAASovD,GAE/Ep8D,EAAOyB,QAASgoF,EACjB,CAIKhtB,GACJz8D,EAAOyB,KACNg7D,EAAO7B,sBAAuB4uB,EAAO3uB,eAAgB2uB,EAAOtyC,eAAgBsyC,EAAOx8E,SAAS,GAAS,IAIvG,OAAOhN,CACR,CAzEgB0pF,CAA4BlsF,EAAEmpB,MAAOnsB,GAGpD,OAAOkzC,EAAO7uC,KAAK8nB,GAAS,IAAI49D,GAAoB59D,EAAOnpB,EAAE5E,IAAK4E,EAAE6X,SAAU7X,EAAE8X,SAAU9X,EAAEwkF,cAAe,IAwE5GkD,GAAmBX,GAAoBR,IAAgB,CAAEvmF,EAAGhD,KAe3D,GAAKgD,EAAEmpB,MAAM6b,IAAI1c,QAAStrB,EAAE2/C,mBAK3B,OAJM3/C,EAAE0gE,mBACP19D,EAAEmpB,MAAM6b,IAAI/kB,SAGN,CAAEjgB,GAiBV,GAAKA,EAAEmpB,MAAM4b,MAAM63B,gBAAiB5/D,EAAEwgE,gBAAmBx9D,EAAEmpB,MAAMymB,iBAAkB5yC,EAAEwgE,eAAkB,CACtG,MAAM4nB,EAAaplF,EAAEonB,QAUrB,OARAg+D,EAAWj8D,MAAQ,IAAI,GACtBnsB,EAAEygE,mBAAmBr2C,QACrBpnB,EAAEmpB,MAAM6b,IAAIu4B,aAAcvgE,EAAEwgE,cAAexgE,EAAEygE,qBAG5Cz9D,EAAEmpB,MAAe6b,IAAMhoC,EAAEwgE,cAAcp2C,QACzCpnB,EAAEmpB,MAAM6b,IAAI02B,WAAa,aAElB,CAAE17D,EAAGolF,EACb,CAMA,OAFAplF,EAAEmpB,MAAQnpB,EAAEmpB,MAAM8zC,gCAAiCjgE,GAE5C,CAAEgD,EAAG,IAGb0nF,GAAmBvB,GAAiBY,IAAoB,CAAE/mF,EAAGhD,KAC5D,MAAMwF,EAA2B,CAAExC,GAYnC,GAAKA,EAAEomF,yBAA2BpmF,EAAE8pB,SAAS8yC,gBAAiB5/D,EAAEmsB,MAAM4b,QAAW/nC,EAAEmsB,MAAMymB,iBAAkB5vC,EAAE8pB,UAAa,CACzH,MAAM6+D,EAAKgD,GAAsC3rF,EAAGhD,EAAE5B,IAAK4B,EAAE8a,UAExD6wE,GACJnmF,EAAOyB,KAAM0kF,EAEf,CAKA,OAAOnmF,CAAM,IAGdklF,GAAmBvB,GAAiBA,IAAiB,CAAEnmF,EAAGhD,EAAG4U,KAUvD5R,EAAE8pB,SAASxB,QAAStrB,EAAE8sB,WAAclY,EAAQ+4E,YAMjD3qF,EAAE8pB,SAAW9pB,EAAE8pB,SAASizC,iCAAkC//D,IALlD,CAAEgD,MAUX0nF,GAAmBvB,GAAiBV,IAAe,CAAEzlF,EAAGhD,KAGvDgD,EAAE8pB,SAAW9pB,EAAE8pB,SAASkzC,+BAAgChgE,GAEjD,CAAEgD,MAGV0nF,GAAmBvB,GAAiBI,IAAgB,CAAEvmF,EAAGhD,KAGxDgD,EAAE8pB,SAAW9pB,EAAE8pB,SAASmzC,gCAAiCjgE,GAElD,CAAEgD,MAGV0nF,GAAmBvB,GAAiBK,IAAgB,CAAExmF,EAAGhD,KACxDgD,EAAE8pB,SAAW9pB,EAAE8pB,SAASozC,gCAAiClgE,GAElD,CAAEgD,MAKV0nF,GAAmBd,GAAiBT,IAAiB,CAAEnmF,EAAGhD,KACpDgD,EAAE4mE,WACN5mE,EAAE4mE,SAAW5mE,EAAE4mE,SAAS7J,iCAAkC//D,GAAK,IAG3DgD,EAAEw7C,WACNx7C,EAAEw7C,SAAWx7C,EAAEw7C,SAASuhB,iCAAkC//D,GAAK,IAGzD,CAAEgD,MAGV0nF,GAAmBd,GAAiBA,IAAiB,CAAE5mF,EAAGhD,EAAG4U,KAC5D,GAAK5R,EAAEqG,MAAQrJ,EAAEqJ,KAAO,CACvB,IAAKuL,EAAQ+4E,UAGZ,MAAO,CAAE,IAAI3D,GAAa,IAF1BhnF,EAAE4mE,SAAW5pE,EAAEw+C,SAAWx+C,EAAEw+C,SAASp0B,QAAU,IAIjD,CAEA,MAAO,CAAEpnB,EAAG,IAGb0nF,GAAmBd,GAAiBJ,IAAgB,CAAExmF,EAAGhD,KACnDgD,EAAE4mE,WACN5mE,EAAE4mE,SAAW5mE,EAAE4mE,SAAS1J,gCAAiClgE,IAGrDgD,EAAEw7C,WACNx7C,EAAEw7C,SAAWx7C,EAAEw7C,SAAS0hB,gCAAiClgE,IAGnD,CAAEgD,MAGV0nF,GAAmBd,GAAiBnB,IAAe,CAAEzlF,EAAGhD,EAAG4U,KAK1D,GAJK5R,EAAE4mE,WACN5mE,EAAE4mE,SAAW,GAAMvH,kBAAmBr/D,EAAE4mE,SAAS5J,+BAAgChgE,KAG7EgD,EAAEw7C,SAAW,CACjB,GAAK5pC,EAAQm5E,WAAa,CACzB,MAAMoB,EAAY,GAAM9sB,kBAAmBr/D,EAAEw7C,SAASwhB,+BAAgChgE,IAEtF,GAAgC,QAA3B4U,EAAQm5E,WAAWT,MAAkBttF,EAAE08C,eAAepxB,QAAStoB,EAAEw7C,SAASzW,OAI9E,OAHE/kC,EAAEw7C,SAAkBxW,IAAMmnD,EAAUnnD,IACpChlC,EAAEw7C,SAASzW,MAAejhC,KAAO8N,EAAQm5E,WAAWjnF,KAE/C,CAAE9D,GACH,GAAgC,SAA3B4R,EAAQm5E,WAAWT,MAAmBttF,EAAE08C,eAAepxB,QAAStoB,EAAEw7C,SAASxW,KAItF,OAHEhlC,EAAEw7C,SAAkBzW,MAAQonD,EAAUpnD,MACtC/kC,EAAEw7C,SAASxW,IAAalhC,KAAO8N,EAAQm5E,WAAWjnF,KAE7C,CAAE9D,EAEX,CAEAA,EAAEw7C,SAAW,GAAM6jB,kBAAmBr/D,EAAEw7C,SAASwhB,+BAAgChgE,GAClF,CAEA,MAAO,CAAEgD,EAAG,IAGb0nF,GAAmBd,GAAiBL,IAAgB,CAAEvmF,EAAGhD,EAAG4U,KAK3D,GAJK5R,EAAE4mE,WACN5mE,EAAE4mE,SAAW5mE,EAAE4mE,SAAS3J,gCAAiCjgE,IAGrDgD,EAAEw7C,SAAW,CACjB,GAAK5pC,EAAQm5E,WAAa,CACzB,MAAMoB,EAAYnsF,EAAEw7C,SAASyhB,gCAAiCjgE,GAgB9D,OAdKgD,EAAEw7C,SAASzW,MAAMzc,QAAStrB,EAAEwgE,gBAAmB5rD,EAAQm5E,WAAWP,4BACpExqF,EAAEw7C,SAAkBzW,MAAQ,GAASkI,UAAWjwC,EAAE2/C,mBACzC38C,EAAEw7C,SAASzW,MAAMzc,QAAStrB,EAAEwgE,iBAAoB5rD,EAAQm5E,WAAWR,mBAC5EvqF,EAAEw7C,SAAkBzW,MAAQ,GAASkI,UAAWjwC,EAAEygE,qBAGhDz9D,EAAEw7C,SAASxW,IAAI1c,QAAStrB,EAAEwgE,gBAAmB5rD,EAAQm5E,WAAWL,kBAClE1qF,EAAEw7C,SAAkBxW,IAAM,GAASiI,UAAWjwC,EAAEygE,oBACvCz9D,EAAEw7C,SAASxW,IAAI1c,QAAStrB,EAAEwgE,gBAAmB5rD,EAAQm5E,WAAWN,0BACzEzqF,EAAEw7C,SAAkBxW,IAAM,GAASiI,UAAWjwC,EAAE2/C,mBAEhD38C,EAAEw7C,SAAkBxW,IAAMmnD,EAAUnnD,IAGhC,CAAEhlC,EACV,CAEAA,EAAEw7C,SAAWx7C,EAAEw7C,SAASyhB,gCAAiCjgE,EAC1D,CAEA,MAAO,CAAEgD,EAAG,IAKb0nF,GAAmBlB,GAAgBL,IAAiB,CAAEnmF,EAAGhD,KACnDgD,EAAEq9D,eAAeT,gBAAiB5/D,EAAE8sB,YACxC9pB,EAAEwP,SAAWxS,EAAEwS,SAGhBxP,EAAEq9D,eAAiBr9D,EAAEq9D,eAAeN,iCAAkC//D,GACtEgD,EAAE05C,eAAiB15C,EAAE05C,eAAeqjB,iCAAkC//D,GAE/D,CAAEgD,MAGV0nF,GAAmBlB,GAAgBA,IAAgB,CAAExmF,EAAGhD,EAAG4U,KAQ1D,GAAK5R,EAAEq9D,eAAe/0C,QAAStrB,EAAEqgE,iBAAoBr9D,EAAE05C,eAAepxB,QAAStrB,EAAE08C,gBAAmB,CAYnG,GAAM9nC,EAAQk5E,WAEP,CACN,MAAMhnF,EAAO9G,EAAE0gE,kBAAkB55D,KAAKqC,QAMtC,OALArC,EAAKG,KAAM,GAEXjE,EAAEq9D,eAAiB,IAAI,GAAUrgE,EAAE0gE,kBAAkB/iE,KAAMmJ,GAC3D9D,EAAEwP,QAAU,EAEL,CAAExP,EACV,CATC,MAAO,CAAE,IAAIgnF,GAAa,GAU5B,CA6BA,GACChnF,EAAEq9D,eAAe/0C,QAAStrB,EAAEqgE,kBAAqBr9D,EAAE05C,eAAepxB,QAAStrB,EAAE08C,kBAC5E9nC,EAAQk5E,YAAoC,iBAAtBl5E,EAAQm5E,WAC9B,CACD,MAAMqB,EAAiD,cAAlCpsF,EAAE05C,eAAe/+C,KAAK8+B,SACrC4yD,EAAiD,cAAlCrvF,EAAE08C,eAAe/+C,KAAK8+B,SAW3C,GALgB4yD,IAAiBD,KAHjBA,IAAiBC,IAMUz6E,EAAQ+4E,UAElC,CAChB,MAAMttB,EAAiBrgE,EAAE08C,eAAewjB,gCAAiClgE,GACnE08C,EAAiB15C,EAAE05C,eAAewjB,gCAAiClgE,GAEzE,MAAO,CAAE,IAAIyoF,GAAepoB,EAAgBr9D,EAAEwP,QAASkqC,EAAgB,GACxE,CACC,MAAO,CAAE,IAAIstC,GAAa,GAE5B,CAiBA,OAbKhnF,EAAEq9D,eAAeT,gBAAiB5/D,EAAE08C,kBACxC15C,EAAEwP,SAAWxS,EAAEwS,SAGhBxP,EAAEq9D,eAAiBr9D,EAAEq9D,eAAeH,gCAAiClgE,GACrEgD,EAAE05C,eAAiB15C,EAAE05C,eAAewjB,gCAAiClgE,GAI/DgD,EAAE09D,kBAAkBp1C,QAAStrB,EAAE0gE,oBAAwB9rD,EAAQ+4E,YACpE3qF,EAAE09D,kBAAoB19D,EAAE09D,kBAAkBR,gCAAiClgE,IAGrE,CAAEgD,EAAG,IAGb0nF,GAAmBlB,GAAgBf,IAAe,CAAEzlF,EAAGhD,EAAG4U,KAYzD,MAAM06E,EAAe,GAAM17C,4BAA6B5zC,EAAEqgE,eAAgBrgE,EAAEwS,SAE5E,MAAe,UAAVxS,EAAEsS,OAAqBsC,EAAQk5E,aAAel5E,EAAQ22E,iBACrDvoF,EAAE49D,iBAAiBhB,gBAAiB5/D,EAAEqgE,iBAAoBivB,EAAa18C,iBAAkB5vC,EAAEq9D,gBACxF,CAAE,IAAI2pB,GAAa,KAMvBhnF,EAAEq9D,eAAeT,gBAAiB5/D,EAAE08C,kBACxC15C,EAAEwP,SAAWxS,EAAEwS,SAGXxP,EAAEq9D,eAAeT,gBAAiB5/D,EAAEqgE,kBACxCr9D,EAAEwP,SAAWxS,EAAEwS,SAGhBxP,EAAEq9D,eAAiBr9D,EAAEq9D,eAAeL,+BAAgChgE,GACpEgD,EAAE05C,eAAiB15C,EAAE05C,eAAesjB,+BAAgChgE,GAM9DgD,EAAE09D,kBAAkBp1C,QAAStrB,EAAE08C,kBACpC15C,EAAE09D,kBAAoB19D,EAAE09D,kBAAkBV,+BAAgChgE,IAGpE,CAAEgD,GAAG,IAGb0nF,GAAmBlB,GAAgBD,IAAgB,CAAEvmF,EAAGhD,EAAG4U,KAyE1D,GAxEK5U,EAAE0gE,oBAGN19D,EAAE09D,kBAAoB19D,EAAE09D,kBAAkBC,0BAA2B3gE,EAAE0gE,kBAAmB,GAYrF19D,EAAE49D,iBAAiBt1C,QAAStrB,EAAE0gE,qBAClC19D,EAAEwP,QAAUxS,EAAEwS,UAwDXxP,EAAE05C,eAAepxB,QAAStrB,EAAEwgE,eAAkB,CAClD,MAAM+uB,EAA2B,GAAbvvF,EAAEwS,QAChBg9E,EAAwBxvF,EAAE0gE,mBAAqB19D,EAAE49D,iBAAiBt1C,QAAStrB,EAAE0gE,mBAEnF,GAAK6uB,GAAeC,GAA+C,uBAAtB56E,EAAQm5E,WAGpD,OAFA/qF,EAAEq9D,eAAiBr9D,EAAEq9D,eAAeJ,gCAAiCjgE,GAE9D,CAAEgD,EAEX,CAQA,GAAKA,EAAEq9D,eAAe/0C,QAAStrB,EAAEwgE,eAAkB,CAIlD,GAA2B,uBAAtB5rD,EAAQm5E,WAIZ,OAHA/qF,EAAEwP,QAAU,EACZxP,EAAE05C,eAAiB15C,EAAE05C,eAAeujB,gCAAiCjgE,GAE9D,CAAEgD,GAUV,GAA2B,oBAAtB4R,EAAQm5E,YAAoC/qF,EAAEq9D,eAAep9C,OAAS,EAI1E,OAHAjgB,EAAEq9D,eAAiBrgE,EAAEygE,mBAAmBr2C,QACxCpnB,EAAE05C,eAAiB15C,EAAE05C,eAAeujB,gCAAiCjgE,GAE9D,CAAEgD,EAEX,CAWA,OAPKA,EAAEq9D,eAAeT,gBAAiB5/D,EAAEwgE,iBACxCx9D,EAAEwP,QAAUxS,EAAEwgE,cAAcv9C,QAG7BjgB,EAAEq9D,eAAiBr9D,EAAEq9D,eAAeJ,gCAAiCjgE,GACrEgD,EAAE05C,eAAiB15C,EAAE05C,eAAeujB,gCAAiCjgE,GAE9D,CAAEgD,EAAG,IAKb0nF,GAAmBjC,GAAeU,IAAiB,CAAEnmF,EAAGhD,KACvD,MACM8gE,EADY,GAAMltB,4BAA6B5wC,EAAEq9D,eAAgBr9D,EAAEwP,SAC3CutD,iCAAkC//D,GAAG,GAAS,GAe5E,OAbAgD,EAAEq9D,eAAiBS,EAAY/4B,MAC/B/kC,EAAEwP,QAAUsuD,EAAY94B,IAAI/kB,OAAS69C,EAAY/4B,MAAM9kB,OAQjDjgB,EAAE05C,eAAepxB,QAAStrB,EAAE8sB,YACjC9pB,EAAE05C,eAAiB15C,EAAE05C,eAAeqjB,iCAAkC//D,IAGhE,CAAEgD,EAAG,IAGb0nF,GAAmBjC,GAAeA,IAAe,CAAEzlF,EAAGhD,EAAG4U,KAKxD,MAAMugC,EAAS,GAAMvB,4BAA6B5wC,EAAEq9D,eAAgBr9D,EAAEwP,SAChE4iC,EAAS,GAAMxB,4BAA6B5zC,EAAEqgE,eAAgBrgE,EAAEwS,SAItE,IAcIq2E,EAdA8E,EAAY/4E,EAAQ+4E,UAIpB9/E,GAAgB+G,EAAQ+4E,UA+B5B,GA5B2B,gBAAtB/4E,EAAQm5E,YAAsD,eAAtBn5E,EAAQq5E,WACpDpgF,GAAe,EACkB,eAAtB+G,EAAQm5E,YAAqD,gBAAtBn5E,EAAQq5E,aAC1DpgF,GAAe,GAOfg7E,EADI7lF,EAAE05C,eAAepxB,QAAStrB,EAAE08C,iBAAoB7uC,EAChC7K,EAAE05C,eAAeikB,0BACpC3gE,EAAEqgE,eACFrgE,EAAEwS,SAGiBxP,EAAE05C,eAAe0jB,sBACpCpgE,EAAEqgE,eACFrgE,EAAE08C,eACF18C,EAAEwS,SAUCs8E,GAA2B9rF,EAAGhD,IAAO8uF,GAA2B9uF,EAAGgD,GAGvE,MAAO,CAAEhD,EAAE4oF,eAcZ,GAJoBzzC,EAAOvC,iBAAkB5yC,EAAE08C,iBAI3BvH,EAAOtC,cAAeuC,GAAQ,GAMjD,OAHED,EAAgBpN,MAAQoN,EAAOpN,MAAMq4B,sBAAuBpgE,EAAEqgE,eAAgBrgE,EAAE08C,eAAgB18C,EAAEwS,SAClG2iC,EAAgBnN,IAAMmN,EAAOnN,IAAIo4B,sBAAuBpgE,EAAEqgE,eAAgBrgE,EAAE08C,eAAgB18C,EAAEwS,SAEzFu8E,GAA+B,CAAE55C,GAAU0zC,GAQnD,GAFoBzzC,EAAOxC,iBAAkB5vC,EAAE05C,iBAE3BtH,EAAOvC,cAAesC,GAAQ,GAMjD,OAHEA,EAAgBpN,MAAQoN,EAAOpN,MAAMw4B,aAAcvgE,EAAEqgE,eAAgBrgE,EAAE2oF,sBACvExzC,EAAgBnN,IAAMmN,EAAOnN,IAAIu4B,aAAcvgE,EAAEqgE,eAAgBrgE,EAAE2oF,sBAE9DoG,GAA+B,CAAE55C,GAAU0zC,GAanD,MAAM4G,EAAS1xE,EAAe/a,EAAEq9D,eAAejB,gBAAiBp/D,EAAEqgE,eAAejB,iBAEjF,GAAe,UAAVqwB,GAAgC,aAAVA,EAO1B,OAHEt6C,EAAgBpN,MAAQoN,EAAOpN,MAAMq4B,sBAAuBpgE,EAAEqgE,eAAgBrgE,EAAE08C,eAAgB18C,EAAEwS,SAClG2iC,EAAgBnN,IAAMmN,EAAOnN,IAAIo4B,sBAAuBpgE,EAAEqgE,eAAgBrgE,EAAE08C,eAAgB18C,EAAEwS,SAEzFu8E,GAA+B,CAAE55C,GAAU0zC,GAcpC,UAAV7lF,EAAEsP,MAA8B,UAAVtS,EAAEsS,MAAqBsC,EAAQg5E,YAAeh5E,EAAQ22E,gBAE3D,UAAVvoF,EAAEsP,MAA8B,UAAVtS,EAAEsS,MAAqBsC,EAAQk5E,YAAel5E,EAAQ22E,kBACvFoC,GAAY,GAFZA,GAAY,EAOb,MAAMz6C,EAAS,GAIT8uB,EAAa7sB,EAAOlC,cAAemC,GAEzC,UAAYjpB,KAAS61C,EAAa,CAE/B71C,EAAe4b,MAAQ5b,EAAM4b,MAAM44B,0BAA2B3gE,EAAEqgE,eAAgBrgE,EAAEwS,SAClF2Z,EAAe6b,IAAM7b,EAAM6b,IAAI24B,0BAA2B3gE,EAAEqgE,eAAgBrgE,EAAEwS,SAGhF,MAAMk9E,EAAuG,QAAxF3xE,EAAeoO,EAAM4b,MAAMq3B,gBAAiBp/D,EAAE2oF,qBAAqBvpB,iBAClFnpB,EAAY9pB,EAAMg0C,2BAA4BngE,EAAE2oF,qBAAsB3oF,EAAEwS,QAASk9E,GAEvFx8C,EAAOjsC,QAASgvC,EACjB,CAGA,MAAMgsB,EAAS9sB,EAAO5qB,gBAAiB6qB,GA+BvC,OA7BgB,OAAX6sB,GAAmB0rB,IAErB1rB,EAAgBl6B,MAAQk6B,EAAOl6B,MAAMw4B,aAAcvgE,EAAEqgE,eAAgBrgE,EAAE2oF,sBACvE1mB,EAAgBj6B,IAAMi6B,EAAOj6B,IAAIu4B,aAAcvgE,EAAEqgE,eAAgBrgE,EAAE2oF,sBAQ9C,IAAlBz1C,EAAOvzC,OACXuzC,EAAOjsC,KAAMg7D,GAGa,GAAjB/uB,EAAOvzC,OACXy1C,EAAOrN,MAAMhE,SAAUoR,EAAOpN,QAAWqN,EAAOrN,MAAMzc,QAAS6pB,EAAOpN,OAC1EmL,EAAOzsC,QAASw7D,GAEhB/uB,EAAOjsC,KAAMg7D,GAMd/uB,EAAOrkC,OAAQ,EAAG,EAAGozD,IAIA,IAAlB/uB,EAAOvzC,OAGJ,CAAE,IAAIqqF,GAAahnF,EAAEwkF,cAGtBuH,GAA+B77C,EAAQ21C,EAAoB,IAGnE6B,GAAmBjC,GAAec,IAAgB,CAAEvmF,EAAGhD,EAAG4U,KACzD,IAAIi0E,EAAoB7lF,EAAE05C,eAAetyB,QAKnCpnB,EAAE05C,eAAepxB,QAAStrB,EAAE2/C,oBAAwB3/C,EAAE0gE,mBAA2C,mBAAtB9rD,EAAQm5E,aACxFlF,EAAoB7lF,EAAE05C,eAAeujB,gCAAiCjgE,IAUvE,MAAM8hE,EAAY,GAAMluB,4BAA6B5wC,EAAEq9D,eAAgBr9D,EAAEwP,SAEzE,GAAKsvD,EAAU95B,IAAI1c,QAAStrB,EAAE2/C,mBAS7B,OANM3/C,EAAE0gE,mBACP19D,EAAEwP,UAGHxP,EAAE05C,eAAiBmsC,EAEZ,CAAE7lF,GAmBV,GAAK8+D,EAAU/5B,MAAM63B,gBAAiB5/D,EAAEwgE,gBAAmBsB,EAAUlvB,iBAAkB5yC,EAAEwgE,eAAkB,CAC1G,IAAImvB,EAAa,IAAI,GAAO3vF,EAAEwgE,cAAesB,EAAU95B,KACvD2nD,EAAaA,EAAW1vB,gCAAiCjgE,GAOzD,OAAO+uF,GALQ,CACd,IAAI,GAAOjtB,EAAU/5B,MAAO/nC,EAAEwgE,eAC9BmvB,GAG6C9G,EAC/C,CAOK7lF,EAAE05C,eAAepxB,QAAStrB,EAAEwgE,gBAAyC,kBAAtB5rD,EAAQm5E,aAC3DlF,EAAoB7oF,EAAEygE,oBAwBlBz9D,EAAE05C,eAAepxB,QAAStrB,EAAE2/C,oBAA6C,iBAAtB/qC,EAAQm5E,aAC/DlF,EAAoB7lF,EAAE05C,gBAKvB,MACMxJ,EAAS,CADK4uB,EAAU7B,gCAAiCjgE,IAO/D,GAAKA,EAAE0gE,kBAAoB,CAC1B,MAAMkvB,EAAwB9tB,EAAU/5B,MAAMzc,QAAStrB,EAAE0gE,oBAAuBoB,EAAUlvB,iBAAkB5yC,EAAE0gE,mBAEzG19D,EAAEwP,QAAU,GAAKo9E,IAA0Bh7E,EAAQg5E,YACvD16C,EAAOjsC,KAAM,GAAM2sC,4BAA6B5zC,EAAE2/C,kBAAmB,GAEvE,CAEA,OAAOovC,GAA+B77C,EAAQ21C,EAAmB,IAGlE6B,GAAmBjC,GAAee,IAAgB,CAAExmF,EAAGhD,EAAG4U,KACzD,MAAM0rD,EAAa,GAAM1sB,4BAA6B5wC,EAAEq9D,eAAgBr9D,EAAEwP,SAE1E,GAAKxS,EAAE4gE,iBAAiBhB,gBAAiB58D,EAAEq9D,iBAAoBC,EAAW1tB,iBAAkB5yC,EAAEqgE,gBAC7F,GAAe,UAAVr9D,EAAEsP,MAAqBsC,EAAQ22E,iBA6CnC,GAAkB,GAAbvoF,EAAEwP,QACN,OAAMoC,EAAQk5E,YAGb9qF,EAAEq9D,eAAiBrgE,EAAE0gE,kBAAkBt2C,QACvCpnB,EAAE05C,eAAiB15C,EAAE05C,eAAewjB,gCAAiClgE,GAE9D,CAAEgD,IALF,CAAE,IAAIgnF,GAAa,SArC5B,IAAMp1E,EAAQg5E,WAAa,CAC1B,MAAMtoD,EAAU,GAEhB,IAAIuqD,EAAe7vF,EAAE0gE,kBAAkBt2C,QACnC0lE,EAAuB9vF,EAAE08C,eAAewjB,gCAAiClgE,GAExEgD,EAAEwP,QAAU,IAChB8yB,EAAQr+B,KAAM,IAAIwhF,GAAezlF,EAAEq9D,eAAgBr9D,EAAEwP,QAAU,EAAGxP,EAAE05C,eAAgB,IAEpFmzC,EAAeA,EAAazvB,sBAAuBp9D,EAAEq9D,eAAgBr9D,EAAE05C,eAAgB15C,EAAEwP,QAAU,GACnGs9E,EAAuBA,EAAqB1vB,sBAAuBp9D,EAAEq9D,eAAgBr9D,EAAE05C,eAAgB15C,EAAEwP,QAAU,IAGpH,MAAMu9E,EAAe/vF,EAAE4gE,iBAAiBL,aAAcv9D,EAAEq9D,eAAgBr9D,EAAE05C,gBACpEszC,EAAS,IAAIvH,GAAeoH,EAAc,EAAGE,EAAc,GAE3DE,EAA2BD,EAAOrH,qBAAqB7hF,KAAKqC,QAClE8mF,EAAyBhpF,KAAM,GAE/B,MAAMipF,EAAuB,IAAI,GAAUF,EAAOtzC,eAAe/+C,KAAMsyF,GACvEH,EAAuBA,EAAqB1vB,sBAAuByvB,EAAcE,EAAc,GAC/F,MAAMI,EAAiB,IAAI1H,GAAeqH,EAAsB9vF,EAAEwS,QAAS09E,EAAsB,GAKjG,OAHA5qD,EAAQr+B,KAAM+oF,GACd1qD,EAAQr+B,KAAMkpF,GAEP7qD,CACR,CAuBF,MACMw7B,EADY,GAAMltB,4BAA6B5wC,EAAEq9D,eAAgBr9D,EAAEwP,SAC3C0tD,gCAAiClgE,GAM/D,OAJAgD,EAAEq9D,eAAiBS,EAAY/4B,MAC/B/kC,EAAEwP,QAAUsuD,EAAY94B,IAAI/kB,OAAS69C,EAAY/4B,MAAM9kB,OACvDjgB,EAAE05C,eAAiB15C,EAAE05C,eAAewjB,gCAAiClgE,GAE9D,CAAEgD,EAAG,IAKb0nF,GAAmBT,GAAiBd,IAAiB,CAAEnmF,EAAGhD,KACzDgD,EAAE8pB,SAAW9pB,EAAE8pB,SAASizC,iCAAkC//D,GAEnD,CAAEgD,MAGV0nF,GAAmBT,GAAiBT,IAAgB,CAAExmF,EAAGhD,IAKnDgD,EAAE8pB,SAASxB,QAAStrB,EAAE4gE,mBAC1B59D,EAAE8pB,SAAW9sB,EAAE0gE,kBAAkBt2C,QACjCpnB,EAAE8pB,SAAS4xC,WAAa,SAEjB,CAAE17D,KAGVA,EAAE8pB,SAAW9pB,EAAE8pB,SAASozC,gCAAiClgE,GAElD,CAAEgD,MAGV0nF,GAAmBT,GAAiBxB,IAAe,CAAEzlF,EAAGhD,KACvDgD,EAAE8pB,SAAW9pB,EAAE8pB,SAASkzC,+BAAgChgE,GAEjD,CAAEgD,MAGV0nF,GAAmBT,GAAiBA,IAAiB,CAAEjnF,EAAGhD,EAAG4U,KAC5D,GAAK5R,EAAE8pB,SAASxB,QAAStrB,EAAE8sB,UAAa,CACvC,IAAKlY,EAAQ+4E,UAGZ,MAAO,CAAE,IAAI3D,GAAa,IAF1BhnF,EAAEknF,QAAUlqF,EAAE2+C,OAIhB,CAEA,MAAO,CAAE37C,EAAG,IAGb0nF,GAAmBT,GAAiBV,IAAgB,CAAEvmF,EAAGhD,KAiBxD,GAA+C,QAA1C+d,EAHc/a,EAAE8pB,SAAShmB,KACZ9G,EAAEwgE,cAAcpB,mBAEwBp/D,EAAE0gE,kBAAoB,CAC/E,MAAM0vB,EAAc,IAAInG,GAAiBjnF,EAAE8pB,SAAS6kB,aAAc,GAAK3uC,EAAEknF,QAASlnF,EAAE27C,QAAS,GAE7F,MAAO,CAAE37C,EAAGotF,EACb,CAMA,OAFAptF,EAAE8pB,SAAW9pB,EAAE8pB,SAASmzC,gCAAiCjgE,GAElD,CAAEgD,EAAG,IAKb0nF,GAAmBP,GAAwBA,IAAwB,CAAEnnF,EAAGhD,EAAG4U,KAC1E,GAAK5R,EAAErF,OAASqC,EAAErC,MAAQqF,EAAE5E,MAAQ4B,EAAE5B,IAAM,CAC3C,IAAMwW,EAAQ+4E,WAAa3qF,EAAE8X,WAAa9a,EAAE8a,SAC3C,MAAO,CAAE,IAAIkvE,GAAa,IAE1BhnF,EAAE6X,SAAW7a,EAAE8a,QAEjB,CAEA,MAAO,CAAE9X,EAAG,IAKb0nF,GAAmBN,GAAeA,IAAe,CAAEpnF,EAAGhD,IAChDgD,EAAEy5B,WAAaz8B,EAAEy8B,UAAYz5B,EAAEqnF,QAAUrqF,EAAEqqF,MACxC,CAAE,IAAIL,GAAa,IAGpB,CAAEhnF,KAKV0nF,GAAmBnB,GAAgBJ,IAAiB,CAAEnmF,EAAGhD,KAGnDgD,EAAEw9D,cAAcZ,gBAAiB5/D,EAAE8sB,WAAc9pB,EAAEw9D,cAAcv9C,OAASjjB,EAAE8sB,SAAS7J,SACzFjgB,EAAEwP,SAAWxS,EAAEwS,SAGhBxP,EAAEw9D,cAAgBx9D,EAAEw9D,cAAcT,iCAAkC//D,GACpEgD,EAAE28C,kBAAoB38C,EAAE28C,kBAAkBogB,iCAAkC//D,GAErE,CAAEgD,MAGV0nF,GAAmBnB,GAAgBC,IAAgB,CAAExmF,EAAGhD,EAAG4U,KAqD1D,IAAM5R,EAAE09D,oBAAsB9rD,EAAQk5E,YAAc9qF,EAAEw9D,cAAcZ,gBAAiB5/D,EAAEqgE,gBAAmB,CACzG,MAAMgwB,EAAYrwF,EAAE0gE,kBAAkB55D,KAAKqC,QAC3CknF,EAAUppF,KAAM,GAEhB,MAAMu5D,EAAgB,IAAI,GAAUxgE,EAAE0gE,kBAAkB/iE,KAAM0yF,GACxD1wC,EAAoB4pC,GAAeG,qBAAsB,IAAI,GAAU1pF,EAAE0gE,kBAAkB/iE,KAAM0yF,IAEjGC,EAAkB,IAAI/G,GAAgB/oB,EAAe,EAAG7gB,EAAmB,KAAM,GAOvF,OALA38C,EAAEw9D,cAAgBx9D,EAAEw9D,cAAcN,gCAAiClgE,GACnEgD,EAAE28C,kBAAoB4pC,GAAeG,qBAAsB1mF,EAAEw9D,eAC7Dx9D,EAAE09D,kBAAoB4vB,EAAgB3wC,kBAAkBv1B,QACxDpnB,EAAE09D,kBAAkBhC,WAAa,SAE1B,CAAE4xB,EAAiBttF,EAC3B,CAmBA,OAfKA,EAAEw9D,cAAcZ,gBAAiB5/D,EAAE4gE,oBAAuB59D,EAAEw9D,cAAct8B,QAASlkC,EAAE4gE,mBACzF59D,EAAEwP,UAGExP,EAAEw9D,cAAcZ,gBAAiB5/D,EAAE08C,kBACvC15C,EAAEwP,SAAWxS,EAAEwS,SAGhBxP,EAAEw9D,cAAgBx9D,EAAEw9D,cAAcN,gCAAiClgE,GACnEgD,EAAE28C,kBAAoB4pC,GAAeG,qBAAsB1mF,EAAEw9D,eAExDx9D,EAAE09D,oBACN19D,EAAE09D,kBAAoB19D,EAAE09D,kBAAkBR,gCAAiClgE,IAGrE,CAAEgD,EAAG,IAGb0nF,GAAmBnB,GAAgBd,IAAe,CAAEzlF,EAAGhD,EAAG4U,KACzD,MAAM27E,EAAc,GAAM38C,4BAA6B5zC,EAAEqgE,eAAgBrgE,EAAEwS,SAE3E,GAAKxP,EAAE09D,kBAAoB,CAO1B,MAAM8vB,EAAiBD,EAAYxoD,MAAMzc,QAAStoB,EAAE09D,oBAAuB6vB,EAAY39C,iBAAkB5vC,EAAE09D,mBAE3G,IAAM9rD,EAAQk5E,YAAc0C,EAAiB,CAC5C,MAAMnwB,EAAiBr9D,EAAEw9D,cAAcR,+BAAgChgE,GAEjEywF,EAAoBztF,EAAE09D,kBAAkBV,+BAAgChgE,GACxE0wF,EAAgBD,EAAkB3pF,KAAKqC,QAC7CunF,EAAczpF,KAAM,GAEpB,MAAM4hF,EAAoB,IAAI,GAAU4H,EAAkB9yF,KAAM+yF,GAGhE,MAAO,CAFQ,IAAIjI,GAAepoB,EAAgBr9D,EAAEwP,QAASq2E,EAAmB,GAGjF,CAEA7lF,EAAE09D,kBAAoB19D,EAAE09D,kBAAkBV,+BAAgChgE,EAC3E,CASA,MAAM2wF,EAAgB3tF,EAAEw9D,cAAcl1C,QAAStrB,EAAE08C,gBAEjD,GAAKi0C,IAAyC,kBAAtB/7E,EAAQq5E,YAAwD,eAAtBr5E,EAAQm5E,YAKzE,OAJA/qF,EAAEwP,SAAWxS,EAAEwS,QACfxP,EAAEw9D,cAAgBx9D,EAAEw9D,cAAcG,0BAA2B3gE,EAAEqgE,eAAgBrgE,EAAEwS,SACjFxP,EAAE28C,kBAAoB4pC,GAAeG,qBAAsB1mF,EAAEw9D,eAEtD,CAAEx9D,GAGV,GAAK2tF,GAAiB/7E,EAAQm5E,YAAcn5E,EAAQm5E,WAAWv7E,QAAU,CACxE,MAAM,QAAEA,EAAO,OAAEyQ,GAAWrO,EAAQm5E,WAKpC,OAHA/qF,EAAEwP,SAAWA,EACbxP,EAAEw9D,cAAgBx9D,EAAEw9D,cAAc7uB,aAAc1uB,GAEzC,CAAEjgB,EACV,CAmBA,GAAKA,EAAEw9D,cAAcZ,gBAAiB5/D,EAAEqgE,iBAAoBkwB,EAAY39C,iBAAkB5vC,EAAEw9D,eAAkB,CAC7G,MAAMowB,EAAiB5wF,EAAEwS,SAAYxP,EAAEw9D,cAAcv9C,OAASjjB,EAAEqgE,eAAep9C,QAU/E,OATAjgB,EAAEwP,SAAWo+E,EAER5tF,EAAEw9D,cAAcZ,gBAAiB5/D,EAAE08C,iBAAoB15C,EAAEw9D,cAAcv9C,OAASjjB,EAAE08C,eAAez5B,SACrGjgB,EAAEwP,SAAWxS,EAAEwS,SAGhBxP,EAAEw9D,cAAgBxgE,EAAEqgE,eAAej2C,QACnCpnB,EAAE28C,kBAAoB4pC,GAAeG,qBAAsB1mF,EAAEw9D,eAEtD,CAAEx9D,EACV,CA0BA,OArBMhD,EAAEqgE,eAAe/0C,QAAStrB,EAAE08C,kBAC5B15C,EAAEw9D,cAAcZ,gBAAiB5/D,EAAEqgE,iBAAoBr9D,EAAEw9D,cAAcv9C,QAAUjjB,EAAEqgE,eAAep9C,SACtGjgB,EAAEwP,SAAWxS,EAAEwS,SAGXxP,EAAEw9D,cAAcZ,gBAAiB5/D,EAAE08C,iBAAoB15C,EAAEw9D,cAAcv9C,OAASjjB,EAAE08C,eAAez5B,SACrGjgB,EAAEwP,SAAWxS,EAAEwS,UAKjBxP,EAAEw9D,cAAc9B,WAAa,SAC7B17D,EAAEw9D,cAAgBx9D,EAAEw9D,cAAcR,+BAAgChgE,GAClEgD,EAAEw9D,cAAc9B,WAAa,SAExB17D,EAAE09D,kBACN19D,EAAE28C,kBAAoB38C,EAAE28C,kBAAkBqgB,+BAAgChgE,GAE1EgD,EAAE28C,kBAAoB4pC,GAAeG,qBAAsB1mF,EAAEw9D,eAGvD,CAAEx9D,EAAG,IAGb0nF,GAAmBnB,GAAgBA,IAAgB,CAAEvmF,EAAGhD,EAAG4U,KAiB1D,GAAK5R,EAAEw9D,cAAcl1C,QAAStrB,EAAEwgE,eAAkB,CACjD,IAAMx9D,EAAE09D,oBAAsB1gE,EAAE0gE,kBAC/B,MAAO,CAAE,IAAIspB,GAAa,IAG3B,GAAKhnF,EAAE09D,mBAAqB1gE,EAAE0gE,mBAAqB19D,EAAE09D,kBAAkBp1C,QAAStrB,EAAE0gE,mBACjF,MAAO,CAAE,IAAIspB,GAAa,IAK3B,GAA2B,eAAtBp1E,EAAQm5E,WASZ,OAPA/qF,EAAEwP,QAAU,EAKZxP,EAAE09D,kBAAoB19D,EAAE09D,kBAAmBT,gCAAiCjgE,GAErE,CAAEgD,EAEX,CAcA,GAAKA,EAAE09D,mBAAqB1gE,EAAE0gE,mBAAqB19D,EAAE09D,kBAAkBp1C,QAAStrB,EAAE0gE,mBAAsB,CACvG,MAAMmwB,EAAgD,cAAjC7tF,EAAEw9D,cAAc7iE,KAAK8+B,SACpCq0D,EAAgD,cAAjC9wF,EAAEwgE,cAAc7iE,KAAK8+B,SAW1C,GALgBq0D,IAAiBD,KAHjBA,IAAiBC,IAMUl8E,EAAQ+4E,UAElC,CAChB,MAAMnoF,EAAS,GAcf,OAVKxF,EAAEwS,SACNhN,EAAOyB,KAAM,IAAIwhF,GAAezoF,EAAEygE,mBAAoBzgE,EAAEwS,QAASxS,EAAEwgE,cAAe,IAK9Ex9D,EAAEwP,SACNhN,EAAOyB,KAAM,IAAIwhF,GAAezlF,EAAEw9D,cAAex9D,EAAEwP,QAASxP,EAAEy9D,mBAAoB,IAG5Ej7D,CACR,CACC,MAAO,CAAE,IAAIwkF,GAAa,GAE5B,CAWA,GATKhnF,EAAE09D,oBACN19D,EAAE09D,kBAAoB19D,EAAE09D,kBAAkBT,gCAAiCjgE,IAQvEgD,EAAEw9D,cAAcl1C,QAAStrB,EAAE2/C,oBAA6C,eAAtB/qC,EAAQm5E,WAG9D,OAFA/qF,EAAEwP,UAEK,CAAExP,GAOV,GAAKhD,EAAEwgE,cAAcl1C,QAAStoB,EAAE28C,oBAA6C,eAAtB/qC,EAAQq5E,WAA8B,CAC5F,MAAM8C,EAAkB/wF,EAAE2/C,kBAAkB74C,KAAKqC,QACjD4nF,EAAgB9pF,KAAM,GAEtB,MAAMk2C,EAAc,IAAI,GAAUn9C,EAAE2/C,kBAAkBhiD,KAAMozF,GAG5D,MAAO,CAAE/tF,EAFM,IAAIylF,GAAezlF,EAAE28C,kBAAmB,EAAGxC,EAAa,GAGxE,CAWA,OAPKn6C,EAAEw9D,cAAcZ,gBAAiB5/D,EAAEwgE,gBAAmBx9D,EAAEw9D,cAAcv9C,OAASjjB,EAAEwgE,cAAcv9C,SACnGjgB,EAAEwP,SAAWxS,EAAEwS,SAGhBxP,EAAEw9D,cAAgBx9D,EAAEw9D,cAAcP,gCAAiCjgE,GACnEgD,EAAE28C,kBAAoB4pC,GAAeG,qBAAsB1mF,EAAEw9D,eAEtD,CAAEx9D,EAAG,ICzvEE,MAAM,WAAqByT,EAAc,KAWvD,WAAArN,CAAazL,EAAmBmJ,EAAqB43D,EAAiC,UAGrF,GAFA7pD,MAAOlX,EAAMmJ,EAAM43D,IAEbr3D,KAAK1J,KAAK4X,GAAI,eAMnB,MAAM,IAAI,EAAe,0CAA2C5X,GAGrE,GAAiB+K,KAAMrB,KACxB,CAOO,MAAA8gB,GACN9gB,KAAK2P,eACN,CAKO,UAAAg6E,GACN,OAAO,IAAI,GAAU3pF,KAAK1J,KAAM0J,KAAKP,KAAKqC,QAAS9B,KAAKq3D,WACzD,CAKA,mBAAcuyB,CAAcnkE,EAAoB4xC,GAC/C,OAAO,IAAIr3D,KAAMylB,EAASnvB,KAAqBmvB,EAAShmB,KAAKqC,QAASu1D,GAA0B5xC,EAAS4xC,WAC1G,EAqCD,SAAS,KACRr3D,KAAKwP,SACJxP,KAAK1J,KAAKkN,SAAUpL,MACpB,kBACA,CAAEkX,EAAOpT,KACR,MAAMu8D,EAAYv8D,EAAM,GAElBu8D,EAAU2L,qBAIhB,GAAU/iE,KAAMrB,KAAMy4D,EAAW,GAElC,CAAE7rD,SAAU,OAEd,CAKA,SAAS,GAA+B6rD,GACvC,MAAMt6D,EAAS6B,KAAKw4D,0BAA2BC,GAE/C,IAAMz4D,KAAKikB,QAAS9lB,GAAW,CAC9B,MAAM0rF,EAAc7pF,KAAK2pF,aAEvB3pF,KAAcP,KAAOtB,EAAOsB,KAC5BO,KAAc1J,KAAO6H,EAAO7H,KAE9B0J,KAAKwQ,KAA+B,SAAUq5E,EAC/C,CACD,CAzCA,GAAahoF,UAAUqM,GAAK,SAAUjD,GACrC,MAAgB,iBAATA,GAAoC,uBAATA,GAEzB,YAARA,GAA+B,mBAATA,CACxB,ECpFe,MAAM6+E,GAkCpB,WAAA/nF,CAAakJ,EAAkB,CAAC,GACV,iBAATA,IACXA,EAAgB,gBAATA,EAAyB,CAAE0yE,YAAY,GAAU,CAAC,EASzD,EAAY,6CAGb,MAAM,WAAEA,GAAa,EAAI,QAAEoM,GAAU,EAAI,OAAEC,GAAS,EAAK,SAAEC,GAAW,GAAUh/E,EAEhFjL,KAAKq6D,WAAa,GAClBr6D,KAAK29E,WAAaA,EAClB39E,KAAK+pF,QAAUA,EACf/pF,KAAKgqF,OAASA,EACdhqF,KAAKiqF,SAAWA,CACjB,CAcA,QAAWh/E,GASV,OAFA,EAAY,yBAEL,SACR,CAMA,eAAWk1E,GACV,UAAYmE,KAAMtkF,KAAKq6D,WACtB,GAAwB,OAAnBiqB,EAAGnE,YACP,OAAOmE,EAAGnE,YAIZ,OAAO,IACR,CAQO,YAAA+J,CAAczxB,GAIpB,OAHAA,EAAUyN,MAAQlmE,KAClBA,KAAKq6D,WAAWz6D,KAAM64D,GAEfA,CACR,E,yZChGc,MAAM0xB,GAyEpB,WAAApoF,CAAaqoF,GA7Db,KAAiBC,kBAAwE,IAAI94E,IAO7F,KAAiB+4E,kBACd,IAAI/4E,IAUP,KAAiBg5E,gBAAyF,IAAIh5E,IAO9G,KAAiBi5E,cAA2C,IAAIj5E,IAMhE,KAAQk5E,aAAuB,EAS/B,KAAQC,eAAyC,KASjD,KAAQC,4BAAsD,KAK9D,KAAQC,gBAA6B,IAAIh3E,IAQxC5T,KAAK6qF,kBAAoBT,CAC1B,CAKA,WAAWjoD,GACV,OAAsC,GAA/BniC,KAAKqqF,kBAAkBx2E,MAA0C,GAA7B7T,KAAKuqF,gBAAgB12E,MAAwC,GAA3B7T,KAAKwqF,cAAc32E,IACjG,CAOO,eAAAi3E,CAAiBC,GAKvB,MAAMtyB,EAAYsyB,EAclB,OAAStyB,EAAUxtD,MAClB,IAAK,SACJ,GAAKjL,KAAKgrF,qBAAsBvyB,EAAUhzC,SAAS7mB,QAClD,OAGDoB,KAAKirF,YAAaxyB,EAAUhzC,SAAS7mB,OAAQ65D,EAAUhzC,SAAS7J,OAAQ68C,EAAUvxB,MAAMmvB,WAExF,MAED,IAAK,eACL,IAAK,kBACL,IAAK,kBACJ,UAAYp2D,KAAQw4D,EAAU3zC,MAAMonB,SAAU,CAAEpD,SAAS,IACnD9oC,KAAKgrF,qBAAsB/qF,EAAKrB,SAIrCoB,KAAKkrF,eAAgBjrF,GAGtB,MAED,IAAK,SACL,IAAK,OACL,IAAK,WAAY,CAGhB,GACCw4D,EAAUO,eAAe/0C,QAASw0C,EAAUpjB,iBAC5CojB,EAAUO,eAAe1uB,aAAcmuB,EAAUttD,SAAU8Y,QAASw0C,EAAUpjB,gBAE9E,OAGD,MAAM81C,EAAuBnrF,KAAKgrF,qBAAsBvyB,EAAUO,eAAep6D,QAC3EwsF,EAAuBprF,KAAKgrF,qBAAsBvyB,EAAUpjB,eAAez2C,QAE3EusF,GACLnrF,KAAKqrF,YAAa5yB,EAAUO,eAAep6D,OAAQ65D,EAAUO,eAAep9C,OAAQ68C,EAAUttD,SAGzFigF,GACLprF,KAAKirF,YAAaxyB,EAAUpjB,eAAez2C,OAAQ65D,EAAU6oB,qBAAqB1lE,OAAQ68C,EAAUttD,SAGrG,KACD,CACA,IAAK,SAAU,CACd,GAAKnL,KAAKgrF,qBAAsBvyB,EAAUhzC,SAAS7mB,QAClD,OAGDoB,KAAKqrF,YAAa5yB,EAAUhzC,SAAS7mB,OAAQ65D,EAAUhzC,SAAS7J,OAAQ,GACxE5b,KAAKirF,YAAaxyB,EAAUhzC,SAAS7mB,OAAQ65D,EAAUhzC,SAAS7J,OAAQ,GAExE,MAAMkJ,EAAQ,GAAMynB,4BAA6BksB,EAAUhzC,SAAU,GAErE,UAAY27C,KAAUphE,KAAK6qF,kBAAkBS,4BAA6BxmE,GAAU,CACnF,MAAMymE,EAAanqB,EAAOzO,UAE1B3yD,KAAKwrF,mBAAoBpqB,EAAOp/D,KAAMupF,EAAYA,EACnD,CAEA,KACD,CACA,IAAK,QAAS,CACb,MAAMnJ,EAAe3pB,EAAUU,cAAcv6D,OAGvCoB,KAAKgrF,qBAAsB5I,IAChCpiF,KAAKqrF,YAAajJ,EAAc3pB,EAAUU,cAAcv9C,OAAQ68C,EAAUttD,SAIrEnL,KAAKgrF,qBAAsBvyB,EAAUngB,kBAAkB15C,SAC5DoB,KAAKirF,YAAaxyB,EAAUngB,kBAAkB15C,OAAQ65D,EAAUngB,kBAAkB18B,OAAQ,GAItF68C,EAAUY,mBACdr5D,KAAKqrF,YAAa5yB,EAAUY,kBAAkBz6D,OAAQ65D,EAAUY,kBAAkBz9C,OAAQ,GAG3F,KACD,CACA,IAAK,QAAS,CAEb,MAAM0mE,EAAgB7pB,EAAUO,eAAep6D,OAEzCoB,KAAKgrF,qBAAsB1I,EAAc1jF,SAC9CoB,KAAKqrF,YAAa/I,EAAc1jF,OAAW0jF,EAAuBt4C,YAAa,GAIhF,MAAMyhD,EAAkBhzB,EAAUY,kBAAkBz6D,OAEpDoB,KAAKirF,YAAaQ,EAAiBhzB,EAAUY,kBAAkBz9C,OAAQ,GAGvE,MAAM8vE,EAAoBjzB,EAAUpjB,eAAez2C,OAE7CoB,KAAKgrF,qBAAsBU,IAChC1rF,KAAKirF,YAAaS,EAAmBjzB,EAAUpjB,eAAez5B,OAAQ0mE,EAAcjsB,WAGrF,KACD,CACA,IAAK,aACL,IAAK,UAAW,CACf,MAAM//D,EAAOmiE,EAAU4oB,mBAEvB,IAAM/qF,EAAKq1F,UACV,OAID,GAAKr1F,EAAKilC,cAAgBk9B,EAAUuqB,MACnC,OAGDhjF,KAAK4rF,uBAAwBnzB,EAAUrjC,SAAUqjC,EAAUuqB,OAE3D,KACD,CACA,IAAK,mBACL,IAAK,sBACL,IAAK,sBAAuB,CAC3B,IAAMvqB,EAAUniE,KAAKq1F,UACpB,OAGD,MAAMv2D,EAAWqjC,EAAUniE,KAAK8+B,SAEhCp1B,KAAK6rF,2BAA4Bz2D,EAAUqjC,EAAU1hE,IAAK0hE,EAAUjlD,SAAUilD,EAAUhlD,UAExF,KACD,EAIDzT,KAAK0qF,eAAiB,IACvB,CASO,kBAAAc,CAAoBlvB,EAAoBwvB,EAA2BC,GACpED,EAAchnE,OAASgnE,EAAchnE,MAAMxuB,KAAK4X,GAAI,iBAAoB49E,EAAchnE,MAAMxuB,KAAKq1F,YACrGG,EAAchnE,MAAQ,MAGlBinE,EAAcjnE,OAASinE,EAAcjnE,MAAMxuB,KAAK4X,GAAI,iBAAoB69E,EAAcjnE,MAAMxuB,KAAKq1F,YACrGI,EAAcjnE,MAAQ,MAGvB,IAAIknE,EAAWhsF,KAAKuqF,gBAAgBniF,IAAKk0D,GAEnC0vB,EAKLA,EAASD,cAAgBA,GAJzBC,EAAW,CAAED,gBAAeD,iBAE5B9rF,KAAKuqF,gBAAgB94E,IAAK6qD,EAAY0vB,IAKF,MAAhCA,EAASF,cAAchnE,OAAwC,MAAvBinE,EAAcjnE,OAG1D9kB,KAAKuqF,gBAAgB54E,OAAQ2qD,EAE/B,CAOO,kBAAAyD,GACN,MAAM5hE,EAAS,GAEf,UAAc6D,EAAMu4B,KAAYv6B,KAAKuqF,gBACD,MAA9BhwD,EAAOuxD,cAAchnE,OACzB3mB,EAAOyB,KAAM,CAAEoC,OAAM8iB,MAAOyV,EAAOuxD,cAAchnE,QAInD,OAAO3mB,CACR,CAOO,eAAA0iE,GACN,MAAM1iE,EAAS,GAEf,UAAc6D,EAAMu4B,KAAYv6B,KAAKuqF,gBACD,MAA9BhwD,EAAOwxD,cAAcjnE,OACzB3mB,EAAOyB,KAAM,CAAEoC,OAAM8iB,MAAOyV,EAAOwxD,cAAcjnE,QAInD,OAAO3mB,CACR,CAKO,iBAAA8tF,GAON,OAAOtrF,MAAMrB,KAAMU,KAAKuqF,iBAAkBvtF,KAAK,EAAIgF,EAAMu4B,MAAO,CAE9Dv4B,OACAc,KAAM,CACLy/D,SAAUhoC,EAAOuxD,cAAchnE,MAC/BqyB,SAAU5c,EAAOwxD,cAAcjnE,UAInC,CAaO,cAAAonE,GACN,GAAKlsF,KAAKqqF,kBAAkBx2E,KAAO,EAClC,OAAO,EAGR,GAAK7T,KAAKwqF,cAAc32E,KAAO,EAC9B,OAAO,EAGR,UAAY,cAAEk4E,EAAa,cAAED,KAAmB9rF,KAAKuqF,gBAAgBr/E,SAAW,CAC/E,GAAK6gF,EAAc/2D,cAAgB82D,EAAc92D,YAChD,OAAO,EAGR,GAAK+2D,EAAc/2D,YAAc,CAChC,MAAMm3D,EAAcJ,EAAcjnE,QAAUgnE,EAAchnE,MACpDsnE,GAAiBL,EAAcjnE,OAASgnE,EAAchnE,MACtDunE,EAAgBN,EAAcjnE,OAASgnE,EAAchnE,QAAUinE,EAAcjnE,MAAMb,QAAS6nE,EAAchnE,OAEhH,GAAKqnE,GAAeC,GAAiBC,EACpC,OAAO,CAET,CACD,CAEA,OAAO,CACR,CAoBO,UAAAlsB,CAAY57D,EAAmD,CAAC,GAEtE,GAAKvE,KAAK0qF,eACT,OAAKnmF,EAAQ+nF,0BACLtsF,KAAK2qF,4BAA6B7oF,QAElC9B,KAAK0qF,eAAe5oF,QAK7B,IAAIyqF,EAA8C,GAGlD,UAAY90E,KAAWzX,KAAKqqF,kBAAkBpzF,OAAS,CAEtD,MAAMgpE,EAAUjgE,KAAKqqF,kBAAkBjiF,IAAKqP,GAAW0J,MAAM,CAAExlB,EAAGhD,IAC5DgD,EAAEigB,SAAWjjB,EAAEijB,OACdjgB,EAAEsP,MAAQtS,EAAEsS,KAIC,UAAVtP,EAAEsP,MAAoB,EAAI,EAG3B,EAGDtP,EAAEigB,OAASjjB,EAAEijB,QAAU,EAAI,IAI7B4wE,EAAmBxsF,KAAKsqF,kBAAkBliF,IAAKqP,GAE/Cg1E,EAAkBC,GAAsBj1E,EAAQ+jB,eAGhDkiB,EAAUivC,GAA6BH,EAAiBl0F,OAAQ2nE,GAEtE,IAAIzlE,EAAI,EACJ8/D,EAAI,EAGR,UAAYvc,KAAUL,EACrB,GAAgB,MAAXK,EAEJwuC,EAAQ3sF,KAAMI,KAAK4sF,eAAgBn1E,EAASjd,EAAGiyF,EAAiBjyF,KAEhEA,SACM,GAAgB,MAAXujD,EAEXwuC,EAAQ3sF,KAAMI,KAAK6sF,eAAgBp1E,EAASjd,EAAGgyF,EAAkBlyB,KAEjEA,SACM,GAAgB,MAAXvc,EAAiB,CAE5B,MAAM+uC,EAAoBL,EAAiBjyF,GAAI+K,WACzCwnF,EAAqBP,EAAkBlyB,GAAI/0D,WACjD,IAAIuf,EAEJ,GAAkC,SAA7B2nE,EAAiBjyF,GAAIwH,KACzB8iB,EAAQ,IAAI,GAAO,GAAS8jB,UAAWnxB,EAASjd,GAAK,GAASouC,UAAWnxB,EAASjd,EAAI,QAChF,CACN,MAAMoK,EAAQ6S,EAAQi/C,cAAel8D,GACrCsqB,EAAQ,IAAI,GAAO,GAAS8jB,UAAWnxB,EAASjd,GAAK,GAASouC,UAAWnxB,EAAQokB,SAAUj3B,GAAU,GACtG,CAIA2nF,EAAQ3sF,QAASI,KAAKgtF,mBAAoBloE,EAAOioE,EAAoBD,IAErEtyF,IACA8/D,GACD,MAEC9/D,IACA8/D,GAGH,CAGAiyB,EAAQprE,MAAM,CAAExlB,EAAGhD,IAIbgD,EAAE8pB,SAAUnvB,MAAQqC,EAAE8sB,SAAUnvB,KAC7BqF,EAAE8pB,SAAUnvB,KAAK8+B,SAAYz8B,EAAE8sB,SAAUnvB,KAAK8+B,UAAa,EAAI,EAIlEz5B,EAAE8pB,SAAUxB,QAAStrB,EAAE8sB,UAEpB9pB,EAAEsxF,YAAet0F,EAAEs0F,YAIpBtxF,EAAE8pB,SAAUiX,SAAU/jC,EAAE8sB,WAAe,EAAI,IAInD,QAAUjrB,EAAI,EAAG0yF,EAAY,EAAG1yF,EAAI+xF,EAAQj0F,OAAQkC,IAAM,CACzD,MAAM2yF,EAAWZ,EAASW,GACpBE,EAAWb,EAAS/xF,GAGpB6yF,EACY,UAAjBF,EAASliF,MAAqC,UAAjBmiF,EAASniF,MACrB,SAAjBkiF,EAASnrF,MAAoC,SAAjBorF,EAASprF,MACrCmrF,EAAS1nE,SAASxB,QAASmpE,EAAS3nE,UAG/B6nE,EACY,UAAjBH,EAASliF,MAAqC,UAAjBmiF,EAASniF,MACrB,SAAjBkiF,EAASnrF,MAAoC,SAAjBorF,EAASprF,MACrCmrF,EAAS1nE,SAAS7mB,QAAUwuF,EAAS3nE,SAAS7mB,QAC9CuuF,EAAS1nE,SAAS7J,OAASuxE,EAAS70F,QAAU80F,EAAS3nE,SAAS7J,OAG3D2xE,EACY,aAAjBJ,EAASliF,MAAwC,aAAjBmiF,EAASniF,MACzCkiF,EAAS1nE,SAAU7mB,QAAUwuF,EAAS3nE,SAAU7mB,QAChDuuF,EAASroE,MAAMhH,QAAUsvE,EAAStoE,MAAMhH,QACtCqvE,EAAS1nE,SAAU7J,OAASuxE,EAAS70F,QAAa80F,EAAS3nE,SAAU7J,QACvEuxE,EAAS1oC,cAAgB2oC,EAAS3oC,cAClC0oC,EAAS3sB,mBAAqB4sB,EAAS5sB,mBACvC2sB,EAAS1sB,mBAAqB2sB,EAAS3sB,kBAEnC4sB,GAA2BC,GAAwBC,GACvDJ,EAAS70F,SAEJi1F,IACFJ,EAASroE,MAAe6b,IAAMwsD,EAASroE,MAAM6b,IAAI2J,aAAc,IAGlEiiD,EAAS/xF,GAAM,MAEf0yF,EAAY1yF,CAEd,CAEA+xF,EAAUA,EAAQxmF,QAAQzM,GAAKA,IAG/B,UAAY2G,KAAQssF,SACZtsF,EAAKgtF,YAEM,aAAbhtF,EAAKgL,cACFhL,EAAKwlB,gBACLxlB,EAAK3H,QAUd,OANA0H,KAAKyqF,aAAe,EAGpBzqF,KAAK2qF,4BAA8B4B,EACnCvsF,KAAK0qF,eAAiB6B,EAAQxmF,OAAQynF,IAEjCjpF,EAAQ+nF,0BACLtsF,KAAK2qF,4BAA4B7oF,QAEjC9B,KAAK0qF,eAAe5oF,OAE7B,CAOO,eAAA2rF,GACN,OAAO9sF,MAAMrB,KAAMU,KAAKwqF,cAAct/E,UAAWlO,KAAK0wF,IACrD,MAAMh1E,EAAQ,MAAKg1E,GAanB,YAXqB,IAAhBh1E,EAAMi1E,cAQHj1E,EAAMnT,WAGPmT,CAAK,GAEd,CAKO,iBAAAonD,GACN,OAAO,IAAIlsD,IAAK5T,KAAK4qF,gBACtB,CAKO,KAAAgD,GACN5tF,KAAKqqF,kBAAkBz4E,QACvB5R,KAAKsqF,kBAAkB14E,QACvB5R,KAAKuqF,gBAAgB34E,QACrB5R,KAAKwqF,cAAc54E,QACnB5R,KAAK4qF,gBAAkB,IAAIh3E,IAC3B5T,KAAK0qF,eAAiB,IACvB,CAKQ,sBAAAkB,CAAwBx2D,EAAkBmG,GACjD,IAAMv7B,KAAKwqF,cAAc58E,IAAKwnB,GAG7B,YAFAp1B,KAAKwqF,cAAc/4E,IAAK2jB,EAAU,CAAEpzB,KAAMozB,EAAUu4D,MAAOpyD,EAAa,WAAa,aAKtF,MAAMmyD,EAAW1tF,KAAKwqF,cAAcpiF,IAAKgtB,QAEjB,IAAnBs4D,EAASC,cAIND,EAASC,WAEa,IAAxBD,EAASnoF,YAEbvF,KAAKwqF,cAAc74E,OAAQyjB,IAG5Bs4D,EAASC,MAAQpyD,EAAa,WAAa,UAE7C,CAKQ,0BAAAswD,CAA4Bz2D,EAAkBr+B,EAAayc,EAAmBC,GACrF,MAAMi6E,EAAyB1tF,KAAKwqF,cAAcpiF,IAAKgtB,IAAc,CAAEpzB,KAAMozB,GACvEgQ,EAAkEsoD,EAASnoF,YAAc,CAAC,EAEhG,GAAK6/B,EAAOruC,GAAQ,CAEnB,MAAM82F,EAAYzoD,EAAOruC,GAEpB0c,IAAao6E,EAAUr6E,gBAEpB4xB,EAAOruC,GAGd82F,EAAUp6E,SAAWA,CAEvB,MAEC2xB,EAAOruC,GAAQ,CAAEyc,WAAUC,YAGY,IAAnCzc,OAAOyhB,QAAS2sB,GAAQ9sC,eAErBo1F,EAASnoF,gBAEQ,IAAnBmoF,EAASC,OAEb3tF,KAAKwqF,cAAc74E,OAAQyjB,KAI5Bs4D,EAASnoF,WAAa6/B,EAEtBplC,KAAKwqF,cAAc/4E,IAAK2jB,EAAUs4D,GAEpC,CASO,YAAA3Z,CAAc9zE,GACpB,GAAKD,KAAKgrF,qBAAsB/qF,EAAKrB,QACpC,OAGDoB,KAAKqrF,YAAaprF,EAAKrB,OAASqB,EAAK+pC,YAAc/pC,EAAK09B,YACxD39B,KAAKirF,YAAahrF,EAAKrB,OAASqB,EAAK+pC,YAAc/pC,EAAK09B,YAExD39B,KAAK4qF,gBAAgB/8E,IAAK5N,GAE1B,MAAM6kB,EAAQ,GAAM2nB,UAAWxsC,GAE/B,UAAYmhE,KAAUphE,KAAK6qF,kBAAkBS,4BAA6BxmE,GAAU,CACnF,MAAMymE,EAAanqB,EAAOzO,UAE1B3yD,KAAKwrF,mBAAoBpqB,EAAOp/D,KAAMupF,EAAYA,EACnD,CAGAvrF,KAAK0qF,eAAiB,IACvB,CAcO,eAAAoD,CAAiBx3F,GACvB,GAAMA,EAAKilC,aAAX,CAIAv7B,KAAK4rF,uBAAwBt1F,EAAK8+B,UAAU,GAC5Cp1B,KAAKirF,YAAa30F,EAAM,EAAGA,EAAK+/D,WAMhC,UAAYt/D,KAAOT,EAAKgoC,mBACvBt+B,KAAK6rF,2BAA4Bv1F,EAAK8+B,SAAUr+B,EAAK,KAAMT,EAAK+kC,aAActkC,IAG/E,UAAYqqE,KAAUphE,KAAK6qF,kBAC1B,GAAKzpB,EAAOT,WAAWrqE,MAAQA,EAAO,CACrC,MAAMi1F,EAAanqB,EAAOzO,UAE1B3yD,KAAKwrF,mBAAoBpqB,EAAOp/D,M,EAAM,MAAKupF,G,QAAL,CAAiBzmE,MAAO,SAAQymE,EACvE,C,KAlBD,CAoBD,CAKQ,WAAAN,CAAarsF,EAAoCgd,EAAgBzQ,GACxE,GAAKvM,EAAOtI,KAAK4X,GAAI,iBAAoBtP,EAAOtI,KAAKq1F,UACpD,OAGD,MAAMoC,EAAa,CAAE9iF,KAAM,SAAU2Q,SAAQzQ,UAASxG,MAAO3E,KAAKyqF,gBAElEzqF,KAAKguF,YAAapvF,EAAQmvF,EAC3B,CAKQ,WAAA1C,CAAazsF,EAAoCgd,EAAgBzQ,GACxE,GAAKvM,EAAOtI,KAAK4X,GAAI,iBAAoBtP,EAAOtI,KAAKq1F,UACpD,OAGD,MAAMoC,EAAa,CAAE9iF,KAAM,SAAU2Q,SAAQzQ,UAASxG,MAAO3E,KAAKyqF,gBAElEzqF,KAAKguF,YAAapvF,EAAQmvF,GAE1B/tF,KAAKiuF,wBAAyBrvF,EAAQgd,EAAQzQ,EAC/C,CAKQ,cAAA+/E,CAAgBjrF,GACvB,GAAKA,EAAK3J,KAAK4X,GAAI,iBAAoBjO,EAAK3J,KAAKq1F,UAChD,OAGD,MAAMoC,EAAa,CAAE9iF,KAAM,YAAa2Q,OAAQ3b,EAAK+pC,YAAc7+B,QAASlL,EAAK09B,WAAYh5B,MAAO3E,KAAKyqF,gBAEzGzqF,KAAKguF,YAAa/tF,EAAKrB,OAAmBmvF,EAC3C,CAKQ,WAAAC,CAAapvF,EAAoCmvF,GAExD/tF,KAAKkuF,cAAetvF,GAGpB,MAAMqhE,EAAUjgE,KAAKmuF,sBAAuBvvF,GAG5CoB,KAAKouF,cAAeL,EAAY9tB,GAGhCA,EAAQrgE,KAAMmuF,GAId,QAAUvzF,EAAI,EAAGA,EAAIylE,EAAQ3nE,OAAQkC,IAC/BylE,EAASzlE,GAAI2Q,QAAU,IAC3B80D,EAAQz4D,OAAQhN,EAAG,GAEnBA,IAGH,CAKQ,qBAAA2zF,CAAuB12E,GAC9B,IAAIwoD,EAUJ,OARKjgE,KAAKqqF,kBAAkBz8E,IAAK6J,GAChCwoD,EAAUjgE,KAAKqqF,kBAAkBjiF,IAAKqP,IAEtCwoD,EAAU,GAEVjgE,KAAKqqF,kBAAkB54E,IAAKgG,EAASwoD,IAG/BA,CACR,CAKQ,aAAAiuB,CAAez2E,GAChBzX,KAAKsqF,kBAAkB18E,IAAK6J,IACjCzX,KAAKsqF,kBAAkB74E,IAAKgG,EAASi1E,GAAsBj1E,EAAQ+jB,eAErE,CASQ,aAAA4yD,CAAeC,EAAiBpuB,GAiBvCouB,EAAIC,cAAgBD,EAAIljF,QAExB,UAAYojF,KAAOtuB,EAAU,CAC5B,MAAMuuB,EAASH,EAAIzyE,OAASyyE,EAAIljF,QAC1BsjF,EAASF,EAAI3yE,OAAS2yE,EAAIpjF,QAEhC,GAAiB,UAAZkjF,EAAIpjF,OACS,UAAZsjF,EAAItjF,OACHojF,EAAIzyE,QAAU2yE,EAAI3yE,OACtB2yE,EAAI3yE,QAAUyyE,EAAIljF,QACPkjF,EAAIzyE,OAAS6yE,IACxBF,EAAIpjF,SAAWkjF,EAAIC,cACnBD,EAAIC,cAAgB,IAIL,UAAZC,EAAItjF,MACHojF,EAAIzyE,OAAS2yE,EAAI3yE,SACrB2yE,EAAI3yE,QAAUyyE,EAAIljF,SAIH,aAAZojF,EAAItjF,MACR,GAAKojF,EAAIzyE,QAAU2yE,EAAI3yE,OACtB2yE,EAAI3yE,QAAUyyE,EAAIljF,aACZ,GAAKkjF,EAAIzyE,OAAS6yE,EAAS,CAWjC,MAAMtjF,EAAUojF,EAAIpjF,QAEpBojF,EAAIpjF,QAAUkjF,EAAIzyE,OAAS2yE,EAAI3yE,OAI/BqkD,EAAQ7gE,QAAS,CAChB6L,KAAM,YACN2Q,OAAQ4yE,EACRrjF,QAASA,EAAUojF,EAAIpjF,QACvBxG,MAAO3E,KAAKyqF,gBAEd,CAIF,GAAiB,UAAZ4D,EAAIpjF,KAAmB,CAC3B,GAAiB,UAAZsjF,EAAItjF,KACR,GAAKujF,GAAUD,EAAI3yE,OAClB2yE,EAAI3yE,QAAUyyE,EAAIljF,aACZ,GAAKqjF,GAAUC,EACrB,GAAKJ,EAAIzyE,OAAS2yE,EAAI3yE,OAAS,CAC9B,MAAM8yE,EAAqBF,EAASD,EAAI3yE,OAExC2yE,EAAI3yE,OAASyyE,EAAIzyE,OAEjB2yE,EAAIpjF,SAAWujF,EACfL,EAAIC,eAAiBI,CACtB,MACCH,EAAIpjF,SAAWkjF,EAAIC,cACnBD,EAAIC,cAAgB,OAGrB,GAAKD,EAAIzyE,QAAU2yE,EAAI3yE,OACtByyE,EAAIC,eAAiBC,EAAIpjF,QACzBojF,EAAIpjF,QAAU,OACR,GAAKkjF,EAAIzyE,OAAS6yE,EAAS,CACjC,MAAMC,EAAqBD,EAASJ,EAAIzyE,OAExC2yE,EAAIpjF,SAAWujF,EACfL,EAAIC,eAAiBI,CACtB,CAaF,GATiB,UAAZH,EAAItjF,OACHujF,GAAUD,EAAI3yE,OAClB2yE,EAAI3yE,QAAUyyE,EAAIljF,QACPkjF,EAAIzyE,OAAS2yE,EAAI3yE,SAC5ByyE,EAAIC,eAAiBC,EAAIpjF,QACzBojF,EAAIpjF,QAAU,IAIC,aAAZojF,EAAItjF,KACR,GAAKujF,GAAUD,EAAI3yE,OAClB2yE,EAAI3yE,QAAUyyE,EAAIljF,aACZ,GAAKkjF,EAAIzyE,OAAS2yE,EAAI3yE,OAAS,CACrC,MAAM8yE,EAAqBF,EAASD,EAAI3yE,OAExC2yE,EAAI3yE,OAASyyE,EAAIzyE,OACjB2yE,EAAIpjF,SAAWujF,CAChB,MAAO,GAAKL,EAAIzyE,OAAS6yE,EACxB,GAAKD,GAAUC,EAAS,CAMvB,MAAMtjF,EAAUojF,EAAIpjF,QAEpBojF,EAAIpjF,QAAUkjF,EAAIzyE,OAAS2yE,EAAI3yE,OAE/B,MAAM+yE,EAAexjF,EAAUojF,EAAIpjF,QAAUkjF,EAAIC,cAIjDruB,EAAQ7gE,QAAS,CAChB6L,KAAM,YACN2Q,OAAQyyE,EAAIzyE,OACZzQ,QAASwjF,EACThqF,MAAO3E,KAAKyqF,gBAEd,MACC8D,EAAIpjF,SAAWsjF,EAASJ,EAAIzyE,MAIhC,CAEA,GAAiB,aAAZyyE,EAAIpjF,KAAsB,CAE9B,GAAiB,UAAZsjF,EAAItjF,KACR,GAAKojF,EAAIzyE,OAAS2yE,EAAI3yE,QAAU4yE,EAASD,EAAI3yE,OAAS,CACrD,GAAK4yE,EAASC,EAAS,CAOtB,MAAMG,EAAgB,CACrB3jF,KAAM,YACN2Q,OAAQ6yE,EACRtjF,QAASqjF,EAASC,EAClB9pF,MAAO3E,KAAKyqF,gBAGbzqF,KAAKouF,cAAeQ,EAAe3uB,GAEnCA,EAAQrgE,KAAMgvF,EACf,CAEAP,EAAIC,cAAgBC,EAAI3yE,OAASyyE,EAAIzyE,OACrCyyE,EAAIljF,QAAUkjF,EAAIC,aACnB,MAAYD,EAAIzyE,QAAU2yE,EAAI3yE,QAAUyyE,EAAIzyE,OAAS6yE,IAC/CD,EAASC,GACbJ,EAAIC,cAAgBE,EAASC,EAC7BJ,EAAIzyE,OAAS6yE,GAEbJ,EAAIC,cAAgB,GAKvB,GAAiB,UAAZC,EAAItjF,MAGHojF,EAAIzyE,OAAS2yE,EAAI3yE,QAAU4yE,EAASD,EAAI3yE,OAAS,CACrD,MAAMgzE,EAAgB,CACrB3jF,KAAM,YACN2Q,OAAQ2yE,EAAI3yE,OACZzQ,QAASqjF,EAASD,EAAI3yE,OACtBjX,MAAO3E,KAAKyqF,gBAGbzqF,KAAKouF,cAAeQ,EAAe3uB,GAEnCA,EAAQrgE,KAAMgvF,GAEdP,EAAIC,cAAgBC,EAAI3yE,OAASyyE,EAAIzyE,OACrCyyE,EAAIljF,QAAUkjF,EAAIC,aACnB,CAGgB,aAAZC,EAAItjF,OAEHojF,EAAIzyE,QAAU2yE,EAAI3yE,QAAU4yE,GAAUC,GAE1CJ,EAAIC,cAAgB,EACpBD,EAAIljF,QAAU,EACdkjF,EAAIzyE,OAAS,GACFyyE,EAAIzyE,QAAU2yE,EAAI3yE,QAAU4yE,GAAUC,IAEjDF,EAAIpjF,QAAU,GAGjB,CACD,CAEAkjF,EAAIljF,QAAUkjF,EAAIC,qBACXD,EAAIC,aACZ,CAUQ,cAAA1B,CACPhuF,EACAgd,EACAizE,GAEA,MAAO,CACN5jF,KAAM,SACNwa,SAAU,GAASmjB,UAAWhqC,EAAQgd,GACtC5Z,KAAM6sF,EAAgB7sF,KACtBuD,WAAY,IAAIgM,IAAKs9E,EAAgBtpF,YACrCjN,OAAQ,EACR20F,YAAajtF,KAAKyqF,eAEpB,CAUQ,cAAAoC,CACPjuF,EACAgd,EACAizE,GAEA,MAAO,CACN5jF,KAAM,SACNwa,SAAU,GAASmjB,UAAWhqC,EAAQgd,GACtC5Z,KAAM6sF,EAAgB7sF,KACtBuD,WAAY,IAAIgM,IAAKs9E,EAAgBtpF,YACrCjN,OAAQ,EACR20F,YAAajtF,KAAKyqF,eAEpB,CAUQ,kBAAAuC,CACPloE,EACAuiD,EACAF,GAGA,MAAMygB,EAAqD,GAG3DzgB,EAAgB,IAAI51D,IAAK41D,GAGzB,UAAcpwE,EAAKyc,KAAc6zD,EAAgB,CAEhD,MAAM5zD,EAAW0zD,EAAcv5D,IAAK7W,GAAQowE,EAAc/+D,IAAKrR,GAAQ,KAGlE0c,IAAaD,GAEjBo0E,EAAMhoF,KAAM,CACXqL,KAAM,YACNwa,SAAUX,EAAM4b,MAChB5b,MAAOA,EAAM/B,QACbzqB,OAAQ,EACRmsD,aAAc1tD,EACdypE,kBAAmBhtD,EACnBitD,kBAAmBhtD,EACnBw5E,YAAajtF,KAAKyqF,iBAKpBtjB,EAAcx1D,OAAQ5a,EACvB,CAGA,UAAcA,EAAK0c,KAAc0zD,EAEhCygB,EAAMhoF,KAAM,CACXqL,KAAM,YACNwa,SAAUX,EAAM4b,MAChB5b,MAAOA,EAAM/B,QACbzqB,OAAQ,EACRmsD,aAAc1tD,EACdypE,kBAAmB,KACnBC,kBAAmBhtD,EACnBw5E,YAAajtF,KAAKyqF,iBAIpB,OAAO7C,CACR,CAKQ,oBAAAoD,CAAsBvzE,GAC7B,MAAM7Y,EAAS6Y,EAAQ7Y,OAEvB,IAAMA,EACL,OAAO,EAGR,MAAMqhE,EAAUjgE,KAAKqqF,kBAAkBjiF,IAAKxJ,GACtCgd,EAASnE,EAAQuyB,YAEvB,GAAKi2B,EACJ,UAAY1lC,KAAU0lC,EACrB,GAAoB,UAAf1lC,EAAOtvB,MAAoB2Q,GAAW2e,EAAO3e,QAAUA,EAAU2e,EAAO3e,OAAS2e,EAAOpvB,QAC5F,OAAO,EAKV,OAAOnL,KAAKgrF,qBAAsBpsF,EACnC,CAMQ,uBAAAqvF,CAAyBrvF,EAAoCgd,EAAgBzQ,GACpF,MAAM2Z,EAAQ,IAAI,GAAO,GAAS8jB,UAAWhqC,EAAQgd,GAAU,GAASgtB,UAAWhqC,EAAQgd,EAASzQ,IAEpG,UAAYlL,KAAQ6kB,EAAMonB,SAAU,CAAEpD,SAAS,IACzC7oC,EAAKiO,GAAI,aACblO,KAAKsqF,kBAAkB34E,OAAQ1R,GAC/BD,KAAKqqF,kBAAkB14E,OAAQ1R,GAE/BD,KAAKiuF,wBAAyBhuF,EAAM,EAAGA,EAAKo2D,WAG/C,EAeD,SAASq2B,GAAsBp1E,GAC9B,MAAMw3E,EAAW,GAEjB,UAAYn3E,KAASL,EACpB,GAAKK,EAAMzJ,GAAI,SACd,QAAU1T,EAAI,EAAGA,EAAImd,EAAM7U,KAAKxK,OAAQkC,IACvCs0F,EAASlvF,KAAM,CACdoC,KAAM,QACNuD,WAAY,IAAIgM,IAAKoG,EAAMquB,wBAI7B8oD,EAASlvF,KAAM,CACdoC,KAAQ2V,EAAmB3V,KAC3BuD,WAAY,IAAIgM,IAAKoG,EAAMquB,mBAK9B,OAAO8oD,CACR,CAiDA,SAASnC,GAA6BoC,EAA2B9uB,GAChE,MAAMviB,EAAyB,GAE/B,IAAI9hC,EAAS,EACTozE,EAAqB,EAGzB,UAAYz0D,KAAU0lC,EAAU,CAE/B,GAAK1lC,EAAO3e,OAASA,EAAS,CAC7B,QAAUphB,EAAI,EAAGA,EAAI+/B,EAAO3e,OAASA,EAAQphB,IAC5CkjD,EAAQ99C,KAAM,KAGfovF,GAAsBz0D,EAAO3e,OAASA,CACvC,CAGA,GAAoB,UAAf2e,EAAOtvB,KAAmB,CAC9B,QAAUzQ,EAAI,EAAGA,EAAI+/B,EAAOpvB,QAAS3Q,IACpCkjD,EAAQ99C,KAAM,KAIfgc,EAAS2e,EAAO3e,OAAS2e,EAAOpvB,OACjC,MAAO,GAAoB,UAAfovB,EAAOtvB,KAAmB,CACrC,QAAUzQ,EAAI,EAAGA,EAAI+/B,EAAOpvB,QAAS3Q,IACpCkjD,EAAQ99C,KAAM,KAIfgc,EAAS2e,EAAO3e,OAEhBozE,GAAsBz0D,EAAOpvB,OAC9B,MACCuyC,EAAQ99C,QAAS,IAAI07C,OAAQ/gB,EAAOpvB,SAAUpO,MAAO,KAGrD6e,EAAS2e,EAAO3e,OAAS2e,EAAOpvB,QAEhC6jF,GAAsBz0D,EAAOpvB,OAE/B,CAIA,GAAK6jF,EAAqBD,EACzB,QAAUv0F,EAAI,EAAGA,EAAIu0F,EAAoBC,EAAqBpzE,EAAQphB,IACrEkjD,EAAQ99C,KAAM,KAIhB,OAAO89C,CACR,CAKA,SAAS8vC,GAA2B90E,GACnC,MAAMu2E,EAAU,aAAcv2E,GAA0C,cAAjCA,EAAM+M,SAAUnvB,KAAK8+B,SACtD85D,EAAY,UAAWx2E,GAAsC,cAA7BA,EAAMoM,MAAMxuB,KAAK8+B,SAEvD,OAAQ65D,IAAYC,CACrB,CCz1Ce,MAAMC,GAArB,cAIC,KAAQC,YAAgC,GASxC,KAAQC,WAAwC,IAAI99E,IAKpD,KAAQ+9E,kBAAoC,IAAI17E,IAKhD,KAAQ27E,6BAAoD,IAAIh+E,IAKhE,KAAQi+E,SAAmB,EAQ3B,KAAQC,MAA6B,IAAIl+E,GAAI,CAU7C,WAAW5C,GACV,OAAO3O,KAAKwvF,QACb,CAEA,WAAW7gF,CAASA,GAGd3O,KAAKovF,YAAY92F,QAAUqW,EAAU3O,KAAKwvF,SAAW,GACzDxvF,KAAKyvF,MAAMh+E,IAAKzR,KAAKwvF,SAAU7gF,GAGhC3O,KAAKwvF,SAAW7gF,CACjB,CAKA,iBAAW+gF,GACV,OAAO1vF,KAAKovF,YAAapvF,KAAKovF,YAAY92F,OAAS,EACpD,CAOO,YAAA4xF,CAAczxB,GACpB,GAAKA,EAAU0nB,cAAgBngF,KAAK2O,QAOnC,MAAM,IAAI,EAAe,wDAAyD3O,KAAM,CACvFy4D,YACAk3B,eAAgB3vF,KAAK2O,UAIvB3O,KAAKovF,YAAYxvF,KAAM64D,GACvBz4D,KAAKwvF,WAELxvF,KAAKuvF,6BAA6B99E,IAAKgnD,EAAU0nB,YAAangF,KAAKovF,YAAY92F,OAAS,EACzF,CAWO,aAAAs3F,CAAeC,EAA0BC,EAAwB9vF,KAAK2O,SAG5E,IAAM3O,KAAKovF,YAAY92F,OACtB,MAAO,GAGR,MAAMy3F,EAAiB/vF,KAAKovF,YAAa,QAEhB,IAApBS,IACJA,EAAkBE,EAAe5P,aAKlC,IAAI6P,EAAcF,EAAgB,EAKlC,UAAcG,EAASC,KAAWlwF,KAAKyvF,MACjCI,EAAkBI,GAAWJ,EAAkBK,IACnDL,EAAkBK,GAGdF,EAAcC,GAAWD,EAAcE,IAC3CF,EAAcC,EAAU,GAK1B,GAAKD,EAAcD,EAAe5P,aAAgB0P,EAAkB7vF,KAAK0vF,cAAevP,YACvF,MAAO,GAGR,IAAIgQ,EAAYnwF,KAAKuvF,6BAA6BnnF,IAAKynF,QAGpC,IAAdM,IACJA,EAAY,GAGb,IAAIC,EAAUpwF,KAAKuvF,6BAA6BnnF,IAAK4nF,GAQrD,YALiB,IAAZI,IACJA,EAAUpwF,KAAKovF,YAAY92F,OAAS,GAI9B0H,KAAKovF,YAAYttF,MACvBquF,EAGAC,EAAU,EAEZ,CAQO,YAAAC,CAAclQ,GACpB,MAAMmQ,EAAiBtwF,KAAKuvF,6BAA6BnnF,IAAK+3E,GAE9D,QAAwB,IAAnBmQ,EAIL,OAAOtwF,KAAKovF,YAAakB,EAC1B,CASO,oBAAAC,CAAsBC,EAA4BC,GACxDzwF,KAAKqvF,WAAW59E,IAAKg/E,EAAkBD,GACvCxwF,KAAKsvF,kBAAkBzhF,IAAK2iF,EAC7B,CAQO,kBAAAE,CAAoBj4B,GAC1B,OAAOz4D,KAAKqvF,WAAWzhF,IAAK6qD,EAC7B,CAQO,iBAAAsuB,CAAmBtuB,GACzB,OAAOz4D,KAAKsvF,kBAAkB1hF,IAAK6qD,EACpC,CAQO,kBAAAyuB,CAAoBuJ,GAC1B,OAAOzwF,KAAKqvF,WAAWjnF,IAAKqoF,EAC7B,CAKO,KAAA7C,GACN5tF,KAAKwvF,SAAW,EAChBxvF,KAAKqvF,WAAa,IAAI99E,IACtBvR,KAAKovF,YAAc,GACnBpvF,KAAKsvF,kBAAoB,IAAI17E,IAC7B5T,KAAKyvF,MAAQ,IAAIl+E,IACjBvR,KAAKuvF,6BAA+B,IAAIh+E,GACzC,ECnOc,MAAMo/E,WAAoB,GA8BxC,WAAA5uF,CAAayB,EAAoBxB,EAAcozB,EAAmB,QACjE5nB,MAAOxL,GAjBR,KAAOkhF,aAAc,EAOrB,KAAOyI,WAAY,EAYlB3rF,KAAK8lE,UAAYtiE,EACjBxD,KAAKo1B,SAAWA,CACjB,CAKA,YAAoB5xB,GACnB,OAAOxD,KAAK8lE,SACb,CAWgB,UAAAvqC,GACf,OAAOv7B,KAAKkjF,WACb,CAOgB,MAAAlmD,GACf,OAAOh9B,KAAKo1B,QACb,EAaDu7D,GAAY9uF,UAAUqM,GAAK,SAAUjD,EAAcjJ,GAClD,OAAMA,EAOCA,IAAShC,KAAKgC,OACX,gBAATiJ,GAAmC,sBAATA,GAEjB,YAATA,GAA+B,kBAATA,GATN,gBAATA,GAAmC,sBAATA,GAEvB,YAATA,GAA+B,kBAATA,GACb,SAATA,GAA4B,eAATA,CAQtB,E,+aCxEA,MAAM2lF,GAAgB,aAkBP,MAAM,WAAiBxhF,KAkDrC,WAAArN,CAAa3J,GACZoV,QAEAxN,KAAK5H,MAAQA,EACb4H,KAAKwlF,QAAU,IAAI2J,GACnBnvF,KAAKi1B,UAAY,IAAI,GAAmBj1B,MACxCA,KAAK6wC,MAAQ,IAAI,GAAY,CAAE1gB,WAAY,aAC3CnwB,KAAK2/D,OAAS,IAAIwqB,GAAQ/xF,EAAMwnE,SAChC5/D,KAAKs1B,YAAa,EAElBt1B,KAAK4wC,YAAc,IAAIh9B,IACvB5T,KAAK6wF,4CAA6C,EAGlD7wF,KAAKijF,WAAY,QAAS2N,IAG1B5wF,KAAKwP,SAAoCpX,EAAO,kBAAkB,CAAEid,EAAKnZ,KACxE,MAAMu8D,EAAYv8D,EAAM,GAEnBu8D,EAAU2L,qBACdpkE,KAAK2/D,OAAOmrB,gBAAiBryB,EAC9B,GACE,CAAE7rD,SAAU,SAGf5M,KAAKwP,SAAoCpX,EAAO,kBAAkB,CAAEid,EAAKnZ,KACxE,MAAMu8D,EAAYv8D,EAAM,GAEnBu8D,EAAU2L,qBACdpkE,KAAKwlF,QAAQ0E,aAAczxB,EAC5B,GACE,CAAE7rD,SAAU,QAGf5M,KAAKwP,SAAgCxP,KAAKi1B,UAAW,UAAU,KAC9Dj1B,KAAK6wF,4CAA6C,CAAI,IAMvD7wF,KAAKwP,SAAuCpX,EAAMwnE,QAAS,UAAU,CAAEvqD,EAAK+rD,EAAQmB,EAAUprB,EAAU20C,KAEvG,MAAMC,EAAgB,SAAK3qB,EAAOzO,WAAZ,CAAuB7tC,MAAOqyB,IAGpDn3C,KAAK2/D,OAAO6rB,mBAAoBpqB,EAAOp/D,KAAM8pF,EAAeC,GAE1C,OAAbxpB,GAEJnB,EAAO/xD,GAAuB,UAAU,CAAEgG,EAAKktD,KAC9C,MAAMgpB,EAAanqB,EAAOzO,UAE1B3yD,KAAK2/D,OAAO6rB,mBACXpqB,EAAOp/D,KACP,SAAKupF,GAAL,CAAiBzmE,MAAOy9C,IACxBgpB,EACA,GAEH,IAcDvrF,KAAK06B,mBAAmBF,IACvB,IAAIr8B,GAAS,EAEb,UAAY7H,KAAQ0J,KAAK6wC,MAClBv6C,EAAKilC,cAAiBjlC,EAAK6rC,UAChC3H,EAAOv0B,OAAQu0B,EAAOud,cAAezhD,IAErC6H,GAAS,GAIX,UAAYijE,KAAUphE,KAAK5H,MAAMwnE,QAC1BwB,EAAOT,WAAWrqE,KAAKilC,eAC5Bf,EAAOs2D,aAAc1vB,GAErBjjE,GAAS,GAIX,OAAOA,CAAM,GAEf,CAWA,WAAWwQ,GACV,OAAO3O,KAAKwlF,QAAQ72E,OACrB,CAEA,WAAWA,CAASA,GACnB3O,KAAKwlF,QAAQ72E,QAAUA,CACxB,CAKA,aAAWirD,GACV,OAAO55D,KAAK8wC,QAAS8/C,GACtB,CAaO,UAAA3N,CAAYt+B,EAAsB,QAASvvB,EAAmB,QACpE,GAAKp1B,KAAK6wC,MAAMzoC,IAAKgtB,GAMpB,MAAM,IAAI,EAAe,wCAAyCp1B,KAAM,CAAEgC,KAAMozB,IAGjF,MAAM9+B,EAAO,IAAIq6F,GAAa3wF,KAAM2kD,EAAavvB,GAGjD,OAFAp1B,KAAK6wC,MAAMhjC,IAAKvX,GAETA,CACR,CAKO,OAAA0vB,GACNhmB,KAAKi1B,UAAUjP,UACfhmB,KAAK2P,eACN,CAWO,OAAAmhC,CAAS9uC,EAAe,QAC9B,OAAOhC,KAAK6wC,MAAMzoC,IAAKpG,EACxB,CAUO,YAAA+uF,CAAcC,GAAkB,GACtC,OAAOhxF,KAAK2tE,SAAUqjB,GAAkBh0F,KAAK1G,GAAQA,EAAK8+B,UAC3D,CAUO,QAAAu4C,CAAUqjB,GAAkB,GAClC,OAAOrwF,MAAMrB,KAAMU,KAAK6wC,OACtB9qC,QAAQzP,GAAQA,GAAQ0J,KAAK45D,YAAeo3B,GAAmB16F,EAAKilC,eAAkBjlC,EAAKq1F,WAC9F,CAuCO,iBAAAjxD,CAAmBqW,GACzB/wC,KAAK4wC,YAAY/iC,IAAKkjC,EACvB,CAOO,MAAA/T,GACN,MAAMC,EAAY,GAAOj9B,MAMzB,OAHAi9B,EAAKhI,UAAY,mCACjBgI,EAAK7kC,MAAQ,uBAEN6kC,CACR,CAYO,kBAAAg0D,CAAoBz2D,GACrBx6B,KAAKkxF,8CACTlxF,KAAKgxC,gBAAiBxW,GAGtBx6B,KAAKi1B,UAAUF,UAEV/0B,KAAK2/D,OAAOusB,iBAChBlsF,KAAKwQ,KAA2B,cAAegqB,EAAO0rC,OAEtDlmE,KAAKwQ,KAA2B,SAAUgqB,EAAO0rC,OAKlDlmE,KAAKi1B,UAAUF,UAEf/0B,KAAK2/D,OAAOiuB,SAGb5tF,KAAK6wF,4CAA6C,CACnD,CASU,yCAAAK,GACT,OAAQlxF,KAAK2/D,OAAOx9B,SAAWniC,KAAK6wF,0CACrC,CAQU,eAAAM,GACT,MAAMtgD,EAAQ7wC,KAAK2tE,WAEnB,OAAO98B,EAAMv4C,OAASu4C,EAAO,GAAM7wC,KAAK45D,SACzC,CAQO,gBAAA2M,GACN,MAAM6qB,EAAcpxF,KAAKmxF,kBACnB/4F,EAAQ4H,KAAK5H,MACbsrE,EAAStrE,EAAMsrE,OAGfj+C,EAAWrtB,EAAMi5F,uBAAwBD,EAAa,CAAE,IAI9D,OAHqB1tB,EAAOiE,yBAA0BliD,IAG/BrtB,EAAMy/C,YAAapyB,EAC3C,CAUO,uBAAAkhD,CAAyB7hD,GAC/B,OAAOwsE,GAA0BxsE,EAAM4b,QAAW4wD,GAA0BxsE,EAAM6b,IACnF,CAOQ,eAAAqQ,CAAiBxW,GACxB,IAAIyW,GAAW,EAEf,GACC,UAAY1hC,KAAYvP,KAAK4wC,YAW5B,GAJA5wC,KAAKi1B,UAAUF,UAEfkc,EAAW1hC,EAAUirB,GAEhByW,EACJ,YAGOA,EACX,EAsDD,SAASqgD,GAA0BC,GAClC,MAAM9zD,EAAW8zD,EAAc9zD,SAE/B,GAAKA,EAAW,CACf,MAAM36B,EAAO26B,EAAS36B,KAChB8Y,EAAS21E,EAAc31E,OAAS6hB,EAASuM,YAE/C,OAAQtW,GAAuB5wB,EAAM8Y,KAAaiY,GAAwB/wB,EAAM8Y,EACjF,CAEA,OAAO,CACR,C,8RC9ee,MAAM41E,WAAyBpiF,KAA9C,kCAIC,KAAQozE,SAAgC,IAAIjxE,GAAI,CAOhD,CAASzQ,OAAOC,YACf,OAAOf,KAAKwiF,SAASt3E,QACtB,CAQO,GAAA0C,CAAK+lE,GACX,MAAMrX,EAAaqX,aAAwB8d,GAAS9d,EAAa3xE,KAAO2xE,EAExE,OAAO3zE,KAAKwiF,SAAS50E,IAAK0uD,EAC3B,CASO,GAAAl0D,CAAKk0D,GACX,OAAOt8D,KAAKwiF,SAASp6E,IAAKk0D,IAAgB,IAC3C,CAoBO,IAAAmmB,CACN9O,EACA7uD,EACA4sE,GAAkC,EAClC18D,GAAuB,GAEvB,MAAMsnC,EAAaqX,aAAwB8d,GAAS9d,EAAa3xE,KAAO2xE,EAExE,GAAKrX,EAAW/sC,SAAU,KAMzB,MAAM,IAAI,EAAe,yCAA0CvvB,MAGpE,MAAM2xF,EAAY3xF,KAAKwiF,SAASp6E,IAAKk0D,GAErC,GAAKq1B,EAAY,CAChB,MAAM7F,EAAgB6F,EAAUh/B,UAE1B4P,EAAWovB,EAAUhxB,WAC3B,IAAIixB,GAAa,EAqBjB,OAnBMrvB,EAASt+C,QAASa,KACvB6sE,EAAUE,iBAAkB,GAAU1tB,UAAWr/C,IACjD8sE,GAAa,GAGTF,GAA0BC,EAAUD,yBACxCC,EAAUG,wBAA0BJ,EACpCE,GAAa,GAGc,kBAAhB58D,GAA6BA,GAAe28D,EAAU38D,cACjE28D,EAAU98D,aAAeG,EACzB48D,GAAa,GAGTA,GACJ5xF,KAAKwQ,KAAmC,UAAW8rD,IAAeq1B,EAAWpvB,EAAUz9C,EAAOgnE,GAGxF6F,CACR,CAEA,MAAMlrB,EAAY,GAAUtC,UAAWr/C,GACjCs8C,EAAS,IAAIqwB,GAAQn1B,EAAYmK,EAAWirB,EAAwB18D,G,MAK1E,OAHAh1B,KAAKwiF,SAAS/wE,IAAK6qD,EAAY8E,GAC/BphE,KAAKwQ,KAAmC,UAAW8rD,IAAe8E,EAAQ,KAAMt8C,G,2HAAO,IAAKs8C,EAAOzO,W,QAAZ,CAAuB7tC,MAAO,UAE9Gs8C,CACR,CAUO,OAAA/vC,CAASsiD,GACf,MAAMrX,EAAaqX,aAAwB8d,GAAS9d,EAAa3xE,KAAO2xE,EAClEge,EAAY3xF,KAAKwiF,SAASp6E,IAAKk0D,GAErC,QAAKq1B,IACJ3xF,KAAKwiF,SAAS7wE,OAAQ2qD,GACtBt8D,KAAKwQ,KAAmC,UAAW8rD,IAAeq1B,EAAWA,EAAUhxB,WAAY,KAAMgxB,EAAUh/B,WAEnH3yD,KAAK+xF,eAAgBJ,IAEd,EAIT,CAWO,QAAA9d,CAAUF,GAChB,MAAMrX,EAAaqX,aAAwB8d,GAAS9d,EAAa3xE,KAAO2xE,EAClEvS,EAASphE,KAAKwiF,SAASp6E,IAAKk0D,GAElC,IAAM8E,EAML,MAAM,IAAI,EAAe,6CAA8CphE,MAGxE,MAAM8kB,EAAQs8C,EAAOT,WAErB3gE,KAAKwQ,KAAmC,UAAW8rD,IAAe8E,EAAQt8C,EAAOA,EAAOs8C,EAAOzO,UAChG,CAKA,qBAAQuO,CAAsBz7C,GAC7B,UAAY27C,KAAUphE,KAChBohE,EAAOT,WAAWp1B,iBAAkB9lB,WAClC27C,EAGT,CAKA,4BAAQkqB,CAA6BxmE,GACpC,UAAYs8C,KAAUphE,KAC+B,OAA/CohE,EAAOT,WAAWz9C,gBAAiB4B,WACjCs8C,EAGT,CAKO,OAAAp7C,GACN,UAAYo7C,KAAUphE,KAAKwiF,SAASt3E,SACnClL,KAAK+xF,eAAgB3wB,GAGtBphE,KAAKwiF,SAAW,KAEhBxiF,KAAK2P,eACN,CAcA,gBAAQqiF,CAAiBC,GACxB,UAAY7wB,KAAUphE,KAAKwiF,SAASt3E,SAC9Bk2D,EAAOp/D,KAAK4iD,WAAYqtC,EAAS,aAC/B7wB,EAGT,CAKQ,cAAA2wB,CAAgB3wB,GACvBA,EAAOzxD,gBACPyxD,EAAO8wB,kBACR,EA0FD,MAAMT,WAAeriF,EAAc,KAkClC,WAAArN,CACCC,EACAykE,EACAirB,EACA18D,GAEAxnB,QAEAxN,KAAKgC,KAAOA,EACZhC,KAAKmyF,WAAanyF,KAAK6xF,iBAAkBprB,GACzCzmE,KAAK8xF,wBAA0BJ,EAC/B1xF,KAAK60B,aAAeG,CACrB,CAOA,0BAAW08D,GACV,IAAM1xF,KAAKmyF,WACV,MAAM,IAAI,EAAe,mBAAoBnyF,MAG9C,OAAOA,KAAK8xF,uBACb,CAKA,eAAW98D,GACV,IAAMh1B,KAAKmyF,WACV,MAAM,IAAI,EAAe,mBAAoBnyF,MAG9C,OAAOA,KAAK60B,YACb,CAKO,OAAA89B,GACN,MAAO,CACN7tC,MAAO9kB,KAAK2gE,WACZ3rC,YAAah1B,KAAKg1B,YAClB08D,uBAAwB1xF,KAAK0xF,uBAE/B,CAKO,QAAAU,GACN,IAAMpyF,KAAKmyF,WACV,MAAM,IAAI,EAAe,mBAAoBnyF,MAG9C,OAAOA,KAAKmyF,WAAWzxD,MAAM3d,OAC9B,CAKO,MAAAsvE,GACN,IAAMryF,KAAKmyF,WACV,MAAM,IAAI,EAAe,mBAAoBnyF,MAG9C,OAAOA,KAAKmyF,WAAWxxD,IAAI5d,OAC5B,CAYO,QAAA49C,GACN,IAAM3gE,KAAKmyF,WACV,MAAM,IAAI,EAAe,mBAAoBnyF,MAG9C,OAAOA,KAAKmyF,WAAWjuB,SACxB,CASO,gBAAA2tB,CAAkBprB,GAWxB,OAVKzmE,KAAKmyF,YACTnyF,KAAKkyF,mBAINzrB,EAAUr1D,SAAU,gBAAiB7R,GAAIS,MACzCymE,EAAUr1D,SAAU,kBAAmB7R,GAAIS,MAE3CA,KAAKmyF,WAAa1rB,EAEXA,CACR,CAOO,gBAAAyrB,GACNlyF,KAAKmyF,WAAYzgF,eAAgB,eAAgB1R,MACjDA,KAAKmyF,WAAYzgF,eAAgB,iBAAkB1R,MACnDA,KAAKmyF,WAAYrxE,SACjB9gB,KAAKmyF,WAAa,IACnB,EAKDV,GAAO5vF,UAAUqM,GAAK,SAAUjD,GAC/B,MAAgB,WAATA,GAA8B,iBAATA,CAC7B,ECtee,MAAMqnF,WAAwBpS,GAsB5C,WAAAn+E,CAAai3D,EAA0B7tD,GACtCqC,MAAO,MAEPxN,KAAKg5D,eAAiBA,EAAej2C,QACrC/iB,KAAKmL,QAAUA,CAChB,CAKA,QAAWF,GACV,MAAO,QACR,CAKA,sBAAWo2E,GACV,OAAO,IACR,CAKgB,MAAArkD,GACf,MAAMC,EAAYzvB,MAAMwvB,SAIxB,OAFAC,EAAK+7B,eAAiBh5D,KAAKg5D,eAAeh8B,SAEnCC,CACR,CAMgB,SAAAmjD,GACf,GAAKpgF,KAAKg5D,eAAe1iE,KAAKkN,SAM7B,MAAM,IAAI,EAAe,oCAAqCxD,KAEhE,CAMO,QAAA6hF,GACNxwD,GAAS,GAAMkb,4BAA6BvsC,KAAKg5D,eAAgBh5D,KAAKmL,SACvE,CAKA,oBAA2Bg7B,GAC1B,MAAO,iBACR,EC3Ec,MAAM,WAAyB,GAgC7C,WAAApkC,CAAauV,GACZ9J,QA3BD,KAAgBoyD,QAA8B,IAAIruD,IAelD,KAAiBm0B,UAAsB,IAAIywB,GAcrC7+C,GACJtX,KAAK2lC,aAAc,EAAGruB,EAExB,CAKA,CAASxW,OAAOC,YACf,OAAOf,KAAKw7B,aACb,CAKA,cAAWF,GACV,OAAOt7B,KAAK0lC,UAAUptC,MACvB,CAKA,aAAW+9D,GACV,OAAOr2D,KAAK0lC,UAAU2wB,SACvB,CAKA,WAAWl0B,GACV,OAA2B,IAApBniC,KAAKs7B,UACb,CAKA,eAAWW,GACV,OAAO,IACR,CAKA,mBAAWrV,GACV,OAAO,IACR,CAKA,QAAWtwB,GACV,OAAO0J,IACR,CAKA,UAAWpB,GACV,OAAO,IACR,CAKA,YAAW4E,GACV,OAAO,IACR,CAKO,UAAA+3B,GACN,OAAO,CACR,CAKO,YAAAY,GACN,MAAO,EACR,CAQO,QAAAN,CAAUj3B,GAChB,OAAO5E,KAAK0lC,UAAU4wB,QAAS1xD,EAChC,CAKO,WAAA42B,GACN,OAAOx7B,KAAK0lC,UAAW5kC,OAAOC,WAC/B,CAQO,aAAAi7B,CAAe78B,GACrB,OAAOa,KAAK0lC,UAAU6wB,aAAcp3D,EACrC,CAUO,mBAAA42D,CAAqB52D,GAC3B,OAAOa,KAAK0lC,UAAU8wB,mBAAoBr3D,EAC3C,CAKO,OAAA+8B,GACN,MAAO,EACR,CAcO,aAAAg7B,CAAeC,GAErB,IAAIh4D,EAAgCa,KAEpC,UAAY4E,KAASuyD,EACpBh4D,EAASA,EAAqC08B,SAAY18B,EAAqCu3D,cAAe9xD,IAG/G,OAAOzF,CACR,CAuBO,aAAAu3D,CAAe96C,GACrB,OAAO5b,KAAK0lC,UAAUgxB,cAAe96C,EACtC,CAQO,MAAAohB,GACN,MAAMC,EAAO,GAEb,UAAY99B,KAAQa,KAAK0lC,UACxBzI,EAAKr9B,KAAMT,EAAK69B,UAGjB,OAAOC,CACR,CASA,eAAcg6B,CAAUh6B,GACvB,MAAM3lB,EAAW,GAEjB,UAAYK,KAASslB,EACftlB,EAAM3V,KAEVsV,EAAS1X,KAAM,GAAQq3D,SAAUt/C,IAGjCL,EAAS1X,KAAM,GAAKq3D,SAAUt/C,IAIhC,OAAO,IAAI,GAAkBL,EAC9B,CAQO,YAAA2vB,CAAcrW,GACpB5wB,KAAK2lC,aAAc3lC,KAAKs7B,WAAY1K,EACrC,CAUO,YAAA+U,CAAc/gC,EAAegsB,GACnC,MAAMsW,EAiFR,SAAoBA,GAEnB,GAAqB,iBAATA,EACX,MAAO,CAAE,IAAI,GAAMA,IAGdvwB,EAAYuwB,KACjBA,EAAQ,CAAEA,IAIX,OAAOvmC,MAAMrB,KAAM4nC,GACjBlqC,KAAKmC,GACe,iBAARA,EACJ,IAAI,GAAMA,GAGbA,aAAgB,GACb,IAAI,GAAMA,EAAK2D,KAAM3D,EAAK6mC,iBAG3B7mC,GAEV,CAxGgB,CAAWyxB,GAEzB,UAAYzxB,KAAQ+nC,EAEE,OAAhB/nC,EAAKP,QACTO,EAAKkyB,UAGJlyB,EAAcP,OAASoB,KAG1BA,KAAK0lC,UAAUyQ,aAAcvxC,EAAOsiC,EACrC,CAWO,eAAApK,CAAiBl4B,EAAeuG,EAAkB,GACxD,MAAM+7B,EAAQlnC,KAAK0lC,UAAUqxB,aAAcnyD,EAAOuG,GAElD,UAAYhM,KAAQ+nC,EACjB/nC,EAAcP,OAAS,KAG1B,OAAOsoC,CACR,EA2CD,GAAiBrlC,UAAUqM,GAAK,SAAUjD,GACzC,MAAgB,qBAATA,GAAwC,2BAATA,CACvC,EC5Te,MAAMsnF,GAmBpB,WAAAxwF,CAAa3J,EAAc8tE,GAC1BlmE,KAAK5H,MAAQA,EACb4H,KAAKkmE,MAAQA,CACd,CAcO,UAAAvyB,CACN7wC,EACAyC,GAEA,OAAO,IAAI,GAAMzC,EAAMyC,EACxB,CAcO,aAAAD,CACNtD,EACAuD,GAEA,OAAO,IAAI,GAASvD,EAAMuD,EAC3B,CAOO,sBAAAmuC,GACN,OAAO,IAAI,EACZ,CAUO,YAAA8+C,CAAc/6E,EAAkBovB,GAAgB,GACtD,OAAOpvB,EAAQ8lB,OAAQsJ,EACxB,CA+CO,MAAAnhC,CACNzF,EACA6qC,EACAlvB,EAAyB,GAIzB,GAFA5b,KAAKyyF,6BAEAxyF,aAAgB,IAAqB,IAAbA,EAAK6C,KACjC,OAGD,MAAM2iB,EAAW,GAASmjB,UAAWkC,EAAgBlvB,GAGrD,GAAK3b,EAAKrB,OAAS,CAElB,GAAK8zF,GAAYzyF,EAAK3J,KAAMmvB,EAASnvB,MAIpC,YAFA0J,KAAKs1C,KAAM,GAAM7I,UAAWxsC,GAAQwlB,GAMpC,GAAKxlB,EAAK3J,KAAKkN,SAOd,MAAM,IAAI,EACT,qCACAxD,MAKDA,KAAKiG,OAAQhG,EAGhB,CAEA,MAAM0O,EAAU8W,EAASnvB,KAAKkN,SAAWiiB,EAASnvB,KAAKkN,SAASmL,QAAU,KAEpEjJ,EAAS,IAAIo8E,GAAiBr8D,EAAUxlB,EAAM0O,GAUpD,GARK1O,aAAgB,KACpByF,EAAOq8E,yBAA0B,GAGlC/hF,KAAKkmE,MAAMgkB,aAAcxkF,GACzB1F,KAAK5H,MAAMu6F,eAAgBjtF,GAGtBzF,aAAgB,GACpB,UAAcq8D,EAAYoE,KAAiBzgE,EAAK2/D,QAAU,CAEzD,MAAMgzB,EAAoB,GAAShqD,UAAW83B,EAAYpqE,KAAM,GAM1DiO,EAAU,CAAEugB,MALJ,IAAI,GACjB47C,EAAYhgC,MAAMw4B,aAAc05B,EAAmBntE,GACnDi7C,EAAY//B,IAAIu4B,aAAc05B,EAAmBntE,IAGzBotE,gBAAgB,EAAM79D,aAAa,GAEvDh1B,KAAK5H,MAAMwnE,QAAQhyD,IAAK0uD,GAC5Bt8D,KAAK8yF,aAAcx2B,EAAY/3D,GAE/BvE,KAAK+yF,UAAWz2B,EAAY/3D,EAE9B,CAEF,CAkEO,UAAAyuF,CACN94D,EACA30B,EACAulC,EACAlvB,GAEKrW,aAAsB,IAAoBA,aAAsB,IAAWA,aAAsB,GACrGvF,KAAK0F,OAAQ1F,KAAK2zC,WAAYzZ,GAAQ30B,EAAYulC,GAElD9qC,KAAK0F,OAAQ1F,KAAK2zC,WAAYzZ,EAAM30B,GAAculC,EAAgBlvB,EAEpE,CAkEO,aAAAitD,CACN7mE,EACAuD,EACA0tF,EACAr3E,GAEKrW,aAAsB,IAAoBA,aAAsB,IAAWA,aAAsB,GACrGvF,KAAK0F,OAAQ1F,KAAKsF,cAAetD,GAAQuD,EAAY0tF,GAErDjzF,KAAK0F,OAAQ1F,KAAKsF,cAAetD,EAAMuD,GAAc0tF,EAAwBr3E,EAE/E,CAiBO,MAAAiqC,CAAQ5lD,EAA+BrB,GAC7CoB,KAAK0F,OAAQzF,EAAMrB,EAAQ,MAC5B,CAkCO,UAAAs0F,CACNh5D,EACA30B,EACA3G,GAEK2G,aAAsB,IAAoBA,aAAsB,GACpEvF,KAAK0F,OAAQ1F,KAAK2zC,WAAYzZ,GAAQ30B,EAAY,OAElDvF,KAAK0F,OAAQ1F,KAAK2zC,WAAYzZ,EAAM30B,GAAc3G,EAAS,MAE7D,CAkCO,aAAAu0F,CACNnxF,EACAuD,EACA3G,GAEK2G,aAAsB,IAAoBA,aAAsB,GACpEvF,KAAK0F,OAAQ1F,KAAKsF,cAAetD,GAAQuD,EAAY,OAErDvF,KAAK0F,OAAQ1F,KAAKsF,cAAetD,EAAMuD,GAAc3G,EAAS,MAEhE,CAUO,YAAA6G,CAAc1O,EAAayB,EAAgB46F,GAGjD,GAFApzF,KAAKyyF,6BAEAW,aAAuB,GAAQ,CACnC,MAAMvnD,EAASunD,EAAYl5B,uBAE3B,UAAYp1C,KAAS+mB,EACpBwnD,GAAqBrzF,KAAMjJ,EAAKyB,EAAOssB,EAEzC,MACCwuE,GAAoBtzF,KAAMjJ,EAAKyB,EAAO46F,EAExC,CAgBO,aAAAG,CACNhuF,EACA6tF,GAEA,UAAcr8F,EAAKuD,KAAS,GAAOiL,GAClCvF,KAAKyF,aAAc1O,EAAKuD,EAAK84F,EAE/B,CASO,eAAA1sF,CAAiB3P,EAAaq8F,GAGpC,GAFApzF,KAAKyyF,6BAEAW,aAAuB,GAAQ,CACnC,MAAMvnD,EAASunD,EAAYl5B,uBAE3B,UAAYp1C,KAAS+mB,EACpBwnD,GAAqBrzF,KAAMjJ,EAAK,KAAM+tB,EAExC,MACCwuE,GAAoBtzF,KAAMjJ,EAAK,KAAMq8F,EAEvC,CAOO,eAAAI,CAAiBJ,GACvBpzF,KAAKyyF,6BAEL,MAAMgB,EAA6BxzF,IAClC,UAAYmW,KAAanW,EAAKq+B,mBAC7Bt+B,KAAK0G,gBAAiB0P,EAAWnW,EAClC,EAGD,GAAQmzF,aAAuB,GAG9B,UAAYnzF,KAAQmzF,EAAYlnD,WAC/BunD,EAA0BxzF,QAH3BwzF,EAA0BL,EAM5B,CA+BO,IAAA99C,CACNxwB,EACAgmB,EACAlvB,GAIA,GAFA5b,KAAKyyF,+BAEG3tE,aAAiB,IAMxB,MAAM,IAAI,EAAe,4BAA6B9kB,MAGvD,IAAM8kB,EAAMhH,OAMX,MAAM,IAAI,EAAe,6BAA8B9d,MAGxD,MAAMylB,EAAW,GAASmjB,UAAWkC,EAAgBlvB,GAGrD,GAAK6J,EAASxB,QAASa,EAAM4b,OAC5B,OAMD,GAFA1gC,KAAK0zF,gCAAiC,OAAQ5uE,IAExC4tE,GAAY5tE,EAAMxuB,KAAMmvB,EAASnvB,MAOtC,MAAM,IAAI,EAAe,iCAAkC0J,MAG5D,MAAM2O,EAAUmW,EAAMxuB,KAAKkN,SAAWshB,EAAMxuB,KAAKkN,SAASmL,QAAU,KAC9D8pD,EAAY,IAAI2oB,GAAet8D,EAAM4b,MAAO5b,EAAM6b,IAAI/kB,OAASkJ,EAAM4b,MAAM9kB,OAAQ6J,EAAU9W,GAEnG3O,KAAKkmE,MAAMgkB,aAAczxB,GACzBz4D,KAAK5H,MAAMu6F,eAAgBl6B,EAC5B,CAOO,MAAAxyD,CAAQmtF,GACdpzF,KAAKyyF,6BAEL,MACM5mD,GADgBunD,aAAuB,GAAQA,EAAc,GAAM3mD,UAAW2mD,IACvDl5B,uBAAuB7uD,UAEpD,UAAYm+D,KAAQ39B,EAEnB7rC,KAAK0zF,gCAAiC,OAAQlqB,GAE9CmqB,GAAsBnqB,EAAK9oC,MAAO8oC,EAAK7oC,IAAI/kB,OAAS4tD,EAAK9oC,MAAM9kB,OAAQ5b,KAAKkmE,MAAOlmE,KAAK5H,MAE1F,CAUO,KAAAw7F,CAAOnuE,GACbzlB,KAAKyyF,6BAEL,MAAMtoD,EAAa1kB,EAAS0kB,WACtBD,EAAYzkB,EAASykB,UAK3B,GAFAlqC,KAAK0zF,gCAAiC,QAASjuE,KAEvC0kB,aAAsB,IAM7B,MAAM,IAAI,EAAe,iCAAkCnqC,MAG5D,KAAQkqC,aAAqB,IAM5B,MAAM,IAAI,EAAe,gCAAiClqC,MAGrDylB,EAASnvB,KAAKkN,SAGnBxD,KAAK6zF,OAAQpuE,GAFbzlB,KAAK8zF,eAAgBruE,EAIvB,CASO,sBAAA4rE,CACN/6F,EACAmJ,EACA43D,GAEA,OAAOr3D,KAAK5H,MAAMi5F,uBAAwB/6F,EAAMmJ,EAAM43D,EACvD,CAOO,gBAAA3f,CACN5M,EACAlvB,GAEA,OAAO5b,KAAK5H,MAAMs/C,iBAAkB5M,EAAgBlvB,EACrD,CAOO,mBAAA+7B,CAAqB13C,GAC3B,OAAOD,KAAK5H,MAAMu/C,oBAAqB13C,EACxC,CAOO,oBAAA23C,CAAsB33C,GAC5B,OAAOD,KAAK5H,MAAMw/C,qBAAsB33C,EACzC,CAQO,WAAA43C,CAAanX,EAAiBC,GACpC,OAAO3gC,KAAK5H,MAAMy/C,YAAanX,EAAOC,EACvC,CAOO,aAAAoX,CAAetgC,GACrB,OAAOzX,KAAK5H,MAAM2/C,cAAetgC,EAClC,CAOO,aAAAqgC,CAAergC,GACrB,OAAOzX,KAAK5H,MAAM0/C,cAAergC,EAClC,CAgBO,eAAAugC,IAAoB97C,GAC1B,OAAO8D,KAAK5H,MAAM4/C,mBAAoB97C,EACvC,CAOQ,cAAA43F,CAAgBruE,GACvB,MAAM0kB,EAAa1kB,EAAS0kB,WACtBD,EAAYzkB,EAASykB,UAE3BlqC,KAAKs1C,KAAM,GAAM9I,UAAWtC,GAAoB,GAAStB,UAAWuB,EAAa,QACjFnqC,KAAKiG,OAAQikC,EACd,CAOQ,MAAA2pD,CAAQpuE,GACf,MAAM4vB,EAAiB,GAASzM,UAAWnjB,EAAS0kB,WAAa,OAC3D6uB,EAAiB,GAASpwB,UAAWnjB,EAASykB,UAAY,GAE1D0vB,EAAYn0C,EAASnvB,KAAKkN,SAAUo2D,UACpCP,EAAoB,IAAI,GAAUO,EAAW,CAAE,IAE/CjrD,EAAU8W,EAASnvB,KAAKkN,SAAUmL,QAElCilF,EAAQ,IAAIzR,GACjBnpB,EACEvzC,EAASykB,UAAmBmsB,UAC9BhhB,EACAgkB,EACA1qD,GAGD3O,KAAKkmE,MAAMgkB,aAAc0J,GACzB5zF,KAAK5H,MAAMu6F,eAAgBiB,EAC5B,CAQO,MAAAv8C,CAAQ5/B,EAAqC6/B,GAGnD,GAFAt3C,KAAKyyF,+BAEGh7E,aAAmB,IAM1B,MAAM,IAAI,EACT,qCACAzX,MAIF,MAAM2O,EAAU8I,EAAQnhB,KAAKkN,SAAWiU,EAAQnhB,KAAKkN,SAASmL,QAAU,KAClEolF,EAAkB,IAAInR,GAAiB,GAAS74C,cAAetyB,GAAWA,EAAQzV,KAAMs1C,EAAS3oC,GAEvG3O,KAAKkmE,MAAMgkB,aAAc6J,GACzB/zF,KAAK5H,MAAMu6F,eAAgBoB,EAC5B,CAeO,KAAAh3F,CAAO0oB,EAAoBoxD,GACjC72E,KAAKyyF,6BAEL,IA4BIuB,EACAC,EA7BA7R,EAAe38D,EAAS7mB,OAE5B,IAAMwjF,EAAaxjF,OAMlB,MAAM,IAAI,EAAe,iCAAkCoB,MAQ5D,GAJM62E,IACLA,EAAeuL,EAAaxjF,SAGvB6mB,EAAS7mB,OAAOu9B,aAAc,CAAEE,aAAa,IAAS9M,SAAUsnD,GAMrE,MAAM,IAAI,EAAe,qCAAsC72E,MAShE,EAAG,CACF,MAAM2O,EAAUyzE,EAAa9rF,KAAKkN,SAAW4+E,EAAa9rF,KAAKkN,SAASmL,QAAU,KAC5ExD,EAAUi3E,EAAa/rB,UAAY5wC,EAAS7J,OAE5C08B,EAAoB4pC,GAAeG,qBAAsB58D,GACzD1oB,EAAQ,IAAImlF,GAAgBz8D,EAAUta,EAASmtC,EAAmB,KAAM3pC,GAE9E3O,KAAKkmE,MAAMgkB,aAAcntF,GACzBiD,KAAK5H,MAAMu6F,eAAgB51F,GAGrBi3F,GAAsBC,IAC3BD,EAAoB5R,EACpB6R,EAAmBxuE,EAAS7mB,OAAOq9B,aAIpCmmD,GADA38D,EAAWzlB,KAAK23C,oBAAqBlyB,EAAS7mB,SACtBA,MACzB,OAAUwjF,IAAiBvL,GAE3B,MAAO,CACNpxD,WACAX,MAAO,IAAI,GAAO,GAAS8jB,UAAWorD,EAAoB,OAAS,GAASprD,UAAWqrD,EAAmB,IAE5G,CAWO,IAAAn9C,CAAMhyB,EAAcovE,GAG1B,GAFAl0F,KAAKyyF,8BAEC3tE,EAAMhH,OAMX,MAAM,IAAI,EAAe,6BAA8B9d,MAGxD,MAAMyX,EAAUy8E,aAA2B,GAAUA,EAAkB,IAAI,GAASA,GAEpF,GAAKz8E,EAAQ6jB,WAAa,EAMzB,MAAM,IAAI,EAAe,gCAAiCt7B,MAG3D,GAAwB,OAAnByX,EAAQ7Y,OAMZ,MAAM,IAAI,EAAe,+BAAgCoB,MAG1DA,KAAK0F,OAAQ+R,EAASqN,EAAM4b,OAG5B,MAAMyzD,EAAe,IAAI,GAAOrvE,EAAM4b,MAAM4J,aAAc,GAAKxlB,EAAM6b,IAAI2J,aAAc,IAEvFtqC,KAAKs1C,KAAM6+C,EAAc,GAASvrD,UAAWnxB,EAAS,GACvD,CAQO,MAAAy/B,CAAQz/B,GAGd,GAFAzX,KAAKyyF,6BAEmB,OAAnBh7E,EAAQ7Y,OAMZ,MAAM,IAAI,EAAe,kCAAmCoB,MAG7DA,KAAKs1C,KAAM,GAAM9I,UAAW/0B,GAAWzX,KAAK23C,oBAAqBlgC,IACjEzX,KAAKiG,OAAQwR,EACd,CA8CO,SAAAs7E,CACN/wF,EACAuC,GAQA,GAFAvE,KAAKyyF,8BAECluF,GAA4C,kBAA1BA,EAAQsuF,eAM/B,MAAM,IAAI,EAAe,qCAAsC7yF,MAGhE,MAAM6yF,EAAiBtuF,EAAQsuF,eACzB/tE,EAAQvgB,EAAQugB,MAChBkQ,OAAsC,IAAxBzwB,EAAQywB,aAAoCzwB,EAAQywB,YAExE,GAAKh1B,KAAK5H,MAAMwnE,QAAQhyD,IAAK5L,GAM5B,MAAM,IAAI,EAAe,iCAAkChC,MAG5D,IAAM8kB,EAML,MAAM,IAAI,EAAe,4BAA6B9kB,MAGvD,OAAM6yF,GAINuB,GAAsBp0F,KAAMgC,EAAM,KAAM8iB,EAAOkQ,GAExCh1B,KAAK5H,MAAMwnE,QAAQx3D,IAAKpG,IALvBhC,KAAK5H,MAAMwnE,QAAQ6iB,KAAMzgF,EAAM8iB,EAAO+tE,EAAgB79D,EAM/D,CAyDO,YAAA89D,CACNnf,EACApvE,GAMAvE,KAAKyyF,6BAEL,MAAMn2B,EAAoC,iBAAhBqX,EAA2BA,EAAeA,EAAa3xE,KAC3E4xE,EAAgB5zE,KAAK5H,MAAMwnE,QAAQx3D,IAAKk0D,GAE9C,IAAMsX,EAML,MAAM,IAAI,EAAe,wCAAyC5zE,MAGnE,IAAMuE,EAeL,OAJA,EAAY,wDAAyD,CAAE+3D,oBAEvEt8D,KAAK5H,MAAMwnE,QAAQiU,SAAUD,GAK9B,MAAMygB,EAA4D,kBAA1B9vF,EAAQsuF,eAC1CyB,EAAmD,kBAAvB/vF,EAAQywB,YAGpCA,EAAcs/D,EAAqB/vF,EAAQywB,YAAc4+C,EAAc5+C,YAE7E,IAAMq/D,IAA6B9vF,EAAQugB,QAAUwvE,EAMpD,MAAM,IAAI,EAAe,oCAAqCt0F,MAG/D,MAAMoxE,EAAewC,EAAcjT,WAC7B4zB,EAAehwF,EAAQugB,MAAQvgB,EAAQugB,MAAQssD,EAEhDijB,GAA4B9vF,EAAQsuF,iBAAmBjf,EAAc8d,uBAEpEntF,EAAQsuF,eAGZuB,GAAsBp0F,KAAMs8D,EAAY,KAAMi4B,EAAcv/D,IAI5Do/D,GAAsBp0F,KAAMs8D,EAAY8U,EAAc,KAAMp8C,GAG5Dh1B,KAAK5H,MAAMwnE,QAAQ6iB,KAAMnmB,EAAYi4B,OAAc,EAAWv/D,IAO3D4+C,EAAc8d,uBAClB0C,GAAsBp0F,KAAMs8D,EAAY8U,EAAcmjB,EAAcv/D,GAEpEh1B,KAAK5H,MAAMwnE,QAAQ6iB,KAAMnmB,EAAYi4B,OAAc,EAAWv/D,EAEhE,CASO,YAAA87D,CAAcnd,GACpB3zE,KAAKyyF,6BAEL,MAAMzwF,EAA8B,iBAAhB2xE,EAA2BA,EAAeA,EAAa3xE,KAE3E,IAAMhC,KAAK5H,MAAMwnE,QAAQhyD,IAAK5L,GAM7B,MAAM,IAAI,EAAe,gCAAiChC,MAG3D,MAAMohE,EAASphE,KAAK5H,MAAMwnE,QAAQx3D,IAAKpG,GAEvC,IAAMo/D,EAAOswB,uBAGZ,YAFA1xF,KAAK5H,MAAMwnE,QAAQvuC,QAASrvB,GAO7BoyF,GAAsBp0F,KAAMgC,EAFXo/D,EAAOT,WAEoB,KAAMS,EAAOpsC,YAC1D,CAYO,OAAAw/D,CAASp/D,EAAkBuvB,EAAc,SAC/C3kD,KAAKyyF,6BAEL,MAAMn8F,EAAO0J,KAAK5H,MAAMoL,SAASstC,QAAS1b,GAE1C,GAAK9+B,GAAQA,EAAKilC,aAMjB,MAAM,IAAI,EAAe,6BAA8Bv7B,MAGxD,MAAMwD,EAAWxD,KAAK5H,MAAMoL,SACtBi1D,EAAY,IAAIsqB,GAAe3tD,EAAUuvB,GAAa,EAAMnhD,EAAUA,EAASmL,SAKrF,OAHA3O,KAAKkmE,MAAMgkB,aAAczxB,GACzBz4D,KAAK5H,MAAMu6F,eAAgBl6B,GAEpBz4D,KAAK5H,MAAMoL,SAASstC,QAAS1b,EACrC,CAiBO,UAAAq/D,CAAYC,GAClB10F,KAAKyyF,6BAEL,MAAMn8F,EAA4B,iBAAdo+F,EAAyB10F,KAAK5H,MAAMoL,SAASstC,QAAS4jD,GAAeA,EAEzF,IAAMp+F,IAASA,EAAKilC,aAMnB,MAAM,IAAI,EAAe,4BAA6Bv7B,MAKvD,UAAYohE,KAAUphE,KAAK5H,MAAMwnE,QAC3BwB,EAAOT,WAAWrqE,OAASA,GAC/B0J,KAAK8wF,aAAc1vB,GAKrB,UAAYrqE,KAAOT,EAAKgoC,mBACvBt+B,KAAK0G,gBAAiB3P,EAAKT,GAI5B0J,KAAKiG,OAAQjG,KAAK+3C,cAAezhD,IAGjC,MAAMkN,EAAWxD,KAAK5H,MAAMoL,SACtBi1D,EAAY,IAAIsqB,GAAezsF,EAAK8+B,SAAU9+B,EAAK0L,MAAM,EAAOwB,EAAUA,EAASmL,SAEzF3O,KAAKkmE,MAAMgkB,aAAczxB,GACzBz4D,KAAK5H,MAAMu6F,eAAgBl6B,EAC5B,CAmFO,YAAAjlB,IAAiBt3C,GACvB8D,KAAKyyF,6BAELzyF,KAAK5H,MAAMoL,SAASyxB,UAAUka,UAAWjzC,EAC1C,CAWO,iBAAAu3C,CACN3I,EACAlvB,GAEA5b,KAAKyyF,6BAELzyF,KAAK5H,MAAMoL,SAASyxB,UAAUma,UAAWtE,EAAgBlvB,EAC1D,CAmCO,qBAAA+4E,CACNC,EACAp8F,GAIA,GAFAwH,KAAKyyF,6BAEiC,iBAA1BmC,EACX50F,KAAK60F,uBAAwBD,EAAuBp8F,QAEpD,UAAczB,EAAKyB,KAAW,GAAOo8F,GACpC50F,KAAK60F,uBAAwB99F,EAAKyB,EAGrC,CAmBO,wBAAAsmF,CAA0BgW,GAGhC,GAFA90F,KAAKyyF,6BAE+B,iBAAxBqC,EACX90F,KAAK+0F,0BAA2BD,QAEhC,UAAY/9F,KAAO+9F,EAClB90F,KAAK+0F,0BAA2Bh+F,EAGnC,CAwBO,wBAAAi+F,GACN,OAAOh1F,KAAK5H,MAAMoL,SAASyxB,UAAUiwC,kBACtC,CAWO,uBAAA+vB,CAAyBr8E,GAC/B5Y,KAAK5H,MAAMoL,SAASyxB,UAAUmwC,gBAAiBxsD,EAChD,CAMQ,sBAAAi8E,CAAwB99F,EAAayB,GAC5C,MAAMy8B,EAAYj1B,KAAK5H,MAAMoL,SAASyxB,UAGtC,GAAKA,EAAU+V,aAAe/V,EAAUyG,OAAQ98B,OAAOujC,QAAU,CAChE,MAAM+yD,EAAW,GAAkB5vB,sBAAuBvuE,GAE1DiJ,KAAKyF,aAAcyvF,EAAU18F,EAAOy8B,EAAUyG,OAAQ98B,OACvD,CAEAq2B,EAAUmS,cAAerwC,EAAKyB,EAC/B,CAKQ,yBAAAu8F,CAA2Bh+F,GAClC,MAAMk+B,EAAYj1B,KAAK5H,MAAMoL,SAASyxB,UAGtC,GAAKA,EAAU+V,aAAe/V,EAAUyG,OAAQ98B,OAAOujC,QAAU,CAChE,MAAM+yD,EAAW,GAAkB5vB,sBAAuBvuE,GAE1DiJ,KAAK0G,gBAAiBwuF,EAAUjgE,EAAUyG,OAAQ98B,OACnD,CAEAq2B,EAAUqS,iBAAkBvwC,EAC7B,CAKQ,0BAAA07F,GAUP,GAAOzyF,KAAK5H,MAAe+8F,iBAAmBn1F,KAC7C,MAAM,IAAI,EAAe,uBAAwBA,KAEnD,CAUQ,+BAAA0zF,CACPzoF,EACA8pC,GAEA,UAAYqsB,KAAUphE,KAAK5H,MAAMwnE,QAAU,CAC1C,IAAMwB,EAAOswB,uBACZ,SAGD,MAAMhxB,EAAcU,EAAOT,WAC3B,IAAIy0B,GAAa,EAEjB,GAAc,SAATnqF,EAAkB,CACtB,MAAM6Z,EAAQiwB,EACdqgD,EACCtwE,EAAMymB,iBAAkBm1B,EAAYhgC,QACpC5b,EAAM4b,MAAMzc,QAASy8C,EAAYhgC,QACjC5b,EAAMymB,iBAAkBm1B,EAAY//B,MACpC7b,EAAM6b,IAAI1c,QAASy8C,EAAY//B,IACjC,KAAO,CAEN,MAAMlb,EAAWsvB,EACXw3B,EAAgB9mD,EAAS0kB,WACzBmiC,EAAe7mD,EAASykB,UAMxBmrD,EAAwB30B,EAAYhgC,MAAM9hC,QAAU2tE,GAAiB7L,EAAYhgC,MAAM8I,QAMvF8rD,EAAyB50B,EAAY//B,IAAI/hC,QAAU0tE,GAA0C,GAA1B5L,EAAY//B,IAAI/kB,OAMnF25E,EAA2B70B,EAAY//B,IAAIuJ,WAAaoiC,EAMxDkpB,EAA6B90B,EAAYhgC,MAAMwJ,WAAaoiC,EAElE8oB,EAAaC,GAAyBC,GAA0BC,GAA4BC,CAC7F,CAEKJ,GACJp1F,KAAK8yF,aAAc1xB,EAAOp/D,KAAM,CAAE8iB,MAAO47C,GAE3C,CACD,EAWD,SAAS2yB,GAAqB74D,EAAgBzjC,EAAayB,EAAgBssB,GAC1E,MAAM1sB,EAAQoiC,EAAOpiC,MACfif,EAAMjf,EAAMoL,SAGlB,IAIIiiB,EAGAgwE,EAGAC,EAVAC,EAAoB7wE,EAAM4b,MAY9B,UAAYpmC,KAAOwqB,EAAM+lB,UAAW,CAAE/B,SAAS,IAC9C4sD,EAAap7F,EAAI2F,KAAKo7B,aAActkC,GAI/B0uB,GAAYgwE,GAAeC,IAE1BD,GAAej9F,GACnB0xF,IAGDyL,EAAoBlwE,GAGrBA,EAAWnrB,EAAI2vC,aACfwrD,EAAcC,EASf,SAASxL,IACR,MAAMplE,EAAQ,IAAI,GAAO6wE,EAAmBlwE,GACtC9W,EAAUmW,EAAMxuB,KAAKkN,SAAW6T,EAAI1I,QAAU,KAC9C8pD,EAAY,IAAIiqB,GAAoB59D,EAAO/tB,EAAK0+F,EAAaj9F,EAAOmW,GAE1E6rB,EAAO0rC,MAAMgkB,aAAczxB,GAC3BrgE,EAAMu6F,eAAgBl6B,EACvB,CAXKhzC,aAAoB,IAAYA,GAAYkwE,GAAqBF,GAAej9F,GACpF0xF,GAWF,CAKA,SAASoJ,GAAoB94D,EAAgBzjC,EAAayB,EAAgByH,GACzE,MAAM7H,EAAQoiC,EAAOpiC,MACfif,EAAMjf,EAAMoL,SACZoyF,EAAgB31F,EAAKo7B,aAActkC,GACzC,IAAI+tB,EAAO2zC,EAEX,GAAKm9B,GAAiBp9F,EAAQ,CAG7B,GAFsByH,EAAK3J,OAAS2J,EAEf,CAEpB,MAAM0O,EAAU1O,EAAKuD,SAAW6T,EAAI1I,QAAU,KAE9C8pD,EAAY,IAAIqqB,GAAwB7iF,EAAalJ,EAAK6+F,EAAep9F,EAAOmW,EACjF,KAAO,CACNmW,EAAQ,IAAI,GAAO,GAASilB,cAAe9pC,GAAQu6B,EAAOmd,oBAAqB13C,IAE/E,MAAM0O,EAAUmW,EAAMxuB,KAAKkN,SAAW6T,EAAI1I,QAAU,KAEpD8pD,EAAY,IAAIiqB,GAAoB59D,EAAO/tB,EAAK6+F,EAAep9F,EAAOmW,EACvE,CAEA6rB,EAAO0rC,MAAMgkB,aAAczxB,GAC3BrgE,EAAMu6F,eAAgBl6B,EACvB,CACD,CAKA,SAAS27B,GACR55D,EACAx4B,EACAugE,EACAprB,EACAniB,GAEA,MAAM58B,EAAQoiC,EAAOpiC,MACfif,EAAMjf,EAAMoL,SAEZi1D,EAAY,IAAI8pB,GAAiBvgF,EAAMugE,EAAUprB,EAAU/+C,EAAMwnE,UAAW5qC,EAAa3d,EAAI1I,SAEnG6rB,EAAO0rC,MAAMgkB,aAAczxB,GAC3BrgE,EAAMu6F,eAAgBl6B,EACvB,CAWA,SAASk7B,GAAsBluE,EAAoBta,EAAiB+6D,EAAc9tE,GACjF,IAAIqgE,EAEJ,GAAKhzC,EAASnvB,KAAKkN,SAAW,CAC7B,MAAM6T,EAAMjf,EAAMoL,SACZ61D,EAAoB,IAAI,GAAUhiD,EAAIuiD,UAAW,CAAE,IAEzDnB,EAAY,IAAI2oB,GAAe37D,EAAUta,EAASkuD,EAAmBhiD,EAAI1I,QAC1E,MACC8pD,EAAY,IAAI65B,GAAiB7sE,EAAUta,GAG5C+6D,EAAMgkB,aAAczxB,GACpBrgE,EAAMu6F,eAAgBl6B,EACvB,CAWA,SAASi6B,GAAYmD,EAAgCC,GAEpD,OAAKD,IAAUC,GAKVD,aAAiBlF,IAAemF,aAAiBnF,EAKvD,CC/yDe,SAASoF,GACvB39F,EACA68B,EACA1wB,EAII,CAAC,GAEL,GAAK0wB,EAAU+V,YACd,OAGD,MAAMgrD,EAAW/gE,EAAUoY,gBAG3B,GAA+B,cAA1B2oD,EAAS1/F,KAAK8+B,SAClB,OAGD,MAAMsuC,EAAStrE,EAAMsrE,OAErBtrE,EAAMmiC,QAAQC,IAGb,IAAMj2B,EAAQ0xF,yBAuehB,SAAqDvyB,EAAgBzuC,GACpE,MAAM4hD,EAAenT,EAAOyO,gBAAiBl9C,GAE7C,IAAMA,EAAUmuC,sBAAuByT,GACtC,OAAO,EAGR,MAAM/xD,EAAQmQ,EAAUoY,gBAExB,GAAKvoB,EAAM4b,MAAM9hC,QAAUkmB,EAAM6b,IAAI/hC,OACpC,OAAO,EAGR,OAAO8kE,EAAOyF,WAAY0N,EAAc,YACzC,CArf2Cqf,CAA4CxyB,EAAQzuC,GAG5F,YAudH,SAA4CuF,EAAgBvF,GAC3D,MAAM4hD,EAAer8C,EAAOpiC,MAAMsrE,OAAOyO,gBAAiBl9C,GAE1DuF,EAAOv0B,OAAQu0B,EAAOud,cAAe8+B,IACrCsf,GAAiB37D,EAAQA,EAAOkd,iBAAkBm/B,EAAc,GAAK5hD,EACtE,CA9dGmhE,CAAmC57D,EAAQvF,GAM5C,MAAMohE,EAA6B,CAAC,EAEpC,IAAM9xF,EAAQ+xF,mBAAqB,CAClC,MAAMpmD,EAAkBjb,EAAU+Y,qBAE7BkC,GACJl5C,OAAO+wB,OAAQsuE,EAA4B3yB,EAAO4T,0BAA2BpnC,EAAiB,iBAAiB,GAEjH,CAGA,MAAQzH,EAAeiQ,GAoDzB,SAA4C5zB,GAC3C,MAAM1sB,EAAQ0sB,EAAMxuB,KAAKkN,SAAUpL,MAE7BqwC,EAAgB3jB,EAAM4b,MAC5B,IAAIgY,EAAc5zB,EAAM6b,IAIxB,GAAKvoC,EAAMylF,WAAY/4D,EAAO,CAAEyxE,eAAe,IAAW,CACzD,MAAMrzB,EAqCR,SAAyBz9C,GACxB,MAAMhO,EAAUgO,EAAS7mB,OACnB8kE,EAASjsD,EAAQnhB,KAAKkN,SAAUpL,MAAMsrE,OACtCtnC,EAAY3kB,EAAQ0kB,aAAc,CAAEG,aAAa,EAAMD,aAAa,IAE1E,UAAY5kB,KAAW2kB,EAAY,CAClC,GAAKsnC,EAAOI,QAASrsD,GACpB,OAAO,KAGR,GAAKisD,EAAOC,QAASlsD,GACpB,OAAOA,CAET,CACD,CAnDmB,CAAgBihC,GAEjC,GAAKwqB,GAAYxqB,EAAYwf,WAAY9/D,EAAMs/C,iBAAkBwrB,EAAU,IAAQ,CAElF,MAAMjuC,EAAY78B,EAAM4/C,gBAAiBlzB,GAIzC1sB,EAAMo+F,gBAAiBvhE,EAAW,CAAE1Q,UAAW,aAE/C,MAAMkyE,EAAiBxhE,EAAUuY,kBAS3BkpD,EAAet+F,EAAMy/C,YAAa4+C,EAAgB/9C,GAElDtgD,EAAMylF,WAAY6Y,EAAc,CAAEH,eAAe,MACtD79C,EAAc+9C,EAEhB,CACD,CAEA,MAAO,CACN,GAAa7M,aAAcnhD,EAAe,cAC1C,GAAamhD,aAAclxC,EAAa,UAE1C,CA5FyCi+C,CAAmCX,GAGpEvtD,EAAcyvB,WAAYxf,IAC/Ble,EAAOv0B,OAAQu0B,EAAOqd,YAAapP,EAAeiQ,IAW7Cn0C,EAAQqyF,iBAuGhB,SAAwBp8D,EAAgBiO,EAAyBiQ,GAChE,MAAMtgD,EAAQoiC,EAAOpiC,MAGrB,IAAMy+F,GAAkBr8D,EAAOpiC,MAAMsrE,OAAQj7B,EAAeiQ,GAC3D,OA4BD,MAAQo+C,EAAeC,GA0PxB,SAA8CC,EAAqBC,GAClE,MAAMz6D,EAAaw6D,EAAU76D,eACvBM,EAAaw6D,EAAU96D,eAE7B,IAAI3hC,EAAI,EAER,KAAQgiC,EAAYhiC,IAAOgiC,EAAYhiC,IAAOiiC,EAAYjiC,IACzDA,IAGD,MAAO,CAAEgiC,EAAYhiC,GAAKiiC,EAAYjiC,GACvC,CArQwC08F,CAAqCzuD,EAAeiQ,GAU3F,IAAMo+C,IAAkBC,EACvB,QAGK3+F,EAAMylF,WAAYiZ,EAAe,CAAEP,eAAe,KAAYn+F,EAAMylF,WAAYkZ,EAAa,CAAER,eAAe,IACnHY,GAAoB38D,EAAQiO,EAAeiQ,EAAao+C,EAAcl4F,QAEtEw4F,GAAmB58D,EAAQiO,EAAeiQ,EAAao+C,EAAcl4F,OAEvE,CA1JGy4F,CAAe78D,EAAQiO,EAAeiQ,GAQtCgrB,EAAOyT,2BAA4B1uC,EAAc7pC,OAAO48B,cAAehB,IAGxE88D,GAAqB98D,EAAQvF,EAAWwT,IAKlClkC,EAAQ+xF,oBA8XhB,SAA8B5yB,EAAgBj+C,GAC7C,MAAM8xE,EAAgB7zB,EAAOyF,WAAY1jD,EAAU,SAC7C+xE,EAAqB9zB,EAAOyF,WAAY1jD,EAAU,aAExD,OAAQ8xE,GAAiBC,CAC1B,CAnYsCC,CAAqB/zB,EAAQj7B,IAChE0tD,GAAiB37D,EAAQiO,EAAexT,EAAWohE,GAGpD5tD,EAAc3nB,SACd43B,EAAY53B,QAAQ,GAEtB,CAyJA,SAASs2E,GACR58D,EACAiO,EACAiQ,EACAg/C,GAEA,MAAMrrD,EAAe5D,EAAc7pC,OAC7B0tC,EAAaoM,EAAY95C,OAG/B,GAAKytC,GAAgBqrD,GAAkBprD,GAAcorD,EAArD,CA4CA,IAvCAjvD,EAAgBjO,EAAOmd,oBAAqBtL,IAC5CqM,EAAcle,EAAOod,qBAAsBtL,IAGzBroB,QAASwkB,IAS1BjO,EAAO90B,OAAQ4mC,EAAY7D,GAe5BjO,EAAOo5D,MAAOnrD,GAWNiQ,EAAY95C,OAAOujC,SAAU,CACpC,MAAMw1D,EAAiBj/C,EAAY95C,OAEnC85C,EAAcle,EAAOod,qBAAsB+/C,GAE3Cn9D,EAAOv0B,OAAQ0xF,EAChB,CAGMd,GAAkBr8D,EAAOpiC,MAAMsrE,OAAQj7B,EAAeiQ,IAK5D0+C,GAAmB58D,EAAQiO,EAAeiQ,EAAag/C,EAxDvD,CAyDD,CAsBA,SAASP,GACR38D,EACAiO,EACAiQ,EACAg/C,GAEA,MAAMrrD,EAAe5D,EAAc7pC,OAC7B0tC,EAAaoM,EAAY95C,OAG/B,GAAKytC,GAAgBqrD,GAAkBprD,GAAcorD,EAArD,CA8BA,IAzBAjvD,EAAgBjO,EAAOmd,oBAAqBtL,IAC5CqM,EAAcle,EAAOod,qBAAsBtL,IAGzBroB,QAASwkB,IAS1BjO,EAAO90B,OAAQ2mC,EAAcqM,GAYtBjQ,EAAc7pC,OAAOujC,SAAU,CACtC,MAAMw1D,EAAiBlvD,EAAc7pC,OAErC6pC,EAAgBjO,EAAOod,qBAAsB+/C,GAE7Cn9D,EAAOv0B,OAAQ0xF,EAChB,CAGAj/C,EAAcle,EAAOod,qBAAsBtL,GA2B5C,SAAqB9R,EAAgB/U,GACpC,MAAM4mB,EAAoB5mB,EAAS0kB,WAC7BmC,EAAkB7mB,EAASykB,UAE5BmC,EAAarqC,MAAQsqC,EAAWtqC,MACpCw4B,EAAO6c,OAAQhL,EAAcC,EAAWtqC,MAGzCw4B,EAAOg5D,gBAAiBnnD,GACxB7R,EAAO+4D,cAAev8F,OAAOo2B,YAAakf,EAAWtG,iBAAmBqG,GAExE7R,EAAOo5D,MAAOnuE,EACf,CA1BCmyE,CAAYp9D,EAAQke,GAGdm+C,GAAkBr8D,EAAOpiC,MAAMsrE,OAAQj7B,EAAeiQ,IAK5Dy+C,GAAoB38D,EAAQiO,EAAeiQ,EAAag/C,EA1DxD,CA2DD,CAuBA,SAASb,GAAkBnzB,EAAgBj7B,EAAyBiQ,GACnE,MAAMrM,EAAe5D,EAAc7pC,OAC7B0tC,EAAaoM,EAAY95C,OAI/B,OAAKytC,GAAgBC,KAKhBo3B,EAAOI,QAASz3B,KAAkBq3B,EAAOI,QAASx3B,IAyCxD,SAAiCurD,EAAmBC,EAAoBp0B,GACvE,MAAMq0B,EAAe,IAAI,GAAOF,EAASC,GAEzC,UAAYt/F,KAASu/F,EAAaltD,YACjC,GAAK64B,EAAOI,QAAStrE,EAAMyH,MAC1B,OAAO,EAIT,OAAO,CACR,CA5CQ+3F,CAAwBvvD,EAAeiQ,EAAagrB,GAC5D,CA6CA,SAASyyB,GACR37D,EACA/U,EACAwP,EACA1vB,EAAa,CAAC,GAEd,MAAMyoE,EAAYxzC,EAAOl1B,cAAe,aAExCk1B,EAAOpiC,MAAMsrE,OAAOwT,qBAAsBlJ,EAAWzoE,EAAYi1B,GAEjEA,EAAO90B,OAAQsoE,EAAWvoD,GAE1B6xE,GAAqB98D,EAAQvF,EAAWuF,EAAOkd,iBAAkBs2B,EAAW,GAC7E,CAmCA,SAASspB,GACR98D,EACAvF,EACA8f,GAEK9f,aAAqB,GACzBuF,EAAOgZ,aAAcuB,GAErB9f,EAAUmN,MAAO2S,EAEnB,CCveA,SAASkjD,GAAoBnzE,EAAc0V,GAC1C,MAAM09D,EAAoD,GAE1Dv3F,MAAMrB,KAAMwlB,EAAMonB,SAAU,CAAE3nB,UAAW,cAGvCvnB,KAAKiD,GAAQu6B,EAAOsd,cAAe73C,KAKnC8F,QAAQoyF,IAGLA,EAAUz3D,MAAM7D,QAAS/X,EAAM4b,QAAWy3D,EAAUz3D,MAAMzc,QAASa,EAAM4b,UACzEy3D,EAAUx3D,IAAIjE,SAAU5X,EAAM6b,MAASw3D,EAAUx3D,IAAI1c,QAASa,EAAM6b,QAIvE/iC,SAASu6F,IACTD,EAAet4F,KAAMu4F,EAAUz3D,MAAM9hC,QAErC47B,EAAOv0B,OAAQkyF,EAAW,IAK5BD,EAAet6F,SAASw6F,IACvB,IAAIx5F,EAASw5F,EAEb,KAAQx5F,EAAOA,QAAUA,EAAOujC,SAAU,CACzC,MAAMk2D,EAAc79D,EAAOsd,cAAel5C,GAE1CA,EAASA,EAAOA,OAEhB47B,EAAOv0B,OAAQoyF,EAChB,IAEF,CCyEA,MAAMC,GA2EL,WAAAv2F,CAAa3J,EAAcoiC,EAAgB/U,GA7B3C,KAAQ8yE,WAA0B,KAKlC,KAAQC,UAAyB,KAKjC,KAAQC,mBAAqC,KAK7C,KAAQC,oBAAmC,GAK3C,KAAQC,eAAsC,KAK9C,KAAQC,aAAoC,KAE5C,KAAQC,cAA6B,KAGpC74F,KAAK5H,MAAQA,EACb4H,KAAKw6B,OAASA,EACdx6B,KAAKylB,SAAWA,EAChBzlB,KAAK84F,aAAe,IAAIllF,IAAK,CAAE5T,KAAKylB,SAAS7mB,SAC7CoB,KAAK0jE,OAAStrE,EAAMsrE,OAEpB1jE,KAAK+4F,kBAAoBv+D,EAAOkZ,yBAChC1zC,KAAKg5F,0BAA4Bx+D,EAAOkd,iBAAkB13C,KAAK+4F,kBAAmB,EACnF,CAOO,WAAAE,CAAa/xD,GACnB,UAAY/nC,KAAQwB,MAAMrB,KAAM4nC,GAC/BlnC,KAAKk5F,YAAa/5F,GAInBa,KAAKm5F,yBAGAn5F,KAAKy4F,oBACTz4F,KAAKo5F,iCAAkCp5F,KAAKy4F,oBAK7Cz4F,KAAKq5F,gBAGLr5F,KAAK0jE,OAAOyT,2BAA4Bn3E,KAAK04F,oBAAqB14F,KAAKw6B,QACvEx6B,KAAK04F,oBAAsB,EAC5B,CAOQ,gCAAAU,CAAkCj6F,GACzC,MAAMm6F,EAAwBt5F,KAAKw6B,OAAOmd,oBAAqB33C,KAAKw4F,WAC9De,EAAoBv5F,KAAKw6B,OAAOmd,oBAAqBx4C,GAG3D,GAAKo6F,EAAkB18D,QAASy8D,GAA0B,CAIzD,GAHAt5F,KAAKw4F,UAAYr5F,EAGZa,KAAKylB,SAAS7mB,QAAUO,IAASa,KAAKylB,SAAS+jB,QAInD,MAAM,IAAI,EAAe,2CAA4CxpC,MAGtEA,KAAKylB,SAAW8zE,EAChBv5F,KAAKw5F,uBAAwBx5F,KAAKylB,SACnC,CACD,CAMO,iBAAAg0E,GACN,OAAKz5F,KAAK64F,cACF,GAAMpsD,UAAWzsC,KAAK64F,eAGvB74F,KAAK5H,MAAMsrE,OAAOiE,yBAA0B3nE,KAAKylB,SACzD,CAMO,gBAAAi0E,GACN,OAAM15F,KAAK24F,eAIJ,IAAI,GAAO34F,KAAK24F,eAAgB34F,KAAK44F,cAHpC,IAIT,CAKO,OAAA5yE,GACDhmB,KAAK24F,gBACT34F,KAAK24F,eAAe73E,SAGhB9gB,KAAK44F,cACT54F,KAAK44F,aAAa93E,QAEpB,CAKQ,WAAAo4E,CAAa/5F,GAIpB,GAAKa,KAAK0jE,OAAOgE,SAAUvoE,GAG1B,YAFAa,KAAK25F,cAAex6F,GASrB,IAAIy6F,EAAY55F,KAAK65F,wCAAyC16F,GAExDy6F,IAGLA,EAAY55F,KAAK85F,gCAAiC36F,GAE5Cy6F,IAQP55F,KAAK+5F,kBAAmB56F,GAGlBa,KAAKu4F,aACVv4F,KAAKu4F,WAAap5F,GAGnBa,KAAKw4F,UAAYr5F,GAdfa,KAAKg6F,sBAAuB76F,EAe/B,CAKQ,sBAAAg6F,GACP,GAAKn5F,KAAK+4F,kBAAkB52D,QAC3B,OAGD,MAAM83D,EAAe,GAAarQ,aAAc5pF,KAAKylB,SAAU,UAE/DzlB,KAAKw5F,uBAAwBx5F,KAAKylB,UAK7BzlB,KAAK+4F,kBAAkBl9D,SAAU,IAAO77B,KAAKu4F,aACjDv4F,KAAKw6B,OAAO90B,OAAQ1F,KAAKu4F,WAAav4F,KAAKylB,UAI3CzlB,KAAKk6F,eAELl6F,KAAKylB,SAAWw0E,EAAatQ,cAIxB3pF,KAAK+4F,kBAAkB52D,SAC5BniC,KAAKw6B,OAAO90B,OAAQ1F,KAAK+4F,kBAAmB/4F,KAAKylB,UAGlDzlB,KAAKg5F,0BAA4Bh5F,KAAKw6B,OAAOkd,iBAAkB13C,KAAK+4F,kBAAmB,GAEvF/4F,KAAKylB,SAAWw0E,EAAatQ,aAC7BsQ,EAAan5E,QACd,CAKQ,aAAA64E,CAAex6F,GAEjBa,KAAK85F,gCAAiC36F,GAC1Ca,KAAK+5F,kBAAmB56F,GAIxBa,KAAKm6F,qBAAsBh7F,EAE7B,CAKQ,qBAAA66F,CAAuB76F,GAEzBA,EAAK+O,GAAI,WACblO,KAAKi5F,YAAa95F,EAAKq8B,eAIvBx7B,KAAKm6F,qBAAsBh7F,EAE7B,CAOQ,iBAAA46F,CAAmB56F,GAE1B,IAAMa,KAAK0jE,OAAOyF,WAAYnpE,KAAKylB,SAAUtmB,GAW5C,MAAM,IAAI,EACT,+BACAa,KACA,CAAEb,OAAMsmB,SAAUzlB,KAAKylB,WAIzBzlB,KAAKw6B,OAAO90B,OAAQvG,EAAMa,KAAKg5F,2BAC/Bh5F,KAAKg5F,0BAA4Bh5F,KAAKg5F,0BAA0B1uD,aAAcnrC,EAAKw+B,YAG9E39B,KAAK0jE,OAAOgE,SAAUvoE,KAAWa,KAAK0jE,OAAOyF,WAAYnpE,KAAKylB,SAAU,SAC5EzlB,KAAK64F,cAAgB15F,EAErBa,KAAK64F,cAAgB,KAGtB74F,KAAK04F,oBAAoB94F,KAAMT,EAChC,CASQ,sBAAAq6F,CAAwB/zE,GAIzBzlB,KAAK24F,iBACV34F,KAAK24F,eAAiB,GAAa/O,aAAcnkE,EAAU,eAOtDzlB,KAAK44F,eAAgB54F,KAAK44F,aAAal8D,SAAUjX,KACjDzlB,KAAK44F,cACT54F,KAAK44F,aAAa93E,SAGnB9gB,KAAK44F,aAAe,GAAahP,aAAcnkE,EAAU,UAE3D,CAQQ,YAAAy0E,GACP,MAAM/6F,EAAOa,KAAKu4F,WAElB,KAAQp5F,aAAgB,IACvB,OAGD,IAAMa,KAAKo6F,cAAej7F,GACzB,OAGD,MAAMk7F,EAAe,GAAatwD,cAAe5qC,GACjDk7F,EAAahjC,WAAa,SAE1B,MAAM4iC,EAAe,GAAarQ,aAAc5pF,KAAKylB,SAAU,UAc1DzlB,KAAK24F,eAAgB10E,QAASo2E,KAClCr6F,KAAK24F,eAAgB73E,SACrB9gB,KAAK24F,eAAiB,GAAa/vD,UAAWyxD,EAAalwD,WAAa,MAAO,eAY3EnqC,KAAKu4F,aAAev4F,KAAKw4F,YAC7Bx4F,KAAKu4F,WAAa8B,EAAalwD,WAC/BnqC,KAAKw4F,UAAY6B,EAAalwD,YAG/BnqC,KAAKw6B,OAAOo5D,MAAOyG,GAUdA,EAAap2E,QAASjkB,KAAK44F,eAAmB54F,KAAKu4F,aAAev4F,KAAKw4F,YAC3Ex4F,KAAK44F,aAAc93E,SACnB9gB,KAAK44F,aAAe,GAAahwD,UAAWyxD,EAAalwD,WAAa,MAAO,WAG9EnqC,KAAKylB,SAAWw0E,EAAatQ,aAC7BsQ,EAAan5E,SAIb9gB,KAAK04F,oBAAoB94F,KAAMI,KAAKylB,SAAS7mB,QAE7Cy7F,EAAav5E,QACd,CAQQ,aAAAu4E,GACP,MAAMl6F,EAAOa,KAAKw4F,UAElB,KAAQr5F,aAAgB,IACvB,OAGD,IAAMa,KAAKs6F,eAAgBn7F,GAC1B,OAGD,MAAMo7F,EAAgB,GAAa9wD,aAActqC,GAIjD,GAHAo7F,EAAcljC,WAAa,UAGrBr3D,KAAKylB,SAASxB,QAASs2E,GAa5B,MAAM,IAAI,EAAe,2CAA4Cv6F,MAKtEA,KAAKylB,SAAW,GAASmjB,UAAW2xD,EAAcpwD,WAAa,OAK/D,MAAM8vD,EAAe,GAAarQ,aAAc5pF,KAAKylB,SAAU,cAG1DzlB,KAAK44F,aAAc30E,QAASs2E,KAChCv6F,KAAK44F,aAAc93E,SACnB9gB,KAAK44F,aAAe,GAAahwD,UAAW2xD,EAAcpwD,WAAa,MAAO,WAY1EnqC,KAAKu4F,aAAev4F,KAAKw4F,YAC7Bx4F,KAAKu4F,WAAagC,EAAcpwD,WAChCnqC,KAAKw4F,UAAY+B,EAAcpwD,YAGhCnqC,KAAKw6B,OAAOo5D,MAAO2G,GAGdA,EAAcjwD,cAAe,GAAIrmB,QAASjkB,KAAK24F,iBAAqB34F,KAAKu4F,aAAev4F,KAAKw4F,YACjGx4F,KAAK24F,eAAgB73E,SACrB9gB,KAAK24F,eAAiB,GAAa/vD,UAAW2xD,EAAcpwD,WAAa,EAAG,eAG7EnqC,KAAKylB,SAAWw0E,EAAatQ,aAC7BsQ,EAAan5E,SAIb9gB,KAAK04F,oBAAoB94F,KAAMI,KAAKylB,SAAS7mB,QAE7C27F,EAAcz5E,QACf,CAOQ,aAAAs5E,CAAej7F,GACtB,MAAMynB,EAAkBznB,EAAKynB,gBAE7B,OAASA,aAA2B,IACnC5mB,KAAK84F,aAAalrF,IAAKgZ,IACvB5mB,KAAK5H,MAAMsrE,OAAOmS,WAAYjvD,EAAiBznB,EACjD,CAOQ,cAAAm7F,CAAgBn7F,GACvB,MAAM88B,EAAc98B,EAAK88B,YAEzB,OAASA,aAAuB,IAC/Bj8B,KAAK84F,aAAalrF,IAAKquB,IACvBj8B,KAAK5H,MAAMsrE,OAAOmS,WAAY12E,EAAM88B,EACtC,CAOQ,oBAAAk+D,CAAsBh7F,GAC7B,MAAM6uE,EAAYhuE,KAAKw6B,OAAOl1B,cAAe,aAKxCtF,KAAKw6F,cAAex6F,KAAKylB,SAAS7mB,OAAeovE,IAAehuE,KAAK0jE,OAAOyF,WAAY6E,EAAW7uE,KACvG6uE,EAAU/mC,aAAc9nC,GACxBa,KAAKk5F,YAAalrB,GAEpB,CASQ,uCAAA6rB,CAAyC16F,GAChD,GAAKa,KAAK0jE,OAAOyF,WAAYnpE,KAAKylB,SAAS7mB,OAAeO,GACzD,OAAO,EAMR,IAAMa,KAAK0jE,OAAOyF,WAAYnpE,KAAKylB,SAAS7mB,OAAe,eAAkBoB,KAAK0jE,OAAOyF,WAAY,YAAahqE,GACjH,OAAO,EAIRa,KAAKm5F,yBAGL,MAAMnrB,EAAYhuE,KAAKw6B,OAAOl1B,cAAe,aAQ7C,OANAtF,KAAKw6B,OAAO90B,OAAQsoE,EAAWhuE,KAAKylB,UACpCzlB,KAAKw5F,uBAAwBx5F,KAAKylB,UAElCzlB,KAAKy4F,mBAAqBzqB,EAC1BhuE,KAAKylB,SAAWzlB,KAAKw6B,OAAOkd,iBAAkBs2B,EAAW,IAElD,CACR,CAMQ,+BAAA8rB,CAAiC36F,GACxC,MAAM66E,EAAYh6E,KAAKw6F,cAAex6F,KAAKylB,SAAS7mB,OAAeO,GAEnE,IAAM66E,EACL,OAAO,EAQR,IAJKA,GAAah6E,KAAKylB,SAAS7mB,QAC/BoB,KAAKm5F,yBAGEnf,GAAah6E,KAAKylB,SAAS7mB,QAClC,GAAKoB,KAAKylB,SAASqkB,UAAY,CAG9B,MAAMlrC,EAAkBoB,KAAKylB,SAAS7mB,OAEtCoB,KAAKylB,SAAWzlB,KAAKw6B,OAAOod,qBAAsBh5C,GAW7CA,EAAOujC,SAAWvjC,EAAOA,SAAWo7E,GACxCh6E,KAAKw6B,OAAOv0B,OAAQrH,EAEtB,MAAO,GAAKoB,KAAKylB,SAAS+jB,QAGzBxpC,KAAKylB,SAAWzlB,KAAKw6B,OAAOmd,oBAAqB33C,KAAKylB,SAAS7mB,YACzD,CACN,MAAM67F,EAAUz6F,KAAKw6B,OAAOmd,oBAAqB33C,KAAKylB,SAAS7mB,QAE/DoB,KAAKw5F,uBAAwBx5F,KAAKylB,UAClCzlB,KAAKw6B,OAAOz9B,MAAOiD,KAAKylB,UAExBzlB,KAAKylB,SAAWg1E,EAEhBz6F,KAAK84F,aAAajrF,IAAK7N,KAAKylB,SAASykB,UACtC,CAGD,OAAO,CACR,CAQQ,aAAAswD,CAAeE,EAAyBC,GAC/C,OAAK36F,KAAK0jE,OAAOyF,WAAYuxB,EAAgBC,GACrCD,EAQH16F,KAAK0jE,OAAOI,QAAS42B,GAClB,KAGD16F,KAAKw6F,cAAeE,EAAe97F,OAAe+7F,EAC1D,ECt1BM,SAASC,GACf3lE,EACA78B,EACAyiG,EAAqC,QAErC,MAAM3qD,EAAkBjb,EAAU+Y,qBAElC,GAAKkC,GAAmB93C,EAAMsrE,OAAOgE,SAAUx3B,KAAsB93C,EAAMsrE,OAAOkE,SAAU13B,GAC3F,MAAc,UAAT2qD,GAA8B,SAATA,EAClBziG,EAAMy/C,YAAaz/C,EAAMs/C,iBAAkBxH,EAAiB2qD,IAG7DziG,EAAM0/C,cAAe5H,GAG7B,MAAM4qD,EAAa,GAAO7lE,EAAU0tC,qBAGpC,IAAMm4B,EACL,OAAO1iG,EAAMy/C,YAAa5iB,EAAUgY,OAKrC,GAAK6tD,EAAW34D,QACf,OAAO/pC,EAAMy/C,YAAaz/C,EAAMs/C,iBAAkBojD,EAAY,IAG/D,MAAMC,EAAgB3iG,EAAMu/C,oBAAqBmjD,GAGjD,OAAK7lE,EAAUgY,MAAOirB,WAAY6iC,GAC1B3iG,EAAMy/C,YAAakjD,GAIpB3iG,EAAMy/C,YAAaz/C,EAAMw/C,qBAAsBkjD,GACvD,CCzBe,SAASE,GACvB5iG,EACAgd,EACA64B,EACA1pC,EAGI,CAAC,GAEL,IAAMnM,EAAMsrE,OAAOgE,SAAUtyD,GAS5B,MAAM,IAAI,EAAe,qCAAsChd,EAAO,CAAEgd,WAIzE,MAAM6lF,EAAmDhtD,GAA0B71C,EAAMoL,SAASyxB,UAGlG,IAAIimE,EAAqBD,EAEpB12F,EAAQ42F,qBAAuB/iG,EAAMsrE,OAAOC,QAASvuD,KACzD8lF,EAAqB9iG,EAAM4/C,gBAAiB4iD,GAA2BK,EAAmB7iG,EAAOmM,EAAQ42F,uBAI1G,MAAMC,EAAqB,GAAOH,EAAkBt4B,qBAC9C04B,EAAmB,CAAC,EAM1B,OAJKD,GACJpkG,OAAO+wB,OAAQszE,EAAkBjjG,EAAMsrE,OAAO4T,0BAA2B8jB,EAAoB,iBAAiB,IAGxGhjG,EAAMmiC,QAAQC,IAGd0gE,EAAmBlwD,aACxB5yC,EAAM29F,cAAemF,EAAoB,CAAE5E,oBAAoB,IAGhE,IAAIgF,EAAkBlmF,EACtB,MAAMmmF,EAA0BL,EAAmBx/D,OAAQ98B,QAIzDxG,EAAMsrE,OAAOyF,WAAYoyB,EAAgCnmF,IAC1Dhd,EAAMsrE,OAAOyF,WAAYoyB,EAAgC,cACzDnjG,EAAMsrE,OAAOyF,WAAY,YAAa/zD,KAEtCkmF,EAAkB9gE,EAAOl1B,cAAe,aAExCk1B,EAAO90B,OAAQ0P,EAAQkmF,IAIxBljG,EAAMsrE,OAAOwT,qBAAsBokB,EAAiBD,EAAkB7gE,GAGtE,MAAMghE,EAAgBpjG,EAAMqjG,cAAeH,EAAiBJ,GAG5D,OAAKM,EAAcxwD,aAIdzmC,EAAQivC,cAkBf,SACChZ,EACAkgE,EACAG,EACAa,GAEA,MAAMtjG,EAAQoiC,EAAOpiC,MAErB,GAAc,MAATyiG,EAGJ,YAFArgE,EAAOgZ,aAAcknD,EAAgB,MAKtC,GAAc,SAATG,EASJ,MAAM,IAAI,EAAe,6CAA8CziG,GAGxE,IAAIujG,EAAcjB,EAAez+D,YAEjC,GAAK7jC,EAAMsrE,OAAOkE,SAAU8yB,GAG3B,YAFAlgE,EAAOgZ,aAAcknD,EAAgB,SAMtC,MAAMkB,EAAkBD,GAAevjG,EAAMsrE,OAAOyF,WAAYwyB,EAAa,UAGvEC,GAAmBxjG,EAAMsrE,OAAOyF,WAAYuxB,EAAe97F,OAAe,eAC/E+8F,EAAcnhE,EAAOl1B,cAAe,aAEpClN,EAAMsrE,OAAOwT,qBAAsBykB,EAAaD,EAAqBlhE,GACrEpiC,EAAMqjG,cAAeE,EAAanhE,EAAOmd,oBAAqB+iD,KAI1DiB,GACJnhE,EAAOgZ,aAAcmoD,EAAa,EAEpC,CAlEGE,CAAiBrhE,EAAQplB,EAAQ7Q,EAAQivC,aAAc6nD,GAJhDG,CAOY,GAEtB,CC1GA,MAAMM,GAAyB,cAqF/B,SAASC,GACRj5F,EAOAtK,GAEA,MAAM,UAAEwjG,EAAS,OAAEtlD,EAAM,KAAE/vB,EAAI,OAAE+8C,EAAM,uBAAEu4B,GAA2Bn5F,GAC9D,KAAEmI,EAAI,KAAEhL,EAAI,aAAEgqC,GAAiBzxC,EAIrC,GAAa,QAARyS,EACJ,MAAmB,SAAdnI,EAAK6jB,KAqEZ,SAAsC+vB,EAAoBslD,GACzD,IAAIv+D,EAAwBiZ,EAAOjxB,SAASgY,SAEtCA,IACLA,EAAWu+D,EAAYtlD,EAAOjxB,SAASykB,UAAYwM,EAAOjxB,SAAS0kB,YAGpE,KAAQ1M,GAAYA,EAASvvB,GAAI,UAAY,CAC5C,MAAM0N,EAAS86B,EAAOjxB,SAAS7J,OAAS6hB,EAASuM,YAKjD,GAAKkyD,GAAkBz+D,EAAU7hB,EAAQogF,GACxCv+D,EAAWu+D,EAAYtlD,EAAOjxB,SAASykB,UAAYwM,EAAOjxB,SAAS0kB,eACpE,IAEUgyD,GAAkB1+D,EAAS36B,KAAM8Y,EAAQogF,GAClD,MAIAtlD,EAAOp1C,MACR,CACD,CAEA,OAAOo1C,EAAOjxB,QACf,CA/FU22E,CAA6B1lD,EAAQslD,GAuC/C,SACCtlD,EACA/vB,EACAs1E,GAEA,MAAMx+D,EAAWiZ,EAAOjxB,SAASgY,SAEjC,GAAKA,EAAW,CACf,MAAM36B,EAAO26B,EAAS36B,KACtB,IAAI8Y,EAAS86B,EAAOjxB,SAAS7J,OAAS6hB,EAASuM,YAE/C,KACCtW,GAAuB5wB,EAAM8Y,IACnB,aAAR+K,GAAuBkN,GAAwB/wB,EAAM8Y,IACrDqgF,GAA0B/nE,GAAuBpxB,EAAM8Y,IAEzD86B,EAAOp1C,OAEPsa,EAAS86B,EAAOjxB,SAAS7J,OAAS6hB,EAASuM,WAE7C,CAEA,OAAO0M,EAAOjxB,QACf,CA3DS42E,CAAoB3lD,EAAQ/vB,EAAMs1E,GAI1C,GAAKhxF,IAAU+wF,EAAY,eAAiB,cAAiB,CAE5D,GAAKt4B,EAAO+O,aAAcxyE,GACzB,OAAO,GAAS2oC,UAAW3oC,EAAM+7F,EAAY,QAAU,UAIxD,GAAKt4B,EAAOyF,WAAYl/B,EAAc,SACrC,OAAOA,CAET,KAEK,CAEJ,GAAKy5B,EAAOI,QAAS7jE,GAIpB,YAFAy2C,EAAOxN,MAAM,KAAM,IAMpB,GAAKw6B,EAAOyF,WAAYl/B,EAAc,SACrC,OAAOA,CAET,CACD,CAgEA,SAASqyD,GAAgB57D,EAAiBs7D,GACzC,MAAM1lG,EAAOoqC,EAAMpqC,KACbimG,EAAY,GAAS3zD,UAAWtyC,EAAM0lG,EAAY,MAAQ,GAEhE,OAAKA,EACG,IAAI,GAAOt7D,EAAO67D,GAElB,IAAI,GAAOA,EAAW77D,EAE/B,CAKA,SAASy7D,GAAkBr5F,EAAc8Y,EAAgBogF,GAExD,MAAMQ,EAAgB5gF,GAAWogF,EAAY,GAAK,GAElD,OAAOF,GAAuBvsE,SAAUzsB,EAAKuqB,OAAQmvE,GACtD,CAKA,SAASN,GAAkBz+D,EAAgB7hB,EAAgBogF,GAC1D,OAAOpgF,KAAaogF,EAAYv+D,EAASE,WAAa,EACvD,CCxMe,MAAM,WAAcxqB,KA8BlC,WAAApR,GACCyL,QAEAxN,KAAK4/D,QAAU,IAAI4xB,GACnBxxF,KAAKwD,SAAW,IAAI,GAAUxD,MAC9BA,KAAK0jE,OAAS,IAAImR,GAElB70E,KAAKy8F,gBAAkB,GACvBz8F,KAAKm1F,eAAiB,KAEpB,CAAE,gBAAiB,kBAAmB,qBAAsB,kBAC5Dv3F,SAASsX,GAAclV,KAAKiV,SAAUC,KAIxClV,KAAKqP,GAA8B,kBAAkB,CAAEgG,EAAKnZ,KACzCA,EAAM,GAEdkkF,WAAW,GACnB,CAAExzE,SAAU,YAGf5M,KAAK0jE,OAAOwR,SAAU,QAAS,CAC9BpR,SAAS,IAGV9jE,KAAK0jE,OAAOwR,SAAU,aAAc,CACnCqD,QAAS,CAAE,QAAS,gBAGrBv4E,KAAK0jE,OAAOwR,SAAU,SAAU,CAC/BqD,QAAS,CAAE,QAAS,cACpB5U,SAAS,IAGV3jE,KAAK0jE,OAAOwR,SAAU,eAAgB,CACrC6D,WAAY,SACZpV,SAAS,EACT+D,UAAU,IAGX1nE,KAAK0jE,OAAOwR,SAAU,gBAAiB,CACtC6D,WAAY,QACZC,kBAAmB,QACnBpR,UAAU,EACVF,UAAU,IAGX1nE,KAAK0jE,OAAOwR,SAAU,QAAS,CAC9BqD,QAAS,SACT3Q,UAAU,EACV8N,WAAW,IAGZ11E,KAAK0jE,OAAOwR,SAAU,mBAAoB,CACzC4D,eAAgB,QAChBG,cAAe,QACfnV,SAAS,IAGV9jE,KAAK0jE,OAAOwR,SAAU,oBAAqB,CAC1C4D,eAAgB,QAChBG,cAAe,QACfnV,SAAS,IAOV9jE,KAAK0jE,OAAOwR,SAAU,WACtBl1E,KAAK0jE,OAAOsS,eAAe,CAAEzoE,EAASmvF,KACrC,GAA8B,YAAzBA,EAAgB16F,KACpB,OAAO,CACR,IAGD8uE,GAA0B9wE,MAG1BA,KAAKwD,SAASk3B,kBAAmBgzC,IAGjC1tE,KAAKqP,GAA6B,iBAAiB,CAAEgG,GAAOnV,EAAS+tC,MACpE54B,EAAIlE,OJvGQ,SACd/Y,EACA8H,EACA+tC,GAEA,OAAO71C,EAAMmiC,QAAQC,IACpB,MAAMvF,EAA2CgZ,GAA0B71C,EAAMoL,SAASyxB,UAEpFA,EAAU+V,aACf5yC,EAAM29F,cAAe9gE,EAAW,CAAEqhE,oBAAoB,IAGvD,MAAMqG,EAAY,IAAIrE,GAAWlgG,EAAOoiC,EAAQvF,EAAUyG,QACpDkhE,EAAqB,GAE3B,IAAIC,EAEJ,GAAK38F,EAAQgO,GAAI,oBAAuB,CAEvC,GAAKhO,EAAQ0/D,QAAQ/rD,KAAO,CAC3B,MAAMipF,EAAkB,GAExB,UAAc96F,EAAM8iB,KAAW5kB,EAAQ0/D,QAAU,CAChD,MAAM,MAAEl/B,EAAK,IAAEC,GAAQ7b,EACjBkmB,EAActK,EAAMzc,QAAS0c,GAEnCm8D,EAAgBl9F,KACf,CAAE6lB,SAAUib,EAAO1+B,OAAMgpC,eACzB,CAAEvlB,SAAUkb,EAAK3+B,OAAMgpC,eAEzB,CAIA8xD,EAAgB37E,MAAM,EAAIsE,SAAUs3E,IAAUt3E,SAAUu3E,KAAYD,EAAKrgE,SAAUsgE,GAAS,GAAK,IAEjG,UAAY,SAAEv3E,EAAQ,KAAEzjB,EAAI,YAAEgpC,KAAiB8xD,EAAkB,CAChE,IAAItjD,EAAc,KACdlH,EAAY,KAChB,MAAM2qD,EAAgBx3E,EAAS7mB,SAAWsB,GAAWulB,EAASqkB,UACxDN,EAAU/jB,EAAS7mB,SAAWsB,GAAWulB,EAAS+jB,QAWlDyzD,GAAkBzzD,EAGZwB,IAGXsH,EAAY2qD,EAAgB,QAAmB,QAL/CzjD,EAAchf,EAAOl1B,cAAe,WACpCk1B,EAAO90B,OAAQ8zC,EAAa/zB,IAO7Bm3E,EAAmBh9F,KAAM,CACxBoC,OACAyV,QAAS+hC,EACTlH,aAEF,CACD,CAEAuqD,EAAgB38F,EAAQs7B,aACzB,MACCqhE,EAAgB,CAAE38F,GAGnBy8F,EAAU1D,YAAa4D,GAEvB,IAAI1lD,EAAWwlD,EAAUlD,oBAEzB,GAAKv5F,EAAQgO,GAAI,qBAAwB0uF,EAAmBtkG,OAAS,CAIpE,MAAM4kG,EAAqB/lD,EAAW,GAAUgtB,UAAWhtB,GAAa,KAGlEgmD,EAA+C,CAAC,EAKtD,QAAU3iG,EAAIoiG,EAAmBtkG,OAAS,EAAGkC,GAAK,EAAGA,IAAM,CAC1D,MAAM,KAAEwH,EAAI,QAAEyV,EAAO,UAAE66B,GAAcsqD,EAAoBpiG,GACnD4iG,GAAmBD,EAAan7F,GAMtC,GAJKo7F,IACJD,EAAan7F,GAAS,IAGlByV,EAAU,CAEd,MAAM4lF,EAAkB7iE,EAAOkd,iBAAkBjgC,EAAS,UAE1D0lF,EAAan7F,GAAOpC,KAAMy9F,GAE1B7iE,EAAOv0B,OAAQwR,EAChB,KAAO,CAEN,MAAM6lF,EAAmBX,EAAUjD,mBAEnC,IAAM4D,EAAmB,CAInBhrD,GACJ6qD,EAAan7F,GAAOpC,KAAM+8F,EAAUl3E,UAGrC,QACD,CAEK6sB,EAGJ6qD,EAAan7F,GAAOpC,KAAM09F,EAAkBhrD,IAE5C6qD,EAAan7F,GAAOpC,KAAMw9F,EAAkBE,EAAiB58D,MAAQ48D,EAAiB38D,IAExF,CACD,CAEA,UAAc3+B,GAAQ0+B,EAAOC,MAAW3pC,OAAOyhB,QAAS0kF,GAGlDz8D,GAASC,GAAOD,EAAMpqC,OAASqqC,EAAIrqC,MACvCkkC,EAAOu4D,UAAW/wF,EAAM,CACvB6wF,gBAAgB,EAChB79D,aAAa,EACblQ,MAAO,IAAI,GAAO4b,EAAOC,KAKvBu8D,IACJ/lD,EAAW+lD,EAAmBh5B,UAC9Bg5B,EAAmBp8E,SAErB,CAGKq2B,IACCliB,aAAqB,GACzBuF,EAAOgZ,aAAc2D,GAErBliB,EAAUmN,MAAO+U,IASnB,MAAMqkD,EAAgBmB,EAAUjD,oBAAsBthG,EAAMy/C,YAAa5iB,EAAUyG,QAInF,OAFAihE,EAAU32E,UAEHw1E,CAAa,GAEtB,CIhEgBC,CAAez7F,KAAME,EAAS+tC,EAAY,IAIxDjuC,KAAKqP,GAA4B,gBAAgB,CAAEgG,GAAOoC,EAASwd,EAAW1wB,MAC7E8Q,EAAIlE,OAAS6pF,GAAch7F,KAAMyX,EAASwd,EAAW1wB,EAAS,IAI/DvE,KAAKqP,GAAyB,aAAagG,IAC1C,MAAMggB,GAAar1B,KAAKwD,SAAS8xB,WAEjCjgB,EAAIlE,OAASkkB,EAEPA,GAELhgB,EAAIlJ,MACL,GAUF,CA2CO,MAAAouB,CAAiBhrB,GACvB,IACC,OAAqC,IAAhCvP,KAAKy8F,gBAAgBnkG,QAEzB0H,KAAKy8F,gBAAgB78F,KAAM,CAAEsmE,MAAO,IAAI4jB,GAASv6E,aAE1CvP,KAAKu9F,qBAAsB,IAG3BhuF,EAAUvP,KAAKm1F,eAExB,CAAE,MAAQ3zF,GAGT,EAAc2M,uBAAwB3M,EAAKxB,KAC5C,CACD,CAwFO,aAAAomE,CACNo3B,EACAjuF,GAEA,IACOiuF,EAE6B,mBAAhBA,GAClBjuF,EAAWiuF,EACXA,EAAc,IAAI1T,IACJ0T,aAAuB1T,KACrC0T,EAAc,IAAI1T,GAAO0T,IALzBA,EAAc,IAAI1T,GAQnB9pF,KAAKy8F,gBAAgB78F,KAAM,CAAEsmE,MAAOs3B,EAAajuF,aAEb,GAA/BvP,KAAKy8F,gBAAgBnkG,QACzB0H,KAAKu9F,oBAEP,CAAE,MAAQ/7F,GAGT,EAAc2M,uBAAwB3M,EAAKxB,KAC5C,CACD,CAaO,cAAA2yF,CAAgBl6B,GAMtBA,EAAUopB,UACX,CA4JO,aAAA4Z,CACNv7F,EACA+tC,EACAC,KACGuvD,GAEH,MAAMxoE,EAAYyoE,GAAqBzvD,EAAYC,GAGnD,OAAOluC,KAAKwQ,KAA+B,gBAAiB,CAAEtQ,EAAS+0B,EAAWiZ,KAAkBuvD,GACrG,CAqFO,YAAAzC,CACNvjF,EACAw2B,EACAC,EACA3pC,KAIGk5F,GAEH,MAAMxoE,EAAYyoE,GAAqBzvD,EAAYC,GAInD,OAAOluC,KAAKwQ,KAA8B,eAAgB,CAAEiH,EAASwd,EAAW1wB,EAASA,KAAYk5F,GACtG,CAiDO,aAAA1H,CACN9gE,EACA1wB,GAQAwxF,GAAe/1F,KAAMi1B,EAAW1wB,EACjC,CA8BO,eAAAiyF,CACNvhE,EACA1wB,IDxqBa,SACdnM,EACA68B,EACA1wB,EAII,CAAC,GAEL,MAAMm/D,EAAStrE,EAAMsrE,OACfs4B,EAAiC,YAArBz3F,EAAQggB,UACpBoC,EAAOpiB,EAAQoiB,KAAOpiB,EAAQoiB,KAAO,YACrCs1E,IAA2B13F,EAAQ03F,uBAEnChvD,EAAQhY,EAAUgY,MAElByJ,EAAS,IAAI,GAAY,CAC9BlO,WAAY8zD,GAAgBrvD,EAAO+uD,GACnCnzD,kBAAkB,EAClBtkB,UAAWy3E,EAAY,UAAY,aAG9Bl5F,EAAO,CAAE4zC,SAAQgtB,SAAQs4B,YAAWr1E,OAAMs1E,0BAEhD,IAAI36F,EAEJ,KAAUA,EAAOo1C,EAAOp1C,QAAW,CAClC,GAAKA,EAAKC,KACT,OAGD,MAAMkkB,EAAWs2E,GAAgBj5F,EAAMxB,EAAK9I,OAE5C,GAAKitB,EASJ,YARKwP,aAAqB,GACzB78B,EAAMmiC,QAAQC,IACbA,EAAOiZ,kBAAmBhuB,EAAU,IAGrCwP,EAAUwZ,SAAUhpB,GAKvB,CACD,CCioBE+wE,CAAiBx2F,KAAMi1B,EAAW1wB,EACnC,CA6BO,kBAAAo5F,CAAoB1oE,GAC1B,OLnuBa,SACd78B,EACA68B,GAEA,OAAO78B,EAAMmiC,QAAQC,IACpB,MAAMojE,EAAOpjE,EAAOkZ,yBACd5uB,EAAQmQ,EAAUoY,gBAExB,IAAMvoB,GAASA,EAAMkmB,YACpB,OAAO4yD,EAGR,MAAMtnG,EAAOwuB,EAAM4b,MAAMpqC,KACnBunG,EAAa/4E,EAAM4b,MAAMs3B,cAAelzC,EAAM6b,KAC9Cm9D,EAAexnG,EAAK4gE,cAAe2mC,GAezC,IAAIE,EAIHA,EAFIj5E,EAAM4b,MAAM9hC,QAAUkmB,EAAM6b,IAAI/hC,OAEjBkmB,EAEA0V,EAAOqd,YACzBrd,EAAOkd,iBAAkBomD,EAAch5E,EAAM4b,MAAMjhC,KAAMo+F,EAAWvlG,SACpEkiC,EAAOkd,iBAAkBomD,EAAch5E,EAAM6b,IAAIlhC,KAAMo+F,EAAWvlG,QAAW,IAI/E,MAAM6S,EAAU4yF,EAAiBp9D,IAAI/kB,OAASmiF,EAAiBr9D,MAAM9kB,OAGrE,UAAY3b,KAAQ89F,EAAiB7xD,SAAU,CAAEpD,SAAS,IACpD7oC,EAAKiO,GAAI,cACbssB,EAAO04D,WAAYjzF,EAAK6C,KAAM7C,EAAK+lC,gBAAiB43D,GAEpDpjE,EAAOqrB,OAAQrrB,EAAOg4D,aAAcvyF,GAAiB,GAAQ29F,GAmB/D,GAAKG,GAAoBj5E,EAAQ,CAEhC,MAAMqyB,EAAWryB,EAAMi0C,sBAAuBglC,EAAiBr9D,MAAOlG,EAAOkd,iBAAkBkmD,EAAM,GAAKzyF,GAAW,GAE/G6yF,EAAkBxjE,EAAOqd,YAAard,EAAOkd,iBAAkBkmD,EAAM,GAAKzmD,EAASzW,OAGzFu3D,GAFyBz9D,EAAOqd,YAAaV,EAASxW,IAAKnG,EAAOkd,iBAAkBkmD,EAAM,QAEpDpjE,GACtCy9D,GAAoB+F,EAAiBxjE,EACtC,CAEA,OAAOojE,CAAI,GAEb,CKmpBSD,CAAoB39F,KAAMi1B,EAClC,CAsBO,UAAA4oD,CACNogB,EACA15F,EAGI,CAAC,GAEL,MAAMugB,EAAQm5E,aAA0B,GAAaA,EAAiB,GAAWzxD,UAAWyxD,GAE5F,GAAKn5E,EAAMkmB,YACV,OAAO,EAGR,MAAM,kBAAE8yC,GAAoB,EAAK,cAAEyY,GAAgB,GAAUhyF,EAG7D,IAAMgyF,EACL,UAAY2H,KAAsBl+F,KAAK4/D,QAAQ0rB,4BAA6BxmE,GAC3E,GAAKo5E,EAAmBlpE,YACvB,OAAO,EAKV,UAAY/0B,KAAQ6kB,EAAMonB,WACzB,GAAKlsC,KAAK0jE,OAAOgS,UAAWz1E,GAAS,CACpC,IAAKA,EAAKiO,GAAI,cAOb,OAAO,EANP,IAAM4vE,EACL,OAAO,EACD,IAAmC,IAA9B79E,EAAK6C,KAAK4G,OAAQ,MAC7B,OAAO,CAKV,CAGD,OAAO,CACR,CAeO,SAAA2rB,CAAW4Y,GACjB,MAAMhZ,EAAYyoE,GAAqBzvD,GAEvC,OAAOjuC,KAAKwQ,KAA2B,YAAa,CAAEykB,GACvD,CAYO,sBAAAo8D,CACN/6F,EACAmJ,EACA43D,GAEA,OAAO,IAAI,GAAe/gE,EAAMmJ,EAAM43D,EACvC,CAsBO,gBAAA3f,CACN5M,EACAlvB,GAEA,OAAO,GAAcgtB,UAAWkC,EAAgBlvB,EACjD,CAUO,mBAAA+7B,CAAqB13C,GAC3B,OAAO,GAAcwpC,aAAcxpC,EACpC,CAUO,oBAAA23C,CAAsB33C,GAC5B,OAAO,GAAc8pC,cAAe9pC,EACrC,CAiBO,WAAA43C,CAAanX,EAAsBC,GACzC,OAAO,IAAI,GAAYD,EAAOC,EAC/B,CAiBO,aAAAoX,CAAetgC,GACrB,OAAO,GAAW+0B,UAAW/0B,EAC9B,CAgBO,aAAAqgC,CAAe73C,GACrB,OAAO,GAAWwsC,UAAWxsC,EAC9B,CA+EO,eAAA+3C,IAAoB97C,GAC1B,OAAO,IAAI,MAAmBA,EAC/B,CAYO,WAAAiiG,CAAalzF,GACnB,OAAO,IAAI6+E,GAAO7+E,EACnB,CASO,uBAAAmzF,CAAyBnhE,GAC/B,OAAOkmD,GAAiBlsB,SAAUh6B,EAAMj9B,KAAKwD,SAC9C,CAKO,OAAAwiB,GACNhmB,KAAKwD,SAASwiB,UACdhmB,KAAK2P,eACN,CAMQ,kBAAA4tF,GACP,MAAMc,EAAM,GAEZr+F,KAAKwQ,KAAM,kBAEX,IACC,KAAQxQ,KAAKy8F,gBAAgBnkG,QAAS,CAErC,MAAMgmG,EAAet+F,KAAKy8F,gBAAiB,GAAIv2B,MAC/ClmE,KAAKm1F,eAAiB,IAAI5C,GAAQvyF,KAAMs+F,GAGxC,MAAMC,EAAsBv+F,KAAKy8F,gBAAiB,GAAIltF,SAAUvP,KAAKm1F,gBACrEkJ,EAAIz+F,KAAM2+F,GAEVv+F,KAAKwD,SAASytF,mBAAoBjxF,KAAKm1F,gBAEvCn1F,KAAKy8F,gBAAgBtwE,QACrBnsB,KAAKm1F,eAAiB,IACvB,CACD,CAAE,QACDn1F,KAAKy8F,gBAAgBnkG,OAAS,EAC9B0H,KAAKm1F,eAAiB,KAEtBn1F,KAAKwQ,KAAM,gBACZ,CAEA,OAAO6tF,CACR,EAMD,SAASX,GACRzvD,EACAC,GAEA,GAAMD,EAIN,OAAKA,aAAsB,IAAkBA,aAAsB,GAC3DA,EAGHA,aAAsB,GACrBC,GAAmC,IAAlBA,EACd,IAAI,GAAgBD,EAAYC,GAC5BD,EAAW//B,GAAI,eACnB,IAAI,GAAgB+/B,EAAY,MAEhC,IAAI,GAAgBA,EAAY,MAIlC,IAAI,GAAgBA,EAC5B,CClnCe,MAAMuwD,WAAsBrzC,GAA3C,kCAIC,KAAgBC,aAAe,QAKxB,UAAAC,CAAYJ,GAClBjrD,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAC3B,ECZc,MAAM,WAAsBE,GAA3C,kCAIC,KAAgBC,aAAe,CAAE,YAAa,UAAW,YAAa,WAAW,CAK1E,UAAAC,CAAYJ,GAClBjrD,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAC3B,ECiBc,MAAM,GASpB,WAAAlpD,CAAayB,GACZxD,KAAKwD,SAAWA,CACjB,CAQO,sBAAAkwC,CAAwBp8B,GAC9B,OAAO,IAAI+7B,GAAkBrzC,KAAKwD,SAAU8T,EAC7C,CAkBO,aAAAhS,CACNtD,EACAojC,EACA9tB,GAEA,OAAO,IAAI,GAAStX,KAAKwD,SAAUxB,EAAMojC,EAAO9tB,EACjD,CAQO,UAAAq8B,CAAY7wC,GAClB,OAAO,IAAIo6B,GAAMl9B,KAAKwD,SAAUV,EACjC,CAWO,KAAAigB,CAAOtL,EAAkBovB,GAAgB,GAC/C,OAAOpvB,EAAQ8lB,OAAQsJ,EACxB,CAWO,WAAAlhC,CAAairB,EAAgDnZ,GACnE,OAAOA,EAAQwvB,aAAcrW,EAC9B,CAYO,WAAA6tE,CAAa75F,EAAegsB,EAA8BnZ,GAChE,OAAOA,EAAQkuB,aAAc/gC,EAAOgsB,EACrC,CAWO,cAAA8tE,CAAgB95F,EAAeuG,EAAiBsM,GACtD,OAAOA,EAAQqlB,gBAAiBl4B,EAAOuG,EACxC,CAQO,MAAAlF,CAAQwR,GACd,MAAM7Y,EAAS6Y,EAAQ7Y,OAEvB,OAAKA,EACGoB,KAAK0+F,eAAgB9/F,EAAOo9B,cAAevkB,GAAW,EAAG7Y,GAG1D,EACR,CASO,OAAAqa,CAAS0lF,EAAqBxpD,GACpC,MAAMv2C,EAAS+/F,EAAW//F,OAE1B,GAAKA,EAAS,CACb,MAAMgG,EAAQhG,EAAOo9B,cAAe2iE,GAKpC,OAHA3+F,KAAK0+F,eAAgB95F,EAAO,EAAGhG,GAC/BoB,KAAKy+F,YAAa75F,EAAOuwC,EAAYv2C,IAE9B,CACR,CAEA,OAAO,CACR,CAQO,aAAAu6C,CAAe1hC,GACrB,MAAM7Y,EAAS6Y,EAAQ7Y,OAEvB,GAAKA,EAAS,CACb,MAAMgG,EAAQhG,EAAOo9B,cAAevkB,GAEpCzX,KAAKiG,OAAQwR,GACbzX,KAAKy+F,YAAa75F,EAAO6S,EAAQ+jB,cAAe58B,EACjD,CACD,CAYO,MAAAy4C,CAAQC,EAAiB7/B,GAC/B,MAAM09B,EAAa,IAAI,GAASn1C,KAAKwD,SAAU8zC,EAAS7/B,EAAQuuB,gBAAiBvuB,EAAQ+jB,eAEzF,OAAOx7B,KAAKiZ,QAASxB,EAAS09B,GAAeA,EAAa,IAC3D,CAcO,YAAA1vC,CAAc1O,EAAayB,EAAgBif,GACjDA,EAAQ2vB,cAAerwC,EAAKyB,EAC7B,CAaO,eAAAkO,CAAiB3P,EAAa0gB,GACpCA,EAAQ6vB,iBAAkBvwC,EAC3B,CAcO,QAAA6kC,CAAUuK,EAAmC1uB,GACnDA,EAAQ8vB,UAAWpB,EACpB,CAcO,WAAApL,CAAaoL,EAAmC1uB,GACtDA,EAAQ+vB,aAAcrB,EACvB,CA0CO,QAAAuO,CAAUthC,EAA2CwrF,EAAkCnnF,GACxF,GAAerE,SAA0B,IAAZqE,EAC/BmnF,EAA4Bn3D,UAAWr0B,GAEzCqE,EAASgwB,UAAWr0B,EAAoBwrF,EAE1C,CAkBO,WAAAjqD,CAAavhC,EAAkCqE,GACrDA,EAAQiwB,aAAct0B,EACvB,CAWO,iBAAAwhC,CAAmB79C,EAAsByB,EAAgBif,GAC/DA,EAAQkwB,mBAAoB5wC,EAAKyB,EAClC,CAUO,oBAAAq8C,CAAsB99C,EAAsB0gB,GAClD,OAAOA,EAAQmwB,sBAAuB7wC,EACvC,CAiBO,gBAAA2gD,CAAkB5M,EAAiClvB,GACzD,OAAO+sB,GAASC,UAAWkC,EAAgBlvB,EAC5C,CAOO,mBAAA+7B,CAAqB13C,GAC3B,OAAO0oC,GAASc,aAAcxpC,EAC/B,CAOO,oBAAA23C,CAAsB33C,GAC5B,OAAO0oC,GAASoB,cAAe9pC,EAChC,CAUO,WAAA43C,CAAanX,EAAiBC,GACpC,OAAO,IAAIoK,GAAOrK,EAAOC,EAC1B,CAKO,aAAAmX,CAAe73C,GACrB,OAAO8qC,GAAM0B,UAAWxsC,EACzB,CAQO,aAAA83C,CAAetgC,GACrB,OAAOszB,GAAMyB,UAAW/0B,EACzB,CAgGO,eAAAugC,IAAoB97C,GAC1B,OAAO,IAAIwwC,MAAcxwC,EAC1B,EClhBc,MAAM2iG,GASpB,WAAA98F,GACC/B,KAAK8+F,UAAY,IAAIvtF,GACtB,CAOO,GAAA1D,CACN+rB,EACAlE,GAEA11B,KAAK8+F,UAAUrtF,IAAKmoB,EAAalE,EAClC,CAOO,GAAAttB,CAA2BwxB,GACjC,OAAO55B,KAAK8+F,UAAU12F,IAAKwxB,EAC5B,CASO,OAAArE,CACNqE,KACGmlE,GAEH,MAAMrpE,EAAU11B,KAAKoI,IAAKwxB,GAE1B,IAAMlE,EAOL,MAAM,IAAI,EAAe,sCAAuC11B,KAAM,CAAE45B,gBAGzE,OAAOlE,EAAQH,WAAYwpE,EAC5B,CAKA,MAAQpgF,SACA3e,KAAK8+F,UAAU7nG,MACvB,CAKA,SAAQ+nG,SACAh/F,KAAK8+F,UAAU5zF,QACvB,CAOA,CAASpK,OAAOC,YACf,OAAOf,KAAK8+F,UAAWh+F,OAAOC,WAC/B,CAKO,OAAAilB,GACN,UAAY0P,KAAW11B,KAAKg/F,WAC3BtpE,EAAQ1P,SAEV,ECnDc,MAAei5E,WAAe9rF,KA0N5C,WAAApR,CAAainB,EAAuB,CAAC,GACpCxb,QAEA,MAAMzL,EAAc/B,KAAK+B,YAGnB2sB,EAAW1F,EAAO0F,UAAc3sB,EAAYg3B,eAAiBh3B,EAAYg3B,cAAcrK,SAE7F1uB,KAAKm2B,SAAWnN,EAAOzb,SAAW,IAAIsrB,GAAS,CAAEnK,aACjD1uB,KAAKm2B,SAASmD,WAAYt5B,MAAOgpB,EAAOzb,SAIxC,MAAMyoB,EAAmBr1B,MAAMrB,KAAMyC,EAAYi3B,gBAAkB,IAEnEh5B,KAAKgpB,OAAS,IAAI7K,GAAsB6K,EAAQjnB,EAAYg3B,eAC5D/4B,KAAKgpB,OAAOtyB,OAAQ,UAAWs/B,GAC/Bh2B,KAAKgpB,OAAOtyB,OAAQsJ,KAAKm2B,SAASsD,oBAElCz5B,KAAK22B,QAAU,IAAIZ,GAA0B/1B,KAAMg2B,EAAkBh2B,KAAKm2B,SAASQ,SAEnF32B,KAAKk5B,OAASl5B,KAAKm2B,SAAS+C,OAC5Bl5B,KAAKhF,EAAIgF,KAAKk5B,OAAOl+B,EAErBgF,KAAKk/F,eAAiB,IAAItrF,IAE1B5T,KAAKg/F,SAAW,IAAIH,GAEpB7+F,KAAKyR,IAAK,QAAS,gBACnBzR,KAAKyP,KAAwB,SAAS,IAAQzP,KAAK2tF,MAAQ,SAAW,CAAE/gF,SAAU,SAClF5M,KAAKyP,KAA0B,WAAW,IAAQzP,KAAK2tF,MAAQ,aAAe,CAAE/gF,SAAU,SAE1F5M,KAAK5H,MAAQ,IAAI,GAEjB4H,KAAKqP,GAAI,qBAAqB,KAC7BrP,KAAK5H,MAAMoL,SAAS8xB,WAAat1B,KAAKs1B,UAAU,IAGjD,MAAMyQ,EAAkB,IAAIpC,GAE5B3jC,KAAK8C,KAAO,IAAIw6E,GAAgBt9E,KAAK5H,MAAO2tC,GAE5C/lC,KAAKm/F,QAAU,IAAI/rB,GAAmBpzE,KAAK5H,MAAO2tC,GAClD/lC,KAAKm/F,QAAQllE,KAAKz2B,SAASwD,KAAM,cAAezH,GAAIS,MAEpDA,KAAK3B,WAAa,IAAI6gF,GAAY,CAAEl/E,KAAKm/F,QAAQ9rB,mBAAoBrzE,KAAK8C,KAAKuwE,oBAAsBrzE,KAAK8C,KAAKy6E,kBAC/Gv9E,KAAK3B,WAAWqhF,SAAU,eAAgB1/E,KAAK8C,KAAKuwE,oBACpDrzE,KAAK3B,WAAWqhF,SAAU,kBAAmB1/E,KAAKm/F,QAAQ9rB,oBAE1DrzE,KAAKo/F,WAAa,IAAIzlE,GAAyB35B,MAC/CA,KAAKo/F,WAAW5vF,SAAUxP,KAAKm/F,QAAQllE,KAAKz2B,SAC7C,CAuBA,cAAW8xB,GACV,OAAOt1B,KAAKk/F,eAAerrF,KAAO,CACnC,CAEA,cAAWyhB,CAAY98B,GAsBtB,MAAM,IAAI,EAAe,kCAC1B,CA8CO,kBAAA6mG,CAAoBC,GAC1B,GAAuB,iBAAXA,GAAyC,iBAAXA,EAMzC,MAAM,IAAI,EAAe,mCAAoC,KAAM,CAAEA,WAGjEt/F,KAAKk/F,eAAetxF,IAAK0xF,KAI9Bt/F,KAAKk/F,eAAerxF,IAAKyxF,GAES,IAA7Bt/F,KAAKk/F,eAAerrF,MAExB7T,KAAKwQ,KAAsC,oBAAqB,cAAc,GAAM,GAEtF,CASO,mBAAA+uF,CAAqBD,GAC3B,GAAuB,iBAAXA,GAAyC,iBAAXA,EACzC,MAAM,IAAI,EAAe,mCAAoC,KAAM,CAAEA,WAGhEt/F,KAAKk/F,eAAetxF,IAAK0xF,KAI/Bt/F,KAAKk/F,eAAevtF,OAAQ2tF,GAEM,IAA7Bt/F,KAAKk/F,eAAerrF,MAExB7T,KAAKwQ,KAAsC,oBAAqB,cAAc,GAAO,GAEvF,CAOO,WAAAqnB,GACN,MAAM7O,EAAShpB,KAAKgpB,OACd2N,EAAU3N,EAAO5gB,IAAK,WACtBo3F,EAAgBx2E,EAAO5gB,IAAK,kBAAqB,GACjDq3F,EAAez2E,EAAO5gB,IAAK,iBAAoB,GAC/CsvB,EAAoB1O,EAAO5gB,IAAK,sBAAyB,GAE/D,OAAOpI,KAAK22B,QAAQD,KAAMC,EAAQx2B,OAAQs/F,GAAgBD,EAAe9nE,EAC1E,CAWO,OAAA1R,GACN,IAAI05E,EAAiCjnE,QAAQlc,UAM7C,MAJmB,gBAAdvc,KAAK2tF,QACT+R,EAAe,IAAIjnE,SAASlc,GAAWvc,KAAKyP,KAAwB,QAAS8M,MAGvEmjF,EACL5nE,MAAM,KACN93B,KAAKwQ,KAA0B,WAC/BxQ,KAAK2P,gBACL3P,KAAKg/F,SAASh5E,SAAS,IAEvB8R,MAAM,IAAM93B,KAAK22B,QAAQ3Q,YACzB8R,MAAM,KACN93B,KAAK5H,MAAM4tB,UACXhmB,KAAK8C,KAAKkjB,UACVhmB,KAAKm/F,QAAQn5E,UACbhmB,KAAKo/F,WAAWp5E,SAAS,IAIzB8R,MAAM,IAAM93B,KAAKm2B,SAASqD,cAAex5B,OAC5C,CAeO,OAAAu1B,CACNqE,KACGmlE,GAEH,IACC,OAAO/+F,KAAKg/F,SAASzpE,QAASqE,KAAgBmlE,EAC/C,CAAE,MAAQv9F,GAGT,EAAc2M,uBAAwB3M,EAAKxB,KAC5C,CACD,CAWO,KAAAitC,GACNjtC,KAAKm/F,QAAQllE,KAAKgT,OACnB,CAeA,aAAc3vB,IAAWphB,GACxB,MAAM,IAAI7D,MAAO,8BAClB,ECxiBc,SAASsnG,GAAgDj7F,GAevE,OAdA,cAA6BA,EACrB,OAAAkuD,CAAS9vD,GACf9C,KAAK8C,KAAK2O,IAAK3O,EAChB,CAEO,OAAA6vD,CAASpuD,GAKf,OAAOvE,KAAK8C,KAAKsF,IAAK7D,EACvB,EAIF,CAGA,CACC,MAAMosC,EAAUgvD,GAAuB3oG,QAErC2oG,GAAsB/sC,QAAUjiB,EAAM9uC,UAAU+wD,QAChD+sC,GAAsBhtC,QAAUhiB,EAAM9uC,UAAU8wD,OACnD,CCnBe,SAASitC,GAAmDl7F,GAqC1E,OApCA,cAA6BA,EAGrB,mBAAAm7F,CAAqB/8F,EAAe9C,KAAK8C,KAAKsF,OACpD,IAAMpI,KAAKyhF,cASV,MAAM,IAAI,EACT,+BACAzhF,MAIF,MAAM8/F,EAA4B9/F,KAAKgpB,OAAO5gB,IAAK,gCAC7C23F,EAA0B//F,KAAKyhF,yBAAyBj7D,oBAY9DF,GAAkBtmB,KAAKyhF,cANjBqe,GAA8BC,EAMEj9F,EALC,GAMxC,EAIF,CAGE88F,GAAyBC,oBAAwBD,GAA0B5oG,QAAS6K,UAAUg+F,oBCEjF,MAAMG,WAAuBtmE,GAiB3C,qBAAkBpD,GACjB,MAAO,gBACR,CAKO,IAAAI,GACN12B,KAAKyR,IAAK,UAAU,GAEpBzR,KAAKigG,SAAW,IAAI,GAAY,CAAE9vE,WAAY,QAC9CnwB,KAAKigG,SAAS7uF,SAAU,MAAO,UAAW7R,GAAIS,KAC/C,CAWO,GAAA6N,CAAKO,GACX,GAAwB,iBAAZA,EAMX,MAAM,IAAI,EAAe,qCAAsCpO,MAGhE,MAAM+9C,EAAS,IAAM5qC,KAMrB,OAJA4qC,EAAOtsC,IAAK,UAAWrD,GACvBpO,KAAKigG,SAASpyF,IAAKkwC,GACnB/9C,KAAKkgG,QAAS,EAEPniD,CACR,CAOO,MAAA93C,CAAQ83C,GACd/9C,KAAKigG,SAASh6F,OAAQ83C,GACtB/9C,KAAKkgG,SAAWlgG,KAAKigG,SAAS3nG,MAC/B,CAOA,SAAWm4B,GACV,OAAOzwB,KAAKigG,SAAS73F,IAAK,EAC3B,CAKA,CAAStH,OAAOC,YACf,OAAOf,KAAKigG,SAAUn/F,OAAOC,WAC9B,ECnJD,MC2Ea,GAAQ,CACpBo/F,KC5ED,suBD6EC1sE,OD7ED,wRC8EC2sE,QE9ED,sVF+ECC,MG/ED,8OHgFCC,IIhFD,ymBJiFCC,OKjFD,qVLkFCvtC,MMlFD,2XNmFCwtC,UOnFD,6lDPoFCC,aQpFD,wkBRqFCzyB,USrFD,yMTsFC0yB,KUtFD,+KVuFCxmE,KWvFD,qXXyFCymE,YYzFD,4TZ0FCC,Ya1FD,qiBb2FCC,Sc3FD,+Td4FCC,Ue5FD,8Xf6FCC,YgB7FD,oYhB8FCC,WiB9FD,iYjB+FCC,akB/FD,6XlBiGCC,WmBjGD,4SnBkGCC,aoBlGD,sPpBmGCC,YqBnGD,6RrBoGCC,gBsBpGD,8OtBqGCC,auBrGD,0QvBsGCC,gBwBtGD,sPxBuGCC,iByBvGD,sPzByGCC,e0BzGD,kzB1B0GCC,gB2B1GD,uuB3B2GCC,gB4B3GD,uuB5B4GCC,iB6B5GD,uuB7B8GCC,O8B9GD,if9B+GCC,Q+B/GD,+L/BgHCxhE,MgChHD,mZhCiHCyhE,kBiCjHD,iLC2Be,SAAS,IACvB,QAAEnyF,EAAO,UAAEoyF,EAAS,SAAEzyF,EAAQ,gBAAE0yF,IAOhCryF,EAAQJ,SAAUhM,SAAU,aAAa,CAAE6R,EAAK2L,KAC/C,IAAMghF,IACL,OAKD,MAAMviG,EAAqC,mBAAvBuhB,EAAOkhF,aAA6BlhF,EAAOkhF,eAAiB,GAE1EC,EAAgD,mBAAnBF,EAAgCA,IAAoBA,EAEvF,UAAYvH,KAAkByH,EAC7B,GAAKzH,EAAex2E,SAAUlD,EAAOpd,SAAoBnE,EAAK8vB,SAAUmrE,GACvE,OAIFnrF,GAAU,GAEZ,CCfe,SAAS6yF,GAA4DnoE,GA+BnF,OA7BA,cAA6BA,EAGrB,qBAAAooE,GACNriG,KAAKsiG,2BAA4B,CAClC,CAEO,oBAAAC,GACNviG,KAAKsiG,2BAA4B,CAClC,CAEA,WAAAvgG,IAAgB7F,GACfsR,SAAUtR,GAEV8D,KAAKyR,IAAK,6BAA6B,GACvCzR,KAAKwiG,sCACN,CAEU,oCAAAA,GACTxiG,KAAKyiG,eAAgB,CACpBl9F,WAAY,CACXg5B,MAAO,CACNv+B,KAAK0iG,aAAaC,GAAI,4BAA6B,8BAIvD,EAIF,CCvBe,SAAS,IAAe,KAAE1oE,IACxCA,EAAKzqB,SAAUyqB,EAAKxiB,QAAU,UAAU,CAAEpC,EAAK2L,KAC9CA,EAAOkS,iBACP+G,EAAKzpB,KAA0B,SAAU,GACvC,CAAEkP,YAAY,GAClB,CCMe,MAAM,WAAkD,GAatE,WAAA3d,CAAa6gG,EAAgC,IAC5Cp1F,MAAOo1F,EAAc,CAGpBzyE,WAAY,YAIbnwB,KAAKqP,GAA+B,OAAO,CAAEgG,EAAK4kB,EAAMr1B,KACvD5E,KAAK6iG,gCAAiC5oE,EAAMr1B,EAAO,IAIpD5E,KAAKqP,GAAkC,UAAU,CAAEgG,EAAK4kB,KAClDA,EAAKxiB,SAAWzX,KAAK8iG,gBACzB7oE,EAAKxiB,QAAQxR,QACd,IAGDjG,KAAK8iG,eAAiB,IACvB,CAMO,OAAA98E,GACNhmB,KAAKhD,KAAKi9B,GAAQA,EAAKjU,WACxB,CASO,SAAA+8E,CAAWC,GACjBhjG,KAAK8iG,eAAiBE,EAGtB,UAAY/oE,KAAQj6B,KACnBA,KAAK6iG,gCAAiC5oE,EAExC,CAwCgB,QAAA7oB,IAAaC,GAC5B,IAAMA,EAAO/Y,SAA0B+Y,EAkF7BkF,OAAO5a,GAAiB,iBAALA,IA5E5B,MAAM,IAAI,EACT,0CACAqE,MAIF,MAAO,CACNT,GAAI0jG,IAEH,UAAYhpE,KAAQj6B,KACnB,UAAYkjG,KAAW7xF,EACtB4oB,EAAK7oB,SAAU8xF,GAAU3jG,GAAI0jG,GAK/BjjG,KAAKqP,GAA8B,OAAO,CAAEgG,EAAK4kB,KAChD,UAAYipE,KAAW7xF,EACtB4oB,EAAK7oB,SAAU8xF,GAAU3jG,GAAI0jG,EAC9B,IAIDjjG,KAAKqP,GAAiC,UAAU,CAAEgG,EAAK4kB,KACtD,UAAYipE,KAAW7xF,EACtB4oB,EAAKvoB,eAAgBwxF,EAASD,EAC/B,GACE,EAGN,CAeQ,+BAAAJ,CAAiC5oE,EAAYr1B,GAC9Cq1B,EAAKkpE,YACVlpE,EAAK0X,SAGD1X,EAAKxiB,SAAWzX,KAAK8iG,gBACzB9iG,KAAK8iG,eAAet8F,aAAcyzB,EAAKxiB,QAASzX,KAAK8iG,eAAexrF,SAAU1S,GAEhF,CAYgB,MAAAqB,CAAQmrB,GACvB,OAAO5jB,MAAMvH,OAAQmrB,EACtB,E,eCnOG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ0I,OCiFR,MAAM,WAAyD,GAAiB3mB,MA8F9F,WAAApR,CAAam3B,GACZ1rB,QAEAxN,KAAKyX,QAAU,KACfzX,KAAKmjG,YAAa,EAElBnjG,KAAKk5B,OAASA,EACdl5B,KAAKhF,EAAIk+B,GAAUA,EAAOl+B,EAE1BgF,KAAKojG,iBAAmB,IAAI,GAC5BpjG,KAAKqjG,iBAAmBrjG,KAAKsjG,mBAG7BtjG,KAAKojG,iBAAiB/zF,GAA8B,OAAO,CAAEgG,EAAKkuF,KACjEA,EAAWrqE,OAASA,EACpBqqE,EAAWvoG,EAAIk+B,GAAUA,EAAOl+B,CAAC,IAGlCgF,KAAKiV,SAAU,SAChB,CA6CA,gBAAWytF,GACV,OAAK1iG,KAAKwjG,cACFxjG,KAAKwjG,cAGJxjG,KAAKwjG,cAAgB,GAASx8F,KAAMhH,KAAMA,KACpD,CAiCO,gBAAAsjG,CAAyCG,GAC/C,MAAMF,EAAa,IAAI,GAAmBE,GAI1C,OAFAzjG,KAAKojG,iBAAiBv1F,IAAK01F,GAEpBA,CACR,CAiEO,aAAAG,CAAepsF,GACfX,EAAYW,KACjBA,EAAW,CAAEA,IAGd,UAAYK,KAASL,EACpBtX,KAAKqjG,iBAAiBx1F,IAAK8J,EAE7B,CAUO,eAAAgsF,CAAiBrsF,GACjBX,EAAYW,KACjBA,EAAW,CAAEA,IAGd,UAAYK,KAASL,EACpBtX,KAAKqjG,iBAAiBp9F,OAAQ0R,EAEhC,CAaO,WAAAisF,CAAa17F,GACnBlI,KAAK6jG,SAAW,IAAI,GAAU37F,EAC/B,CAgBO,cAAAu6F,CAAgBv6F,GACtB,GAASgrC,OAAQlzC,KAAK6jG,SAAW37F,EAClC,CA6DO,MAAAypC,GACN,GAAK3xC,KAAKmjG,WAMT,MAAM,IAAI,EAAe,kCAAmCnjG,MAIxDA,KAAK6jG,WACT7jG,KAAKyX,QAAUzX,KAAK6jG,SAASlyD,SAG7B3xC,KAAK0jG,cAAe1jG,KAAK6jG,SAASC,aAGnC9jG,KAAKmjG,YAAa,CACnB,CAUO,OAAAn9E,GACNhmB,KAAK2P,gBAEL3P,KAAKojG,iBAAiBpmG,KAAKvD,GAAKA,EAAEusB,YAG7BhmB,KAAK6jG,UAAc7jG,KAAK6jG,SAAkBE,aAC9C/jG,KAAK6jG,SAASllC,OAAQ3+D,KAAKyX,QAE7B,ECpcc,MAAM,WAAiBrI,KAsDrC,WAAArN,CAAa0zE,GACZjoE,QAEAxW,OAAO+wB,OAAQ/nB,KAAM,GAAW,GAAOy1E,KAEvCz1E,KAAKgkG,aAAc,EACnBhkG,KAAK+jG,YAAc,IACpB,CAWO,MAAApyD,GACN,MAAMxyC,EAAOa,KAAKikG,YAAa,CAC9BC,cAAc,IAKf,OAFAlkG,KAAKgkG,aAAc,EAEZ7kG,CACR,CA2CO,KAAA0R,CAAO1R,GAUb,OATAa,KAAK+jG,YAm0CC,CACNzsF,SAAU,GACVtD,SAAU,GACVzO,WAAY,CAAC,GAp0CbvF,KAAKikG,YAAa,CACjB9kG,OACA+kG,cAAc,EACdC,YAAY,EACZC,WAAYpkG,KAAK+jG,cAGX5kG,CACR,CAQO,MAAAw/D,CAAQx/D,GACd,IAAMa,KAAK+jG,YAMV,MAAM,IAAI,EACT,iCACA,CAAE/jG,KAAMb,IAIVa,KAAKqkG,wBAAyBllG,EAAMa,KAAK+jG,YAC1C,CA8BA,SAAQD,SACP,SAAUp6F,EAAQ+rE,GACjB,GAAKA,EAAIn+D,SACR,UAAYK,KAAS89D,EAAIn+D,SACnBgtF,GAAQ3sF,SACNA,EACK4sF,GAAY5sF,WAChBjO,EAAQiO,GAInB,CAEOjO,CAAQ1J,KAChB,CA0CA,WAAuBgH,CACtBsO,EACA1F,GAEA,MAAO,CACNrQ,GAAE,CAAEilG,EAAqCj1F,IACjC,IAAIk1F,GAAmB,CAC7BC,oBAAqBF,EACrBpuF,UAAWouF,EACXlvF,aAAY1F,UAASL,aAIvBozF,GAAE,CAAEvsF,EAAmBuuF,EAAsBp1F,IACrC,IAAIq1F,GAAmB,CAC7BtvF,aAAY1F,UAASwG,YAAWuuF,cAAap1F,aAIjD,CA8DA,aAAc2jC,CAAQ2wD,EAAoBpuB,GACzC,GAAKouB,EAASG,YAQb,MAAM,IAAI,EACT,yBACA,CAAEhkG,KAAM6jG,IAIVpB,GAAgBoB,EAAU,GAAW,GAAOpuB,IAC7C,CAOQ,WAAAwuB,CAAanhG,GACpB,IAAI+hG,EAUJ,GANCA,EAFI/hG,EAAK3D,KAEGa,KAAKmX,KAAOnX,KAAKk6B,KAGjBl6B,KAAKmX,IAAMnX,KAAKk6B,MAAQl6B,KAAKk6B,KAGrC2qE,EAOJ,MAAM,IAAI,EACT,2BACA7kG,MAIF,OAAKA,KAAKk6B,KACFl6B,KAAK8kG,YAAahiG,GAElB9C,KAAK+kG,eAAgBjiG,EAE9B,CAOQ,cAAAiiG,CAAgBjiG,GACvB,IAAI3D,EAAO2D,EAAK3D,KAUhB,OARMA,IACLA,EAAO2D,EAAK3D,KAAOqE,SAASkU,gBAAiB1X,KAAKglG,IA1arC,+BA0aoDhlG,KAAKmX,MAGvEnX,KAAKilG,kBAAmBniG,GACxB9C,KAAKklG,uBAAwBpiG,GAC7B9C,KAAKmlG,gBAAiBriG,GAEf3D,CACR,CAOQ,WAAA2lG,CAAahiG,GACpB,IAAI3D,EAAO2D,EAAK3D,KAoChB,OAjCKA,EACJ2D,EAAKshG,WAAYlqE,KAAO/6B,EAAK4hD,YAE7B5hD,EAAO2D,EAAK3D,KAAOqE,SAAS6C,eAAgB,IAaxC++F,GAAoBplG,KAAKk6B,MAC7Bl6B,KAAKqlG,kBAAmB,CACvB3hC,OAAQ1jE,KAAKk6B,KACbh1B,QAASogG,GAAgBnmG,GACzB2D,SAUD3D,EAAK4hD,YAAc/gD,KAAKk6B,KAAMh9B,KAAM,IAG9BiC,CACR,CAOQ,iBAAA8lG,CAAmBniG,GAC1B,IAAM9C,KAAKuF,WACV,OAGD,MAAMpG,EAAO2D,EAAK3D,KACZilG,EAAathG,EAAKshG,WAExB,UAAY30B,KAAYzvE,KAAKuF,WAAa,CAEzC,MAAMggG,EAAepmG,EAAKk8B,aAAco0C,GAGlC+1B,EAAYxlG,KAAKuF,WAAYkqE,GAG9B20B,IACJA,EAAW7+F,WAAYkqE,GAAa81B,GAUrC,MAAME,EAASC,GAAcF,GAAcA,EAAW,GAAIR,GAAK,KAmB/D,GAAKI,GAAoBI,GAAc,CAQtC,MAAMG,EAAcD,GAAcF,GAAcA,EAAW,GAAIhtG,MAAQgtG,EAIlEpB,GAAcwB,GAAcn2B,IAChCk2B,EAAYvmG,QAASmmG,GAGtBvlG,KAAKqlG,kBAAmB,CACvB3hC,OAAQiiC,EACRzgG,QAAS2gG,GAAqB1mG,EAAMswE,EAAUg2B,GAC9C3iG,QAEF,SASsB,SAAZ2sE,GAAiD,iBAAnB+1B,EAAW,GAClDxlG,KAAK8lG,sBAAuBN,EAAW,GAAY1iG,OAgB/C,CAGCshG,GAAcmB,GAAgBK,GAAcn2B,IAChD+1B,EAAUpmG,QAASmmG,GAGpB,MAAM/sG,EAAagtG,EAUjBxoG,KAAO1C,GAAcA,GAAQA,EAAI9B,OAAiB8B,IAElD2zB,QAAQ,CAAE4nB,EAAMv0C,IAAUu0C,EAAK11C,OAAQmB,IAAQ,IAE/C2sB,OAAQ83E,GAAmB,IAEvBC,GAASxtG,IACd2G,EAAK8mG,eAAgBR,EAAQh2B,EAAUj3E,EAEzC,CACD,CACD,CAgCQ,qBAAAstG,CAAuBjnE,EAA+D/7B,GAC7F,MAAM3D,EAAO2D,EAAK3D,KAElB,UAAY+mG,KAAarnE,EAAS,CACjC,MAAMsnE,EAAatnE,EAAQqnE,GAQtBd,GAAoBe,GACxBnmG,KAAKqlG,kBAAmB,CACvB3hC,OAAQ,CAAEyiC,GACVjhG,QAASkhG,GAAiBjnG,EAAM+mG,GAChCpjG,SAWC3D,EAAckG,MAAO6gG,GAAcC,CAEvC,CACD,CAOQ,sBAAAjB,CAAwBpiG,GAC/B,MAAM3D,EAAO2D,EAAK3D,KACZ2hD,EAAYh+C,EAAKohG,aAAe1gG,SAASkwC,yBAA2Bv0C,EACpEglG,EAAarhG,EAAKqhG,WACxB,IAAIkC,EAAa,EAEjB,UAAY1uF,KAAS3X,KAAKsX,SACzB,GAAKgvF,GAAkB3uF,IACtB,IAAMwsF,EAAa,CAClBxsF,EAAMorF,UAAW5jG,GAGjB,UAAY86B,KAAQtiB,EACnBmpC,EAAUn7C,YAAas0B,EAAKxiB,QAE9B,OACM,GAAK6sF,GAAQ3sF,GACbwsF,IACCxsF,EAAMwrF,YACXxrF,EAAMg6B,SAGPmP,EAAUn7C,YAAagS,EAAMF,eAExB,GAAKwH,GAAQtH,GACnBmpC,EAAUn7C,YAAagS,QAEvB,GAAKwsF,EAAa,CACjB,MACMoC,EAsyBH,CACNjvF,SAAU,GACVtD,SAAU,GACVzO,WAAY,CAAC,GA1yBSzC,EAAKshG,WAGb9sF,SAAS1X,KAAM2mG,GAE1B5uF,EAAMssF,YAAa,CAClBC,cAAc,EACd/kG,KAAM2hD,EAAUx6C,WAAY+/F,KAC5BlC,YAAY,EACZC,WAAYmC,GAEd,MACCzlD,EAAUn7C,YAAagS,EAAMg6B,UAK3B7uC,EAAKohG,cACT/kG,EAAKwG,YAAam7C,EAEpB,CAQQ,eAAAqkD,CAAiBriG,GACxB,GAAM9C,KAAKwmG,eAIX,UAAYzvG,KAAOiJ,KAAKwmG,eAAiB,CACxC,MAAMC,EAAiBzmG,KAAKwmG,eAAgBzvG,GAAMiG,KAAK0pG,IACtD,MAAQC,EAAYC,GAAgB7vG,EAAIgG,MAAO,KAE/C,OAAO2pG,EAAWG,yBAA0BF,EAAYC,EAAa9jG,EAAM,IAGvEA,EAAKshG,YACTthG,EAAKshG,WAAWpwF,SAASpU,KAAM6mG,EAEjC,CACD,CAaQ,iBAAApB,EAAmB,OAAE3hC,EAAM,QAAEx+D,EAAO,KAAEpC,IAK7C,MAAMshG,EAAathG,EAAKshG,WAGxB0C,GAAsBpjC,EAAQx+D,EAASpC,GAEvC,MAAM2jG,EAAiB/iC,EAErB39D,QAAQ9F,IAAS+lG,GAAS/lG,KAE1B8F,QAAU9F,GAAwCA,EAAKqV,aAIvDtY,KAAK+pG,GAAmBA,EAAgBC,0BAA2BtjC,EAAQx+D,EAASpC,KAEjFshG,GACJA,EAAWpwF,SAASpU,KAAM6mG,EAE5B,CAUQ,uBAAApC,CAAyBllG,EAA0BilG,GAC1D,UAAYnwF,KAAWmwF,EAAWpwF,SAWjC,UAAYizF,KAAiBhzF,EAC5BgzF,IAIF,GAAK7C,EAAWlqE,KAGf,YAFA/6B,EAAK4hD,YAAcqjD,EAAWlqE,MAK/B,MAAMziB,EAAUtY,EAEhB,UAAYswE,KAAY20B,EAAW7+F,WAAa,CAC/C,MAAMigG,EAAYpB,EAAW7+F,WAAYkqE,GAGtB,OAAd+1B,EACJ/tF,EAAQ/Q,gBAAiB+oE,GAEzBh4D,EAAQhS,aAAcgqE,EAAU+1B,EAElC,CAEA,QAAUhrG,EAAI,EAAGA,EAAI4pG,EAAW9sF,SAAShf,SAAUkC,EAClDwF,KAAKqkG,wBAAyB5sF,EAAQnR,WAAY9L,GAAY4pG,EAAW9sF,SAAU9c,GAErF,EAWM,MAAe0sG,GAgCrB,WAAAnlG,CAAa0zE,GAMZz1E,KAAKoW,UAAYq/D,EAAIr/D,UACrBpW,KAAKsV,WAAamgE,EAAIngE,WACtBtV,KAAK4P,QAAU6lE,EAAI7lE,QACnB5P,KAAKuP,SAAWkmE,EAAIlmE,QACrB,CAWO,QAAA43F,CAAUhoG,GAChB,MAAM3G,EAAUwH,KAAKsV,WAAqBtV,KAAKoW,WAE/C,OAAOpW,KAAKuP,SAAWvP,KAAKuP,SAAU/W,EAAO2G,GAAS3G,CACvD,CAYO,yBAAAwuG,CACNtjC,EACAx+D,EACApC,GAEA,MAAMyM,EAAW,IAAMu3F,GAAsBpjC,EAAQx+D,EAASpC,GAK9D,OAHA9C,KAAK4P,QAAQJ,SAAiCxP,KAAKsV,WAAY,UAAWtV,KAAKoW,YAAc7G,GAGtF,KACNvP,KAAK4P,QAAQD,cAAe3P,KAAKsV,WAAY,UAAWtV,KAAKoW,YAAc7G,EAAU,CAEvF,EAaM,MAAMk1F,WAA0ByC,GAGtC,WAAAnlG,CAAa0zE,GAGZjoE,MAAOioE,GAEPz1E,KAAK0kG,oBAAsBjvB,EAAIivB,mBAChC,CAWO,wBAAAmC,CACNF,EACAC,EACA9jG,GAEA,MAAMyM,EAAW,CAAE8F,EAAc2L,KAC1B4lF,IAAiB5lF,EAAOpd,OAAoBuwB,QAASyyE,KAClB,mBAA5B5mG,KAAK0kG,oBAChB1kG,KAAK0kG,oBAAqB1jF,GAE1BhhB,KAAKsV,WAAW9E,KAAMxQ,KAAK0kG,oBAAqB1jF,GAElD,EAMD,OAHAhhB,KAAK4P,QAAQJ,SAAU1M,EAAK3D,KAAMwnG,EAAYp3F,GAGvC,KACNvP,KAAK4P,QAAQD,cAAe7M,EAAK3D,KAAMwnG,EAAYp3F,EAAU,CAE/D,EASM,MAAMq1F,WAA0BsC,GAOtC,WAAAnlG,CAAa0zE,GAGZjoE,MAAOioE,GAEPz1E,KAAK2kG,YAAclvB,EAAIkvB,WACxB,CAKgB,QAAAwC,CAAUhoG,GAGzB,OAAO6mG,GAFOx4F,MAAM25F,SAAUhoG,MAEMa,KAAK2kG,cAAe,EACzD,EAOD,SAASS,GAAoB1hC,GAC5B,QAAMA,IAWDA,EAAOlrE,QACXkrE,EAASA,EAAOlrE,OAGZmI,MAAMC,QAAS8iE,GACZA,EAAOrvC,KAAM+wE,IACT1hC,aAAkBwjC,GAK/B,CA4BA,SAASJ,GACRpjC,EACAx+D,GACA,KAAE/F,IAEF,MAAM+L,EAxBP,SAA8Bw4D,EAAsDvkE,GACnF,OAAOukE,EAAO1mE,KAAK0pG,GAEbA,aAAsBQ,GACnBR,EAAWS,SAAUhoG,GAItBunG,GAET,CAcgBU,CAAqB1jC,EAAQvkE,GAC5C,IAAI3G,EAOHA,EADqB,GAAjBkrE,EAAOprE,QAAeorE,EAAQ,aAAekhC,GACzC15F,EAAQ,GAERA,EAAO+iB,OAAQ83E,GAAmB,IAGtCC,GAASxtG,GACb0M,EAAQe,SAERf,EAAQuM,IAAKjZ,EAEf,CAcA,SAAS8sG,GAAgBnmG,GACxB,MAAO,CACN,GAAAsS,CAAKjZ,GACJ2G,EAAK4hD,YAAcvoD,CACpB,EAEA,MAAAyN,GACC9G,EAAK4hD,YAAc,EACpB,EAEF,CAWA,SAAS8kD,GAAqBt/E,EAAakpD,EAAkBu1B,GAC5D,MAAO,CACN,GAAAvzF,CAAKjZ,GACJ+tB,EAAG0/E,eAAgBjB,EAAIv1B,EAAUj3E,EAClC,EAEA,MAAAyN,GACCsgB,EAAG8gF,kBAAmBrC,EAAIv1B,EAC3B,EAEF,CAUA,SAAS22B,GAAiB7/E,EAAS2/E,GAClC,MAAO,CACN,GAAAz0F,CAAKjZ,GACJ+tB,EAAGlhB,MAAO6gG,GAAc1tG,CACzB,EAEA,MAAAyN,GACCsgB,EAAGlhB,MAAO6gG,GAAc,IACzB,EAEF,CAKA,SAAS,GAAOzwB,GAkBf,OAjBc,GAAeA,GAAKj9E,IAYjC,GAAKA,IAAWA,aAAiB0uG,IAAmB3C,GAAY/rG,IAAW8rG,GAAQ9rG,IAAW8tG,GAAkB9tG,IAC/G,OAAOA,CACR,GAIF,CAcA,SAAS,GAAWi9E,GAcnB,GAbmB,iBAAPA,EACXA,EAwHF,SAAuCA,GACtC,MAAO,CACNv7C,KAAM,CAAEu7C,GAEV,CA5HQ6xB,CAA8B7xB,GACzBA,EAAIv7C,MAgJjB,SAAkCu7C,GACjCA,EAAIv7C,KAAO,GAASu7C,EAAIv7C,KACzB,CAjJEqtE,CAAyB9xB,GAGrBA,EAAIpmE,KACRomE,EAAI+wB,eA2FN,SAA6BgB,GAC5B,UAAYtuG,KAAKsuG,EAChBC,GAAUD,EAAWtuG,GAGtB,OAAOsuG,CACR,CAjGuBE,CAAoBjyB,EAAIpmE,WAGtComE,EAAIpmE,KAGNomE,EAAIv7C,KAAO,CACXu7C,EAAIlwE,YAmDX,SAA8BA,GAC7B,UAAY5J,KAAK4J,EACXA,EAAY5J,GAAInD,QACpB+M,EAAY5J,GAAInD,MAAQ,GAAS+M,EAAY5J,GAAInD,QAGlDivG,GAAUliG,EAAY5J,EAExB,CA1DGgsG,CAAqBlyB,EAAIlwE,YAG1B,MAAM+R,EAAW,GAEjB,GAAKm+D,EAAIn+D,SACR,GAAKgvF,GAAkB7wB,EAAIn+D,UAC1BA,EAAS1X,KAAM61E,EAAIn+D,eAEnB,UAAYK,KAAS89D,EAAIn+D,SACnBitF,GAAY5sF,IAAW2sF,GAAQ3sF,IAAWsH,GAAQtH,GACtDL,EAAS1X,KAAM+X,GAEfL,EAAS1X,KAAM,IAAI,GAAU+X,IAMjC89D,EAAIn+D,SAAWA,CAChB,CAEA,OAAOm+D,CACR,CAmIA,SAASgyB,GAAU5iG,EAAU9N,GAC5B8N,EAAK9N,GAAQ,GAAS8N,EAAK9N,GAC5B,CAMA,SAASgvG,GAAmBlwD,EAA2Bn2C,GACtD,OAAKsmG,GAAStmG,GACNm2C,EACImwD,GAASnwD,GACbn2C,EAEA,GAAIm2C,KAAUn2C,GAEvB,CAmBA,SAASkoG,GAAwB/iG,EAAUgjG,GAC1C,UAAYlsG,KAAKksG,EACXhjG,EAAKlJ,GACTkJ,EAAKlJ,GAAIiE,QAASioG,EAAKlsG,IAEvBkJ,EAAKlJ,GAAMksG,EAAKlsG,EAGnB,CAUA,SAAS8mG,GAAgBoB,EAAoBpuB,GAqB5C,GApBKA,EAAIlwE,aACFs+F,EAASt+F,aACds+F,EAASt+F,WAAa,CAAC,GAGxBqiG,GAAwB/D,EAASt+F,WAAYkwE,EAAIlwE,aAG7CkwE,EAAI+wB,iBACF3C,EAAS2C,iBACd3C,EAAS2C,eAAiB,CAAC,GAG5BoB,GAAwB/D,EAAS2C,eAAgB/wB,EAAI+wB,iBAGjD/wB,EAAIv7C,MACR2pE,EAAS3pE,KAAMt6B,QAAS61E,EAAIv7C,MAGxBu7C,EAAIn+D,UAAYm+D,EAAIn+D,SAAShf,OAAS,CAC1C,GAAKurG,EAASvsF,SAAUhf,QAAUm9E,EAAIn+D,SAAShf,OAM9C,MAAM,IAAI,EACT,uCACAurG,GAIF,IAAIwC,EAAa,EAEjB,UAAYpwB,KAAYR,EAAIn+D,SAC3BmrF,GAAgBoB,EAASvsF,SAAW+uF,KAAuBpwB,EAE7D,CACD,CAQA,SAAS+vB,GAASxtG,GACjB,OAAQA,GAAmB,IAAVA,CAClB,CAOA,SAAS8rG,GAAQrkG,GAChB,OAAOA,aAAgB,EACxB,CAOA,SAASskG,GAAYtkG,GACpB,OAAOA,aAAgB,EACxB,CAOA,SAASqmG,GAAkBrmG,GAC1B,OAAOA,aAAgB,EACxB,CAKA,SAASylG,GACRF,GAEA,OAAO,EAAUA,EAAW,KAAWA,EAAW,GAAaR,EAChE,CAoBA,SAASY,GAAcn2B,GACtB,MAAmB,SAAZA,GAAmC,SAAZA,CAC/B,CCh/Ce,MAAMq4B,WAAuB,GAkB3C,WAAA/lG,CAAam3B,EAAgB0pE,EAA+B,IAC3Dp1F,MAAOo1F,GAEP5iG,KAAKk5B,OAASA,CACf,CAKA,2BAAW6uE,GACV,OAAO/nG,KAAKgoG,wBACb,CAMO,WAAAC,GACNjoG,KAAKgoG,yBAA2B,IAAI,GAAU,CAC7C7wF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,eACA,UACA,sBAEDxyB,IAAK/L,KAAKk5B,OAAOvJ,qBAElBrY,SAAUtX,OACP2xC,SAEJ,IAAI+H,EAAUl2C,SAASM,cAAe,oBAEhC41C,IACLA,EAAU,GAAel2C,SAAU,MAAO,CAAE+6B,MAAO,oBACnD/6B,SAASgiB,KAAK7f,YAAa+zC,IAG5BA,EAAQ/zC,YAAa3F,KAAKgoG,yBAC3B,CAMO,aAAAE,GACN16F,MAAMwY,UAEDhmB,KAAKgoG,0BACThoG,KAAKgoG,yBAAyB/hG,SAG/B,MAAMyzC,EAAUl2C,SAASM,cAAe,oBAEnC41C,GAAwC,GAA7BA,EAAQyuD,mBACvBzuD,EAAQzzC,QAEV,E,eC1GG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ6zB,OAAvB,MCQqB,GAArB,cAAsC,GAkErC,WAAA/3B,GACCyL,QAEA,MAAMxG,EAAOhH,KAAK0iG,aAElB1iG,KAAKyR,IAAK,UAAW,IACrBzR,KAAKyR,IAAK,UAAW,aACrBzR,KAAKyR,IAAK,YAAa,IACvBzR,KAAKyR,IAAK,oBAAoB,GAE9BzR,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL6tF,GAAI,6BACJz/F,WAAY,CACXg5B,MAAO,CACN,KACA,UAIA,wBAIAv3B,EAAK27F,GAAI,mBAAoB,0BAE9ByF,QAASphG,EAAKzH,GAAI,aAGrB,CAKgB,MAAAoyC,GACfnkC,MAAMmkC,SAEN3xC,KAAKqoG,oBACLroG,KAAKsoG,kBAILtoG,KAAKqP,GAA2B,kBAAkB,KACjDrP,KAAKqoG,oBACLroG,KAAKsoG,iBAAiB,IAGvBtoG,KAAKqP,GAA2B,oBAAoB,KACnDrP,KAAKsoG,iBAAiB,GAExB,CAKQ,iBAAAD,GACP,GAAKroG,KAAKE,QAAU,CACnB,MACMqoG,GADS,IAAIxjD,WAAYC,gBAAiBhlD,KAAKE,QAAQ2tB,OAAQ,iBAClD/pB,cAAe,OAC5BskG,EAAUG,EAAIltE,aAAc,WAE7B+sE,IACJpoG,KAAKooG,QAAUA,GAKhB,UAAY,KAAEpmG,EAAI,MAAExJ,KAAWmI,MAAMrB,KAAMipG,EAAIhjG,YACzC,GAASijG,6BAA6Bj5E,SAAUvtB,IACpDhC,KAAKyX,QAAShS,aAAczD,EAAMxJ,GAIpC,KAAQwH,KAAKyX,QAAS9Q,YACrB3G,KAAKyX,QAASlR,YAAavG,KAAKyX,QAAS9Q,YAG1C,KAAQ4hG,EAAIjiG,WAAWhO,OAAS,GAC/B0H,KAAKyX,QAAS9R,YAAa4iG,EAAIjiG,WAAY,GAE7C,CACD,CAKQ,eAAAgiG,GACFtoG,KAAKyoG,WACTzoG,KAAKyX,QAASixF,iBAAkB,kBAAmB9qG,SAAS6B,IACzDA,EAAsB4F,MAAMyF,KAAO9K,KAAKyoG,SAAS,GAGtD,GA9JD,IAAqBE,GAArB,GAAqBA,GAoDLH,6BAA8C,CAC5D,qBAAsB,iBAAkB,YAAa,YAAa,QAAS,sBAC3E,8BAA+B,kBAAmB,SAAU,YAAa,UAAW,oBAAqB,OAAQ,eACjH,YAAa,SAAU,cAAe,gBAAiB,cAAe,YAAa,mBAAoB,eAAgB,aACvH,eAAgB,cAAe,kBAAmB,iBAAkB,iBAAkB,aAAc,aAAc,eAClH,OAAQ,UAAW,WAAY,cAAe,iBAAkB,kBAAmB,aAAc,eAAgB,SACjH,mBAAoB,oBAAqB,iBAAkB,kBAAmB,oBAAqB,iBAAkB,eACrH,cAAe,kBAAmB,gBAAiB,iBAAkB,YAAa,eAAgB,gBAClG,aAAc,cAAe,eAAgB,gB,eC7E3C,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ1uE,OCiCR,MAAM,WAAmB,GAyIvC,WAAA/3B,CAAam3B,GACZ1rB,MAAO0rB,GANR,KAAQ0vE,cAAgD,KAQvD,MAAM5hG,EAAOhH,KAAK0iG,aACZmG,EAAe,IAGrB7oG,KAAKyR,IAAK,mBAAe,GACzBzR,KAAKyR,IAAK,iBAAa,GACvBzR,KAAKyR,IAAK,iBAAkB,yBAA0Bo3F,KACtD7oG,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,kBAAc,GACxBzR,KAAKyR,IAAK,YAAQ,GAClBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,QAAQ,GAClBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,gBAAgB,GAC1BzR,KAAKyR,IAAK,iBAAa,GACvBzR,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,YAAQ,GAClBzR,KAAKyR,IAAK,YAAa,GACvBzR,KAAKyR,IAAK,WAAW,GACrBzR,KAAKyR,IAAK,kBAAmB,KAC7BzR,KAAKyR,IAAK,OAAQ,UAClBzR,KAAKyR,IAAK,YAAY,GACtBzR,KAAKyR,IAAK,iBAAiB,GAE3BzR,KAAKsX,SAAWtX,KAAKsjG,mBACrBtjG,KAAK8oG,UAAY9oG,KAAK+oG,mBAEtB/oG,KAAKgpG,SAAW,IAAIL,GACpB3oG,KAAKgpG,SAASvG,eAAgB,CAC7Bl9F,WAAY,CACXg5B,MAAO,qBAITv+B,KAAKipG,cAAgBjpG,KAAKkpG,uBAE1BlpG,KAAKgH,KAAM,kBAAmBzH,GAC7BS,KAAM,UACNA,KAAM,QACNA,KAAM,YACNA,KAAKmpG,kBAAkBniG,KAAMhH,OAG9B,MAAM6jG,EAAgD,CACrD1sF,IAAK,SAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,YACAv3B,EAAKzH,GAAI,SACTyH,EAAK27F,GAAI,YAAa,eAAenqG,IAAUA,IAC/CwO,EAAK27F,GAAI,YAAa,aAAanqG,IAAUA,IAC7CwO,EAAKzH,GAAI,QAAQ/G,GAASA,EAAQ,QAAU,WAC5CwO,EAAK27F,GAAI,WAAY,uBACrB37F,EAAK27F,GAAI,gBAAiB,6BAE3ByG,KAAMpiG,EAAKzH,GAAI,QACf0L,KAAMjE,EAAKzH,GAAI,QAAQ/G,GAASA,GAAgB,WAChD6wG,SAAUriG,EAAKzH,GAAI,YACnB,aAAcyH,EAAKzH,GAAI,aACvB,kBAAmByH,EAAKzH,GAAI,kBAC5B,gBAAiByH,EAAK27F,GAAI,aAAa,GAAMnqG,IAAUA,IACvD,eAAgBwO,EAAKzH,GAAI,QACzB,eAAgByH,EAAKzH,GAAI,QAAQ/G,KAASwH,KAAKspG,cAAeruF,SAAUziB,KACxE,wBAAyBwO,EAAKzH,GAAI,kBAClC,4BAA6ByH,EAAKzH,GAAI,oBAGvC+X,SAAUtX,KAAKsX,SAEfjI,GAAI,CACHk6F,MAAOviG,EAAKzH,IAAI8V,IAGVrV,KAAK00B,UACT10B,KAAKwQ,KAA0B,WAI/B6E,EAAI6d,gBACL,MAOE,EAAIhqB,WACFlJ,KAAK4oG,gBACV5oG,KAAK4oG,cAAgB,IAAO,IAAM5oG,KAAKitC,SAAS,IAGjD42D,EAASx0F,GAAGm6F,UAAYxiG,EAAKzH,IAAI,KAChCS,KAAK4oG,eAAgB,IAGtB/E,EAASx0F,GAAGo6F,QAAUziG,EAAKzH,IAAI,KAC9BS,KAAK4oG,cAAen1E,QAAQ,KAI9BzzB,KAAK4jG,YAAaC,EACnB,CAKgB,MAAAlyD,GACfnkC,MAAMmkC,SAED3xC,KAAK0pG,OACT1pG,KAAKgpG,SAAShiG,KAAM,WAAYzH,GAAIS,KAAM,QAC1CA,KAAKsX,SAASzJ,IAAK7N,KAAKgpG,WAGzBhpG,KAAKsX,SAASzJ,IAAK7N,KAAK8oG,WAEnB9oG,KAAK2pG,eAAiB3pG,KAAK4tB,WAC/B5tB,KAAKsX,SAASzJ,IAAK7N,KAAKipG,cAE1B,CAKO,KAAAh8D,GACNjtC,KAAKyX,QAASw1B,OACf,CAKgB,OAAAjnB,GACVhmB,KAAK4oG,eACT5oG,KAAK4oG,cAAcn1E,SAGpBjmB,MAAMwY,SACP,CAKQ,gBAAA+iF,GACP,MAAMD,EAAY,IAAI,GAChB9hG,EAAOhH,KAAK0iG,aAqBlB,OAnBAoG,EAAUlF,YAAa,CACtBzsF,IAAK,OAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,oBAEDl5B,MAAO2B,EAAKzH,GAAI,cAChBM,GAAIG,KAAK4pG,gBAGVtyF,SAAU,CACT,CACC4iB,KAAMlzB,EAAKzH,GAAI,aAKXupG,CACR,CAMQ,oBAAAI,GACP,MAAMD,EAAgB,IAAI,GAmB1B,OAjBAA,EAAcrF,YAAa,CAC1BzsF,IAAK,OAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,yBAIFjnB,SAAU,CACT,CACC4iB,KAAMl6B,KAAK0iG,aAAanjG,GAAI,aAAa26B,GAAQ/L,GAAqB+L,SAKlE+uE,CACR,CAYQ,iBAAAE,CACPU,EACAt7D,EACA3gB,GAEA,OAAKi8E,EACmB,iBAAXA,EACJA,GAEFj8E,IACJA,EAAYO,GAAqBP,IAG7Bi8E,aAAmBvhG,SAChBuhG,EAASt7D,EAAQ3gB,GAEjB,GAAI2gB,IAAU3gB,EAAY,KAAMA,KAAgB,MAKnD,EACR,E,eC3ZG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQkM,OCoBR,MAAM,WAAyB,GAS7C,WAAA/3B,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKspG,cAAe,EACpBtpG,KAAK8pG,iBAAmB9pG,KAAK+pG,oBAE7B/pG,KAAKyiG,eAAgB,CACpBl9F,WAAY,CACXg5B,MAAO,oBAGV,CAKgB,MAAAoT,GACfnkC,MAAMmkC,SAEN3xC,KAAKsX,SAASzJ,IAAK7N,KAAK8pG,iBACzB,CAKQ,iBAAAC,GACP,MAAMD,EAAmB,IAAI,GA0B7B,OAxBAA,EAAiBlG,YAAa,CAC7BzsF,IAAK,OAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,sBAIFjnB,SAAU,CACT,CACCH,IAAK,OAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,iCAOEurE,CACR,E,eC3FG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQhwE,O,8BCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCMR,MAAM,WAAkB,GAwBtC,WAAA/3B,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKyR,IAAK,YAAQ,GAClBzR,KAAKyR,IAAK,WAAO,GAEjBzR,KAAKH,GAAK,oBAAqB,MAE/B,MAAMmH,EAAOhH,KAAK0iG,aAElB1iG,KAAK4jG,YAAa,CACjBzsF,IAAK,QACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,YAED1+B,GAAIG,KAAKH,GACT+/E,IAAK54E,EAAKzH,GAAI,QAEf+X,SAAU,CACT,CACC4iB,KAAMlzB,EAAKzH,GAAI,WAInB,E,eCjEG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQu6B,OC6CR,MAAM,WAA2E,GA4H/F,WAAA/3B,CACCm3B,EACA4yC,GAEAt+D,MAAO0rB,GAEP,MAAM8wE,EAAU,yBAA0B,MACpCC,EAAY,gCAAiC,MAEnDjqG,KAAKkqG,UAAYp+B,EAAa9rE,KAAMgqG,EAASC,GAE7CjqG,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,WAAW,GACrBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,YAAa,MACvBzR,KAAKyR,IAAK,WAAY,MACtBzR,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,mBAAe,GAEzBzR,KAAK8oG,UAAY9oG,KAAK+oG,iBAAkBiB,GACxChqG,KAAKmqG,WAAanqG,KAAKoqG,kBAAmBH,GAC1CjqG,KAAKqqG,qBAAuBrqG,KAAKsjG,iBAAkB,CAAEtjG,KAAKkqG,UAAWlqG,KAAK8oG,YAE1E9oG,KAAKgH,KAAM,eAAgBzH,GAC1BS,KAAM,YACNA,KAAM,YACN,CAAEsqG,EAAWC,IAAcD,GAAaC,IAGzC,MAAMvjG,EAAOhH,KAAK0iG,aAElB1iG,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,wBACAv3B,EAAKzH,GAAI,SACTyH,EAAK27F,GAAI,YAAa,eAAenqG,IAAUA,IAC/CwO,EAAK27F,GAAI,UAAW,+BACpB37F,EAAK27F,GAAI,YAAa,iCACtB37F,EAAK27F,GAAI,cAAe,qCACxB37F,EAAK27F,GAAI,YAAa,cAGxBrrF,SAAU,CACT,CACCH,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,yCAGFjnB,SAAUtX,KAAKqqG,sBAEhBrqG,KAAKmqG,aAGR,CAOQ,gBAAApB,CAAkBlpG,GACzB,MAAMipG,EAAY,IAAI,GAAW9oG,KAAKk5B,QAKtC,OAHA4vE,EAAUlpB,IAAM//E,EAChBipG,EAAU9hG,KAAM,QAASzH,GAAIS,KAAM,SAE5B8oG,CACR,CASQ,iBAAAsB,CAAmBH,GAC1B,MAAME,EAAa,IAAI,GAAMnqG,KAAKk5B,QAC5BlyB,EAAOhH,KAAK0iG,aAqBlB,OAnBAyH,EAAWvG,YAAa,CACvBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,gCACAv3B,EAAK27F,GAAI,YAAa,uCACtB37F,EAAK27F,GAAI,cAAe,aAAanqG,IAAUA,KAEhDqH,GAAIoqG,EACJb,KAAMpiG,EAAK27F,GAAI,YAAa,UAE7BrrF,SAAU,CACT,CACC4iB,KAAMlzB,EAAKzH,GAAI,mBAKX4qG,CACR,CAKO,KAAAl9D,GACNjtC,KAAKkqG,UAAUj9D,OAChB,E,eCpSG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQnT,OCUR,MAAM,WAAkB,GAkFtC,WAAA/3B,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,UAAM,GAChBzR,KAAKyR,IAAK,mBAAe,GACzBzR,KAAKyR,IAAK,cAAc,GACxBzR,KAAKyR,IAAK,YAAY,GACtBzR,KAAKyR,IAAK,yBAAqB,GAE/BzR,KAAKwqG,aAAe,IAAI,GAExBxqG,KAAKgH,KAAM,aAAczH,GAAIS,KAAKwqG,cAClCxqG,KAAKyR,IAAK,WAAW,GACrBzR,KAAKyR,IAAK,YAAa,QAEvB,MAAMzK,EAAOhH,KAAK0iG,aAElB1iG,KAAK4jG,YAAa,CACjBzsF,IAAK,QACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,WACAv3B,EAAK27F,GAAI,YAAa,oBACtB37F,EAAK27F,GAAI,UAAW,uBACpB37F,EAAK27F,GAAI,WAAY,aAEtB9iG,GAAImH,EAAKzH,GAAI,MACbq7B,YAAa5zB,EAAKzH,GAAI,eACtBkrG,SAAUzjG,EAAKzH,GAAI,cACnBmrG,UAAW1jG,EAAKzH,GAAI,aACpB,eAAgByH,EAAK27F,GAAI,YAAY,GACrC,mBAAoB37F,EAAKzH,GAAI,sBAE9B8P,GAAI,CACHoN,MAAOzV,EAAKzH,IAAI,IAAKrD,KACpB8D,KAAKwQ,KAAM,WAAYtU,GACvB8D,KAAK2qG,gBAAgB,IAEtBpwE,OAAQvzB,EAAKzH,GAAIS,KAAK2qG,eAAe3jG,KAAMhH,SAG9C,CAKgB,MAAA2xC,GACfnkC,MAAMmkC,SAEN3xC,KAAKwqG,aAAa38F,IAAK7N,KAAKyX,SAE5BzX,KAAK4qG,oBAAqB5qG,KAAKxH,OAC/BwH,KAAK2qG,iBAIL3qG,KAAKqP,GAA2B,gBAAgB,CAAEgG,EAAKrT,EAAMxJ,KAC5DwH,KAAK4qG,oBAAqBpyG,GAC1BwH,KAAK2qG,gBAAgB,GAEvB,CAKgB,OAAA3kF,GACfxY,MAAMwY,UAENhmB,KAAKwqG,aAAaxkF,SACnB,CAKO,MAAA6kF,GACN7qG,KAAKyX,QAASozF,QACf,CAKO,KAAA59D,GACNjtC,KAAKyX,QAASw1B,OACf,CAKQ,cAAA09D,GACP3qG,KAAKmiC,SAA+BniC,KAAKyX,QAYvBjf,KAXnB,CAKQ,mBAAAoyG,CAAqBpyG,GAC5BwH,KAAKyX,QAASjf,MAAWA,GAAmB,IAAVA,EAAqBA,EAAL,EACnD,EC3Lc,MAAM,WAAsB,GAI1C,WAAAuJ,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKyiG,eAAgB,CACpBl9F,WAAY,CACX0F,KAAM,OACNszB,MAAO,CACN,mBAIJ,ECXc,MAAMusE,WAA0B,GA+B9C,WAAA/oG,CAAam3B,GACZ1rB,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAElB1iG,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,WAAY,MAEtBzR,KAAKsX,SAAWtX,KAAKsjG,mBAErBtjG,KAAK4jG,YAAa,CACjBzsF,IAAK,MAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,WACA,qBACAv3B,EAAKzH,GAAI,YAAY/G,GAAS,sBAAuBA,MACrDwO,EAAK27F,GAAI,YAAa,+BAEvB0G,SAAU,MAGX/xF,SAAUtX,KAAKsX,SAEfjI,GAAI,CAGH07F,YAAa/jG,EAAKzH,IAAI8V,IAC8C,UAA5DA,EAAIzR,OAAwB+9C,QAAQqpD,qBAI3C31F,EAAI6d,gBAAgB,MAIxB,CAOO,KAAA+Z,GACN,GAAKjtC,KAAKsX,SAAShf,OAAS,CAC3B,MAAMqO,EAAkB3G,KAAKsX,SAASmZ,MAEL,mBAArB9pB,EAAWsmC,MACtBtmC,EAAWsmC,QAiBX,EAAY,8CAA+C,CAAEoZ,UAAWrmD,KAAKsX,SAASmZ,MAAOw6E,cAAejrG,MAE9G,CACD,CAOO,SAAAkrG,GACN,GAAKlrG,KAAKsX,SAAShf,OAAS,CAC3B,MAAM2vC,EAAiBjoC,KAAKsX,SAASoZ,KAED,mBAAxBuX,EAAUijE,UACrBjjE,EAAUijE,YAEVjjE,EAAUgF,OAEZ,CACD,E,eCvIG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQnT,OAAvB,MCoEqB,GAArB,cAA0C,GAiHzC,WAAA/3B,CACCm3B,EACAiyE,EACAC,GAEA59F,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAElB1iG,KAAKmrG,WAAaA,EAClBnrG,KAAKorG,UAAYA,EAEjBprG,KAAKyR,IAAK,UAAU,GACpBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,UAAM,GAChBzR,KAAKyR,IAAK,gBAAiB,QAE3BzR,KAAKo/F,WAAa,IAAI,GACtBp/F,KAAKwqG,aAAe,IAAI,GAExBxqG,KAAK4jG,YAAa,CACjBzsF,IAAK,MAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,cACAv3B,EAAKzH,GAAI,SACTyH,EAAK27F,GAAI,YAAa,eAAenqG,IAAUA,KAEhDqH,GAAImH,EAAKzH,GAAI,MACb,mBAAoByH,EAAKzH,GAAI,sBAG9B+X,SAAU,CACT6zF,EACAC,KAIFD,EAAW1I,eAAgB,CAC1Bl9F,WAAY,CACXg5B,MAAO,CACN,uBAED,4BAA6Bv3B,EAAKzH,GAAI,YAGzC,CAKgB,MAAAoyC,GACfnkC,MAAMmkC,SAEN3xC,KAAKwqG,aAAa38F,IAAK7N,KAAKmrG,WAAW1zF,SACvCzX,KAAKwqG,aAAa38F,IAAK7N,KAAKorG,UAAU3zF,SAGtCzX,KAAKwP,SAAmCxP,KAAKmrG,WAAY,QAAQ,KAChEnrG,KAAKqrG,QAAUrrG,KAAKqrG,MAAM,IAI3BrrG,KAAKorG,UAAUpkG,KAAM,aAAczH,GAAIS,KAAM,UAI7CA,KAAKqP,GAAoC,iBAAiB,CAAEgG,EAAKrT,EAAMqpG,KAChEA,IAMsB,SAAvBrrG,KAAKsrG,cACTtrG,KAAKorG,UAAU3lF,SAAW,GAAa8lF,oBAAqB,CAC3D9zF,QAASzX,KAAKorG,UAAU3zF,QACxB7T,OAAQ5D,KAAKmrG,WAAW1zF,QACxB6P,eAAe,EACfF,UAAWpnB,KAAKwrG,kBACbxpG,KAEJhC,KAAKorG,UAAU3lF,SAAWzlB,KAAKsrG,cAChC,IAIDtrG,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,SAE/B,MAAMg0F,EAAgB,CAAE3oG,EAAe2wB,KACjCzzB,KAAKqrG,SACTrrG,KAAKqrG,QAAS,EACd53E,IACD,EAIDzzB,KAAKo/F,WAAW3tF,IAAK,aAAa,CAAE3O,EAAM2wB,KAEpCzzB,KAAKmrG,WAAWz2E,YAAc10B,KAAKqrG,SACvCrrG,KAAKqrG,QAAS,EACd53E,IACD,IAIDzzB,KAAKo/F,WAAW3tF,IAAK,cAAc,CAAE3O,EAAM2wB,KACrCzzB,KAAKqrG,QACT53E,GACD,IAIDzzB,KAAKo/F,WAAW3tF,IAAK,YAAag6F,GAClCzrG,KAAKo/F,WAAW3tF,IAAK,MAAOg6F,EAC7B,CAKO,KAAAx+D,GACNjtC,KAAKmrG,WAAWl+D,OACjB,CAOA,mBAAYu+D,GACX,MAAM,MACLE,EAAK,MAAEC,EAAK,UACZC,EAAS,UAAEC,EAAS,UACpBC,EAAS,UAAEC,EAAS,gBACpBC,EAAe,gBAAEC,EAAe,gBAChCC,EAAe,gBAAEC,GACd,GAAaC,sBAEjB,MAA0C,QAArCpsG,KAAKk5B,OAAQvJ,oBACV,CACNi8E,EAAWC,EAAWG,EAAiBC,EAAiBP,EACxDI,EAAWC,EAAWG,EAAiBC,EAAiBR,GAGlD,CACNE,EAAWD,EAAWK,EAAiBD,EAAiBN,EACxDK,EAAWD,EAAWK,EAAiBD,EAAiBP,EAG3D,GAzQD,IAAqBU,GAArB,GAAqBA,GAuXND,sBAA6D,CAC1EV,MAAO,CAAEY,EAAYC,KACb,CACN9qF,IAAK6qF,EAAWzqF,OAChBE,KAAMuqF,EAAWvqF,MAASwqF,EAAU1pF,MAAQypF,EAAWzpF,OAAU,EACjE7gB,KAAM,MAGR4pG,UAAWU,IACH,CACN7qF,IAAK6qF,EAAWzqF,OAChBE,KAAMuqF,EAAWvqF,KACjB/f,KAAM,OAGR6pG,UAAW,CAAES,EAAYC,KACjB,CACN9qF,IAAK6qF,EAAWzqF,OAChBE,KAAMuqF,EAAWvqF,KAAOwqF,EAAU1pF,MAAQypF,EAAWzpF,MACrD7gB,KAAM,OAGRgqG,gBAAiB,CAAEM,EAAYC,KACvB,CACN9qF,IAAK6qF,EAAWzqF,OAChBE,KAAMuqF,EAAWvqF,MAASwqF,EAAU1pF,MAAQypF,EAAWzpF,OAAU,EACjE7gB,KAAM,QAGRiqG,gBAAiB,CAAEK,EAAYC,KACvB,CACN9qF,IAAK6qF,EAAWzqF,OAChBE,KAAMuqF,EAAWvqF,KAAgD,GAAvCwqF,EAAU1pF,MAAQypF,EAAWzpF,OAAc,EACrE7gB,KAAM,QAGR2pG,MAAO,CAAEW,EAAYC,KACb,CACN9qF,IAAK6qF,EAAW7qF,IAAM8qF,EAAUzpF,OAChCf,KAAMuqF,EAAWvqF,MAASwqF,EAAU1pF,MAAQypF,EAAWzpF,OAAU,EACjE7gB,KAAM,MAGR8pG,UAAW,CAAEQ,EAAYC,KACjB,CACN9qF,IAAK6qF,EAAW7qF,IAAM8qF,EAAUzpF,OAChCf,KAAMuqF,EAAWvqF,KACjB/f,KAAM,OAGR+pG,UAAW,CAAEO,EAAYC,KACjB,CACN9qF,IAAK6qF,EAAW7qF,IAAM8qF,EAAUzpF,OAChCf,KAAMuqF,EAAWvqF,KAAOwqF,EAAU1pF,MAAQypF,EAAWzpF,MACrD7gB,KAAM,OAGRkqG,gBAAiB,CAAEI,EAAYC,KACvB,CACN9qF,IAAK6qF,EAAW7qF,IAAM8qF,EAAUzpF,OAChCf,KAAMuqF,EAAWvqF,MAASwqF,EAAU1pF,MAAQypF,EAAWzpF,OAAU,EACjE7gB,KAAM,QAGRmqG,gBAAiB,CAAEG,EAAYC,KACvB,CACN9qF,IAAK6qF,EAAW7qF,IAAM8qF,EAAUzpF,OAChCf,KAAMuqF,EAAWvqF,KAAgD,GAAvCwqF,EAAU1pF,MAAQypF,EAAWzpF,OAAc,EACrE7gB,KAAM,SA3bWqqG,GAmcLd,oBAAsBpkF,GCnhBtC,2NCoCe,MAAM,WAA2B,GAS/C,WAAAplB,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKwsG,UAAYxsG,KAAKysG,mBAEtBzsG,KAAKyiG,eAAgB,CACpBl9F,WAAY,CACX,iBAAiB,EACjB,gBAAiBvF,KAAK0iG,aAAanjG,GAAI,QAAQ/G,GAASyiB,OAAQziB,QAKlEwH,KAAKoR,SAAU,WAAY7R,GAAIS,KAAM,OACtC,CAKgB,MAAA2xC,GACfnkC,MAAMmkC,SAEN3xC,KAAKsX,SAASzJ,IAAK7N,KAAKwsG,UACzB,CAKQ,gBAAAC,GACP,MAAMD,EAAY,IAAI7D,GAUtB,OARA6D,EAAUtsG,QAAU,GAEpBssG,EAAU/J,eAAgB,CACzBl9F,WAAY,CACXg5B,MAAO,wBAIFiuE,CACR,ECdc,MAAM,GAyCpB,WAAAzqG,CAAawC,GAWZ,GALAvE,KAAK0sG,WAAanoG,EAAQmoG,WAC1B1sG,KAAKwqG,aAAejmG,EAAQimG,aAC5BxqG,KAAK2sG,iBAAmBpoG,EAAQooG,iBAChC3sG,KAAK09C,QAAUn5C,EAAQm5C,QAElBn5C,EAAQm5C,SAAWn5C,EAAQooG,iBAC/B,UAAYz3F,KAAc3Q,EAAQm5C,QAAU,CAC3C,IAAIA,EAAUn5C,EAAQm5C,QAASxoC,GAER,iBAAXwoC,IACXA,EAAU,CAAEA,IAGb,UAAY9vB,KAAa8vB,EACxBn5C,EAAQooG,iBAAiBl7F,IAAKmc,GAAW,CAAE9qB,EAAM2wB,KAChDzzB,KAAMkV,KACNue,GAAQ,GAGX,CAEF,CAQA,SAAWhD,GACV,OAASzwB,KAAK0sG,WAAWn7E,KAAMq7E,KAAiB,IACjD,CAQA,QAAWl8E,GACV,OAAS1wB,KAAK0sG,WAAW3mG,OAAQ6mG,IAAc9qG,OAAQ,GAAK,IAAO,IACpE,CAQA,QAAWR,GACV,OAAOtB,KAAK6sG,kBAAmB,EAChC,CAQA,YAAWC,GACV,OAAO9sG,KAAK6sG,mBAAoB,EACjC,CAMA,WAAW9tG,GACV,IAAI6F,EAAuB,KAG3B,OAA0C,OAArC5E,KAAKwqG,aAAa53E,eACf,MAGR5yB,KAAK0sG,WAAWn7E,MAAM,CAAE0I,EAAM8yE,KAC7B,MAAMC,EAAU/yE,EAAKxiB,UAAYzX,KAAKwqG,aAAa53E,eAMnD,OAJKo6E,IACJpoG,EAAQmoG,GAGFC,CAAO,IAGRpoG,EACR,CAOO,UAAAqoG,GACNjtG,KAAK0yB,OAAQ1yB,KAAKywB,MACnB,CAOO,SAAAy6E,GACNlrG,KAAK0yB,OAAQ1yB,KAAK0wB,KACnB,CAOO,SAAAw8E,GACNltG,KAAK0yB,OAAQ1yB,KAAKsB,KACnB,CAOO,aAAA6rG,GACNntG,KAAK0yB,OAAQ1yB,KAAK8sG,SACnB,CAKQ,MAAAp6E,CAAQuH,GACVA,GACJA,EAAKgT,OAEP,CAQQ,iBAAA4/D,CAAmB91B,GAE1B,MAAMh4E,EAAUiB,KAAKjB,QACfquG,EAAmBptG,KAAK0sG,WAAWp0G,OAEzC,IAAM80G,EACL,OAAO,KAKR,GAAiB,OAAZruG,EACJ,OAAOiB,KAAe,IAAT+2E,EAAa,QAAU,QAIrC,IAAInyE,GAAU7F,EAAUquG,EAAmBr2B,GAASq2B,EAEpD,EAAG,CACF,MAAMnzE,EAAOj6B,KAAK0sG,WAAWtkG,IAAKxD,GAElC,GAAKgoG,GAAa3yE,GACjB,OAAOA,EAIRr1B,GAAUA,EAAQwoG,EAAmBr2B,GAASq2B,CAC/C,OAAUxoG,IAAU7F,GAEpB,OAAO,IACR,EAiBD,SAAS6tG,GAAa3yE,GACrB,SAAWA,EAAKgT,QAAS/lB,GAAW+S,EAAKxiB,SAC1C,CClSe,MAAM41F,WAA6B,GAIjD,WAAAtrG,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAK4jG,YAAa,CACjBzsF,IAAK,OACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,2BAIJ,EChBc,MAAM+uE,WAA6B,GAIjD,WAAAvrG,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAK4jG,YAAa,CACjBzsF,IAAK,OACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,4BAIJ,ECKc,SAASgvE,GACvBvkF,GAMA,OAAKroB,MAAMC,QAASooB,GACZ,CACN4H,MAAO5H,EACPwkF,YAAa,IAITxkF,EAOChyB,OAAO+wB,OAAQ,CACrB6I,MAAO,GACP48E,YAAa,IACXxkF,GATK,CACN4H,MAAO,GACP48E,YAAa,GAQhB,C,eC3DI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ1zE,OAAvB,MCmCM,kBAAEioE,IAAsB,GAExB0L,GAA2D,CAChE3M,UAAW,GAAMA,UACjBX,KAAM,GAAMA,KACZM,aAAc,GAAMA,aACpBzyB,UAAW,GAAMA,UACjB0yB,KAAM,GAAMA,KACZxmE,KAAM,GAAMA,KACZ6nE,kBAAmB,GAAMA,mBAMX,MAAM,WAAoB,GAsHxC,WAAAhgG,CAAam3B,EAAgB30B,GAC5BiJ,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aACZ1nG,EAAIgF,KAAKhF,EAEfgF,KAAKuE,QAAUA,GAAW,CAAC,EAE3BvE,KAAKyR,IAAK,YAAazW,EAAG,mBAC1BgF,KAAKyR,IAAK,WAAY,QAEtBzR,KAAK4wB,MAAQ5wB,KAAKsjG,mBAClBtjG,KAAKwqG,aAAe,IAAI,GACxBxqG,KAAKo/F,WAAa,IAAI,GAEtBp/F,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,aAAa,GAEvBzR,KAAK0tG,UAAY,IAAIC,GAAWz0E,GAChCl5B,KAAKsX,SAAWtX,KAAKsjG,mBACrBtjG,KAAKsX,SAASzJ,IAAK7N,KAAK0tG,WACxB1tG,KAAK0sG,WAAa1sG,KAAKsjG,mBAEvB,MAAMsK,EAAuC,QAA/B10E,EAAOvJ,oBAErB3vB,KAAK6tG,aAAe,IAAI,GAAa,CACpCnB,WAAY1sG,KAAK0sG,WACjBlC,aAAcxqG,KAAKwqG,aACnBmC,iBAAkB3sG,KAAKo/F,WACvB1hD,QAAS,CAERyvD,cAAe,CAAES,EAAQ,aAAe,YAAa,WAGrDV,UAAW,CAAEU,EAAQ,YAAc,aAAc,gBAInD,MAAMlvE,EAAU,CACf,KACA,aACA13B,EAAKzH,GAAI,SACTyH,EAAK27F,GAAI,YAAa,uBC3LV,IAAyB1oE,ED8LjCj6B,KAAKuE,QAAQupG,qBAAuB9tG,KAAKuE,QAAQwpG,YACrDrvE,EAAQ9+B,KAAM,uBAGfI,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAOG,EACP0qE,KAAM,UACN,aAAcpiG,EAAKzH,GAAI,aACvB8F,MAAO,CACN2oG,SAAUhnG,EAAKzH,GAAI,aAEpB8pG,UAAW,GAGZ/xF,SAAUtX,KAAKsX,SAEfjI,GAAI,CAEHm6F,WClNoCvvE,EDkNTj6B,KCjNvBi6B,EAAKyoE,aAAanjG,IAAI8V,IACvBA,EAAIzR,SAAWq2B,EAAKxiB,SACxBpC,EAAI6d,gBACL,QDkNAlzB,KAAKiuG,UAAYjuG,KAAKuE,QAAQupG,oBAAsB,IAAII,GAAiBluG,MAAS,IAAImuG,GAAcnuG,KACrG,CAKgB,MAAA2xC,GACfnkC,MAAMmkC,SAEN3xC,KAAKwqG,aAAa38F,IAAK7N,KAAKyX,SAG5B,UAAYxX,KAAQD,KAAK4wB,MACxB5wB,KAAKwqG,aAAa38F,IAAK5N,EAAKwX,SAG7BzX,KAAK4wB,MAAMvhB,GAA8B,OAAO,CAAEgG,EAAKpV,KACtDD,KAAKwqG,aAAa38F,IAAK5N,EAAKwX,QAAU,IAGvCzX,KAAK4wB,MAAMvhB,GAAiC,UAAU,CAAEgG,EAAKpV,KAC5DD,KAAKwqG,aAAavkG,OAAQhG,EAAKwX,QAAU,IAI1CzX,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,SAE/BzX,KAAKiuG,UAAUt8D,OAAQ3xC,KACxB,CAKgB,OAAAgmB,GAKf,OAJAhmB,KAAKiuG,UAAUjoF,UACfhmB,KAAKwqG,aAAaxkF,UAClBhmB,KAAKo/F,WAAWp5E,UAETxY,MAAMwY,SACd,CAKO,KAAAinB,GACNjtC,KAAK6tG,aAAaZ,YACnB,CAKO,SAAA/B,GACNlrG,KAAK6tG,aAAa3C,WACnB,CAWO,cAAAkD,CACNC,EACA93G,EACAi3G,GAEAxtG,KAAK4wB,MAAMD,QAAS3wB,KAAKsuG,sBAAuBD,EAAe93G,EAASi3G,GACzE,CAUQ,qBAAAc,CACPD,EACA93G,EACAi3G,GAEA,MAAMxkF,EAASukF,GAAwBc,GACjCE,EAAwBf,GAAexkF,EAAOwkF,YAepD,OAdmBxtG,KAAKwuG,yBAA0BxlF,EAAO4H,MAAOr6B,EAASg4G,GACvEvxG,KAAKiD,GACA,EAAUA,GACPD,KAAKyuG,6BAA8BxuG,EAAM1J,EAASg4G,GACrC,MAATtuG,EACJ,IAAIotG,GACS,MAATptG,EACJ,IAAIqtG,GAGL/2G,EAAQ+mB,OAAQrd,KAEvB8F,QAAU9F,KAA0BA,GAGvC,CAWQ,wBAAAuuG,CACP59E,EACAr6B,EACAi3G,GAEA,MAAMkB,EAAgB99E,EACpB7qB,QAAQ,CAAE9F,EAAM45D,EAAKjpC,IACP,MAAT3wB,IAKwC,IAAxCutG,EAAYxkG,QAAS/I,KAIZ,MAATA,GAGCD,KAAKuE,QAAQupG,sBAmBjB,EAAY,qDAAsDl9E,IAE3D,MAOH,EAAU3wB,KAAW1J,EAAQqX,IAAK3N,MAoBvC,EAAY,+BAAgC,CAAEA,UAEvC,MAMV,OAAOD,KAAK2uG,8BAA+BD,EAC5C,CAOQ,6BAAAC,CAA+B/9E,GACtC,MAAMg+E,EAA0B3uG,GAAwC,MAATA,GAAyB,MAATA,EACzE0E,EAAQisB,EAAMt4B,OAGdu2G,EAAwBj+E,EAAMk+E,UAAWF,GAG/C,IAAgC,IAA3BC,EACJ,MAAO,GAIR,MAAME,EAAuBpqG,EAAQisB,EACnC9uB,QACAuJ,UACAyjG,UAAWF,GAEb,OAAOh+E,EAEL9uB,MAAO+sG,EAAuBE,GAE9BhpG,QAAQ,CAAE/D,EAAM63D,EAAKjpC,KAErB,GAAKg+E,EAAuB5sG,GAC3B,OAAO,EAIR,QAFqB63D,EAAM,GAAKjpC,EAAOipC,EAAM,KAAQ73D,EAEjC,GAEvB,CAgBQ,4BAAAysG,CACPvmG,EACA8mG,EACAxB,GAEA,IAAI,MAAEj/D,EAAK,KAAEm7D,EAAI,MAAE94E,EAAK,QAAEi5E,GAAU,EAAI,SAAEoF,GAAW,GAAU/mG,EAK/D,GAHA0oB,EAAQ5wB,KAAKwuG,yBAA0B59E,EAAOo+E,EAAkBxB,IAG1D58E,EAAMt4B,OACX,OAAO,KAGR,MACM42G,EAAe,GADNlvG,KAAKk5B,QA8CpB,OA3CMqV,GAmBL,EAAY,oDAAqDrmC,GAGlEgnG,EAAa3wE,MAAQ,sCACrB2wE,EAAa/D,WAAW15F,IAAK,CAC5B88B,QACAs7D,UACAoF,WAAYA,KAIC,IAATvF,EAEJwF,EAAa/D,WAAWzB,KAAO+D,GAAsB/D,IAAWA,GAAQ3H,GAIxEmN,EAAa/D,WAAW8D,UAAW,EAGpCE,GAAsBD,GAAc,IACnCA,EAAaE,YAAad,sBAAuB19E,EAAOo+E,EAAkBxB,KAGpE0B,CACR,EAwBD,MAAMvB,WAAkB,GASvB,WAAA5rG,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKsX,SAAWtX,KAAKsjG,mBAErBtjG,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,sBAGFjnB,SAAUtX,KAAKsX,UAEjB,EAQD,MAAM62F,GAOL,WAAApsG,CAAak4B,GACZ,MAAMjzB,EAAOizB,EAAKyoE,aAGlBzoE,EAAKxoB,IAAK,cAAc,GAGxBwoB,EAAKyzE,UAAUp2F,SAASpD,OAAQ+lB,EAAKrJ,OAAQkB,OAAO7xB,GAAQA,IAG5Dg6B,EAAKyyE,WAAWx4F,OAAQ+lB,EAAKrJ,OAAQkB,OAAO7xB,GAAQA,IAEpDg6B,EAAKwoE,eAAgB,CACpBl9F,WAAY,CACXg5B,MAAO,CAENv3B,EAAK27F,GAAI,aAAc,0BAI3B,CAKO,MAAAhxD,GAAgB,CAKhB,OAAA3rB,GAAiB,EAqBzB,MAAMkoF,GAqGL,WAAAnsG,CAAak4B,GAjCb,KAAOo1E,eAAwC,KAU/C,KAAOC,cAA+B,KAQtC,KAAOC,kCAA4C,EAgBlDvvG,KAAKi6B,KAAOA,EAEZj6B,KAAKwwD,aAAev2B,EAAK3iB,SACzBtX,KAAKwvG,eAAiBv1E,EAAKyyE,WAC3B1sG,KAAKyvG,cAAgBx1E,EAAKyzE,UAC1B1tG,KAAK0vG,iBAAmBz1E,EAAKuwE,aAC7BxqG,KAAK2vG,WAAa11E,EAAKf,OAEvBl5B,KAAK4vG,eAAiB31E,EAAKqpE,mBAC3BtjG,KAAK6vG,aAAe51E,EAAKqpE,mBACzBtjG,KAAK8vG,qBAAuB9vG,KAAK+vG,8BAGjC91E,EAAKyzE,UAAUp2F,SAASpD,OAAQlU,KAAK4vG,gBAAiB99E,OAAO7xB,GAAQA,IAGrED,KAAK4vG,eAAevgG,GAA2B,SAAUrP,KAAKgwG,2BAA2BhpG,KAAMhH,OAG/Fi6B,EAAK3iB,SAASjI,GAA2B,SAAUrP,KAAKgwG,2BAA2BhpG,KAAMhH,OAMzFi6B,EAAKrJ,MAAMvhB,GAAiC,UAAU,CAAEgG,EAAK46F,KAC5D,MAAMrrG,EAAQqrG,EAAWrrG,MACnBmsB,EAAQpwB,MAAMrB,KAAM2wG,EAAWl/E,OAGrC,UAAYm/E,KAAeD,EAAWj/E,QAChCpsB,GAAS5E,KAAK4vG,eAAet3G,OACjC0H,KAAK6vG,aAAa5pG,OAAQiqG,GAE1BlwG,KAAK4vG,eAAe3pG,OAAQiqG,GAK9B,QAAUC,EAAevrG,EAAOurG,EAAevrG,EAAQmsB,EAAMz4B,OAAQ63G,IAAiB,CACrF,MAAMC,EAAYr/E,EAAOo/E,EAAevrG,GAEnCurG,EAAenwG,KAAK4vG,eAAet3G,OACvC0H,KAAK6vG,aAAahiG,IAAKuiG,EAAWD,EAAenwG,KAAK4vG,eAAet3G,QAErE0H,KAAK4vG,eAAe/hG,IAAKuiG,EAAWD,EAEtC,CAMAnwG,KAAKqwG,iBAAiB,IAGvBp2E,EAAKwoE,eAAgB,CACpBl9F,WAAY,CACXg5B,MAAO,CAEN,yBAIJ,CAOO,MAAAoT,CAAQ1X,GACdj6B,KAAKu3C,YAActd,EAAKxiB,QAExBzX,KAAKswG,0BACLtwG,KAAKuwG,gCAAiCt2E,EACvC,CAKO,OAAAjU,GAGNhmB,KAAK8vG,qBAAqB9pF,UAE1BhmB,KAAKqvG,eAAgBrpF,SACtB,CAUQ,eAAAqqF,GAMP,IAAMrwG,KAAKu3C,YAAan4B,cAAcoG,KAAKtB,SAAUlkB,KAAKu3C,aACzD,OAQD,IAAMrwB,GAAWlnB,KAAKu3C,aAGrB,YAFAv3C,KAAKuvG,kCAAmC,GAOzC,MAAMiB,EAA2BxwG,KAAK6vG,aAAav3G,OACnD,IAAIm4G,EAKJ,KAAQzwG,KAAK0wG,sBACZ1wG,KAAK2wG,iBAELF,GAAmB,EAMpB,IAAMA,GAAoBzwG,KAAK6vG,aAAav3G,OAAS,CAEpD,KAAQ0H,KAAK6vG,aAAav3G,SAAW0H,KAAK0wG,sBACzC1wG,KAAK4wG,oBAOD5wG,KAAK0wG,sBACT1wG,KAAK2wG,gBAEP,CAEK3wG,KAAK6vG,aAAav3G,SAAWk4G,GACjCxwG,KAAKi6B,KAAKzpB,KAA0C,qBAEtD,CAMA,wBAAYkgG,GAEX,IAAM1wG,KAAK4vG,eAAet3G,OACzB,OAAO,EAGR,MAAMmf,EAAUzX,KAAKu3C,YACf5nB,EAAsB3vB,KAAK2vG,WAAWhgF,oBACtCkhF,EAAgB,IAAI,GAAMp5F,EAASwwB,WACnC6oE,EAAc,IAAI,GAAMr5F,GAE9B,IAAMzX,KAAKsvG,cAAgB,CAC1B,MAAMyB,EAAgB,GAAOxtG,OAAOie,iBAAkB/J,GAChDu5F,EAA0C,QAAxBrhF,EAAgC,eAAiB,cAKzE3vB,KAAKsvG,cAAgBlqF,OAAOhoB,SAAU2zG,EAAeC,GACtD,CAEA,MAA6B,QAAxBrhF,EACGkhF,EAAclvF,MAAQmvF,EAAYnvF,MAAQ3hB,KAAKsvG,cAE/CuB,EAAc9uF,KAAO+uF,EAAY/uF,KAAO/hB,KAAKsvG,aAEtD,CAYQ,uBAAAgB,GACP,IAAIW,EAGJjxG,KAAKqvG,eAAiB,IAAI,GAAgBrvG,KAAKu3C,aAAc7+B,IACtDu4F,GAAiBA,IAAkBv4F,EAAMw4F,YAAYruF,QAAS7iB,KAAKuvG,mCACxEvvG,KAAKuvG,kCAAmC,EAExCvvG,KAAKqwG,kBAELY,EAAgBv4F,EAAMw4F,YAAYruF,MACnC,IAGD7iB,KAAKqwG,iBACN,CAMQ,+BAAAE,CAAiCt2E,GACxCA,EAAK5qB,GAA2B,mBAAmB,KAClDrP,KAAKqwG,iBAAiB,GAExB,CAQQ,cAAAM,GACD3wG,KAAK6vG,aAAav3G,SACvB0H,KAAKwwD,aAAa3iD,IAAK,IAAIw/F,IAC3BrtG,KAAKwwD,aAAa3iD,IAAK7N,KAAK8vG,sBAC5B9vG,KAAK0vG,iBAAiB7hG,IAAK7N,KAAK8vG,qBAAqBr4F,UAGtDzX,KAAK6vG,aAAahiG,IAAK7N,KAAK4vG,eAAe3pG,OAAQjG,KAAK4vG,eAAel/E,MAAS,EACjF,CAQQ,iBAAAkgF,GACP5wG,KAAK4vG,eAAe/hG,IAAK7N,KAAK6vG,aAAa5pG,OAAQjG,KAAK6vG,aAAap/E,QAE/DzwB,KAAK6vG,aAAav3G,SACvB0H,KAAKwwD,aAAavqD,OAAQjG,KAAK8vG,sBAC/B9vG,KAAKwwD,aAAavqD,OAAQjG,KAAKwwD,aAAa9/B,MAC5C1wB,KAAK0vG,iBAAiBzpG,OAAQjG,KAAK8vG,qBAAqBr4F,SAE1D,CAMQ,2BAAAs4F,GACP,MAAM72E,EAASl5B,KAAK2vG,WACd30G,EAAIk+B,EAAOl+B,EACXm2G,EAAW,GAAgBj4E,GAiBjC,OAfAi4E,EAAS5yE,MAAQ,+BAIjB4yE,EAAS7F,cAA+C,QAA/BpyE,EAAOvJ,oBAAgC,KAAO,KAEvEw/E,GAAsBgC,EAAUnxG,KAAK6vG,cAErCsB,EAAShG,WAAW15F,IAAK,CACxB88B,MAAOvzC,EAAG,mBACV6uG,SAAS,EACTuH,gBAAgD,QAA/Bl4E,EAAOvJ,oBAAgC,KAAO,KAC/D+5E,KAAM3H,KAGAoP,CACR,CAYQ,0BAAAnB,GACPhwG,KAAKwvG,eAAe59F,QAEpB5R,KAAK4vG,eAAe5yG,KAAKiD,IACxBD,KAAKwvG,eAAe3hG,IAAK5N,EAAM,IAG3BD,KAAK6vG,aAAav3G,QACtB0H,KAAKwvG,eAAe3hG,IAAK7N,KAAK8vG,qBAEhC,E,eE7iCG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQh2E,OCiBR,MAAMu3E,WAAiB,GAsCrC,WAAAtvG,CAAam3B,GACZ1rB,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAElB1iG,KAAK4wB,MAAQ5wB,KAAKsjG,mBAClBtjG,KAAKwqG,aAAe,IAAI,GACxBxqG,KAAKo/F,WAAa,IAAI,GAEtBp/F,KAAK6tG,aAAe,IAAI,GAAa,CACpCnB,WAAY1sG,KAAK4wB,MACjB45E,aAAcxqG,KAAKwqG,aACnBmC,iBAAkB3sG,KAAKo/F,WACvB1hD,QAAS,CAERyvD,cAAe,UAGfD,UAAW,eAIbltG,KAAKyR,IAAK,iBAAa,GACvBzR,KAAKyR,IAAK,YAAQ,GAElBzR,KAAK4jG,YAAa,CACjBzsF,IAAK,KAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,WACA,WAED6qE,KAAMpiG,EAAKzH,GAAI,QACf,aAAcyH,EAAKzH,GAAI,cAGxB+X,SAAUtX,KAAK4wB,OAEjB,CAKgB,MAAA+gB,GACfnkC,MAAMmkC,SAGN,UAAY1xC,KAAQD,KAAK4wB,MACxB5wB,KAAKwqG,aAAa38F,IAAK5N,EAAKwX,SAG7BzX,KAAK4wB,MAAMvhB,GAAsC,OAAO,CAAEgG,EAAKpV,KAC9DD,KAAKwqG,aAAa38F,IAAK5N,EAAKwX,QAAU,IAGvCzX,KAAK4wB,MAAMvhB,GAAyC,UAAU,CAAEgG,EAAKpV,KACpED,KAAKwqG,aAAavkG,OAAQhG,EAAKwX,QAAU,IAI1CzX,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,QAChC,CAKgB,OAAAuO,GACfxY,MAAMwY,UAENhmB,KAAKwqG,aAAaxkF,UAClBhmB,KAAKo/F,WAAWp5E,SACjB,CAKO,KAAAinB,GACNjtC,KAAK6tG,aAAaZ,YACnB,CAKO,SAAA/B,GACNlrG,KAAK6tG,aAAa3C,WACnB,ECvIc,MAAMoG,WAAqB,GAkBzC,WAAAvvG,CAAam3B,GACZ1rB,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAElB1iG,KAAKyR,IAAK,aAAa,GAEvBzR,KAAKsX,SAAWtX,KAAKsjG,mBAErBtjG,KAAK4jG,YAAa,CACjBzsF,IAAK,KAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,gBACAv3B,EAAK27F,GAAI,YAAa,aAAanqG,IAAUA,KAE9C4wG,KAAM,gBAGP9xF,SAAUtX,KAAKsX,UAEjB,CAKO,KAAA21B,GACJjtC,KAAKsX,SAASmZ,MAAyBwc,OAC1C,ECnDc,MAAMskE,WAA0B,GAI9C,WAAAxvG,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAK4jG,YAAa,CACjBzsF,IAAK,KACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,wBAIJ,E,eC7BG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQzE,OCgCR,MAAM,WAAwB,GAgI5C,WAAA/3B,CAAam3B,GACZ1rB,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAGlB1iG,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,kBAAc,GACxBzR,KAAKyR,IAAK,YAAQ,GAClBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,QAAQ,GAClBzR,KAAKyR,IAAK,gBAAgB,GAC1BzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,iBAAa,GACvBzR,KAAKyR,IAAK,iBAAiB,GAC3BzR,KAAKyR,IAAK,aAAS,GACnBzR,KAAKyR,IAAK,YAAa,GACvBzR,KAAKyR,IAAK,WAAW,GACrBzR,KAAKyR,IAAK,kBAAmB,KAC7BzR,KAAKyR,IAAK,OAAQ,UAClBzR,KAAKyR,IAAK,YAAY,GAEtBzR,KAAKsX,SAAWtX,KAAKsjG,mBACrBtjG,KAAKwxG,WAAaxxG,KAAKyxG,oBACvBzxG,KAAKwsG,UAAYxsG,KAAKysG,mBACtBzsG,KAAKo/F,WAAa,IAAI,GACtBp/F,KAAKwqG,aAAe,IAAI,GAExBxqG,KAAK4jG,YAAa,CACjBzsF,IAAK,MAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,iBACAv3B,EAAKzH,GAAI,SACTyH,EAAK27F,GAAI,YAAa,aAAanqG,IAAUA,IAC7CwH,KAAKwsG,UAAU9J,aAAaC,GAAI,OAAQ,yBAI1CrrF,SAAUtX,KAAKsX,UAEjB,CAKgB,MAAAq6B,GACfnkC,MAAMmkC,SAEN3xC,KAAKsX,SAASzJ,IAAK7N,KAAKwxG,YACxBxxG,KAAKsX,SAASzJ,IAAK7N,KAAKwsG,WAExBxsG,KAAKwqG,aAAa38F,IAAK7N,KAAKwxG,WAAW/5F,SACvCzX,KAAKwqG,aAAa38F,IAAK7N,KAAKwsG,UAAU/0F,SAEtCzX,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,SAG/BzX,KAAKo/F,WAAW3tF,IAAK,cAAc,CAAE4D,EAAKoe,KACpCzzB,KAAKwqG,aAAa53E,iBAAmB5yB,KAAKwxG,WAAW/5F,UACzDzX,KAAKwsG,UAAUv/D,QAEfxZ,IACD,IAIDzzB,KAAKo/F,WAAW3tF,IAAK,aAAa,CAAE4D,EAAKoe,KACnCzzB,KAAKwqG,aAAa53E,iBAAmB5yB,KAAKwsG,UAAU/0F,UACxDzX,KAAKwxG,WAAWvkE,QAEhBxZ,IACD,GAEF,CAKgB,OAAAzN,GACfxY,MAAMwY,UAENhmB,KAAKwqG,aAAaxkF,UAClBhmB,KAAKo/F,WAAWp5E,SACjB,CAKO,KAAAinB,GACNjtC,KAAKwxG,WAAWvkE,OACjB,CAMQ,iBAAAwkE,GACP,MAAMD,EAAa,IAAI,GAwBvB,OAtBAA,EAAWxqG,KACV,OACA,YACA,OACA,eACA,YACA,QACA,WACA,UACA,kBACA,OACA,YACCzH,GAAIS,MAENwxG,EAAW/O,eAAgB,CAC1Bl9F,WAAY,CACXg5B,MAAO,4BAITizE,EAAWpgG,SAAU,WAAY7R,GAAIS,MAE9BwxG,CACR,CAMQ,gBAAA/E,GACP,MAAMD,EAAY,IAAI,GAChBxlG,EAAOwlG,EAAU9J,aAqBvB,OAnBA8J,EAAU9C,KAAO,GAEjB8C,EAAU/J,eAAgB,CACzBl9F,WAAY,CACXg5B,MAAO,CACN,yBAED,4BAA6Bv3B,EAAKzH,GAAI,QACtC,iBAAiB,EACjB,gBAAiByH,EAAKzH,GAAI,QAAQ/G,GAASyiB,OAAQziB,QAIrDg0G,EAAUxlG,KAAM,aAAczH,GAAIS,MAClCwsG,EAAUxlG,KAAM,SAAUzH,GAAIS,MAC9BwsG,EAAUxlG,KAAM,WAAYzH,GAAIS,MAEhCwsG,EAAUp7F,SAAU,WAAY7R,GAAIS,KAAM,QAEnCwsG,CACR,E,eCpUG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ1yE,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCkGhB,SAAS,GACfZ,EACAw4E,EAAyE,IAEzE,MAAMvG,EAAa,IAAIuG,EAAax4E,GAE9BkyE,EAAY,IAAIN,GAAmB5xE,GACnCg2E,EAAe,IAAI7C,GAAcnzE,EAAQiyE,EAAYC,GAY3D,OAVAD,EAAWnkG,KAAM,aAAczH,GAAI2vG,GAE9B/D,aAAsB,GAC1BA,EAAWqB,UAAUxlG,KAAM,QAASzH,GAAI2vG,EAAc,UAEtD/D,EAAWnkG,KAAM,QAASzH,GAAI2vG,EAAc,UAuT9C,SAA6BA,IAY7B,SAAsCA,GACrCA,EAAa7/F,GAAuB,UAAU,KAC7C,GAAqB,CACpBO,QAASs/F,EACTlN,UAAW,IAAMkN,EAAa7D,OAC9B97F,SAAU,KACT2/F,EAAa7D,QAAS,CAAK,EAE5BpJ,gBAAiB,IAAM,CACtBiN,EAAaz3F,WACRy3F,EAAa1E,aAAah4E,YAE9B,GAEL,EAzBCm/E,CAA6BzC,GA8B9B,SAAiCA,GAEhCA,EAAa7/F,GAAwB,WAAWgG,IAE1CA,EAAInS,kBAAkB,KAI3BgsG,EAAa7D,QAAS,EAAK,GAE7B,CAvCCuG,CAAwB1C,GA4CzB,SAA8BA,GAC7BA,EAAa1E,aAAan7F,GAAoC,oBAAoB,CAAEgG,EAAKrT,EAAM8wB,KACzFo8E,EAAa7D,SAAWv4E,IAC5Bo8E,EAAa7D,QAAS,EACvB,GAEF,CAjDCwG,CAAqB3C,GAsDtB,SAAwCA,GAEvCA,EAAa9P,WAAW3tF,IAAK,aAAa,CAAE3O,EAAM2wB,KAC5Cy7E,EAAa7D,SACjB6D,EAAa9D,UAAUn+D,QACvBxZ,IACD,IAIDy7E,EAAa9P,WAAW3tF,IAAK,WAAW,CAAE3O,EAAM2wB,KAC1Cy7E,EAAa7D,SACjB6D,EAAa9D,UAAUF,YACvBz3E,IACD,GAEF,CArECq+E,CAA+B5C,GA2EhC,SAAqCA,GACpCA,EAAa7/F,GAAoC,iBAAiB,CAAEgG,EAAKrT,EAAMqpG,KAC9E,GAAKA,EACJ,OAGD,MAAM5zF,EAAUy3F,EAAa9D,UAAU3zF,QAKlCA,GAAWA,EAAQyM,SAAU,GAAO1gB,SAAS0+C,gBACjDgtD,EAAa/D,WAAWl+D,OACzB,GAEF,CAzFC8kE,CAA4B7C,GA8F7B,SAAmCA,GAClCA,EAAa7/F,GAAoC,iBAAiB,CAAEgG,EAAKrT,EAAMqpG,KACxEA,GAKN6D,EAAa9D,UAAUn+D,OAAO,GAK5B,CAAErgC,SAAU,OAChB,CA1GColG,CAA0B9C,EAC3B,CA3TC+C,CAAoB/C,GAEbA,CACR,CA8CO,SAASC,GACfD,EACAgD,EACA3tG,EAOI,CAAC,GAEL2qG,EAAazM,eAAgB,CAC5Bl9F,WAAY,CACXg5B,MAAO,CAAE,0BAIN2wE,EAAa7D,OACjB8G,GAA0BjD,EAAcgD,EAAmB3tG,GAE3D2qG,EAAaz/F,KACZ,iBACA,IAAM0iG,GAA0BjD,EAAcgD,EAAmB3tG,IACjE,CAAEqI,SAAU,YAITrI,EAAQ6tG,qCAEZ,GAA0BlD,GAAc,IAAMA,EAAaE,YAAax+E,MAAMW,MAAQtxB,GAAeA,EAAKoyG,QAE5G,CAKA,SAASF,GACRjD,EACAgD,EACA3tG,GAQA,MAAM20B,EAASg2E,EAAah2E,OACtBl+B,EAAIk+B,EAAOl+B,EAEXo0G,EAAcF,EAAaE,YAAc,IAAI,GAAal2E,GAC1Do5E,EAAsC,mBAArBJ,EAAkCA,IAAsBA,EAE/E9C,EAAYmD,UAAYhuG,EAAQguG,WAAav3G,EAAG,oBAE3CuJ,EAAQypG,WACZoB,EAAYpB,SAAWzpG,EAAQypG,UAG3BzpG,EAAQg6B,QACZ6wE,EAAY7wE,MAAQh6B,EAAQg6B,OAGxBh6B,EAAQiuG,YACZpD,EAAYoD,UAAYjuG,EAAQiuG,WAG5BjuG,EAAQkuG,aACZrD,EAAYqD,YAAa,GAGrBH,aAAmB,GACvBlD,EAAYx+E,MAAM1c,OAAQo+F,GAAUxgF,OAAO7xB,GAAQA,IAEnDmvG,EAAYx+E,MAAMD,QAAS2hF,GAG5BpD,EAAa9D,UAAU9zF,SAASzJ,IAAKuhG,GACrCA,EAAYx+E,MAAMxf,SAAU,WAAY7R,GAAI2vG,EAC7C,CAoDO,SAAS,GACfA,EACAwD,EACAnuG,EAGI,CAAC,GAEA2qG,EAAa7D,OACjBsH,GAAuBzD,EAAcwD,EAAiBnuG,GAEtD2qG,EAAaz/F,KACZ,iBACA,IAAMkjG,GAAuBzD,EAAcwD,EAAiBnuG,IAC5D,CAAEqI,SAAU,YAKd,GAA0BsiG,GAAc,IAAMA,EAAa0D,SAAUhiF,MAAMW,MAAMtxB,GAC3EA,aAAgBqxG,IACXrxG,EAAKqX,SAASmZ,MAAe4hF,QAKzC,CAKA,SAASM,GACRzD,EACAwD,EACAnuG,GAKA,MAAM20B,EAASg2E,EAAah2E,OAEtB05E,EAAW1D,EAAa0D,SAAW,IAAIvB,GAAUn4E,GACjDtI,EAAkC,mBAAnB8hF,EAAgCA,IAAoBA,EAEzEE,EAASL,UAAYhuG,EAAQguG,UAC7BK,EAASxJ,KAAO7kG,EAAQ6kG,KAExBwJ,EAAShiF,MAAM1c,OAAQ0c,GAAQkB,OAAO2jD,IACrC,GAAkB,cAAbA,EAAIxqE,KACR,OAAO,IAAIsmG,GAAmBr4E,GACxB,GAAkB,WAAbu8C,EAAIxqE,MAAkC,iBAAbwqE,EAAIxqE,KAA0B,CAClE,MAAM4nG,EAAe,IAAIvB,GAAcp4E,GACvC,IAAIiyE,EAcJ,OAXCA,EADiB,WAAb11B,EAAIxqE,KACK,IAAI,GAAYiuB,GAEhB,IAAI,GAAkBA,GAIpCiyE,EAAWnkG,QAAShQ,OAAOC,KAAMw+E,EAAIr9E,QAAqCmH,GAAIk2E,EAAIr9E,OAClF+yG,EAAW/5F,SAAU,WAAY7R,GAAIszG,GAErCA,EAAav7F,SAASzJ,IAAKs9F,GAEpB0H,CACR,CAEA,OAAO,IAAI,IAGZ3D,EAAa9D,UAAU9zF,SAASzJ,IAAK+kG,GAErCA,EAAShiF,MAAMxf,SAAU,WAAY7R,GAAI2vG,EAC1C,CAYO,SAAS,GACfA,EACA4D,GAEA5D,EAAa7/F,GAA2B,iBAAiB,KACxD,IAAM6/F,EAAa7D,OAClB,OAGD,MAAM0H,EAA+DD,IAE/DC,IAI6B,mBAAvBA,EAAa9lE,MACxB8lE,EAAa9lE,QAab,EAAY,sDAAuD,CAAEhT,KAAM84E,IAC5E,GAIE,CAAEnmG,SAAU,EAAWI,IAAM,IACjC,CClYO,SAAS,GACfgmG,EACAhJ,EACAC,GAEA,MAAMgJ,EAAY,IAAI,GAAeD,EAAiB95E,QAkBtD,OAhBA+5E,EAAUxhG,IAAK,CACd5R,GAAImqG,EACJkJ,kBAAmBjJ,IAGpBgJ,EAAUjsG,KAAM,cAAezH,GAAIyzG,EAAkB,aAAax6G,IAAUA,IAC5Ey6G,EAAUjsG,KAAM,YAAazH,GAAIyzG,EAAkB,aAAax6G,KAAWA,IAE3Ey6G,EAAU5jG,GAAyB,SAAS,KAG3C2jG,EAAiB1I,UAAY,IAAI,IAGlC0I,EAAiBhsG,KAAM,UAAW,YAAa,eAAgBzH,GAAI0zG,GAE5DA,CACR,CCjEO,MAAM,GAAQ,CAAC5yE,EAAQznC,EAAM,EAAGE,EAAM,IAClCunC,EAASvnC,EAAMA,EAAMunC,EAASznC,EAAMA,EAAMynC,EAExC,GAAQ,CAACA,EAAQ8yE,EAAS,EAAGzuG,EAAO7L,KAAKu6G,IAAI,GAAID,KACnDt6G,KAAKuD,MAAMsI,EAAO27B,GAAU37B,ECI1B2uG,IAHGx6G,KAAKgD,GAGKhE,IACP,MAAXA,EAAI,KACJA,EAAMA,EAAI+E,UAAU,IACpB/E,EAAIS,OAAS,EACN,CACHG,EAAG2E,SAASvF,EAAI,GAAKA,EAAI,GAAI,IAC7Ba,EAAG0E,SAASvF,EAAI,GAAKA,EAAI,GAAI,IAC7Bc,EAAGyE,SAASvF,EAAI,GAAKA,EAAI,GAAI,IAC7B8D,EAAkB,IAAf9D,EAAIS,OAAe,GAAM8E,SAASvF,EAAI,GAAKA,EAAI,GAAI,IAAM,IAAK,GAAK,GAGvE,CACHY,EAAG2E,SAASvF,EAAI+E,UAAU,EAAG,GAAI,IACjClE,EAAG0E,SAASvF,EAAI+E,UAAU,EAAG,GAAI,IACjCjE,EAAGyE,SAASvF,EAAI+E,UAAU,EAAG,GAAI,IACjCjB,EAAkB,IAAf9D,EAAIS,OAAe,GAAM8E,SAASvF,EAAI+E,UAAU,EAAG,GAAI,IAAM,IAAK,GAAK,KA6BrE02G,GAAa,EAAGt6G,IAAGC,IAAGK,IAAGqC,QAClC,MAAM43G,GAAO,IAAMt6G,GAAKK,EAAK,IAC7B,MAAO,CACHN,EAAG,GAAMA,GACTC,EAAG,GAAMs6G,EAAK,GAAKA,EAAK,IAAQt6G,EAAIK,EAAK,KAAOi6G,GAAM,IAAMA,EAAK,IAAMA,GAAO,IAAM,GACpFr6G,EAAG,GAAMq6G,EAAK,GACd53G,EAAG,GAAMA,EAAG,GACf,EAUQ63G,GAAmBC,IAC5B,MAAM,EAAEz6G,EAAC,EAAEC,EAAC,EAAEC,GAAMo6G,GAAWG,GAC/B,MAAO,OAAOz6G,MAAMC,OAAOC,KAAK,EAMvBw6G,GAAa,EAAG16G,IAAGC,IAAGK,IAAGqC,QAClC3C,EAAKA,EAAI,IAAO,EAChBC,GAAQ,IACRK,GAAQ,IACR,MAAMi6G,EAAK16G,KAAK+B,MAAM5B,GAAIL,EAAIW,GAAK,EAAIL,GAAIQ,EAAIH,GAAK,GAAKN,EAAIu6G,GAAMt6G,GAAIgP,EAAI3O,GAAK,GAAK,EAAIN,EAAIu6G,GAAMt6G,GAAIxC,EAAS88G,EAAK,EACrH,MAAO,CACH96G,EAAG,GAAmC,IAA7B,CAACa,EAAGG,EAAGd,EAAGA,EAAGsP,EAAG3O,GAAG7C,IAC5BiC,EAAG,GAAmC,IAA7B,CAACuP,EAAG3O,EAAGA,EAAGG,EAAGd,EAAGA,GAAGlC,IAC5BkC,EAAG,GAAmC,IAA7B,CAACA,EAAGA,EAAGsP,EAAG3O,EAAGA,EAAGG,GAAGhD,IAC5BkF,EAAG,GAAMA,EAAG,GACf,EAoCCg4G,GAAUtzE,IACZ,MAAMxoC,EAAMwoC,EAAO3jC,SAAS,IAC5B,OAAO7E,EAAIS,OAAS,EAAI,IAAMT,EAAMA,CAAG,EAE9B+7G,GAAY,EAAGn7G,IAAGC,IAAGC,IAAGgD,QACjC,MAAMk4G,EAAWl4G,EAAI,EAAIg4G,GAAO,GAAU,IAAJh4G,IAAY,GAClD,MAAO,IAAMg4G,GAAOl7G,GAAKk7G,GAAOj7G,GAAKi7G,GAAOh7G,GAAKk7G,CAAQ,EAEhDC,GAAa,EAAGr7G,IAAGC,IAAGC,IAAGgD,QAClC,MAAM7C,EAAMD,KAAKC,IAAIL,EAAGC,EAAGC,GACrBI,EAAQD,EAAMD,KAAKD,IAAIH,EAAGC,EAAGC,GAE7B46G,EAAKx6G,EACLD,IAAQL,GACHC,EAAIC,GAAKI,EACVD,IAAQJ,EACJ,GAAKC,EAAIF,GAAKM,EACd,GAAKN,EAAIC,GAAKK,EACtB,EACN,MAAO,CACHC,EAAG,GAAM,IAAMu6G,EAAK,EAAIA,EAAK,EAAIA,IACjCt6G,EAAG,GAAMH,EAAOC,EAAQD,EAAO,IAAM,GACrCQ,EAAG,GAAOR,EAAM,IAAO,KACvB6C,IACH,ECrJQo4G,GAAoB,CAACtjF,EAAOujF,KACrC,GAAIvjF,IAAUujF,EACV,OAAO,EACX,IAAK,MAAMzrG,KAAQkoB,EAMf,GAAIA,EAAMloB,KACNyrG,EAAOzrG,GACP,OAAO,EAEf,OAAO,CAAI,ECdTy3B,GAAQ,CAAC,EACFi0E,GAAOnvD,IAChB,IAAI++C,EAAW7jE,GAAM8kB,GAMrB,OALK++C,IACDA,EAAWrgG,SAAS8B,cAAc,YAClCu+F,EAASp9E,UAAYq+B,EACrB9kB,GAAM8kB,GAAQ++C,GAEXA,CAAQ,EAENrzF,GAAO,CAAC5M,EAAQqH,EAAMipG,KAC/BtwG,EAAOuwG,cAAc,IAAIC,YAAYnpG,EAAM,CACvCopG,SAAS,EACTH,WACD,ECZP,IAAII,IAAa,EAEjB,MAAMC,GAAWrwG,GAAM,YAAaA,EAU9BswG,GAAc,CAAC5wG,EAAQ0L,KACzB,MAAMmlG,EAAUF,GAAQjlG,GAASA,EAAMolG,QAAQ,GAAKplG,EAC9C8T,EAAOxf,EAAO2iB,GAAG7D,wBACvBlS,GAAK5M,EAAO2iB,GAAI,OAAQ3iB,EAAO+wG,QAAQ,CACnCz6G,EAAG,IAAOu6G,EAAQG,OAASxxF,EAAKrB,KAAOxe,OAAOsxG,cAAgBzxF,EAAKP,OACnE5oB,EAAG,IAAOw6G,EAAQK,OAAS1xF,EAAK3B,IAAMle,OAAOwxG,cAAgB3xF,EAAKN,UACnE,EAgCA,MAAMkyF,GACT,WAAAjzG,CAAYzL,EAAMwoB,EAAMm2F,EAAMC,GAC1B,MAAMrR,EAAWoQ,GAAI,yCAAyCn1F,MAASm2F,gBAAmBn2F,2BAC1FxoB,EAAKqP,YAAYk+F,EAAS3jG,QAAQi1G,WAAU,IAC5C,MAAM5uF,EAAKjwB,EAAKwN,cAAc,SAASgb,MACvCyH,EAAGjW,iBAAiB,YAAatQ,MACjCumB,EAAGjW,iBAAiB,aAActQ,MAClCumB,EAAGjW,iBAAiB,UAAWtQ,MAC/BA,KAAKumB,GAAKA,EACVvmB,KAAKk1G,GAAKA,EACVl1G,KAAKknC,MAAQ,CAAC3gB,EAAG5f,WAAY4f,EACjC,CACA,YAAI6uF,CAASznB,GACT,MAAM0nB,EAAc1nB,EAAQnqF,SAAS8M,iBAAmB9M,SAAS+M,oBACjE8kG,EAAYf,GAAa,YAAc,YAAat0G,MACpDq1G,EAAYf,GAAa,WAAa,UAAWt0G,KACrD,CACA,WAAAs1G,CAAYhmG,GACR,OAAQA,EAAMrE,MACV,IAAK,YACL,IAAK,aAGD,GAFAqE,EAAM4jB,kBAlEN,CAAC5jB,KACTglG,KAAeC,GAAQjlG,KAEtBglG,KACDA,GAAaC,GAAQjlG,IAClB,IA+DUimG,CAAQjmG,KAAYglG,IAA8B,GAAhBhlG,EAAMkmG,OACzC,OACJx1G,KAAKumB,GAAG0mB,QACRunE,GAAYx0G,KAAMsP,GAClBtP,KAAKo1G,UAAW,EAChB,MACJ,IAAK,YACL,IAAK,YACD9lG,EAAM4jB,iBACNshF,GAAYx0G,KAAMsP,GAClB,MACJ,IAAK,UACL,IAAK,WACDtP,KAAKo1G,UAAW,EAChB,MACJ,IAAK,UApED,EAACxxG,EAAQ0L,KAErB,MAAMge,EAAUhe,EAAMge,QAElBA,EAAU,IAAO1pB,EAAOsxG,IAAM5nF,EAAU,IAAOA,EAAU,KAG7Dhe,EAAM4jB,iBAEN1iB,GAAK5M,EAAO2iB,GAAI,OAAQ3iB,EAAO+wG,QAAQ,CACnCz6G,EAAe,KAAZozB,EACG,IACY,KAAZA,GACK,IACW,KAAZA,EACI,IACY,KAAZA,GACK,IACW,KAAZA,EACI,EACY,KAAZA,GACK,EACD,EAC1BrzB,EAAe,KAAZqzB,EACG,IACY,KAAZA,GACK,IACD,IACX,IAAM,EAyCGmoF,CAAQz1G,KAAMsP,GAG1B,CACA,KAAAjK,CAAMw5B,GACFA,EAAOjhC,SAAQ,CAACyH,EAAO7K,KACnB,IAAK,MAAMM,KAAKuK,EACZrF,KAAKknC,MAAM1sC,GAAG6K,MAAMqwG,YAAY56G,EAAGuK,EAAMvK,GAC7C,GAER,EClGG,MAAM66G,WAAYX,GACrB,WAAAjzG,CAAYzL,GACRkX,MAAMlX,EAAM,MAAO,0DAA0D,EACjF,CACA,MAAAwQ,EAAO,EAAE9N,IACLgH,KAAKhH,EAAIA,EACTgH,KAAKqF,MAAM,CACP,CACI0c,KAAU/oB,EAAI,IAAO,IAAf,IACNsD,MAAOk3G,GAAgB,CAAEx6G,IAAGC,EAAG,IAAKK,EAAG,IAAKqC,EAAG,OAGvDqE,KAAKumB,GAAG9gB,aAAa,gBAAiB,GAAG,GAAMzM,KACnD,CACA,OAAA27G,CAAQ/4F,EAAQ7kB,GAEZ,MAAO,CAAEiC,EAAGjC,EAAM,GAAMiJ,KAAKhH,EAAe,IAAX4iB,EAAO1hB,EAAS,EAAG,KAAO,IAAM0hB,EAAO1hB,EAC5E,ECjBG,MAAM07G,WAAmBZ,GAC5B,WAAAjzG,CAAYzL,GACRkX,MAAMlX,EAAM,aAAc,sBAAsB,EACpD,CACA,MAAAwQ,CAAO2sG,GACHzzG,KAAKyzG,KAAOA,EACZzzG,KAAKqF,MAAM,CACP,CACIoc,IAAQ,IAAMgyF,EAAKn6G,EAAd,IACLyoB,KAAM,GAAG0xF,EAAKx6G,KACdqD,MAAOk3G,GAAgBC,IAE3B,CACI,mBAAoBD,GAAgB,CAAEx6G,EAAGy6G,EAAKz6G,EAAGC,EAAG,IAAKK,EAAG,IAAKqC,EAAG,OAG5EqE,KAAKumB,GAAG9gB,aAAa,iBAAkB,cAAc,GAAMguG,EAAKx6G,mBAAmB,GAAMw6G,EAAKn6G,MAClG,CACA,OAAAq7G,CAAQ/4F,EAAQ7kB,GAEZ,MAAO,CACHkC,EAAGlC,EAAM,GAAMiJ,KAAKyzG,KAAKx6G,EAAe,IAAX2iB,EAAO1hB,EAAS,EAAG,KAAkB,IAAX0hB,EAAO1hB,EAC9DZ,EAAGvC,EAAM,GAAMiJ,KAAKyzG,KAAKn6G,EAAe,IAAXsiB,EAAO3hB,EAAS,EAAG,KAAOpB,KAAKuD,MAAM,IAAiB,IAAXwf,EAAO3hB,GAEvF,EC3BJ,MCOM47G,GAAU/0G,OAAO,QACjBg1G,GAASh1G,OAAO,SAChBi1G,GAAQj1G,OAAO,QACfk1G,GAAUl1G,OAAO,UACjBm1G,GAASn1G,OAAO,SACTo1G,GAAOp1G,OAAO,OACdq1G,GAAWr1G,OAAO,WACxB,MAAMs1G,WAAoBxyF,YAC7B,6BAAWyyF,GACP,MAAO,CAAC,QACZ,CACA,IAAKH,MACD,MAAO,CDnBA,8wBEAA,kKCAA,8SFoBX,CACA,IAAKC,MACD,MAAO,CAACP,GAAYD,GACxB,CACA,SAAIr5G,GACA,OAAO0D,KAAK81G,GAChB,CACA,SAAIx5G,CAAMg6G,GACN,IAAKt2G,KAAK61G,IAASS,GAAW,CAC1B,MAAMC,EAAUv2G,KAAKw2G,WAAWC,OAAOH,GACvCt2G,KAAKg2G,IAASO,GACdv2G,KAAK81G,IAAUQ,CACnB,CACJ,CACA,WAAAv0G,GACIyL,QACA,MAAMq2F,EAAWoQ,GAAI,UAAUj0G,KAAKk2G,IAAMh5G,KAAK,eACzC5G,EAAO0J,KAAK02G,aAAa,CAAEC,KAAM,SACvCrgH,EAAKqP,YAAYk+F,EAAS3jG,QAAQi1G,WAAU,IAC5C7+G,EAAKga,iBAAiB,OAAQtQ,MAC9BA,KAAKi2G,IAAUj2G,KAAKm2G,IAAUn5G,KAAK45G,GAAW,IAAIA,EAAOtgH,IAC7D,CACA,iBAAAugH,GAII,GAAI72G,KAAKwI,eAAe,SAAU,CAC9B,MAAMhQ,EAAQwH,KAAK1D,aACZ0D,KAAY,MACnBA,KAAK1D,MAAQ9D,CACjB,MACUwH,KAAK1D,QACX0D,KAAK1D,MAAQ0D,KAAKw2G,WAAWM,aAErC,CACA,wBAAAC,CAAyBC,EAAOC,EAASC,GACrC,MAAM56G,EAAQ0D,KAAKw2G,WAAWW,SAASD,GAClCl3G,KAAK61G,IAASv5G,KACf0D,KAAK1D,MAAQA,EAErB,CACA,WAAAg5G,CAAYhmG,GAER,MAAM8nG,EAAUp3G,KAAK+1G,IACfQ,EAAU,IAAKa,KAAY9nG,EAAM4kG,QAEvC,IAAIoC,EADJt2G,KAAKg2G,IAASO,GAETxC,GAAkBwC,EAASa,IAC3Bp3G,KAAK61G,IAAUS,EAAWt2G,KAAKw2G,WAAWa,SAASd,MACpDv2G,KAAK81G,IAAUQ,EACf9lG,GAAKxQ,KAAM,gBAAiB,CAAExH,MAAO89G,IAE7C,CACA,CAACT,IAASv5G,GACN,OAAO0D,KAAK1D,OAAS0D,KAAKw2G,WAAW14D,MAAMxhD,EAAO0D,KAAK1D,MAC3D,CACA,CAAC05G,IAASvC,GACNzzG,KAAK+1G,IAAStC,EACdzzG,KAAKi2G,IAAQr4G,SAASkhB,GAASA,EAAKhY,OAAO2sG,IAC/C,EG5EJ,MAAM+C,GAAa,CACfM,aAAc,OACdL,OVKsB5+G,GAAQi8G,GAAWT,GAAUx7G,IUJnDw/G,SAAU,EAAGr+G,IAAGC,IAAGK,OVgDYs6G,GAAUF,GUhDJ,CAAE16G,IAAGC,IAAGK,IAAGqC,EAAG,KACnDmiD,MTYoB,CAACrtB,EAAOujF,IACxBvjF,EAAM7nB,gBAAkBorG,EAAOprG,eAG5BmrG,GAAkBV,GAAU5iF,GAAQ4iF,GAAUW,ISfrDmD,SAAW76G,GAAUA,GAElB,MAAMg7G,WAAgBlB,GACzB,cAAII,GACA,OAAOA,EACX,ECKJe,eAAe7gH,OAAO,mBAFf,cAA6B4gH,M,eCbhC,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQx9E,OCS8B3mB,EAAiB,I,wIClBlE,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ2mB,OC0BR,MAAM09E,GAgBpB,WAAAz1G,CAAauyB,GAPb,KAAiBmjF,YAAc,IAAIlmG,IAQlCvR,KAAKs0B,OAASA,CACf,CAKA,MAAQ3V,GACP,UAAYnmB,KAASwH,KAAKy3G,YAAYvsG,eAC/B1S,EAAMk/G,YAEd,CAWO,GAAA7pG,CAAK7L,EAAcuN,GACzBvP,KAAKy3G,YAAYhmG,IAAK2xB,GAAephC,GAAQ,CAAEuN,WAAUmoG,aAAc11G,GACxE,CAYO,MAAAsb,CAAQtb,GACd,IAAMhC,KAAK4N,IAAK5L,GASf,MAAM,IAAI,EACT,gCACAhC,KACA,CAAEgC,SAIJ,OAAOhC,KAAKy3G,YAAYrvG,IAAKg7B,GAAephC,IAAUuN,SAAUvP,KAAKs0B,OAAO4E,OAC7E,CAOO,GAAAtrB,CAAK5L,GACX,OAAOhC,KAAKy3G,YAAY7pG,IAAKw1B,GAAephC,GAC7C,EAMD,SAASohC,GAAephC,GACvB,OAAOiZ,OAAQjZ,GAAO4G,aACvB,C,eC1HI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQkxB,O,0VCgBvB,MAAM69E,GAAOjxF,GAAQ,MACfkxF,GAAwB,GAAOp0G,SAASgiB,KAyCzB,GAArB,cAA8C,GAyE7C,WAAAzjB,CAAam3B,GACZ1rB,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAElB1iG,KAAKyR,IAAK,MAAO,GACjBzR,KAAKyR,IAAK,OAAQ,GAClBzR,KAAKyR,IAAK,WAAY,YACtBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,aAAa,GACvBzR,KAAKyR,IAAK,aAAS,GAEnBzR,KAAK63G,0BAA4B,KACjC73G,KAAKE,QAAUF,KAAKsjG,mBAEpBtjG,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,mBACAv3B,EAAKzH,GAAI,YAAY/G,GAAS,oBAAqBA,MACnDwO,EAAK27F,GAAI,YAAa,4BACtB37F,EAAK27F,GAAI,YAAa,+BACtB37F,EAAKzH,GAAI,UAGV8F,MAAO,CACNoc,IAAKza,EAAKzH,GAAI,MAAOo4G,IACrB51F,KAAM/a,EAAKzH,GAAI,OAAQo4G,MAIzBrgG,SAAUtX,KAAKE,SAEjB,CAOO,IAAA43G,GACN93G,KAAKknB,WAAY,CAClB,CAOO,IAAA6wF,GACN/3G,KAAKknB,WAAY,CAClB,CAkCO,QAAA8wF,CAAUzzG,GAChBvE,KAAK83G,OAEL,MAAMG,EAAmB,GAAiBA,iBACpChwF,EAAkBjxB,OAAO+wB,OAAQ,CAAC,EAAG,CAC1CtQ,QAASzX,KAAKyX,QACd2P,UAAW,CACV6wF,EAAiBC,gBACjBD,EAAiBE,0BACjBF,EAAiBG,0BACjBH,EAAiBI,oBACjBJ,EAAiBK,oBACjBL,EAAiBM,gBACjBN,EAAiBO,0BACjBP,EAAiBQ,0BACjBR,EAAiBS,oBACjBT,EAAiBU,oBACjBV,EAAiBW,qBAElBvxF,QAASuwF,GACTtwF,eAAe,GACb/iB,GAEGs0G,EAAkB,GAAiBtN,oBAAqBtjF,GAIxDlG,EAAO3kB,SAAUy7G,EAAgB92F,MACjCN,EAAMrkB,SAAUy7G,EAAgBp3F,KAEhCgE,EAAWozF,EAAgB72G,KAC3BgnB,EAAkC6vF,EAAgB7vF,QAAU,CAAC,GAC7D,UAAE8vF,GAAY,GAAS9vF,EAE7BhpB,KAAKyhB,IAAMA,EACXzhB,KAAK+hB,KAAOA,EACZ/hB,KAAKylB,SAAWA,EAChBzlB,KAAK84G,UAAYA,CAClB,CAoCO,GAAAC,CAAKx0G,GACXvE,KAAKg5G,QAELh5G,KAAK63G,0BAA4B,KAC3B73G,KAAKknB,UACTlnB,KAAKi5G,cAAe10G,GAEpBvE,KAAKk5G,cACN,EAGDl5G,KAAKi5G,cAAe10G,GAKpBvE,KAAKwP,SAAiCxP,KAAM,mBAAoBA,KAAK63G,0BACtE,CAKO,KAAAmB,GACDh5G,KAAK63G,4BAET73G,KAAKk5G,eAILl5G,KAAK2P,cAAe3P,KAAM,mBAAoBA,KAAK63G,2BAEnD73G,KAAK63G,0BAA4B,KAEjC73G,KAAK+3G,OAEP,CAOQ,aAAAkB,CAAe10G,GACtBvE,KAAKg4G,SAAUzzG,GAEf,MAAMm9E,EAAgBy3B,GAAe50G,EAAQX,QACvConB,EAAiBzmB,EAAQ8iB,QAAU8xF,GAAe50G,EAAQ8iB,SAAYuwF,GAG5E53G,KAAKwP,SAAU,GAAOhM,SAAU,UAAU,CAAE6R,EAAK2L,KAChD,MAAMo4F,EAAep4F,EAAOpd,OAGtBy1G,EAAuB33B,GAAiB03B,EAAal1F,SAAUw9D,GAG/D43B,EAA8BtuF,GAAkBouF,EAAal1F,SAAU8G,IAIxEquF,IAAwBC,GAAgC53B,GAAkB12D,GAC9EhrB,KAAKg4G,SAAUzzG,EAChB,GACE,CAAEmb,YAAY,IAGjB1f,KAAKwP,SAAU,GAAOjM,OAAQ,UAAU,KACvCvD,KAAKg4G,SAAUzzG,EAAS,GAE1B,CAKQ,YAAA20G,GACPl5G,KAAK2P,cAAe,GAAOnM,SAAU,UACrCxD,KAAK2P,cAAe,GAAOpM,OAAQ,SACpC,GAvTD,IAAqBg2G,GAArB,GAmwBA,SAASJ,GAAe/jG,GACvB,OAAK,GAAWA,GACRA,EAGHkM,GAASlM,GACNA,EAAOuO,wBAGO,mBAAVvO,EACJ+jG,GAAe/jG,KAGhB,IACR,CAsBO,SAASokG,GAAmBj1G,EAK/B,CAAC,GACJ,MAAM,WACLk1G,EAAaF,GAAiBG,gBAAe,aAC7CC,EAAeJ,GAAiBK,kBAAiB,qBACjDC,EAAuBN,GAAiBM,qBAAoB,OAC5D7wF,GACGzkB,EAEJ,MAAO,CAGNu1G,wBAAyB,CAAElyF,EAAYmyF,IAAmB,IACzDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAAO03F,EACxBz3G,KAAM,YACDgnB,GAAU,CAAEA,WAGlBixF,8BAA+B,CAAEryF,EAAYmyF,IAAmB,IAC/Dt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAA6B,IAApBg4F,EAAYl3F,MAAgB42F,EACtDz3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlBkxF,oBAAqB,CAAEtyF,EAAYmyF,IAAmB,IACrDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAAOg4F,EAAYl3F,MAAQ,EAC5C7gB,KAAM,WACDgnB,GAAU,CAAEA,WAGlBmxF,8BAA+B,CAAEvyF,EAAYmyF,IAAmB,IAC/Dt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAA6B,IAApBg4F,EAAYl3F,MAAgB42F,EACtDz3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlBoxF,wBAAyB,CAAExyF,EAAYmyF,IAAmB,IACzDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAAOg4F,EAAYl3F,MAAQ42F,EAC5Cz3G,KAAM,YACDgnB,GAAU,CAAEA,WAKlB0vF,oBAAqB,CAAE9wF,EAAYmyF,IAAmB,IACrDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAAI42F,EAC/Cz3G,KAAM,YACDgnB,GAAU,CAAEA,WAGlBwvF,0BAA2B,CAAE5wF,EAAYmyF,IAAmB,IAC3Dt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAA0B,IAApBk3F,EAAYl3F,MAAgB42F,EAC7Ez3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlBuvF,gBAAiB,CAAE3wF,EAAYmyF,IAAmB,IACjDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAAIk3F,EAAYl3F,MAAQ,EACnE7gB,KAAM,WACDgnB,GAAU,CAAEA,WAGlByvF,0BAA2B,CAAE7wF,EAAYmyF,IAAmB,IAC3Dt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAA0B,IAApBk3F,EAAYl3F,MAAgB42F,EAC7Ez3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlB2vF,oBAAqB,CAAE/wF,EAAYmyF,IAAmB,IACrDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAAIk3F,EAAYl3F,MAAQ42F,EACnEz3G,KAAM,YACDgnB,GAAU,CAAEA,WAKlBqxF,wBAAyB,CAAEzyF,EAAYmyF,IAAmB,IACzDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAWjG,MAAQ83F,EACzBz3G,KAAM,YACDgnB,GAAU,CAAEA,WAGlBsxF,8BAA+B,CAAE1yF,EAAYmyF,IAAmB,IAC/Dt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAWjG,MAA8B,IAApBo4F,EAAYl3F,MAAgB42F,EACvDz3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlBuxF,oBAAqB,CAAE3yF,EAAYmyF,IAAmB,IACrDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAWjG,MAAQo4F,EAAYl3F,MAAQ,EAC7C7gB,KAAM,WACDgnB,GAAU,CAAEA,WAGlBwxF,8BAA+B,CAAE5yF,EAAYmyF,IAAmB,IAC/Dt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAWjG,MAA8B,IAApBo4F,EAAYl3F,MAAgB42F,EACvDz3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlByxF,wBAAyB,CAAE7yF,EAAYmyF,IAAmB,IACzDt4F,IAAKu4F,EAAapyF,EAAYmyF,GAC9Bh4F,KAAM6F,EAAWjG,MAAQo4F,EAAYl3F,MAAQ42F,EAC7Cz3G,KAAM,YACDgnB,GAAU,CAAEA,WAKlB0xF,wBAAyB9yF,GAAgB,IACxCnG,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAAO03F,EACxBz3G,KAAM,YACDgnB,GAAU,CAAEA,WAGlB4xF,8BAA+B,CAAEhzF,EAAYmyF,IAAmB,IAC/Dt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAA6B,IAApBg4F,EAAYl3F,MAAgB42F,EACtDz3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlB6xF,oBAAqB,CAAEjzF,EAAYmyF,IAAmB,IACrDt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAAOg4F,EAAYl3F,MAAQ,EAC5C7gB,KAAM,WACDgnB,GAAU,CAAEA,WAGlB8xF,8BAA+B,CAAElzF,EAAYmyF,IAAmB,IAC/Dt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAA6B,IAApBg4F,EAAYl3F,MAAgB42F,EACtDz3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlB+xF,wBAAyB,CAAEnzF,EAAYmyF,IAAmB,IACzDt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAAOg4F,EAAYl3F,MAAQ42F,EAC5Cz3G,KAAM,YACDgnB,GAAU,CAAEA,WAKlBqvF,oBAAqBzwF,GAAgB,IACpCnG,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAAI42F,EAC/Cz3G,KAAM,YACDgnB,GAAU,CAAEA,WAGlBmvF,0BAA2B,CAAEvwF,EAAYmyF,IAAmB,IAC3Dt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAA0B,IAApBk3F,EAAYl3F,MAAiB42F,EAC9Ez3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlBkvF,gBAAiB,CAAEtwF,EAAYmyF,IAAmB,IACjDt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAAIk3F,EAAYl3F,MAAQ,EACnE7gB,KAAM,WACDgnB,GAAU,CAAEA,WAGlBovF,0BAA2B,CAAExwF,EAAYmyF,IAAmB,IAC3Dt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAA0B,IAApBk3F,EAAYl3F,MAAiB42F,EAC9Ez3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlBsvF,oBAAqB,CAAE1wF,EAAYmyF,IAAmB,IACrDt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAAIk3F,EAAYl3F,MAAQ42F,EACnEz3G,KAAM,YACDgnB,GAAU,CAAEA,WAKlBgyF,wBAAyBpzF,GAAgB,IACxCnG,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAWjG,MAAQ83F,EACzBz3G,KAAM,YACDgnB,GAAU,CAAEA,WAGlBiyF,8BAA+B,CAAErzF,EAAYmyF,IAAmB,IAC/Dt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAWjG,MAA8B,IAApBo4F,EAAYl3F,MAAgB42F,EACvDz3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlBkyF,oBAAqB,CAAEtzF,EAAYmyF,IAAmB,IACrDt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAWjG,MAAQo4F,EAAYl3F,MAAQ,EAC7C7gB,KAAM,WACDgnB,GAAU,CAAEA,WAGlBmyF,8BAA+B,CAAEvzF,EAAYmyF,IAAmB,IAC/Dt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAWjG,MAA8B,IAApBo4F,EAAYl3F,MAAgB42F,EACvDz3G,KAAM,aACDgnB,GAAU,CAAEA,WAGlBoyF,wBAAyB,CAAExzF,EAAYmyF,IAAmB,IACzDt4F,IAAKk5F,EAAa/yF,GAClB7F,KAAM6F,EAAWjG,MAAQo4F,EAAYl3F,MAAQ42F,EAC7Cz3G,KAAM,YACDgnB,GAAU,CAAEA,WAKlBqyF,cAAe,CAAEzzF,EAAYmyF,IAAmB,IAC/Ct4F,IAAKmG,EAAWnG,IAAMmG,EAAW9E,OAAS,EAAIi3F,EAAYj3F,OAAS,EACnEf,KAAM6F,EAAW7F,KAAOg4F,EAAYl3F,MAAQ82F,EAC5C33G,KAAM,WACDgnB,GAAU,CAAEA,WAKlBsyF,cAAe,CAAE1zF,EAAYmyF,IAAmB,IAC/Ct4F,IAAKmG,EAAWnG,IAAMmG,EAAW9E,OAAS,EAAIi3F,EAAYj3F,OAAS,EACnEf,KAAM6F,EAAWjG,MAAQg4F,EACzB33G,KAAM,WACDgnB,GAAU,CAAEA,WAKlB4vF,oBAAqB,CAAEhxF,EAAYmyF,EAAajyF,IACzCF,EAAW1E,gBAAiB4E,GAI3B,CACNrG,IAAKqG,EAAcrG,IAAMo4F,EACzB93F,KAAM6F,EAAW7F,KAAO6F,EAAW/E,MAAQ,EAAIk3F,EAAYl3F,MAAQ,EACnE7gB,KAAM,YACNgnB,OAAQ,IACP8vF,WAAW,GACR9vF,IATG,MAqBV,SAASgxF,EAAapyF,EAAkBmyF,GACvC,OAAOnyF,EAAWnG,IAAMs4F,EAAYj3F,OAAS62F,CAC9C,CAOA,SAASgB,EAAa/yF,GACrB,OAAOA,EAAW/F,OAAS83F,CAC5B,CACD,CA1kCqBJ,GA2UNG,gBAAkB,GA3UZH,GA2WNK,kBAAoB,GA3WdL,GAoYNM,qBAAuB,GApYjBN,GAyYLhO,oBAAsBpkF,GAzYjBoyF,GA4vBNtB,iBAAmBuB,K,eC/zB9B,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ1/E,OAAvB,MCiBMyhF,GAAgB,aAwDD,GAArB,cAA4C,MA8D3C,WAAAx5G,CAAauyB,GAOZ,GANA9mB,QAvCD,KAAQguG,2BAAiD,KAKzD,KAAQC,wBAAkD,KAQ1D,KAAQC,gBAAyC,KA4BhD,GAAeC,SAAS9tG,IAAKymB,GAIxB,GAAesnF,UACnB,OAAO,GAAeA,UAGvB,GAAeA,UAAY57G,KAE3BA,KAAK67G,gBAAkB,IAAI,GAAMvnF,EAAO4E,QACxCl5B,KAAK67G,gBAAgBpqG,IAAK,OAAQ,IAClCzR,KAAK67G,gBAAgBjY,YAAa,CACjCzsF,IAAK,OACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,qBAGFjnB,SAAU,CACT,CACC4iB,KAAMl6B,KAAK67G,gBAAgBnZ,aAAanjG,GAAI,YAK/CS,KAAK87G,iBAAmB,IAAIvC,GAAkBjlF,EAAO4E,QACrDl5B,KAAK87G,iBAAiBv9E,MAAQg9E,GAC9Bv7G,KAAK87G,iBAAiB57G,QAAQ2N,IAAK7N,KAAK67G,iBAExC77G,KAAK+7G,qBAAuB,GAAU/7G,KAAKg8G,YAAa,KAExDh8G,KAAKwP,SAAU,GAAOhM,SAAU,aAAcxD,KAAKi8G,gBAAgBj1G,KAAMhH,MAAQ,CAAE0f,YAAY,IAC/F1f,KAAKwP,SAAU,GAAOhM,SAAU,aAAcxD,KAAKk8G,eAAel1G,KAAMhH,MAAQ,CAAE0f,YAAY,IAE9F1f,KAAKwP,SAAU,GAAOhM,SAAU,QAASxD,KAAKi8G,gBAAgBj1G,KAAMhH,MAAQ,CAAE0f,YAAY,IAC1F1f,KAAKwP,SAAU,GAAOhM,SAAU,OAAQxD,KAAKk8G,eAAel1G,KAAMhH,MAAQ,CAAE0f,YAAY,IAExF1f,KAAKwP,SAAU,GAAOhM,SAAU,SAAUxD,KAAKm8G,UAAUn1G,KAAMhH,MAAQ,CAAE0f,YAAY,IAOrF1f,KAAKo8G,mBAAoB,CAC1B,CASO,OAAAp2F,CAASsO,GACf,MAAM+nF,EAA2B/nF,EAAO6E,GAAGc,MAAQ3F,EAAO6E,GAAGc,KAAKzU,KAElE,GAAem2F,SAAShqG,OAAQ2iB,GAChCt0B,KAAK2P,cAAe2kB,EAAO6E,IAItBkjF,GAA4BA,EAAyBzuG,IAAK5N,KAAK87G,mBACnEO,EAAyBp2G,OAAQjG,KAAK87G,kBAGjC,GAAeH,SAAS9nG,OAC7B7T,KAAKs8G,gBACLt8G,KAAK87G,iBAAiB91F,UACtBhmB,KAAK2P,gBAEL,GAAeisG,UAAY,KAE7B,CASA,8BAAcW,CAAyB92F,GACtC,MAAMwyF,EAAmB,GAAeuE,wBAExC,MAAO,CAENvjH,EAAG,CACFg/G,EAAiBC,gBACjBD,EAAiBK,oBACjBL,EAAiBI,qBAElB/8G,EAAG,CAAE28G,EAAiBM,iBACtBr0G,EAAG,CAAE+zG,EAAiBqD,eACtB5hH,EAAG,CAAEu+G,EAAiBoD,eACtBoB,GAAI,CAAExE,EAAiBK,qBACvBoE,GAAI,CAAEzE,EAAiBI,sBACrB5yF,EACJ,CAQQ,eAAAw2F,CAAiB5mG,GAAc,OAAEzR,IACxC,MAAM+4G,EAA8BC,GAA0Bh5G,GA6LhE,IAAyB6T,EA1LjBklG,IAODA,IAAgC38G,KAAKw7G,6BAI1Cx7G,KAAKs8G,gBAELt8G,KAAK+7G,qBAAsBY,EA8KrB,CACNziF,MAFuBziB,EA7KiDklG,GA+K1D5hE,QAAQ8hE,eACtBp3F,SAAYhO,EAAQsjC,QAAQ+hE,oBAAsB,IAClDC,SAAUtlG,EAAQsjC,QAAQiiE,iBAAmB,MAhL9C,CAQQ,cAAAd,CAAgB7mG,GAAgB,OAAEzR,EAAM,cAAE8wD,IACjD,GAAkB,eAAbr/C,EAAIrT,KAAwB,CAEhC,IAAM,GAAW4B,GAChB,OAMD,GAAK5D,KAAKw7G,4BAA8B53G,IAAW5D,KAAKw7G,2BACvD,OAGD,MAAMyB,EAAwBL,GAA0Bh5G,GAClDs5G,EAA+BN,GAA0BloD,GAK1DuoD,GAAyBA,IAA0BC,GACvDl9G,KAAKs8G,eAEP,KACK,CAGJ,GAAKt8G,KAAKw7G,4BAA8B53G,IAAW5D,KAAKw7G,2BACvD,OAKDx7G,KAAKs8G,eACN,CACD,CAQQ,SAAAH,CAAW9mG,GAAc,OAAEzR,IAE5B5D,KAAKw7G,6BAON53G,EAAOsgB,SAAUlkB,KAAK87G,iBAAiBrkG,UAAa7T,EAAOsgB,SAAUlkB,KAAKw7G,6BAI/Ex7G,KAAKs8G,gBACN,CASQ,WAAAN,CACPmB,GACA,KAAEjjF,EAAI,SAAEzU,EAAQ,SAAEs3F,IAGlB,MAAMK,EAAqB,GAAO,GAAezB,SAASzwG,UAAYiuB,GAAGc,KAAKzU,KAExE43F,EAAmBxvG,IAAK5N,KAAK87G,mBAClCsB,EAAmBvvG,IAAK7N,KAAK87G,kBAG9B97G,KAAK67G,gBAAgB3hF,KAAOA,EAE5Bl6B,KAAK87G,iBAAiB/C,IAAK,CAC1Bn1G,OAAQu5G,EACR/1F,UAAW,GAAem1F,wBAAyB92F,KAGpDzlB,KAAK07G,gBAAkB,IAAI,GAAgByB,GAAkB,KAGtDj2F,GAAWi2F,IAChBn9G,KAAKs8G,eACN,IAGDt8G,KAAK87G,iBAAiBv9E,MAAQ,CAAEg9E,GAAewB,GAC7Ch3G,QAAQogC,GAAaA,IACrBjpC,KAAM,KAKR,UAAYo3B,KAAU,GAAeqnF,SACpC37G,KAAKwP,SAA+B8kB,EAAO6E,GAAI,SAAUn5B,KAAKq9G,uBAAuBr2G,KAAMhH,MAAQ,CAAE4M,SAAU,QAGhH5M,KAAKw7G,2BAA6B2B,EAClCn9G,KAAKy7G,wBAA0Bh2F,CAChC,CAKQ,aAAA62F,GACPt8G,KAAK+7G,qBAAqBtoF,SAE1BzzB,KAAK87G,iBAAiB9C,QAEtB,UAAY1kF,KAAU,GAAeqnF,SACpC37G,KAAK2P,cAAe2kB,EAAO6E,GAAI,UAGhCn5B,KAAKw7G,2BAA6B,KAClCx7G,KAAKy7G,wBAA0B,KAE1Bz7G,KAAK07G,iBACT17G,KAAK07G,gBAAgB11F,SAEvB,CAOQ,sBAAAq3F,GAGDn2F,GAAWlnB,KAAKw7G,4BAMtBx7G,KAAK87G,iBAAiB/C,IAAK,CAC1Bn1G,OAAQ5D,KAAKw7G,2BACbp0F,UAAW,GAAem1F,wBAAyBv8G,KAAKy7G,2BAPxDz7G,KAAKs8G,eASP,GAxVD,IAAqBgB,GAArB,GA6VA,SAASV,GAA0BnlG,GAClC,OAAM,GAAWA,GAIVA,EAAQ8lG,QAAS,4DAHhB,IAIT,CAnWqBD,GAeNd,wBAA0BhD,GAAmB,CAC1DG,aAAc,EACdF,WAAY,KAjBO6D,GAmDL3B,SAAW,IAAI/nG,IAnDV0pG,GAyDL1B,UAAmC,KC1EnD,SAlBA,SAAkBhkG,EAAM0b,EAAM/uB,GAC5B,IAAIgoD,GAAU,EACVE,GAAW,EAEf,GAAmB,mBAAR70C,EACT,MAAM,IAAIzV,UAnDQ,uBAyDpB,OAJI,EAASoC,KACXgoD,EAAU,YAAahoD,IAAYA,EAAQgoD,QAAUA,EACrDE,EAAW,aAAcloD,IAAYA,EAAQkoD,SAAWA,GAEnD,GAAS70C,EAAM0b,EAAM,CAC1B,QAAWi5B,EACX,QAAWj5B,EACX,SAAYm5B,GAEhB,E,0VCzCA,MAIM+wD,GAA+B,GAC/BC,GAA8B,IAC9BC,GAAgB,aAChBC,GAA0B,CAC/Bl8F,KAAM,MACNM,MAAO,MACP/f,KAAM,UACNgnB,OAAQ,CACP8vF,WAAW,IAYE,MAAM8E,WAAkB,MA6BtC,WAAA77G,CAAauyB,GACZ9mB,QAEAxN,KAAKs0B,OAASA,EACdt0B,KAAK69G,aAAe,KACpB79G,KAAK89G,4BAA8B,KACnC99G,KAAK+9G,sBAAwB,GAAU/9G,KAAKg+G,aAAah3G,KAAMhH,MAAQ,GAAI,CAAEusD,SAAS,IAEtFj4B,EAAOjlB,GAAI,QAASrP,KAAKi+G,mBAAmBj3G,KAAMhH,MACnD,CAKO,OAAAgmB,GACN,MAAMk4F,EAAUl+G,KAAK69G,aAEhBK,IAGJA,EAAQlF,QACRh5G,KAAK69G,aAAe,MAGrB79G,KAAK+9G,sBAAsBtqF,SAC3BzzB,KAAK2P,eACN,CAKQ,kBAAAsuG,GACP,MAAM3pF,EAASt0B,KAAKs0B,UACGA,EAAOtL,OAAO5gB,IAAK,8BAGmC,UC5FhE,SAAwB+1G,GAYtC,SAASC,EAAeD,GACvB,OAAKA,EAAM7lH,QAAU,IAAM6lH,EAAM7lH,QAAU,IACnC,QAEA,SAET,CAIA,IAAM6lH,EACL,MAAO,UAGR,IAAIE,EAAgB,GAEpB,IACCA,EAAgB36G,KAAMy6G,EACvB,CAAE,MAAQj6G,GACT,MAAO,SACR,CAEA,MAAMo6G,EAAwBD,EAActhH,MAAO,KAE7CwhH,EAAeD,EAAuB,GACtCE,EAAgBF,EAAuB,GAE7C,IAAME,EACL,OAAOJ,EAAeD,GAGvB,IACCz6G,KAAM86G,EACP,CAAE,MAAQt6G,GACT,IAGC,GAFAR,KAAM66G,IAEA76G,KAAM66G,GAAejmH,OAC1B,OAAO8lH,EAAeD,EAExB,CAAE,MAAQj6G,GACT,OAAOk6G,EAAeD,EACvB,CACD,CAEA,GAAKI,EAAajmH,OAAS,IAAMimH,EAAajmH,OAAS,IACtD,MAAO,UAGR,IAAImmH,EAAyB,GAE7B,IACC/6G,KAAM66G,GACNE,EAAyB/6G,KAAM86G,EAChC,CAAE,MAAQt6G,GACT,MAAO,SACR,CAEA,GAAuC,IAAlCu6G,EAAuBnmH,OAC3B,MAAO,UAGR,MAAMomH,EAAOt5F,OAAQq5F,EAAuB7hH,UAAW,EAAG,IACpD+hH,EAAav5F,OAAQq5F,EAAuB7hH,UAAW,EAAG,IAAQ,EAClEgiH,EAAMx5F,OAAQq5F,EAAuB7hH,UAAW,EAAG,IAEnDiiH,EAAO,IAAIhwG,KAAM6vG,EAAMC,EAAYC,GAEzC,OAAKC,EAAOjwG,GAAekwG,MAAO15F,OAAQy5F,IAClC,UAGD,OACR,CDOwBE,CAAezqF,EAAOtL,OAAO5gB,IAAK,iBAKlDksB,EAAO6E,GAAGc,OAIhB3F,EAAO6E,GAAGqxE,aAAan7F,GAAI,oBAAoB,CAAEgG,EAAKvS,EAAMgwB,KAC3D9yB,KAAKg/G,oCAEAlsF,EACJ9yB,KAAKg+G,eAELh+G,KAAKi/G,cACN,IAGD3qF,EAAO6E,GAAGqxE,aAAan7F,GAAI,yBAAyB,CAAEgG,EAAKvS,EAAM8vB,KAChE5yB,KAAKg/G,oCAEApsF,GACJ5yB,KAAKg+G,cACN,IAGD1pF,EAAO6E,GAAG9pB,GAAI,UAAU,KACvBrP,KAAK+9G,uBAAuB,IAE9B,CAMQ,kBAAAmB,GACP,MAAM5qF,EAASt0B,KAAKs0B,OACd4pF,EAAUl+G,KAAK69G,aAAe,IAAItE,GAClC4F,EAAkBC,GAAqB9qF,GACvC2F,EAAO,IAAIolF,GAAe/qF,EAAO4E,OAAQimF,EAAgB5wE,OAE/D2vE,EAAQh+G,QAAQ2N,IAAKosB,GACrBikF,EAAQzsG,IAAK,CACZ8sB,MAAO,0BAGRjK,EAAO6E,GAAGc,KAAKzU,KAAK3X,IAAKqwG,GACzB5pF,EAAO6E,GAAGqxE,aAAa38F,IAAKqwG,EAAQzmG,SAEpCzX,KAAK69G,aAAeK,CACrB,CAKQ,YAAAF,GACP,IAAMh+G,KAAK89G,4BACV,OAGD,MAAMwB,EAkHR,SAAkChrF,EAAgBirF,GACjD,MAAMJ,EAAkBC,GAAqB9qF,GACvCjM,EAA+C,UAAzB82F,EAAgBl5B,KAU7C,SAAsCs5B,EAAqCv2F,GAC1E,OAAOw2F,GAAwBD,EAAwBv2F,GAAQ,CAAEy2F,EAAU1F,IACnE0F,EAAS19F,KAAO09F,EAAS58F,MAAQk3F,EAAYl3F,MAAQmG,EAAO02F,kBAErE,CAbEC,CAA6BJ,EAAwBJ,GAevD,SAAqCI,EAAqCv2F,GACzE,OAAOw2F,GAAwBD,EAAwBv2F,GAAQy2F,GAAYA,EAAS19F,KAAOiH,EAAO02F,kBACnG,CAhBEE,CAA4BL,EAAwBJ,GAErD,MAAO,CACNv7G,OAAQ27G,EACRn4F,UAAW,CAAEiB,GAEf,CA5HwBw3F,CAAyB7/G,KAAKs0B,OAAQt0B,KAAK89G,6BAE5DwB,IACEt/G,KAAK69G,cACV79G,KAAKk/G,qBAGNl/G,KAAK69G,aAAc9E,IAAKuG,GAE1B,CAKQ,YAAAL,GACFj/G,KAAK69G,cACT79G,KAAK69G,aAAc7E,OAErB,CAKQ,iCAAAgG,GACP,MAAM1qF,EAASt0B,KAAKs0B,OACdxB,EAAYwB,EAAO6E,GAAGqxE,aAAa13E,UACnCF,EAAiB0B,EAAO6E,GAAGqxE,aAAa53E,eAE9C,IAAME,IAAcF,EAGnB,YAFA5yB,KAAK89G,4BAA8B,MAKpC,MAAMgC,EAAyBn/G,MAAMrB,KAAMg1B,EAAO6E,GAAG4mF,4BAA6B/iH,KAAKgF,GAC/EsyB,EAAO6E,GAAG6mF,mBAAoBh+G,KAGjC89G,EAAuBvwF,SAAUqD,GACrC5yB,KAAK89G,4BAA8BlrF,EAInC5yB,KAAK89G,4BAA8BgC,EAAwB,EAE7D,EAMD,MAAMT,WAAsB,GAO3B,WAAAt9G,CAAam3B,EAAgBqV,GAC5B/gC,MAAO0rB,GAEP,MAAM8vE,EAAW,IAAIL,GACf3hG,EAAOhH,KAAK0iG,aAElBsG,EAASv3F,IAAK,CACbvR,QEhPH,80LFiPG+/G,kBAAkB,IAGnBjX,EAASvG,eAAgB,CACxBl9F,WAAY,CACXF,MAAO,CACNwd,MAAOq9F,OACPp9F,OAAQq9F,WAKXngH,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CAAE,KAAM,iBACf,eAAe,GAEhBjnB,SAAU,CACT,CACCH,IAAK,IACL5R,WAAY,CACX66G,KAAM,yHAENx8G,OAAQ,SACRylG,SAAU,MAEX/xF,SAAU,IACNi3B,EAAQ,CACV,CACCp3B,IAAK,OACL5R,WAAY,CACXg5B,MAAO,CAAE,KAAM,yBAEhBjnB,SAAU,CAAEi3B,KAEV,GACJy6D,GAED35F,GAAI,CACHgxG,UAAWr5G,EAAKzH,IAAI8V,GAAOA,EAAI6d,uBAKpC,EAyBD,SAASssF,GACRD,EACAv2F,EACAs3F,GAEA,MAAO,CAAEC,EAA2BxG,KACnC,MAAMyG,EAA6BD,EAAoBh9F,aAGvD,IAAMi9F,EACL,OAAO7C,GAGR,GAAK4C,EAAoB19F,MAAQ46F,IAA+B8C,EAAoBz9F,OAAS06F,GAC5F,OAAOG,GAGR,IAAI8C,EAGHA,EADwB,WAApBz3F,EAAOvD,SACE86F,EAAoB1+F,OAASk4F,EAAYj3F,OAGzCy9F,EAAoB1+F,OAASk4F,EAAYj3F,OAAS,EAGhE29F,GAAcz3F,EAAO03F,eAErB,MAAMC,EAAcL,EAAgBC,EAAqBxG,GAEzD,GAAyB,WAApB/wF,EAAOvD,SAAwB,CACnC,MAAMm7F,EAAiB7G,EAAYh3F,QAAQC,OAAQ29F,EAAaF,GAGhE,GAAKG,EAAev9F,oBAAqBm9F,GAA+BI,EAAet9F,UACtF,OAAOq6F,EAET,KACK,CACJ,MAAMkD,EG7UM,SAAwC9uE,GACtD,IAAIt6B,EAAUs6B,EAAWjrB,cACzB,IAAMrP,EACL,OAAO,KAGR,KAA2B,QAAnBA,EAAQkqC,SAAoB,CACnC,MAAM79B,EAAWrM,EAAQpS,MAAMy7G,WAAa,GAAOv9G,OAAOie,iBAAkB/J,GAAUqpG,UAEtF,GAAkB,SAAbh9F,GAAoC,WAAbA,EAC3B,MAKD,GAFArM,EAAUA,EAAQqP,eAEZrP,EACL,OAAO,IAET,CAEA,OAAOA,CACR,CHwTkD,CAA+B8nG,GAE9E,GAAKsB,EAAyC,CAC7C,MAAME,EAA6C,IAAI,GAAMF,GACvDG,EAAuBR,EAA2B3+F,OAASk4F,EAAYj3F,OAAS,EACtFi+F,EAA2Cl/F,OACrCo/F,EAAyC,SAAhBj4F,EAAOi9D,KACrCs6B,EAAoBx+F,KAAOg/F,EAA2Ch/F,KACtEw+F,EAAoB5+F,MAAQo/F,EAA2Cp/F,MAGxE,GAAKq/F,GAAwBC,EAC5B,OAAOtD,EAET,CACD,CACA,MAAO,CACNl8F,IAAKg/F,EACL1+F,KAAM4+F,EACN3+G,KAAM,YAAagnB,EAAOvD,iBAAmBuD,EAAOi9D,OACpDj9D,OAAQ,CACP8vF,WAAW,GAEZ,CAEH,CAEA,SAASsG,GAAqB9qF,GAC7B,MAAM4sF,EAAa5sF,EAAOtL,OAAO5gB,IAAK,gBAChCqd,EAAWy7F,GAAcA,EAAWz7F,UAAY,SAEtD,OAAO,IACNA,WACA8oB,MAAOmvE,GACPgD,eAA6B,WAAbj7F,EAAwB,EAAI,EAC5Ci6F,iBAAkB,EAElBz5B,KAAiD,QAA3C3xD,EAAO4E,OAAO3K,yBAAqC,QAAU,QAChE2yF,EAEL,C,+NItWe,MAAeC,WAAiBhuG,KA0F9C,WAAApR,CAAauyB,GACZ9mB,QAzDD,KAAO4zG,SAAmB,EA4C1B,KAAQC,qBAAuB,IAAI9vG,IAKnC,KAAQ+vG,6BAAkE,GAUzE,MAAMC,EAAcjtF,EAAO6qE,QAAQllE,KAEnCj6B,KAAKs0B,OAASA,EACdt0B,KAAKgvG,iBAAmB,IAAIwI,GAAkBljF,GAC9Ct0B,KAAKwqG,aAAe,IAAI,GACxBxqG,KAAKwhH,eAAiB,IAAIlE,GAAgBhpF,GAC1Ct0B,KAAKyhH,UAAY,IAAI7D,GAAWtpF,GAEhCt0B,KAAKyR,IAAK,iBAAkBzR,KAAK0hH,iCAEjC1hH,KAAKyP,KAA0B,SAAS,KACvCzP,KAAKohH,SAAU,CAAI,IAIpBphH,KAAKwP,SAA0C+xG,EAAY/9G,SAAU,gBAAiBxD,KAAK8G,OAAOE,KAAMhH,OACxGA,KAAKwP,SAAyC+xG,EAAa,uBAAwBvhH,KAAK2hH,4BAA4B36G,KAAMhH,OAE1HA,KAAK4hH,oBACN,CAcA,WAAWnqG,GACV,OAAO,IACR,CAQO,MAAA3Q,GACN9G,KAAKwQ,KAA2B,SACjC,CAKO,OAAAwV,GACNhmB,KAAK2P,gBAEL3P,KAAKwqG,aAAaxkF,UAClBhmB,KAAKwhH,eAAex7F,QAAShmB,KAAKs0B,QAClCt0B,KAAKyhH,UAAUz7F,UAGf,UAAY+rB,KAAc/xC,KAAKqhH,qBAAqBn2G,SACjD6mC,EAAoB8vE,iBAAmB,KACzC7hH,KAAKs0B,OAAO8qE,WAAWzvF,cAAeoiC,GAGvC/xC,KAAKqhH,qBAAuB,IAAI9vG,IAChCvR,KAAKshH,6BAA+B,EACrC,CAWO,kBAAAQ,CAAoB1sF,EAAkB2c,GAC5C/xC,KAAKqhH,qBAAqB5vG,IAAK2jB,EAAU2c,GAMjCA,EAAoB8vE,mBACzB9vE,EAAoB8vE,iBAAmB7hH,KAAKs0B,QAI/Ct0B,KAAKwqG,aAAa38F,IAAKkkC,GAEvB,MAAMgwE,EAAwB,KAGxB/hH,KAAKs0B,OAAO6qE,QAAQllE,KAAKg7B,WAAY7/B,IAI1Cp1B,KAAKs0B,OAAO8qE,WAAW5vF,SAAUuiC,EAAY,EAIzC/xC,KAAKohH,QACTW,IAIA/hH,KAAKyP,KAA0B,QAASsyG,EAE1C,CAOO,qBAAAC,CAAuB5sF,GAC7B,MAAM2c,EAAa/xC,KAAKqhH,qBAAqBj5G,IAAKgtB,GAE5C2c,IAIN/xC,KAAKqhH,qBAAqB1vG,OAAQyjB,GAElCp1B,KAAKs0B,OAAO8qE,WAAWzvF,cAAeoiC,GACtC/xC,KAAKwqG,aAAavkG,OAAQ8rC,GAExBA,EAAoB8vE,iBAAmB,KAC1C,CAOO,kBAAA7B,CAAoB5qF,EAAmB,QAC7C,OAAOp1B,KAAKqhH,qBAAqBj5G,IAAKgtB,EACvC,CAKO,wBAAA2qF,GACN,OAAO//G,KAAKqhH,qBAAqBpqH,MAClC,CAUO,UAAAgrH,CAAY7S,EAA0B7qG,EAAmC,CAAC,GAC3E6qG,EAAYjM,YAChBnjG,KAAKwqG,aAAa38F,IAAKuhG,EAAY33F,SACnCzX,KAAKs0B,OAAO8qE,WAAW5vF,SAAU4/F,EAAY33F,UAE7C23F,EAAY3/F,KAAyB,UAAU,KAC9CzP,KAAKwqG,aAAa38F,IAAKuhG,EAAY33F,SACnCzX,KAAKs0B,OAAO8qE,WAAW5vF,SAAU4/F,EAAY33F,QAAU,IAIzDzX,KAAKshH,6BAA6B1hH,KAAM,CAAEwvG,cAAa7qG,WACxD,CAOA,qBAAc29G,GAeb,OALA5zG,QAAQC,KACP,8IAEA,CAAE4zG,SAAUniH,OAENA,KAAKqhH,oBACb,CAgBQ,6BAAAK,GACP,MAAMptF,EAASt0B,KAAKs0B,OACd/M,EAAuB+M,EAAOtL,OAAO5gB,IAAK,qBAEhD,GAAKmf,EACJ,OAAOA,EAIR,MAAM66F,EAAqB9tF,EAAOtL,OAAO5gB,IAAK,6BAG9C,OAAKg6G,GAQJ9zG,QAAQC,KACP,0MAKM,CAAEkT,IAAK2gG,IAIR,CAAE3gG,IAAK,EACf,CAOQ,kBAAAmgG,GACP,MAAMttF,EAASt0B,KAAKs0B,OACditF,EAAcjtF,EAAO6qE,QAAQllE,KAEnC,IAAIooF,EACAC,EAGJhuF,EAAO8qE,WAAW3tF,IAAK,WAAW,CAAE3O,EAAM2wB,KACzC,MAAMb,EAAiB5yB,KAAKwqG,aAAa53E,eAMxCjyB,MAAMrB,KAAMU,KAAKqhH,qBAAqBn2G,UAAWqkB,SAAUqD,KAC1DjyB,MAAMrB,KAAMiiH,EAAYvtD,SAAS9oD,UAAWqkB,SAAUqD,KAEvDyvF,EAA4BzvF,GAG7B,MAAM2vF,EAAkCviH,KAAKwiH,sCAOvCD,GAAoCD,IACzCA,EAAuBtiH,KAAKyiH,4CAK7B,QAAUjoH,EAAI,EAAGA,EAAI8nH,EAAqBhqH,OAAQkC,IAAM,CACvD,MAAMkoH,EAAsBJ,EAAqBn2F,QAQjD,GAJAm2F,EAAqB1iH,KAAM8iH,GAK1BA,IAAwBH,GACxBviH,KAAK2iH,gCAAiCD,GACrC,CAEIH,GAAmCA,EAAgCh+G,QAAQq+G,WAC/EL,EAAgCh+G,QAAQq+G,YAGzC,KACD,CACD,CAEAnvF,GAAQ,IAITa,EAAO8qE,WAAW3tF,IAAK,OAAO,CAAE3O,EAAM2wB,KACrC,MAAMovF,EAAoB7iH,KAAKwiH,sCAEzBK,IAMDR,GACJA,EAA0Bp1E,QAC1Bo1E,EAA4B,MAO5B/tF,EAAO6qE,QAAQllE,KAAKgT,QAIhB41E,EAAkBt+G,QAAQq+G,WAC9BC,EAAkBt+G,QAAQq+G,YAG3BnvF,IAAQ,GAEV,CAaQ,wCAAAgvF,GACP,MAAMK,EAAiD,GAEvD,UAAYC,KAAc/iH,KAAKshH,6BAA+B,CAC7D,MAAM,YAAElS,EAAW,QAAE7qG,GAAYw+G,GAE5B77F,GAAWkoF,EAAY33F,UAAalT,EAAQy+G,cAChDF,EAAYljH,KAAMmjH,EAEpB,CAMA,OAFAD,EAAY3hG,MAAM,CAAE8hG,EAAMC,IAAUC,GAA4BF,GAASE,GAA4BD,KAE9FJ,CACR,CAOQ,mCAAAN,GACP,UAAYt6G,KAAclI,KAAKshH,6BAC9B,GAAKp5G,EAAWknG,YAAY33F,SAAWvP,EAAWknG,YAAY33F,QAAQyM,SAAUlkB,KAAKwqG,aAAa53E,gBACjG,OAAO1qB,EAIT,OAAO,IACR,CAQQ,+BAAAy6G,CAAiCS,GACxC,MAAM,YAAEhU,EAAa7qG,SAAS,YAAEy+G,IAAkBI,EAOlD,OALKJ,GACJA,MAIK97F,GAAWkoF,EAAY33F,WAI7B23F,EAAYniE,SAEL,EACR,CASQ,2BAAA00E,CACPtsG,EACAvS,GAEA,MAAMugH,E,yHAA2B,EAChC5hG,IAAK,EACLI,OAAQ,EACRE,KAAM,EACNJ,MAAO,GACJ3hB,KAAKiqB,gBAGTnnB,EAAKmnB,eAAexI,KAAO4hG,EAAyB5hG,IACpD3e,EAAKmnB,eAAepI,QAAUwhG,EAAyBxhG,OACvD/e,EAAKmnB,eAAelI,MAAQshG,EAAyBthG,KACrDjf,EAAKmnB,eAAetI,OAAS0hG,EAAyB1hG,KACvD,EA+ED,SAASwhG,GAA4BJ,GACpC,MAAM,YAAE3T,EAAW,QAAE7qG,GAAYw+G,EACjC,IAAIO,EAAS,GAYb,OATKp8F,GAAWkoF,EAAY33F,UAC3B6rG,IAII/+G,EAAQg/G,cACZD,IAGMA,CACR,C,eC/nBI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQxpF,OCQR,MAAe,WAAqB,GAiBlD,WAAA/3B,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKwlB,KAAO,IAAIsiF,GAAgB5uE,EACjC,CAKgB,MAAAyY,GACfnkC,MAAMmkC,SAEN3xC,KAAKwlB,KAAKyiF,aACX,CAKgB,OAAAjiF,GAGf,OAFAhmB,KAAKwlB,KAAK0iF,gBAEH16F,MAAMwY,SACd,EC1Cc,MAAMw9F,WAAuB,GA0C3C,WAAAzhH,CACCm3B,EACAqoF,EACAp5E,GAEA36B,MAAO0rB,GA3CR,KAAOl3B,KAAsB,KA6C5BhC,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,aACA,sBACA,sBAEDklF,KAAMvqF,EAAOxJ,gBACb3jB,IAAKmtB,EAAO3K,4BAIdvuB,KAAKyR,IAAK,aAAa,GAEvBzR,KAAK0jH,iBAAmBv7E,EACxBnoC,KAAK2jH,sBAAwB3jH,KAAK0jH,iBAClC1jH,KAAK4jH,aAAerC,CACrB,CAOgB,MAAA5vE,GACfnkC,MAAMmkC,SAED3xC,KAAK2jH,oBACT3jH,KAAK6jG,SAAUhzF,MAAO7Q,KAAKyX,QAAUzX,KAAK0jH,kBAE1C1jH,KAAK0jH,iBAAmB1jH,KAAKyX,QAG9BzX,KAAKqP,GAA2B,oBAAoB,IAAMrP,KAAK6jH,4BAC/D7jH,KAAK6jH,yBACN,CAKgB,OAAA79F,GACVhmB,KAAK2jH,qBACT3jH,KAAK6jG,SAAUllC,OAAQ3+D,KAAK0jH,kBAG7Bl2G,MAAMwY,SACP,CAMA,sBAAW89F,GACV,OAAO9jH,KAAK2jH,mBACb,CAMQ,uBAAAE,GACP,MAAMtC,EAAcvhH,KAAK4jH,aAQzB,SAAS98G,EAAQmzB,GAChBsnF,EAAYhnF,QAAQC,IACnB,MAAMo6B,EAAW2sD,EAAY/9G,SAASstC,QAAS7W,EAAKj4B,MAEpDw4B,EAAOoB,SAAU3B,EAAKnH,UAAY,aAAe,aAAc8hC,GAC/Dp6B,EAAOO,YAAad,EAAKnH,UAAY,aAAe,aAAc8hC,EAAU,GAE9E,CAbK2sD,EAAY5rD,sBAoBjB,SAASouD,EAAmB9pF,GAC3BsnF,EAAY9xG,KAAsC,gCAAgC,CAAE4F,EAAKrT,EAAMxJ,KACxFA,EAGLurH,EAAmB9pF,GAFnBnzB,EAAQmzB,EAGT,GAEF,CA3BC8pF,CAAmB/jH,MAEnB8G,EAAQ9G,KA0BV,EC/Ic,MAAMgkH,WAA6BR,GAoBjD,WAAAzhH,CACCm3B,EACAqoF,EACAp5E,EACA5jC,EAAgE,CAAC,GAEjEiJ,MAAO0rB,EAAQqoF,EAAap5E,GAE5B,MAAMntC,EAAIk+B,EAAOl+B,EAEjBgF,KAAKyiG,eAAgB,CACpBl9F,WAAY,CACX6jG,KAAM,UACN7qE,MAAO,gCAITv+B,KAAKikH,eAAiB1/G,EAAQgqC,OAAS,KAAQvzC,EAAG,0BAA2BgF,KAAKgC,MACnF,CAKgB,MAAA2vC,GACfnkC,MAAMmkC,SAEN,MAAM4vE,EAAcvhH,KAAK4jH,aAEzBrC,EAAYhnF,QAAQC,IACnB,MAAMo6B,EAAW2sD,EAAY/9G,SAASstC,QAAS9wC,KAAKgC,MAEpDw4B,EAAO/0B,aAAc,aAAczF,KAAKikH,eAAgBjkH,MAAQ40D,EAAW,GAE7E,E,eCnEG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ96B,OCWR,MAAMoqF,WAAqBxqF,GAIzC,qBAAkBpD,GACjB,MAAO,cACR,CAKO,IAAAI,GAEN12B,KAAKqP,GAAkC,gBAAgB,CAAEgG,EAAKvS,KAC7DS,OAAO4gH,MAAOrhH,EAAKsL,QAAS,GAC1B,CAAExB,SAAU,UAChB,CA6BO,WAAAw3G,CACNh2G,EACAtL,EAA+C,CAAC,GAEhD9C,KAAKqkH,kBAAmB,CACvBj2G,UACAnD,KAAM,UACNsM,UAAWzU,EAAKyU,UAChB+sG,MAAOxhH,EAAKwhH,OAEd,CA6BO,QAAAC,CACNn2G,EACAtL,EAA+C,CAAC,GAEhD9C,KAAKqkH,kBAAmB,CACvBj2G,UACAnD,KAAM,OACNsM,UAAWzU,EAAKyU,UAChB+sG,MAAOxhH,EAAKwhH,OAEd,CAwDO,WAAAE,CACNp2G,EACAtL,EAA+C,CAAC,GAEhD9C,KAAKqkH,kBAAmB,CACvBj2G,UACAnD,KAAM,UACNsM,UAAWzU,EAAKyU,UAChB+sG,MAAOxhH,EAAKwhH,OAEd,CAWQ,iBAAAD,CAAmBvhH,GAM1B,MAAMwM,EAAQxM,EAAKyU,UAClB,QAASzU,EAAKmI,QAAUnI,EAAKyU,YAC7B,QAASzU,EAAKmI,OAEfjL,KAAKwQ,KAAmDlB,EAAO,CAC9DlB,QAAStL,EAAKsL,QACdnD,KAAMnI,EAAKmI,KACXq5G,MAAOxhH,EAAKwhH,OAAS,IAEvB,EClMc,MAAM,WAAcnxG,KASlC,WAAApR,CAAawD,EAAsC+N,GAClD9F,QAGK8F,GACJ,GAAQtT,KAAMsT,GAIV/N,GACJvF,KAAKyR,IAAKlM,EAEZ,E,eCjCG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQu0B,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OAAvB,MCoBM,GAAOpT,GAAQ,MAoCN,MAAM,WAA0B,GA4E9C,WAAA3kB,CAAauyB,GACZ9mB,MAAO8mB,GAlCR,KAAQmwF,aAAe,IAAIlzG,IAK3B,KAAQmzG,WAAa,IAAInzG,IAKzB,KAAQozG,MAAiC,KAMzC,KAAQC,aAAmC,KAK3C,KAAQC,gBAAyC,KAehD7kH,KAAK8kH,gBAAkB,KACtB,MAAM7qF,EAAOj6B,KAAKs0B,OAAO6qE,QAAQllE,KAE3BkO,EADelO,EAAKz2B,SACWyxB,UAAUkT,gBAE/C,OAAKA,EACGlO,EAAK4X,aAAayK,aAAcnU,EAAgB7xC,MAGjD,IAAI,EAGZ0J,KAAKyR,IAAK,cAAe,MACzBzR,KAAKyR,IAAK,kBAAmB,GAC7BzR,KAAKyR,IAAK,mBAAmB,EAC9B,CAzBA,qBAAkB6kB,GACjB,MAAO,mBACR,CA4BgB,OAAAtQ,GACfxY,MAAMwY,UAEDhmB,KAAK2kH,OACT3kH,KAAK2kH,MAAM3+F,UAGPhmB,KAAK4kH,cACT5kH,KAAK4kH,aAAa5+F,UAGdhmB,KAAK6kH,iBACT7kH,KAAK6kH,gBAAgB7+F,SAEvB,CAKA,QAAWiU,GAKV,OAJMj6B,KAAK2kH,OACV3kH,KAAK+kH,mBAGC/kH,KAAK2kH,KACb,CAKO,OAAAK,CAAS/qF,GACf,OAAOt5B,MAAMrB,KAAMU,KAAKykH,aAAaxtH,QAASs4B,SAAU0K,EACzD,CAcO,GAAApsB,CAAK/K,GAKX,GAJM9C,KAAK2kH,OACV3kH,KAAK+kH,mBAGD/kH,KAAKglH,QAASliH,EAAKm3B,MAMvB,MAAM,IAAI,EACT,mCACA,CAAEj6B,KAAM8C,IAIV,MAAMmiH,EAAUniH,EAAKmiH,SAAW,OAGhC,IAAMjlH,KAAK0kH,WAAW92G,IAAKq3G,GAS1B,OARAjlH,KAAK0kH,WAAWjzG,IAAKwzG,EAAS,IAAI1zG,IAAK,CAAE,CAAEzO,EAAKm3B,KAAMn3B,MACtD9C,KAAKykH,aAAahzG,IAAK3O,EAAKm3B,KAAMj6B,KAAK0kH,WAAWt8G,IAAK68G,IACvDjlH,KAAKklH,gBAAkBllH,KAAK0kH,WAAW7wG,UAEjC7T,KAAKmlH,gBAAiBriH,EAAKsiH,gBAChCplH,KAAKqlH,UAAWJ,IAMlB,MAAM52G,EAAQrO,KAAK0kH,WAAWt8G,IAAK68G,GAE9BniH,EAAKsiH,gBACTplH,KAAKqlH,UAAWJ,GAIjB52G,EAAMoD,IAAK3O,EAAKm3B,KAAMn3B,GACtB9C,KAAKykH,aAAahzG,IAAK3O,EAAKm3B,KAAM5rB,GAG7BA,IAAUrO,KAAKmlH,eACnBnlH,KAAKslH,UAAWxiH,EAElB,CAUO,MAAAmD,CAAQg0B,GACd,IAAMj6B,KAAKglH,QAAS/qF,GAMnB,MAAM,IAAI,EACT,0CACA,CAAEj6B,KAAMi6B,IAIV,MAAM5rB,EAAQrO,KAAKykH,aAAar8G,IAAK6xB,GAEhCj6B,KAAKulH,iBAAmBvlH,KAAKwlH,cAAgBvrF,IACjDj6B,KAAKulH,iBAAkB,GAKnBvlH,KAAKwlH,cAAgBvrF,IACL,IAAf5rB,EAAMwF,KACL7T,KAAK0kH,WAAW7wG,KAAO,EAC3B7T,KAAKylH,kBAELzlH,KAAKi6B,KAAK89E,OACV/3G,KAAKwlH,YAAc,KACnBxlH,KAAK4kH,aAAcc,YAGpB1lH,KAAKslH,UAAW3kH,MAAMrB,KAAM+O,EAAMnD,UAAYmD,EAAMwF,KAAO,KAIzC,IAAfxF,EAAMwF,MACV7T,KAAK0kH,WAAW/yG,OAAQ3R,KAAK2lH,YAAat3G,IAC1CrO,KAAKklH,gBAAkBllH,KAAK0kH,WAAW7wG,MAEvCxF,EAAMsD,OAAQsoB,GAGfj6B,KAAKykH,aAAa9yG,OAAQsoB,EAC3B,CAQO,cAAA2rF,CAAgBngG,GACjBA,IACJzlB,KAAKmlH,cAAc/8G,IAAKpI,KAAKwlH,aAAgB//F,SAAWA,GAGzDzlB,KAAKi6B,KAAK8+E,IAAK/4G,KAAK6lH,uBACpB7lH,KAAK6kH,gBAAiBe,gBACvB,CAKO,SAAAP,CAAWxlH,GACjBG,KAAK8lH,aAAejmH,EACpB,MAAMwO,EAAQrO,KAAK0kH,WAAWt8G,IAAKvI,GAEnC,IAAMwO,EAML,MAAM,IAAI,EACT,8CACArO,MAIGA,KAAKmlH,gBAAkB92G,GAI5BrO,KAAKslH,UAAW3kH,MAAMrB,KAAM+O,EAAMnD,UAAWlM,MAC9C,CAKQ,gBAAA+lH,GACP/kH,KAAK2kH,MAAQ,IAAIpL,GAAkBv5G,KAAKs0B,OAAO4E,QAE/Cl5B,KAAKs0B,OAAO6E,GAAGc,KAAKzU,KAAK3X,IAAK7N,KAAK2kH,OACnC3kH,KAAKs0B,OAAO6E,GAAGqxE,aAAa38F,IAAK7N,KAAK2kH,MAAMltG,SAE5CzX,KAAK4kH,aAAe5kH,KAAK+lH,qBACzB/lH,KAAK6kH,gBAAkB7kH,KAAKgmH,uBAC7B,CAKA,iBAAYb,GACX,OAAOnlH,KAAKykH,aAAar8G,IAAKpI,KAAKwlH,YACpC,CAKQ,WAAAG,CAAat3G,GAGpB,OAFc1N,MAAMrB,KAAMU,KAAK0kH,WAAWjsG,WAAY8Y,MAAM7Y,GAASA,EAAO,KAAQrK,IAEtE,EACf,CAKQ,cAAAo3G,GACP,MAAMQ,EAAStlH,MAAMrB,KAAMU,KAAK0kH,WAAWx5G,UAE3C,IAAIg7G,EAAYD,EAAOj9G,QAAShJ,KAAKmlH,eAAkB,EAEjDc,EAAQC,KACbA,EAAY,GAGblmH,KAAKqlH,UAAWrlH,KAAK2lH,YAAaM,EAAQC,IAC3C,CAKQ,cAAAC,GACP,MAAMF,EAAStlH,MAAMrB,KAAMU,KAAK0kH,WAAWx5G,UAE3C,IAAIg7G,EAAYD,EAAOj9G,QAAShJ,KAAKmlH,eAAkB,EAEjDc,EAAQC,KACbA,EAAYD,EAAO3tH,OAAS,GAG7B0H,KAAKqlH,UAAWrlH,KAAK2lH,YAAaM,EAAQC,IAC3C,CAKQ,kBAAAH,GACP,MAAM9rF,EAAO,IAAImsF,GAAapmH,KAAKs0B,OAAO4E,QACpCl+B,EAAIgF,KAAKs0B,OAAO4E,OAAOl+B,EA2C7B,OAzCAgF,KAAKi6B,KAAK/5B,QAAQ2N,IAAKosB,GAGvBA,EAAKjzB,KAAM,uBAAwBzH,GAAIS,KAAM,kBAAmBA,KAAM,mBAAmB,CAAExH,EAAO6tH,KACzFA,GAAoB7tH,EAAQ,IAIrCyhC,EAAK5qB,GAA2B,8BAA8B,IAAQrP,KAAK4lH,kBAAoB,CAAEh5G,SAAU,QAG3GqtB,EAAKjzB,KAAM,WAAYzH,GAAIS,KAAM,cAAeA,KAAM,mBAAmB,CAAEwlH,EAAac,KACvF,GAAKA,EAAiB,EACrB,MAAO,GAGR,MAAMvnH,EAAU4B,MAAMrB,KAAMU,KAAK0kH,WAAWx5G,UAAWlC,QAAShJ,KAAKmlH,eAAkB,EAEvF,OAAOnqH,EAAG,WAAY,CAAE+D,EAASunH,GAAkB,IAGpDrsF,EAAKssF,eAAel3G,GAAwB,WAAW,KAGjD4qB,EAAKuwE,aAAa13E,WACtB9yB,KAAKs0B,OAAO6qE,QAAQllE,KAAKgT,QAG1BjtC,KAAKylH,gBAAgB,IAGtBxrF,EAAKusF,eAAen3G,GAAwB,WAAW,KAGjD4qB,EAAKuwE,aAAa13E,WACtB9yB,KAAKs0B,OAAO6qE,QAAQllE,KAAKgT,QAG1BjtC,KAAKmmH,gBAAgB,IAGflsF,CACR,CAKQ,qBAAA+rF,GACP,MAAM/rF,EAAO,IAAIwsF,GAAgBzmH,KAAKs0B,OAAO4E,OAAQl5B,KAAKi6B,MAa1D,OAXAA,EAAKjzB,KAAM,kBAAmBzH,GAAIS,KAAM,kBAAmBA,KAAM,mBAAmB,CAAEqgC,EAAQgmF,KACzEA,GAAoBhmF,GAAU,EAE9BxnC,KAAKD,IAAKynC,EAAS,EAAG,GAAM,IAGjDpG,EAAKzqB,SAAiCxP,KAAKi6B,KAAM,cAAc,IAAMA,EAAK2rF,mBAC1E3rF,EAAKzqB,SAAiCxP,KAAKi6B,KAAM,eAAe,IAAMA,EAAK2rF,mBAE3E5lH,KAAKs0B,OAAO6E,GAAGc,KAAKzU,KAAK3X,IAAKosB,GAEvBA,CACR,CAWQ,SAAAqrF,EAAW,KAAErrF,EAAI,iBAAEysF,EAAmB,aAAI5N,GAAY,EAAI,eAAEsM,GAAiB,IACpFplH,KAAKi6B,KAAKsE,MAAQmoF,EAClB1mH,KAAKi6B,KAAK6+E,UAAYA,EAEtB94G,KAAK4kH,aAAc+B,SAAU1sF,GAC7Bj6B,KAAKwlH,YAAcvrF,EACnBj6B,KAAKi6B,KAAK8+E,IAAK/4G,KAAK6lH,uBACpB7lH,KAAK6kH,gBAAiBe,iBAEjBR,IACJplH,KAAKulH,iBAAkB,EAEzB,CAMQ,mBAAAM,GACP,IAAIpgG,EAAW9kB,MAAMrB,KAAMU,KAAKmlH,cAAcj6G,UAAWlM,MAAOymB,SAiBhE,OAfKA,IAEEA,EAAS4B,UAEd5B,EAAWzuB,OAAO+wB,OAAQ,CAAC,EAAGtC,EAAU,CACvC4B,QAASrnB,KAAK8kH,mBAKhBr/F,EAAWzuB,OAAO+wB,OAAQ,CAAC,EAAGtC,EAAU,CACvC8B,qBAAsBvnB,KAAKs0B,OAAO6E,GAAGlP,kBAIhCxE,CACR,EAoDM,MAAM2gG,WAAoB,GAoChC,WAAArkH,CAAam3B,GACZ1rB,MAAO0rB,GAEP,MAAMl+B,EAAIk+B,EAAOl+B,EACXgM,EAAOhH,KAAK0iG,aAElB1iG,KAAKyR,IAAK,uBAAuB,GAEjCzR,KAAKwqG,aAAe,IAAI,GACxBxqG,KAAKwmH,eAAiBxmH,KAAK4mH,kBAAmB5rH,EAAG,YCpnBnD,0NDqnBEgF,KAAKumH,eAAiBvmH,KAAK4mH,kBAAmB5rH,EAAG,QErnBnD,4NFsnBEgF,KAAKE,QAAUF,KAAKsjG,mBAEpBtjG,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,sBAED,UAAW,MAEZjnB,SAAU,CACT,CACCH,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,iCACAv3B,EAAKzH,GAAI,uBAAuB/G,GAASA,EAAQ,GAAK,gBAGxD8e,SAAU,CACTtX,KAAKwmH,eACL,CACCrvG,IAAK,OAEL5R,WAAY,CACXg5B,MAAO,CACN,gCAIFjnB,SAAU,CACT,CACC4iB,KAAMlzB,EAAKzH,GAAI,cAIlBS,KAAKumH,iBAGP,CACCpvG,IAAK,MACL5R,WAAY,CACXg5B,MAAO,+BAERjnB,SAAUtX,KAAKE,WAInB,CAKgB,MAAAyxC,GACfnkC,MAAMmkC,SAEN3xC,KAAKwqG,aAAa38F,IAAK7N,KAAKyX,QAC7B,CAKgB,OAAAuO,GACfxY,MAAMwY,UAENhmB,KAAKwqG,aAAaxkF,SACnB,CAOO,QAAA2gG,CAAU1sF,GAChBj6B,KAAK0lH,WACL1lH,KAAKE,QAAQ2N,IAAKosB,EACnB,CAKO,QAAAyrF,GACN1lH,KAAKE,QAAQ0R,OACd,CAQQ,iBAAAg1G,CAAmBr4E,EAAem7D,GACzC,MAAMzvE,EAAO,IAAI,GAAYj6B,KAAKk5B,QAQlC,OANAe,EAAKxoB,IAAK,CACT88B,QACAm7D,OACAG,SAAS,IAGH5vE,CACR,EAMD,MAAMwsF,WAAuB,GAiD5B,WAAA1kH,CAAam3B,EAAgB4iF,GAC5BtuG,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAElB1iG,KAAKyR,IAAK,MAAO,GACjBzR,KAAKyR,IAAK,OAAQ,GAClBzR,KAAKyR,IAAK,SAAU,GACpBzR,KAAKyR,IAAK,QAAS,GACnBzR,KAAKyR,IAAK,iBAAkB,GAE5BzR,KAAKE,QAAUF,KAAKsjG,mBACpBtjG,KAAK6mH,kBAAoB/K,EAEzB97G,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,gBACAv3B,EAAKzH,GAAI,kBAAkB8gC,GAAUA,EAAS,GAAK,eAEpDh7B,MAAO,CACNoc,IAAKza,EAAKzH,GAAI,MAAO,IACrBwiB,KAAM/a,EAAKzH,GAAI,OAAQ,IACvBsjB,MAAO7b,EAAKzH,GAAI,QAAS,IACzBujB,OAAQ9b,EAAKzH,GAAI,SAAU,MAG7B+X,SAAUtX,KAAKE,UAGhBF,KAAKqP,GAAmC,yBAAyB,CAAEgG,EAAKrT,EAAMV,EAAMu0C,KAC9Ev0C,EAAOu0C,EACX71C,KAAK8mH,WAAYxlH,EAAOu0C,GAExB71C,KAAK+mH,cAAelxE,EAAOv0C,GAG5BtB,KAAK4lH,gBAAgB,GAEvB,CAEQ,UAAAkB,CAAYzmF,GACnB,KAAQA,KAAW,CAClB,MAAMpG,EAAO,IAAI,GAEjBA,EAAK2pE,YAAa,CAAEzsF,IAAK,QAEzBnX,KAAKE,QAAQ2N,IAAKosB,GAClBj6B,KAAK0jG,cAAezpE,EACrB,CACD,CAEQ,aAAA8sF,CAAe1mF,GACtB,KAAQA,KAAW,CAClB,MAAMpG,EAAOj6B,KAAKE,QAAQwwB,KAE1B1wB,KAAKE,QAAQ+F,OAAQg0B,GACrBj6B,KAAK2jG,gBAAiB1pE,GACtBA,EAAKjU,SACN,CACD,CAKO,cAAA4/F,GACN,GAAK5lH,KAAKgnH,eAAiB,CAC1B,MAAM,IAAEvlG,EAAG,KAAEM,GAAS/hB,KAAK6mH,mBACrB,MAAEhkG,EAAK,OAAEC,GAAW,IAAI,GAAM9iB,KAAK6mH,kBAAkBpvG,SAE3DzgB,OAAO+wB,OAAQ/nB,KAAM,CAAEyhB,MAAKM,OAAMc,QAAOC,UAC1C,CACD,E,eGz1BG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQgX,OCgBVpT,GAAQ,MCYrB,MAAM,GAAOA,GAAQ,MAON,MAAMugG,WAAuB,GA6D3C,WAAAllH,CAAauyB,GACZ9mB,MAAO8mB,GApCR,KAAQonF,gBAAyC,KAsChD17G,KAAKknH,eAAiB3Z,GAAwBj5E,EAAOtL,OAAO5gB,IAAK,mBACjEpI,KAAKovG,YAAcpvG,KAAKmnH,qBACxBnnH,KAAKwqG,aAAe,IAAI,GAGxBl2E,EAAO6E,GAAG1pB,KAA0B,SAAS,KAC5CzP,KAAKwqG,aAAa38F,IAAKymB,EAAO6E,GAAG6mF,sBACjChgH,KAAKwqG,aAAa38F,IAAK7N,KAAKovG,YAAY33F,QAAU,IAInD6c,EAAO6E,GAAG8oF,WAAYjiH,KAAKovG,YAAa,CACvC4T,YAAa,IAAMhjH,KAAK83G,MAAM,GAC9B8K,UAAW,IAAM5iH,KAAK+3G,OACtBwL,cAAc,IAGfvjH,KAAKonH,SAAW9yF,EAAOqC,QAAQvuB,IAAK,IACpCpI,KAAKqnH,8BAAgC,IAAU,IAAMrnH,KAAKwQ,KAAM,8BAA+B,KAI/FxQ,KAAKiV,SAAU,OAChB,CAxCA,qBAAkBqhB,GACjB,MAAO,gBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GACV,CAoCO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdW,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAGxCj1B,KAAKwP,SAA0CxP,KAAKwqG,aAAc,oBAAoB,CAAEn1F,EAAKrT,EAAM8wB,KAClG,MAAMw0F,EAAmBtnH,KAAKonH,SAAS5B,cAAgBxlH,KAAKovG,aAEtDt8E,GAAaw0F,EAClBtnH,KAAK+3G,OACMjlF,GACX9yB,KAAK83G,MACN,IAID93G,KAAKwP,SAA6CylB,EAAW,gBAAgB,CAAE5f,EAAKvS,MAC9EA,EAAK2/D,cAAgBxtC,EAAU+V,cACnChrC,KAAK+3G,OAKN/3G,KAAKqnH,+BAA+B,IAIrCrnH,KAAKwP,SAAUxP,KAAM,6BAA6B,KAC5CA,KAAKs0B,OAAO6qE,QAAQllE,KAAKz2B,SAASsvB,WACtC9yB,KAAK83G,MACN,IAGK93G,KAAKknH,eAAeK,wBACzBvnH,KAAKwP,SAA4B8kB,EAAQ,SAAS,KACjD,MAAM6T,EAAkB7T,EAAO6E,GAAGc,KAAKoQ,SAAS5yB,QAGhDzX,KAAK07G,gBAAkB,IAAI,GAAgBvzE,GAAiBzvB,IAI3D1Y,KAAKovG,YAAYpB,SAAW,GAAgC,GAA1Bt1F,EAAMw4F,YAAYruF,MAAY,GAC9D,IASL7iB,KAAKwP,SAA8CxP,KAAKovG,YAAa,sBAAsB,KAC1FpvG,KAAKwnH,iBAAiB,GAExB,CAMO,SAAAC,GACN,MAAMlxH,EAAUyJ,KAAKs0B,OAAO6E,GAAG61E,iBAE/BhvG,KAAKovG,YAAYhB,eAAgBpuG,KAAKknH,eAAgB3wH,EACvD,CAKQ,kBAAA4wH,GACP,MAAMnsH,EAAIgF,KAAKs0B,OAAO4E,OAAOl+B,EACvB8yG,GAAuB9tG,KAAKknH,eAAeK,uBAC3CnY,EAAc,IAAI,GAAapvG,KAAKs0B,OAAO4E,OAAQ,CACxD40E,sBACAC,YAAY,IAMb,OAHAqB,EAAYmD,UAAYv3G,EAAG,6BAC3Bo0G,EAAYz9D,SAELy9D,CACR,CAUO,IAAA0I,CAAM4P,GAAqC,GACjD,MAAMpzF,EAASt0B,KAAKs0B,OACdW,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAClCyuC,EAASpvC,EAAOl8B,MAAMsrE,OAGvB1jE,KAAKonH,SAASpC,QAAShlH,KAAKovG,cAK5Bn6E,EAAU+V,cAAgB08E,GAuJjC,SAAmDzyF,EAA8ByuC,GAEhF,GAA8B,IAAzBzuC,EAAUiY,WACd,OAAO,EAGR,MAAO,IAAKjY,EAAUmY,aAAc72B,OAAOuO,IAC1C,MAAMrN,EAAUqN,EAAMmnB,sBAEtB,OAAOx0B,GAAWisD,EAAO+O,aAAch7D,EAAS,GAElD,CA5JOkwG,CAA0C1yF,EAAWyuC,IAMrD/iE,MAAMrB,KAAMU,KAAKovG,YAAYx+E,OAAQra,OAAStW,QAAkC,IAAnBA,EAAKy0B,YAA4Bz0B,EAAKy0B,cAKxG10B,KAAKwP,SAA+BxP,KAAKs0B,OAAO6E,GAAI,UAAU,KAC7Dn5B,KAAKwnH,iBAAiB,IAIvBxnH,KAAKonH,SAASv5G,IAAK,CAClBosB,KAAMj6B,KAAKovG,YACX3pF,SAAUzlB,KAAK4nH,0BACflB,iBAAkB,yBAEpB,CAKO,IAAA3O,GACD/3G,KAAKonH,SAASpC,QAAShlH,KAAKovG,eAChCpvG,KAAK2P,cAAe3P,KAAKs0B,OAAO6E,GAAI,UACpCn5B,KAAKonH,SAASnhH,OAAQjG,KAAKovG,aAE7B,CAMQ,uBAAAwY,GACP,MACM3tF,EADSj6B,KAAKs0B,OACA6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SACpBwzC,EAAgBwc,EAAav+B,UAG7BkY,EAAaqmB,EAAav+B,UAAUkY,WAE1C,MAAO,CAKNvpC,OAAQ,KACP,MAAMkhB,EAAQqoB,EAAa6J,EAAc3J,gBAAkB2J,EAAc1J,eACnEhrB,EAAa,GAAKC,iBAAkB0X,EAAK4X,aAAa2U,eAAgB1hC,IAG5E,OAAKqoB,EACG7qB,EAAY,IAKdA,EAAWhqB,OAAS,GAAmD,IAA9CgqB,EAAYA,EAAWhqB,OAAS,GAAIuqB,OACjEP,EAAWtjB,MAGLsjB,EAAYA,EAAWhqB,OAAS,GACxC,EAED8uB,UAAWpnB,KAAK6nH,qBAAsB16E,GAExC,CAQQ,eAAAq6E,GACPxnH,KAAKonH,SAASxB,eAAgB5lH,KAAK4nH,0BACpC,CAKgB,OAAA5hG,GACfxY,MAAMwY,UAENhmB,KAAK2P,gBACL3P,KAAKqnH,8BAA8B5zF,SACnCzzB,KAAKovG,YAAYppF,UACjBhmB,KAAKwqG,aAAaxkF,UAEbhmB,KAAK07G,iBACT17G,KAAK07G,gBAAgB11F,SAEvB,CAKQ,oBAAA6hG,CAAsB16E,GAC7B,MAGM/lB,EAHgB,EAAIle,UAAY,EAAIC,MAGRqwG,GAAmB,CAGpDG,aAAc9gH,KAAKC,IAClBygH,GAAiBK,kBACjB/gH,KAAKuD,MAAO,GAAK,GAAOmH,OAAOukH,eAAgBC,UAE3CxO,GAAiBtB,iBAEvB,OAAO9qE,EAAa,CACnB/lB,EAAU8yF,oBACV9yF,EAAU0yF,wBACV1yF,EAAUgzF,wBACVhzF,EAAU+yF,8BACV/yF,EAAU6yF,8BACV7yF,EAAUyzF,oBACVzzF,EAAUszF,wBACVtzF,EAAU2zF,wBACV3zF,EAAUwzF,8BACVxzF,EAAU0zF,+BACP,CACH1zF,EAAU8zF,oBACV9zF,EAAUg0F,wBACVh0F,EAAU4zF,wBACV5zF,EAAU+zF,8BACV/zF,EAAU6zF,8BACV7zF,EAAUmzF,oBACVnzF,EAAUqzF,wBACVrzF,EAAUizF,wBACVjzF,EAAUozF,8BACVpzF,EAAUkzF,8BAEZ,E,eC/XG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQxgF,OAAvB,MCGM,GAAOpT,GAAQ,MASN,MAAMshG,WAAwB,GAkB5C,WAAAjmH,CAAam3B,GACZ1rB,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAGlB1iG,KAAKknB,WAAY,EAEjBlnB,KAAKspG,cAAe,EAEpBtpG,KAAKyR,IAAK,MAAO,GACjBzR,KAAKyR,IAAK,OAAQ,GAElBzR,KAAKyiG,eAAgB,CACpBl9F,WAAY,CACXg5B,MAAO,0BACPl5B,MAAO,CACNoc,IAAKza,EAAKzH,GAAI,OAAOjF,GAAO,GAAMA,KAClCynB,KAAM/a,EAAKzH,GAAI,QAAQjF,GAAO,GAAMA,QAIxC,EC5BD,MAAM,GAAOosB,GAAQ,OACbo7E,QAAO,IAAK,GCdL,MAAemmB,GAmF7B,WAAAlmH,CAAainB,GAkBZ,GA1FD,KAAgBk/F,QASX,GAYL,KAAOv6B,MAAuB,eAW9B,KAAQw6B,KAAOt5G,KAAK2yB,IAyCnBxhC,KAAKkoH,QAAU,GACfloH,KAAKooH,kBAAuD,iBAA5Bp/F,EAAOq/F,iBAAgCr/F,EAAOq/F,iBAAmB,EACjGroH,KAAKsoH,2BAAyE,iBAArCt/F,EAAOu/F,0BAAyCv/F,EAAOu/F,0BAA4B,IAE5HvoH,KAAKwoH,mBAAqBnzG,IAEzB,MAAMzL,EAAQ,UAAWyL,EAAMA,EAAIzL,MAAQyL,EAAIozG,OAI1C7+G,aAAiBvR,OACrB2H,KAAK0oH,aAAc9+G,EAAOyL,EAC3B,EAGDrV,KAAK2oH,WAAa,CAAC,GAEX3oH,KAAc4oH,SACrB,MAAM,IAAIvwH,MACT,iLAIH,CAKO,OAAA2tB,GACNhmB,KAAK6oH,qBAEL7oH,KAAK2oH,WAAa,CAAC,CACpB,CAWO,EAAAt5G,CAA8BmC,EAAcjC,GAC5CvP,KAAK2oH,WAAYn3G,KACtBxR,KAAK2oH,WAAYn3G,GAAc,IAGhCxR,KAAK2oH,WAAYn3G,GAAY5R,KAAM2P,EACpC,CAUO,GAAAnD,CAAKoF,EAA2BjC,GACtCvP,KAAK2oH,WAAYn3G,GAAcxR,KAAK2oH,WAAYn3G,GAC9CzL,QAAQ+iH,GAAMA,IAAOv5G,GACxB,CAOU,KAAAw5G,CAAiCv3G,KAAiBtV,GAC3D,MAAMiU,EAAYnQ,KAAK2oH,WAAYn3G,IAAe,GAElD,UAAYjC,KAAYY,EACvBZ,EAASsB,MAAO7Q,KAAM,CAAE,QAAS9D,GAEnC,CAKU,mBAAA8sH,GACTzlH,OAAO+M,iBAAkB,QAAStQ,KAAKwoH,oBACvCjlH,OAAO+M,iBAAkB,qBAAsBtQ,KAAKwoH,mBACrD,CAKU,kBAAAK,GACTtlH,OAAOgN,oBAAqB,QAASvQ,KAAKwoH,oBAC1CjlH,OAAOgN,oBAAqB,qBAAsBvQ,KAAKwoH,mBACxD,CAUQ,YAAAE,CAAc9+G,EAAcyL,GAMnC,GAAKrV,KAAKipH,oBAAqBr/G,GAAU,CACxC5J,KAAKkoH,QAAQtoH,KAAM,CAClBwO,QAASxE,EAAMwE,QACfC,MAAOzE,EAAMyE,MAGb66G,SAAU7zG,aAAe8zG,WAAa9zG,EAAI6zG,cAAW,EACrDE,OAAQ/zG,aAAe8zG,WAAa9zG,EAAI+zG,YAAS,EACjDC,MAAOh0G,aAAe8zG,WAAa9zG,EAAIg0G,WAAQ,EAC/CxK,KAAM7+G,KAAKmoH,SAGZ,MAAMmB,EAAgBtpH,KAAKupH,iBAE3BvpH,KAAK2tF,MAAQ,UACb3tF,KAAK+oH,MAAO,eACZ/oH,KAAK+oH,MAAO,QAAS,CAAEn/G,QAAO0/G,kBAEzBA,EACJtpH,KAAK4oH,YAEL5oH,KAAK2tF,MAAQ,qBACb3tF,KAAK+oH,MAAO,eAEd,CACD,CAOQ,mBAAAE,CAAqBr/G,GAC5B,OACGA,EAAesE,IACftE,EAAesE,GAAI,uBACkB,IAArCtE,EAAyB2D,SAIY,OAArC3D,EAAyB2D,SAGZ,UAAfvN,KAAK2tF,OAEL3tF,KAAKwpH,2BAA4B5/G,EAEnC,CAKQ,cAAA2/G,GACP,GAAKvpH,KAAKkoH,QAAQ5vH,QAAU0H,KAAKooH,kBAChC,OAAO,EAQR,OALsBpoH,KAAKkoH,QAASloH,KAAKkoH,QAAQ5vH,OAAS,GAAIumH,KAC7B7+G,KAAKkoH,QAASloH,KAAKkoH,QAAQ5vH,OAAS,EAAI0H,KAAKooH,mBAAoBvJ,MAEjB7+G,KAAKooH,kBAEnDpoH,KAAKsoH,0BACzC,EC1Qc,SAAS,GAAarkH,EAAewlH,EAAqB,IAAI71G,KAC5E,MAAMszB,EAAQ,CAAEjjC,GAKVylH,EAAW,IAAI91G,IACrB,IAAI+1G,EAAY,EAEhB,KAAQziF,EAAM5uC,OAASqxH,GAAY,CAElC,MAAMxqH,EAAO+nC,EAAOyiF,KAEpB,IAAKD,EAAS97G,IAAKzO,IAAWyqH,GAAsBzqH,KAAUsqH,EAAmB77G,IAAKzO,GAOtF,GAHAuqH,EAAS77G,IAAK1O,GAGT2B,OAAOC,YAAY5B,EAEvB,IACC,UAAY7D,KAAK6D,EAChB+nC,EAAMtnC,KAAMtE,EAMd,CAAE,MAAQkG,GAIV,MAEA,UAAYzK,KAAOoI,EAIL,iBAARpI,GAILmwC,EAAMtnC,KAAMT,EAAMpI,GAOrB,CAIA,OAAO2yH,CACR,CAEA,SAASE,GAAsBzqH,GAC9B,MAAM8L,EAAOjU,OAAO6K,UAAUnF,SAAS2E,KAAMlC,GACvC0qH,SAAoB1qH,EAE1B,QACgB,WAAf0qH,GACe,YAAfA,GACe,WAAfA,GACe,WAAfA,GACe,aAAfA,GACS,kBAAT5+G,GACS,oBAATA,GACS,oBAATA,GAPA4+G,MASA1qH,GAOEA,EAAci9G,mBAGhBj9G,aAAgB2qH,aAChB3qH,aAAgB4qH,MAElB,CChFe,SAASC,GAA+BC,EAAkBC,EAAkBC,EAAgB,IAAIv2G,KAC9G,GAAKq2G,IAAYC,IA0DW,iBADVE,EAzDoBH,IA0DgB,OAAdG,GAzDvC,OAAO,EAwDT,IAAmBA,EAnDlB,MAAMC,EAAY,GAAaJ,EAASE,GAClCG,EAAY,GAAaJ,EAASC,GAExC,UAAYhrH,KAAQkrH,EACnB,GAAKC,EAAU18G,IAAKzO,GACnB,OAAO,EAIT,OAAO,CACR,C,yZCCe,MAAMorH,WAAwDtC,GA8D5E,WAAAlmH,CAAak9F,EAA6DurB,EAAiC,CAAC,GAC3Gh9G,MAAOg9G,GA3DR,KAAQC,QAA0B,KA0BlC,KAAQC,gBAAiB,EAKzB,KAAQC,WAA0C,CAAC,EAgClD3qH,KAAK4qH,eAAiB,GACrB5qH,KAAK6qH,MAAM7jH,KAAMhH,MACsB,iBAAhCwqH,EAAeM,aAA4BN,EAAeM,aAAe,KAI5E7rB,IACJj/F,KAAK+qH,SAAa,CAAEC,EAAehiG,IAAYi2E,EAAO3hF,OAAQ0tG,EAAehiG,IAG9EhpB,KAAKirH,YAAc32F,GAAUA,EAAOtO,SACrC,CAKA,UAAWsO,GACV,OAAOt0B,KAAKyqH,OACb,CAKA,SAAWnoH,GACV,OAAOtC,KAAKyqH,OACb,CAUO,UAAAS,CAAYC,GAClBnrH,KAAK+qH,SAAWI,CACjB,CAmBO,aAAAC,CAAeC,GACrBrrH,KAAKirH,YAAcI,CACpB,CAQmB,QAAAzC,GAClB,OAAOnwF,QAAQlc,UACbub,MAAM,KACN93B,KAAK2tF,MAAQ,eACb3tF,KAAK+oH,MAAO,eAEL/oH,KAAKsrH,cAEZC,OAAO/pH,IACP8M,QAAQ1E,MAAO,kDAAmDpI,EAAK,IAEvEs2B,MAAM,KASN,MAAM0zF,EAAwC,CAAC,EAEzCC,EAA2B,GAE3BC,EAAqD1rH,KAAKse,QAASqtG,iBAAmB,CAAC,EAEvFA,EAAkD,CAAC,EAGzD,UAAcv2F,EAAUw2F,KAAc50H,OAAOyhB,QAASzY,KAAKo9B,MAAOyT,OAC5D+6E,EAASC,UACbL,EAAep2F,GAAa,GAC5Bu2F,EAAiBv2F,GAAas2F,EAAoBt2F,IAAc,CAAC,GAEjEq2F,EAAU7rH,KAAMw1B,GAIlB,MAAM02F,G,EAA8B,MAChC9rH,KAAKse,S,EAD2B,CAEnCmhF,aAAcz/F,KAAKse,QAASmhF,cAAgB,GAC5CgsB,YACAE,kBACAI,qBAAsB/rH,KAAKo9B,O,qBAS5B,cAJO0uF,EAAcrtC,YAErBqtC,EAAcrsB,aAAc7/F,KAAMosH,IAE7BhsH,KAAK0qH,eACF1qH,KAAKsd,OAAQkuG,EAAeM,EAAeA,EAAcv+G,SAO3D,GAAWvN,KAAKisH,gBACbjsH,KAAKsd,OAAQtd,KAAKisH,eAAgBH,EAAeA,EAAcv+G,SAE/DvN,KAAKsd,OAAQtd,KAAK2qH,WAAYmB,EAAeA,EAAcv+G,QAEpE,IAEAuqB,MAAM,KACN93B,KAAK+oH,MAAO,UAAW,GAE1B,CASO,MAAAzrG,CACN0tG,EAA6FhrH,KAAKisH,eAClGjjG,EAAuBhpB,KAAKse,QAC5B/Q,GAEA,OAAOkrB,QAAQlc,UACbub,MAAM,KACNtqB,MAAMw7G,sBAENhpH,KAAKisH,eAAiBjB,EAItBhrH,KAAK0qH,eAAyC,iBAAjBM,GAC1Bh0H,OAAOC,KAAM+zH,GAAgB1yH,OAAS,GAAmD,iBAAvCtB,OAAOkU,OAAQ8/G,GAAiB,GAIrFhrH,KAAKse,QAAUte,KAAKksH,0BAA2BljG,IAAY,CAAC,EAE5DhpB,KAAKse,QAAS/Q,QAAUA,EAEjBvN,KAAK+qH,SAAUC,EAAehrH,KAAKse,YAE1CwZ,MAAMxD,IACNt0B,KAAKyqH,QAAUn2F,EAEfA,EAAOl8B,MAAMoL,SAAS6L,GAAI,cAAerP,KAAK4qH,gBAE9C5qH,KAAKmsH,qBAAuB73F,EAAOl8B,MAAMoL,SAASmL,QAClD3O,KAAKo9B,MAAQp9B,KAAKosH,WAEZpsH,KAAK0qH,iBACV1qH,KAAK2qH,WAAa3qH,KAAKqsH,iBAGxBrsH,KAAK2tF,MAAQ,QACb3tF,KAAK+oH,MAAO,cAAe,GAE9B,CAOgB,OAAA/iG,GACf,OAAOyS,QAAQlc,UACbub,MAAM,KACN93B,KAAK2tF,MAAQ,YACb3tF,KAAK+oH,MAAO,eAEZv7G,MAAMwY,UAEChmB,KAAKsrH,aAEf,CAEQ,QAAAA,GACP,OAAO7yF,QAAQlc,UACbub,MAAM,KACN93B,KAAK6oH,qBAEL7oH,KAAK4qH,eAAen3F,SAEpB,MAAMa,EAASt0B,KAAKyqH,QASpB,OAPAzqH,KAAKyqH,QAAU,KAKfn2F,EAAQl8B,MAAMoL,SAAS4I,IAAK,cAAepM,KAAK4qH,gBAEzC5qH,KAAKirH,YAAa32F,EAAS,GAErC,CAMQ,KAAAu2F,GACP,MAAMl8G,EAAU3O,KAAKyqH,QAASryH,MAAMoL,SAASmL,QAE7C,IACC3O,KAAKo9B,MAAQp9B,KAAKosH,WAEZpsH,KAAK0qH,iBACV1qH,KAAK2qH,WAAa3qH,KAAKqsH,iBAGxBrsH,KAAKmsH,qBAAuBx9G,CAC7B,CAAE,MAAQnN,GACT8M,QAAQ1E,MACPpI,EACA,0GAGF,CACD,CAKO,sBAAA8qH,CAAwB7yG,GAC9BzZ,KAAKusH,eAAiB9yG,CACvB,CAKQ,QAAA2yG,GACP,MAAM93F,EAASt0B,KAAKyqH,QACd55E,EAAQvc,EAAOl8B,MAAMoL,SAASqtC,MAAM9qC,QAAQzP,GAAQA,EAAKilC,cAAiC,cAAjBjlC,EAAK8+B,YAE9E,QAAEuB,GAAYrC,EAEdk4F,EAAqB71F,EAAQ/oB,IAAK,uBAA0B+oB,EAAQvuB,IAAK,sBACzEqkH,EAAe91F,EAAQ/oB,IAAK,iBAAoB+oB,EAAQvuB,IAAK,gBAE7DtF,EAAmB,CACxB+tC,MAAO,CAAC,EACR+uB,QAAS,CAAC,EACV8sD,eAAgB9pH,KAAKC,UAAW,IAChC8pH,YAAa/pH,KAAKC,UAAW,KAG9BguC,EAAMjzC,SAAStH,IACdwM,EAAK+tC,MAAOv6C,EAAK8+B,UAAa,CAC7Bl1B,QAAS0C,KAAKC,UAAWlC,MAAMrB,KAAMhJ,EAAKklC,gBAC1Cj2B,WAAY3C,KAAKC,UAAWlC,MAAMrB,KAAMhJ,EAAK0vC,kBAC7C6lF,SAAUv1H,EAAKq1F,UACf,IAGF,UAAYvqB,KAAU9sC,EAAOl8B,MAAMwnE,QAC5BwB,EAAOvsC,eAIb/xB,EAAK88D,QAASwB,EAAOp/D,MAAS,CAC7B4qH,UAAWxrD,EAAOT,WAAW3jC,SAC7B61D,eAAgBzxB,EAAO0wB,wBACvB98D,YAAaosC,EAAOvsC,eAYtB,OARK23F,IACJ1pH,EAAK4pH,eAAiB9pH,KAAKC,UAAW2pH,EAAmBK,kBAAmB,CAAE7vF,QAAQ,EAAM8vF,iBAAiB,MAGzGL,IACJ3pH,EAAK6pH,YAAc/pH,KAAKC,UAAW4pH,EAAaM,eAAgB,CAAE/vF,QAAQ,EAAM8vF,iBAAiB,MAG3FhqH,CACR,CAKQ,aAAAupH,GACP,MAAMW,EAAyC,CAAC,EAEhD,UAAY53F,KAAYp1B,KAAKs0B,OAAQl8B,MAAMoL,SAASutF,eAAiB,CACpE,MAAM1mD,EAAWrqC,KAAKs0B,OAAQ6E,GAAG6mF,mBAAoB5qF,GAEhDiV,IACJ2iF,EAAW53F,GAAaiV,EAE1B,CAEA,OAAO2iF,CACR,CAQO,0BAAAxD,CAA4B5/G,GAClC,OAAOogH,GAA+BhqH,KAAKyqH,QAAS7gH,EAAM2D,QAASvN,KAAKusH,eACzE,CAKQ,yBAAAL,CAA2BljG,GAClC,OAAO,GAAeA,GAAQ,CAAExwB,EAAOzB,IAEjC,GAAWyB,IAIH,YAARzB,EAHGyB,OAGR,GAIF,EAOD,MAAMwzH,GAKL,WAAAjqH,CAAauyB,GACZt0B,KAAKs0B,OAASA,EAEdt0B,KAAKo9B,MAAQ9I,EAAOtL,OAAO5gB,IAAK,uBACjC,CAKO,IAAAsuB,GAIN12B,KAAKs0B,OAAOxxB,KAAKuM,GAAI,QAAQgG,IAC5BA,EAAIlJ,OAEJnM,KAAKs0B,OAAOl8B,MAAMguE,cAAe,CAAEuX,YAAY,IAASnjD,IACvDx6B,KAAKitH,4BACLjtH,KAAKktH,mBAAoB1yF,EAAQ,IAGlCx6B,KAAKs0B,OAAOxxB,KAAK0N,KAAwB,QAAS,GAGhD,CAAE5D,SAAU,KAChB,CAKQ,WAAAugH,CAAa3yF,EAAgB4yF,GACpC,GAAK,SAAUA,EAAW,CAEzB,MAAM31G,EAAU+iB,EAAOl1B,cAAe8nH,EAASprH,KAAMorH,EAAS7nH,YAE9D,GAAK6nH,EAAS91G,SACb,UAAYK,KAASy1G,EAAS91G,SAC7BG,EAAQwvB,aAAcjnC,KAAKmtH,YAAa3yF,EAAQ7iB,IAIlD,OAAOF,CACR,CAEC,OAAO+iB,EAAOmZ,WAAYy5E,EAAStqH,KAAMsqH,EAAS7nH,WAEpD,CAKQ,kBAAA2nH,CAAoB1yF,GAC3B,MAAMlG,EAASt0B,KAAKs0B,OAEpBt9B,OAAOyhB,QAASzY,KAAKo9B,MAAOyT,OAAQjzC,SAAS,EAAIw3B,GAAYl1B,UAASqF,kBACrE,MAAM8nH,EAAqCzqH,KAAK+7E,MAAOz+E,GACjDotH,EAA+C1qH,KAAK+7E,MAAOp5E,GAE3DgoH,EAAcj5F,EAAOl8B,MAAMoL,SAASstC,QAAS1b,GAEnD,UAAcr+B,EAAKyB,KAAW80H,EAC7B9yF,EAAO/0B,aAAc1O,EAAKyB,EAAO+0H,GAGlC,UAAY51G,KAAS01G,EAAc,CAClC,MAAMluH,EAAOa,KAAKmtH,YAAa3yF,EAAQ7iB,GAEvC6iB,EAAO90B,OAAQvG,EAAMouH,EAAa,MACnC,KAGDv2H,OAAOyhB,QAASzY,KAAKo9B,MAAOwiC,SAAUhiE,SAAS,EAAI0+D,EAAYkxD,MAC9D,MAAM,SAAEhqH,GAAa8wB,EAAOl8B,MAIxB,EAAAo1H,GAFHZ,WAAW,MAAElsF,EAAK,IAAEC,IAEjB,EADAp8B,E,iKAAA,CACA,EADA,CADH,cAIKjO,EAAOkN,EAASstC,QAASpQ,EAAMpqC,MAC/BmyC,EAAgBjO,EAAO62D,uBAAwB/6F,EAAMoqC,EAAMjhC,KAAMihC,EAAM22B,YACvE3e,EAAcle,EAAO62D,uBAAwB/6F,EAAMqqC,EAAIlhC,KAAMkhC,EAAI02B,YAEjEvyC,EAAQ0V,EAAOqd,YAAapP,EAAeiQ,GAEjDle,EAAOu4D,UAAWz2B,EAAY,IAC7Bx3C,SACGvgB,GACD,GAEL,CAKQ,yBAAA0oH,GAEP,MAAMQ,EAAmC7qH,KAAK+7E,MAAO3+E,KAAKo9B,MAAMsvF,gBAC1DgB,EAAgC9qH,KAAK+7E,MAAO3+E,KAAKo9B,MAAMuvF,aAE7Dc,EAAqB7vH,SAAS+vH,IAC7B,MAAMC,EAAY5tH,KAAKs0B,OAAOtL,OAAO5gB,IAAK,2BACpCokH,EAAqBxsH,KAAKs0B,OAAQqC,QAAQvuB,IAAK,sBAErD,GAAKokH,EAAmBqB,iBAAkBF,EAAkBG,UAAa,CAClDtB,EAAmBuB,iBAAkBJ,EAAkBG,UAE/D7nH,QACf,CAEAumH,EAAmBwB,iBAAkB,IAAEJ,aAAcD,GAAqB,IAG3ED,EAAkB9vH,SAASqwH,IAC1B,MAAMC,EAAsBluH,KAAKs0B,OAAQqC,QAAQvuB,IAAK,uBAEtD,GAAK8lH,EAAoBC,cAAeF,EAAepuH,IAAO,CAC1CquH,EAAoBE,cAAeH,EAAepuH,IAE1D0F,WAAa0oH,EAAe1oH,UACxC,MACC2oH,EAAoBG,kBAAmBJ,EACxC,GAEF,ECviBD,MAAMK,GAAcxtH,OAAQ,eAyf5B,MAAMytH,GAAN,cACC,KAAOC,kBAAuC,GAC9C,KAAOC,QAAU,IAAIl9G,IACrB,KAAOm9G,eAAiB,EAOjB,OAAAC,CAASC,GACf5uH,KAAKwuH,kBAAkB5uH,KAAMgvH,EAC9B,CAQO,OAAAC,CAASC,EAAmC/wE,GAClD,MAAMgxE,EAAeD,IAAYR,GAEjCtuH,KAAK0uH,iBAEC1uH,KAAKyuH,QAAQrmH,IAAK0mH,IACvB9uH,KAAKyuH,QAAQh9G,IAAKq9G,EAASr2F,QAAQlc,WAMpC,MAIMyyG,GAJiBD,EACtBt2F,QAAQh1B,IAAKzD,KAAKyuH,QAAQvjH,UAC1ButB,QAAQh1B,IAAK,CAAEzD,KAAKyuH,QAAQrmH,IAAKkmH,IAAetuH,KAAKyuH,QAAQrmH,IAAK0mH,MAE5Bh3F,KAAMimB,GAGvCkxE,EAAgBD,EAAgBzD,OAAO,SAI7C,OAFAvrH,KAAKyuH,QAAQh9G,IAAKq9G,EAASG,GAEpBD,EAAgBE,SAAS,KAC/BlvH,KAAK0uH,iBAEA1uH,KAAKyuH,QAAQrmH,IAAK0mH,KAAcG,GAAyC,IAAxBjvH,KAAK0uH,gBAC1D1uH,KAAKwuH,kBAAkB5wH,SAASkrH,GAAMA,KACvC,GAEF,EASD,SAAS,GAAYqG,GACpB,OAAOxuH,MAAMC,QAASuuH,GAAmBA,EAAiB,CAAEA,EAC7D,CC3jBe,MAAMC,WAAwBjO,GAY5C,WAAAp/G,CAAauyB,EAAgB2F,GAC5BzsB,MAAO8mB,GAEPt0B,KAAKi6B,KAAOA,CACb,CAKA,WAAoBxiB,GACnB,OAAOzX,KAAKi6B,KAAKoQ,SAAS5yB,OAC3B,CAKO,IAAAif,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd2F,EAAOj6B,KAAKi6B,KACZsnF,EAAcjtF,EAAO6qE,QAAQllE,KAC7BoQ,EAAWpQ,EAAKoQ,SAChBglF,EAAc9N,EAAY/9G,SAASstC,UAIzCzG,EAASroC,KAAOqtH,EAAYj6F,SAE5B6E,EAAK0X,SAIL,MAAMxJ,EAAkBkC,EAAS5yB,QAIjCzX,KAAK8hH,mBAAoBz3E,EAASroC,KAAMmmC,GASxCkC,EAASrjC,KAAM,aAAczH,GAAIS,KAAKwqG,cAItC+W,EAAY5sD,cAAexsB,GAE3BnoC,KAAKsvH,mBACLtvH,KAAKwQ,KAA0B,QAChC,CAKgB,OAAAwV,GACfxY,MAAMwY,UAEN,MAAMiU,EAAOj6B,KAAKi6B,KACEj6B,KAAKs0B,OAAO6qE,QAAQllE,KAE5B+6B,cAAe/6B,EAAKoQ,SAASroC,MACzCi4B,EAAKjU,SACN,CAKQ,gBAAAspG,GACP,MAAMh7F,EAASt0B,KAAKs0B,OACditF,EAAcjtF,EAAO6qE,QAAQllE,KAC7Bo1F,EAAc9N,EAAY/9G,SAASstC,UACnClW,EAActG,EAAOtL,OAAO5gB,IAAK,eAEvC,GAAKwyB,EAAc,CAClB,MAAM20F,EAAyC,iBAAhB30F,EAA2BA,EAAcA,EAAay0F,EAAYj6F,UAE5Fm6F,IACJF,EAAYz0F,YAAc20F,EAE5B,CAEA,GAAmB,CAClBt1F,KAAMsnF,EACN9pG,QAAS43G,EACTl1F,cAAc,EACdC,aAAa,GAEf,EC/Gc,MAAMo1F,WAA4B,GAchD,WAAAztH,CACCm3B,EACAqoF,EACAp5E,GAEA36B,MAAO0rB,GAEP,MAAMl+B,EAAIk+B,EAAOl+B,EAEjBgF,KAAKqqC,SAAW,IAAI25E,GAAsB9qF,EAAQqoF,EAAap5E,EAAiB,CAC/EoG,MAAOkhF,GACCz0H,EAAG,qCAAsCy0H,EAAaztH,OAGhE,CAKgB,MAAA2vC,GACfnkC,MAAMmkC,SAEN3xC,KAAK0jG,cAAe1jG,KAAKqqC,SAC1B,ECCc,MAAM,WAAsBs1D,GAAcC,GAAiBX,MAiB/D,WAAAl9F,CAAa2tH,EAA2C1mG,EAAuB,CAAC,GAEzF,IAAM,GAAW0mG,SAAgD,IAAvB1mG,EAAOy1D,YAGhD,MAAM,IAAI,EAAe,6BAA8B,MAGxDjxE,MAAOwb,QAEmC,IAArChpB,KAAKgpB,OAAO5gB,IAAK,gBACrBpI,KAAKgpB,OAAOvX,IAAK,cAuLpB,SAAyBi+G,GACxB,OAAO,GAAWA,ICzPyBnpG,EDyPmBmpG,ECxPzDnpG,aAAcC,oBACXD,EAAG/tB,MAGJ+tB,EAAGE,WDoP4EipG,ECzPxE,IAA6BnpG,CD0P5C,CAzLmCopG,CAAgBD,IAG5C,GAAWA,KACf1vH,KAAKyhF,cAAgBiuC,EE/DT,SAA8Bp7F,EAAgBmtD,GAC5D,GAAKA,EAAcogC,iBAUlB,MAAM,IAAI,EACT,qCACAvtF,GAIFmtD,EAAcogC,iBAAmBvtF,EAEjCA,EAAO7kB,KAA0B,WAAW,YACpCgyE,EAAcogC,gBAAgB,GAEvC,CF0CG+N,CAAqB5vH,KAAM0vH,IAG5B,MAAM/4F,EAAU32B,KAAKgpB,OAAO5gB,IAAK,WACjCuuB,EAAQ/2B,KAAMqnH,IAEdjnH,KAAKgpB,OAAOvX,IAAK,UAAWklB,GAE5B32B,KAAKgpB,OAAOtyB,OAAQ,iBAAkBsJ,KAAKgpB,OAAO5gB,IAAK,YAEvDpI,KAAK5H,MAAMoL,SAASy/E,aAEpB,MAAMhpD,EAAO,IAAIu1F,GAAqBxvH,KAAKk5B,OAAQl5B,KAAKm/F,QAAQllE,KAAMj6B,KAAKyhF,eAC3EzhF,KAAKm5B,GAAK,IAAIi2F,GAAiBpvH,KAAMi6B,GG5ExB,SAAuB3F,GACrC,IAAM,GAAYA,EAAOurE,qBAOxB,MAAM,IAAI,EACT,4CACAvrE,GAIF,MAAMmtD,EAAgBntD,EAAOmtD,cAG7B,GA+BD,SAAqBA,GACpB,QAASA,GAAyD,aAAxCA,EAAc9/B,QAAQ/4C,aACjD,CAjCMinH,CAAYpuC,IAAmBA,EAAcquC,KAAO,CACxD,IAAIC,EACJ,MAAMD,EAAOruC,EAAcquC,KACrBE,EAAW,IAAM17F,EAAOurE,sBAIzB,GAAYiwB,EAAKG,UACrBF,EAAiBD,EAAKG,OAEtBH,EAAKG,OAAS,KACbD,IACAD,EAAel/G,MAAOi/G,EAAM,GAK9BA,EAAKx/G,iBAAkB,SAAU0/G,GAIjC17F,EAAOjlB,GAAwB,WAAW,KACzCygH,EAAKv/G,oBAAqB,SAAUy/G,GAE/BD,IACJD,EAAKG,OAASF,EACf,GAEF,CACD,CHgCEG,CAAclwH,KACf,CASgB,OAAAgmB,GAGf,MAAMljB,EAAO9C,KAAK2yD,UAIlB,OAFA3yD,KAAKm5B,GAAGnT,UAEDxY,MAAMwY,UACX8R,MAAM,KACD93B,KAAKyhF,eACTzhF,KAAK6/F,oBAAqB/8F,EAC3B,GAEH,CAkGA,aAAuBwa,CAAQoyG,EAA2C1mG,EAAuB,CAAC,GACjG,OAAO,IAAIyP,SAASlc,IACnB,GAAK,GAAWmzG,IAAyD,aAAhCA,EAAoB/tE,QAG5D,MAAM,IAAI,EAAe,uBAAwB,MAGlD,MAAMrtB,EAAS,IAAIt0B,KAAM0vH,EAAqB1mG,GAE9CzM,EACC+X,EAAOuD,cACLC,MAAM,IAAMxD,EAAO6E,GAAGzC,SACtBoB,MAAM,IAAMxD,EAAOxxB,KAAK4zB,KAAMpC,EAAOtL,OAAO5gB,IAAK,kBACjD0vB,MAAM,IAAMxD,EAAO9jB,KAAwB,WAC3CsnB,MAAM,IAAMxD,IACd,GAEH,EA4BD,SAAS,GAAW97B,GACnB,OAAO,GAAYA,EACpB,CAzNqB,GAkMNqgC,QAAUA,GAlMJ,GAyMN0xF,eAAiBA,GAzMX,GAgNN4F,gBHhOA,cAAkElI,GAmEhF,WAAAlmH,CACC82B,EACA2xF,EAAiC,CAAC,GAElCh9G,MAAOg9G,GAnER,KAAU4F,WAAa,IAAI7+G,IAU3B,KAAQ4kB,SAA4B,KAMpC,KAAQk6F,cAAgB,IAAIz8G,IAK5B,KAAQ08G,cAAgB,IAAI/B,GAgD3BvuH,KAAKuwH,gBAAkB/F,EAGvBxqH,KAAK+qH,SAAWyF,GAAiB33F,EAAQvb,OAAQkzG,GACjDxwH,KAAKirH,YAAc19G,GAAWA,EAAQyY,UAEtChmB,KAAKswH,cAAc3B,SAAS,KACP,iBAAf3uH,KAAK2tF,QACT3tF,KAAK2tF,MAAQ,QACb3tF,KAAK+oH,MAAO,eACb,GAEF,CAUO,UAAAmC,CAAYC,GAClBnrH,KAAK+qH,SAAWI,CACjB,CAmBO,aAAAC,CAAeC,GACrBrrH,KAAKirH,YAAcI,CACpB,CAMA,WAAW99G,GACV,OAAOvN,KAAKm2B,QACb,CAcO,MAAA7Y,CAAQkzG,EAA+B,CAAC,GAC9C,OAAOxwH,KAAKswH,cAAczB,QAASP,IAAa,KAC/CtuH,KAAKywH,eAAiBD,EAEfxwH,KAAK0wH,YAEd,CAYO,OAAAp4C,CAASznD,GAGf,OAFiB7wB,KAAK2wH,aAAc9/F,GAEpBvuB,KACjB,CAYO,YAAAsuH,CAAc//F,GAGpB,OAFiB7wB,KAAK2wH,aAAc9/F,GAEpB88D,KACjB,CA8CO,GAAA9/E,CAAKgjH,GACX,MAAMC,EAAqB,GAASD,GAEpC,OAAOp4F,QAAQh1B,IAAKqtH,EAAmB9zH,KAAKiD,GACpCD,KAAKswH,cAAczB,QAAS5uH,EAAKJ,IAAI,KAC3C,GAAoB,cAAfG,KAAK2tF,MACT,MAAM,IAAIt1F,MAAO,2CAGlB,IAAM2H,KAAKm2B,SACV,MAAM,IAAI99B,MAAO,6FAGlB,IAAI04H,EAEJ,GAAK/wH,KAAKowH,WAAWxiH,IAAK3N,EAAKJ,IAC9B,MAAM,IAAIxH,MAAO,6CAA8C4H,EAAKJ,QAGrE,GAAmB,WAAdI,EAAKgL,KAkCT,OAjCA8lH,EAAW,IAAIxG,GAAgB,KAAMvqH,KAAKuwH,iBAC1CQ,EAAS7F,WAAYjrH,EAAKkrH,SAC1B4F,EAASzE,uBAAwBtsH,KAAKqwH,eAEjCpwH,EAAKorH,YACT0F,EAAS3F,cAAenrH,EAAKorH,YAG9BrrH,KAAKowH,WAAW3+G,IAAKxR,EAAKJ,GAAIkxH,GAI9BA,EAAS1hH,GAAI,SAAS,CAAEgG,GAAOzL,QAAO0/G,oBACrCtpH,KAAK+oH,MAAO,YAAa,CAAEl4F,OAAQ5wB,EAAKJ,GAAI+J,UAGtC0/G,GAINtpH,KAAKswH,cAAczB,QAAS5uH,EAAKJ,IAAI,IAAM,IAAI44B,SAAeu4F,IAC7D,MAAMC,EAA0B,KAC/BF,EAAS3kH,IAAK,UAAW6kH,GAEzBjxH,KAAK+oH,MAAO,cAAe,CAAEl4F,OAAQ5wB,EAAKJ,KAE1CmxH,GAAK,EAGND,EAAS1hH,GAAI,UAAW4hH,EAAyB,KAC7C,IAGCF,EAASzzG,OAAQrd,EAAKyvH,oBAAqBzvH,EAAK+oB,OAAQhpB,KAAKm2B,UAEpE,MAAM,IAAI99B,MAAO,6BAA8B4H,EAAKgL,SACrD,MAGH,CAiBO,MAAAhF,CAAQirH,GACd,MAAMC,EAAU,GAASD,GAEzB,OAAOz4F,QAAQh1B,IAAK0tH,EAAQn0H,KAAK6zB,GACzB7wB,KAAKswH,cAAczB,QAASh+F,GAAQ,KAC1C,MAAMkgG,EAAW/wH,KAAK2wH,aAAc9/F,GAIpC,OAFA7wB,KAAKowH,WAAWz+G,OAAQkf,GAEjBkgG,EAAS/qG,SAAS,MAG5B,CAUgB,OAAAA,GACf,OAAOhmB,KAAKswH,cAAczB,QAASP,IAAa,KAC/CtuH,KAAK2tF,MAAQ,YACb3tF,KAAK+oH,MAAO,eAEZv7G,MAAMwY,UAEChmB,KAAKsrH,aAEd,CAKU,QAAA1C,GACT,OAAO5oH,KAAKswH,cAAczB,QAASP,IAAa,KAC/CtuH,KAAK2tF,MAAQ,eACb3tF,KAAK+oH,MAAO,eAEL/oH,KAAKsrH,WACVC,OAAO/pH,IACP8M,QAAQ1E,MAAO,4DAA6DpI,EAAK,IAEjFs2B,MAAM,IAAM93B,KAAK0wH,YACjB54F,MAAM,IAAM93B,KAAK+oH,MAAO,eAE5B,CAKQ,OAAA2H,GACP,OAAOj4F,QAAQlc,UACbub,MAAM,KACN93B,KAAKgpH,sBAEEhpH,KAAK+qH,SAAU/qH,KAAKywH,mBAE3B34F,MAAMvqB,IACNvN,KAAKm2B,SAAW5oB,EAChBvN,KAAKqwH,cAAgB,GAAarwH,KAAKm2B,UAEhCsC,QAAQh1B,IACd9C,MAAMrB,KAAMU,KAAKowH,WAAWllH,UAC1BlO,KAAK+zH,IACLA,EAASzE,uBAAwBtsH,KAAKqwH,eAE/BU,EAASzzG,YAAQ,OAAW,EAAWtd,KAAKm2B,gBAIzD,CAKQ,QAAAm1F,GACP,OAAO7yF,QAAQlc,UACbub,MAAM,KACN93B,KAAK6oH,qBAEL,MAAMt7G,EAAUvN,KAAKm2B,SAKrB,OAHAn2B,KAAKm2B,SAAW,KAChBn2B,KAAKqwH,cAAgB,IAAIz8G,IAElB6kB,QAAQh1B,IACd9C,MAAMrB,KAAMU,KAAKowH,WAAWllH,UAC1BlO,KAAK+zH,GAAYA,EAAS/qG,aAG3B8R,MAAM,IAAM93B,KAAKirH,YAAa19G,IAAW,GAE9C,CAOU,YAAAojH,CAAc9/F,GACvB,MAAMkgG,EAAW/wH,KAAKowH,WAAWhoH,IAAKyoB,GAEtC,IAAMkgG,EACL,MAAM,IAAI14H,MAAO,8CAA+Cw4B,MAGjE,OAAOkgG,CACR,CAOO,0BAAAvH,CAA4B5/G,GAClC,UAAYmnH,KAAY/wH,KAAKowH,WAAWllH,SACvC,GAAK6lH,EAASvH,2BAA4B5/G,GACzC,OAAO,EAIT,OAAOogH,GAA+BhqH,KAAKm2B,SAAUvsB,EAAM2D,QAC5D,GOpac,MAAM,WAA0B49C,GAQ9C,WAAAppD,CAAak4B,GACZzsB,MAAOysB,GALR,KAAgBmxB,aAAe,CAC9B,QAAS,OAAQ,MAAO,OAAQ,WAAY,YAAa,UAAW,YAAa,aAMjF,MAAMoI,EAAexzD,KAAKwD,SAM1B,SAAS4tH,EAAanmH,GACrB,MAAO,CAAEoK,EAAgBvS,KACxBA,EAAKowB,iBAEL,MAAMwgC,EAAe5wD,EAAKuuH,UAAY,CAAEvuH,EAAKuuH,WAAc,KACrD3gH,EAAY,IAAIxE,EAAWsnD,EAAcvoD,GAE/CuoD,EAAahjD,KAAME,EAAW,CAC7B+iD,aAAc3wD,EAAK2wD,aACnBl7B,OAAQljB,EAAIrT,KACZ0xD,eACA9vD,OAAQd,EAAKc,OACbqnD,SAAUnoD,EAAKmoD,WAMXv6C,EAAUvE,KAAKF,QACnBnJ,EAAKqwB,iBACN,CAEF,CA1BAnzB,KAAKwP,SAAkCgkD,EAAc,QAAS49D,EAAa,kBAAoB,CAAExkH,SAAU,QAC3G5M,KAAKwP,SAAiCgkD,EAAc,OAAQ49D,EAAa,kBAAoB,CAAExkH,SAAU,QACzG5M,KAAKwP,SAAqCgkD,EAAc,WAAY49D,EAAa,YAAc,CAAExkH,SAAU,OAyB5G,CAEO,UAAAy+C,CAAYJ,GAClB,MAAMoH,EAAqB,kBAAmBpH,EAAWA,EAASqmE,cAAiBrmE,EAASwI,aACtFlB,EAA8B,QAAjBtH,EAAShgD,MAAmC,SAAjBggD,EAAShgD,KAEjD4uB,EAA8B,CACnC45B,aAAc,IAAIrB,GAAcC,EAAoB,CAAEE,gBAGjC,QAAjBtH,EAAShgD,MAAmC,YAAjBggD,EAAShgD,OACxC4uB,EAAQw3F,UAuBX,SAA2Bp3F,EAAYgxB,GACtC,MAAMsmE,EAAWtmE,EAASrnD,OAAiBwb,cACrCllB,EAAI+wD,EAASumE,QACbv3H,EAAIgxD,EAASwmE,QACnB,IAAItwE,EAGCowE,EAAOG,qBAAuBH,EAAOG,oBAAqBx3H,EAAGD,GACjEknD,EAAWowE,EAAOG,oBAAqBx3H,EAAGD,GAGjCgxD,EAAS0mE,cAClBxwE,EAAWowE,EAAO15E,cAClBsJ,EAASyF,SAAUqE,EAAS0mE,YAAa1mE,EAAS2mE,aAClDzwE,EAASlO,UAAU,IAGpB,GAAKkO,EACJ,OAAOlnB,EAAK4X,aAAa6V,eAAgBvG,GAG1C,OAAO,IACR,CA7CuB0wE,CAAkB7xH,KAAKi6B,KAAMgxB,IAGlDjrD,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAAUpxB,EACrC,ECpFD,MAAMi4F,GAAuB,CAAE,aAAc,MAQ9B,SAASC,GAAiBjiD,GACxC,IAAI51C,EAAO,GAEX,GAAK41C,EAAS5hE,GAAI,UAAa4hE,EAAS5hE,GAAI,cAE3CgsB,EAAO41C,EAAShtE,UACV,GAAKgtE,EAAS5hE,GAAI,UAAW,QAAW4hE,EAAS7pC,aAAc,OAErE/L,EAAO41C,EAASz0C,aAAc,YACxB,GAAKy0C,EAAS5hE,GAAI,UAAW,MAEnCgsB,EAAO,SACD,CAGN,IAAI2b,EAAO,KAEX,UAAYl+B,KAAWm4D,EAAiDt0C,cAAgB,CACvF,MAAMw2F,EAAYD,GAAiBp6G,GAG9Bk+B,IAAUA,EAAK3nC,GAAI,qBAAwByJ,EAAMzJ,GAAI,uBAExD4jH,GAAqBviG,SAAYsmB,EAAsB7zC,OACvD8vH,GAAqBviG,SAAY5X,EAAuB3V,MAExDk4B,GAAQ,KAERA,GAAQ,QAIVA,GAAQ83F,EACRn8E,EAAOl+B,CACR,CACD,CAEA,OAAOuiB,CACR,CC0Ee,MAAM,WAA0B,GAI9C,qBAAkB5D,GACjB,MAAO,mBACR,CAKO,IAAAI,GACS12B,KAAKs0B,OACA6qE,QAAQllE,KAEvBu6B,YAAa,IAElBx0D,KAAKiyH,kBACLjyH,KAAKkyH,eACN,CAKQ,eAAAD,GACP,MAAM39F,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf6hC,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SAI1BxD,KAAKwP,SAA2CgkD,EAAc,kBAAkB,CAAEn+C,EAAKvS,KAClE,SAAfA,EAAKy1B,QAAsBjE,EAAOl8B,MAAMi9B,UAAWf,EAAOl8B,MAAMoL,SAASyxB,YAC7E5f,EAAIlJ,MACL,GACE,CAAES,SAAU,YAEf5M,KAAKwP,SAA2CgkD,EAAc,kBAAkB,CAAEn+C,EAAKvS,KACtF,MAAM2wD,EAAe3wD,EAAK2wD,aAC1B,IAAIvzD,EAGJ,GAAK4C,EAAK5C,QACTA,EAAU4C,EAAK5C,YACT,CACN,IAAIiyH,EAAc,GAEb1+D,EAAad,QAAS,aAC1Bw/D,ECvKU,SAAiCrvH,GAC/C,OAAOA,EACLmW,QAAS,2DAA2D,CAAEm5G,EAAWC,IAG3D,GAAjBA,EAAO/5H,OACJ,IAGD+5H,IAGPp5G,QAAS,mBAAoB,GAChC,CD0JmB,CAAwBw6C,EAAad,QAAS,cACjDc,EAAad,QAAS,kBExKrCz4B,GADwCA,EF0KLu5B,EAAad,QAAS,eEvKvD15C,QAAS,KAAM,QACfA,QAAS,KAAM,QAEfA,QAAS,cAAe,WAExBA,QAAS,SAAU,QAEnBA,QAAS,MAAO,4BAEhBA,QAAS,MAAO,UAChBA,QAAS,MAAO,UAEhBA,QAAS,QAAS,YAEVsW,SAAU,YAAe2K,EAAK3K,SAAU,WAEjD2K,EAAO,MAAOA,SFuJXi4F,EEjJGj4F,GFoJJh6B,EAAUF,KAAKs0B,OAAOxxB,KAAK06E,cAAcL,OAAQg1C,EAClD,CE9KY,IAA0Bj4F,EFgLtC,MAAMxpB,EAAY,IAAIxE,EAAWlM,KAAM,uBAEvCA,KAAKwQ,KAAyCE,EAAW,CACxDxQ,UACAuzD,eACAC,aAAc5wD,EAAK4wD,aACnBn7B,OAAQz1B,EAAKy1B,SAMT7nB,EAAUvE,KAAKF,QACnBoJ,EAAIlJ,OAGL8tB,EAAKo7B,sBAAsB,GACzB,CAAEzoD,SAAU,QAEf5M,KAAKwP,SAA6CxP,KAAM,uBAAuB,CAAEqV,EAAKvS,KACrF,GAAKA,EAAK5C,QAAQiiC,QACjB,OAGD,MAKMmwF,EALiBtyH,KAAKs0B,OAAOxxB,KAKE/E,QAAS+E,EAAK5C,QAAS,oBAE3B,GAA5BoyH,EAAch3F,aAInBjmB,EAAIlJ,OAIJ/T,EAAMmiC,QAAQ,KACbv6B,KAAKwQ,KAAsC,mBAAoB,CAC9DtQ,QAASoyH,EACT/5F,OAAQz1B,EAAKy1B,OACbk7B,aAAc3wD,EAAK2wD,aACnBC,aAAc5wD,EAAK4wD,cACjB,IACD,GACD,CAAE9mD,SAAU,QAEf5M,KAAKwP,SAA0CxP,KAAM,oBAAoB,CAAEqV,EAAKvS,KAC/EA,EAAKyvH,YAAcn6H,EAAMqjG,cAAe34F,EAAK5C,QAAS,GACpD,CAAE0M,SAAU,OAChB,CAKQ,aAAAslH,GACP,MAAM59F,EAASt0B,KAAKs0B,OACdk+F,EAAgBl+F,EAAOl8B,MAAMoL,SAE7BgwD,EADOl/B,EAAO6qE,QAAQllE,KACFz2B,SAEpBivH,EAAY,CAAEp9G,EAAgCvS,KACnD,MAAM2wD,EAAe3wD,EAAK2wD,aAE1B3wD,EAAKowB,iBAEL,MAAMhzB,EAAUo0B,EAAOxxB,KAAKq6E,OAAQ7oD,EAAOl8B,MAAMulG,mBAAoB60B,EAAcv9F,YAEnFu+B,EAAahjD,KAAwC,kBAAmB,CACvEijD,eACAvzD,UACAq4B,OAAQljB,EAAIrT,MACV,EAGJhC,KAAKwP,SAAiCgkD,EAAc,OAAQi/D,EAAW,CAAE7lH,SAAU,QACnF5M,KAAKwP,SAAgCgkD,EAAc,OAAO,CAAEn+C,EAAKvS,KAG1DwxB,EAAOl8B,MAAMi9B,UAAWf,EAAOl8B,MAAMoL,SAASyxB,WAGnDw9F,EAAWp9G,EAAKvS,GAFhBA,EAAKowB,gBAGN,GACE,CAAEtmB,SAAU,QAEf5M,KAAKwP,SAA4CgkD,EAAc,mBAAmB,CAAEn+C,EAAKvS,KAClFA,EAAK5C,QAAQiiC,UAClBr/B,EAAK2wD,aAAab,QAAS,YAAa5yD,KAAKs0B,OAAOxxB,KAAK06E,cAAcN,OAAQp6E,EAAK5C,UACpF4C,EAAK2wD,aAAab,QAAS,aAAcm/D,GAAiBjvH,EAAK5C,WAG5C,OAAf4C,EAAKy1B,QACTjE,EAAOl8B,MAAM29F,cAAey8B,EAAcv9F,UAC3C,GACE,CAAEroB,SAAU,OAChB,EG5Pc,MAAM8lH,GA0CpB,WAAA3wH,CAAa3J,EAAcu6H,EAAgB,IAjB3C,KAAQC,OAAuB,KAkB9B5yH,KAAK5H,MAAQA,EACb4H,KAAK6yH,MAAQ,EACb7yH,KAAK2yH,MAAQA,EACb3yH,KAAK8yH,WAAY,EAQjB9yH,KAAK+yH,gBAAkB,CAAE19G,EAAK6wD,KACxBA,EAAM6jB,SAAW7jB,EAAMyX,YAAczX,IAAUlmE,KAAK4yH,QACxD5yH,KAAKgzH,QAAQ,EACd,EAGDhzH,KAAKizH,yBAA2B,KAC/BjzH,KAAKgzH,QAAQ,EAGdhzH,KAAK5H,MAAMoL,SAAS6L,GAAyB,SAAUrP,KAAK+yH,iBAE5D/yH,KAAK5H,MAAMoL,SAASyxB,UAAU5lB,GAAkC,eAAgBrP,KAAKizH,0BACrFjzH,KAAK5H,MAAMoL,SAASyxB,UAAU5lB,GAAkC,mBAAoBrP,KAAKizH,yBAC1F,CAMA,SAAW/sD,GAKV,OAJMlmE,KAAK4yH,SACV5yH,KAAK4yH,OAAS5yH,KAAK5H,MAAM+lG,YAAa,CAAElU,UAAU,KAG5CjqF,KAAK4yH,MACb,CAMA,QAAW/+G,GACV,OAAO7T,KAAK6yH,KACb,CAQO,KAAAp2G,CAAOwwE,GACbjtF,KAAK6yH,OAAS5lC,EAETjtF,KAAK6yH,OAAS7yH,KAAK2yH,OACvB3yH,KAAKgzH,QAAQ,EAEf,CAKA,YAAWE,GACV,OAAOlzH,KAAK8yH,SACb,CAKO,IAAAK,GACNnzH,KAAK8yH,WAAY,CAClB,CAKO,MAAAM,GACNpzH,KAAK8yH,WAAY,CAClB,CAKO,OAAA9sG,GACNhmB,KAAK5H,MAAMoL,SAAS4I,IAAK,SAAUpM,KAAK+yH,iBACxC/yH,KAAK5H,MAAMoL,SAASyxB,UAAU7oB,IAAK,eAAgBpM,KAAKizH,0BACxDjzH,KAAK5H,MAAMoL,SAASyxB,UAAU7oB,IAAK,mBAAoBpM,KAAKizH,yBAC7D,CAOQ,MAAAD,CAAQK,GAAsB,GAC/BrzH,KAAKkzH,WAAYG,IACtBrzH,KAAK4yH,OAAS,KACd5yH,KAAK6yH,MAAQ,EAEf,ECnKc,MAAMS,WAA0B,GAY9C,WAAAvxH,CAAauyB,EAAgBi/F,GAC5B/lH,MAAO8mB,GAEPt0B,KAAKwzH,QAAU,IAAId,GAAcp+F,EAAOl8B,MAAOm7H,GAG/CvzH,KAAK80B,4BAA6B,CACnC,CAKA,UAAW1Z,GACV,OAAOpb,KAAKwzH,OACb,CAKgB,OAAAxtG,GACfxY,MAAMwY,UAENhmB,KAAKwzH,QAAQxtG,SACd,CAUgB,OAAAuP,CAAShxB,EAAoC,CAAC,GAC7D,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MACpBif,EAAMjf,EAAMoL,SACZ02B,EAAO31B,EAAQ21B,MAAQ,GACvBu5F,EAAiBv5F,EAAK5hC,OAE5B,IAAI28B,EAA2C5d,EAAI4d,UASnD,GAPK1wB,EAAQ0wB,UACZA,EAAY1wB,EAAQ0wB,UACT1wB,EAAQugB,QACnBmQ,EAAY78B,EAAM4/C,gBAAiBzzC,EAAQugB,SAItC1sB,EAAMi9B,UAAWJ,GACtB,OAGD,MAAMs9F,EAAchuH,EAAQguH,YAE5Bn6H,EAAMguE,cAAepmE,KAAKwzH,QAAQttD,OAAO1rC,IACxCx6B,KAAKwzH,QAAQL,OAEb/6H,EAAM29F,cAAe9gE,GAEhBiF,GACJ9hC,EAAMqjG,cAAejhE,EAAOmZ,WAAYzZ,EAAM7iB,EAAI4d,UAAU+Q,iBAAmB/Q,GAG3Es9F,EACJ/3F,EAAOgZ,aAAc++E,GACTt9F,EAAU/mB,GAAI,sBAC1BssB,EAAOgZ,aAAcve,GAGtBj1B,KAAKwzH,QAAQJ,SAEbpzH,KAAKwzH,QAAQ/2G,MAAOg3G,EAAgB,GAEtC,EChFD,MAAMC,GAAqB,CAO1B,aAIA,yBAMc,MAAMC,WAA2BhpE,GAU/C,WAAA5oD,CAAak4B,GACZzsB,MAAOysB,GAEPj6B,KAAKuxD,cAAgBt3B,EAAKq3B,YAAaN,IAKlC,EAAI3nD,WACRqqH,GAAmB9zH,KAAM,yBAG1B,MAAM4zD,EAAev5B,EAAKz2B,SAE1BgwD,EAAankD,GAA4B,eAAe,CAAEgG,EAAKvS,KAC9D,IAAM9C,KAAK00B,UACV,OAGD,MAAQ5xB,KAAMo3B,EAAI,aAAEw5B,EAAY,UAAEC,EAAS,SAAE1I,GAAanoD,EAE1D,IAAM4wH,GAAmBnkG,SAAUokC,GAClC,OAKD3zD,KAAKuxD,cAAclE,QAEnB,MAAM38C,EAAY,IAAIxE,EAAWsnD,EAAc,cAE/CA,EAAahjD,KAAME,EAAW,IAAIs6C,GAAc/wB,EAAMgxB,EAAU,CAC/D/wB,OACAjF,UAAWgF,EAAK+d,gBAAiB0b,MAK7BhjD,EAAUvE,KAAKF,QACnBoJ,EAAIlJ,MACL,IAIDqnD,EAAankD,GAAqC,kBAAkB,CAAEgG,GAAOvS,OAAMmoD,eAI5EjrD,KAAK00B,YAAa,EAAIrrB,WAKtBvG,GA0BN0wD,EAAahjD,KAAM,aAAc,IAAIw6C,GAAc/wB,EAAMgxB,EAAU,CAClE/wB,KAAMp3B,EACNmyB,UAAWu+B,EAAav+B,YACpB,GACH,CAAEroB,SAAU,UAChB,CAKO,OAAAmZ,GAAiB,CAKjB,aAAAulC,GAAuB,EC5HhB,MAAMsoE,WAAc,GAIlC,qBAAkBt9F,GACjB,MAAO,OACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf6hC,EAAO3F,EAAO6qE,QAAQllE,KACtBq5C,EAAiBl7E,EAAMoL,SAASyxB,UAEtCgF,EAAKu6B,YAAam/D,IAGlB,MAAME,EAAoB,IAAIP,GAAmBh/F,EAAQA,EAAOtL,OAAO5gB,IAAK,oBAAuB,IAGnGksB,EAAO0qE,SAASnxF,IAAK,aAAcgmH,GACnCv/F,EAAO0qE,SAASnxF,IAAK,QAASgmH,GAE9B7zH,KAAKwP,SAAuCyqB,EAAKz2B,SAAU,cAAc,CAAE6R,EAAKvS,KAGzEm3B,EAAKz2B,SAASm4B,aACnB74B,EAAKowB,iBAGN,MAAM,KAAEgH,EAAMjF,UAAW+hB,EAAeu7E,YAAauB,GAAoBhxH,EAGnEixH,EAAcpzH,MAAMrB,KAAM03C,EAAc5J,aAAcpwC,KAAKypD,GACzDnyB,EAAO6qE,QAAQ3hC,OAAOL,aAAc1W,KAG5C,IAAIusC,EAAa94D,EAIjB,GAAK,EAAI7wB,UAAY,CACpB,MAAM2qH,EAAerzH,MAAMrB,KAAMy0H,EAAa,GAAI7nF,YAAaje,QAAQ,CAAEgmG,EAAW90H,IAC5E80H,GAAc90H,EAAK+O,GAAI,cAAiB/O,EAAK2D,KAAO,KACzD,IAEEkxH,IACCA,EAAa17H,QAAU06F,EAAW16F,OACjC06F,EAAWpuC,WAAYovE,KAC3BhhC,EAAaA,EAAWp2F,UAAWo3H,EAAa17H,QAC9Cy7H,EAAa,GAAarzF,MAAQqzF,EAAa,GAAIrzF,MAAM4J,aAAc0pF,EAAa17H,SAGlF07H,EAAapvE,WAAYouC,KAE3B+gC,EAAa,GAAarzF,MAAQqzF,EAAa,GAAIrzF,MAAM4J,aAAc0oD,EAAW16F,QACpF06F,EAAa,IAIjB,CAEA,MAAMkhC,EAAyE,CAC9Eh6F,KAAM84D,EACN/9D,UAAW78B,EAAM4/C,gBAAiB+7E,IAW9BD,IACJI,EAAsB3B,YAAcj+F,EAAO6qE,QAAQ3hC,OAAOL,aAAc22D,IAGzEx/F,EAAOiB,QAAS,aAAc2+F,GAE9Bj6F,EAAKo7B,sBAAsB,IAGvB,EAAIhsD,UAIRrJ,KAAKwP,SAAUyqB,EAAKz2B,SAAU,WAAW,CAAE6R,EAAKvS,MAC1CwwE,EAAetoC,aAA+B,KAAhBloC,EAAKwqB,SAAmB2M,EAAKz2B,SAASm4B,aAczEw4F,GAAwB/7H,EAAOy7H,EAAmB,IAKnD7zH,KAAKwP,SAAUyqB,EAAKz2B,SAAU,oBAAoB,KAC5C8vE,EAAetoC,aAcpBmpF,GAAwB/7H,EAAOy7H,EAAmB,GAGrD,EAGD,SAASM,GAAwB/7H,EAAcy7H,GAM9C,IAAMA,EAAkBn/F,UACvB,OAGD,MAAMtZ,EAASy4G,EAAkBz4G,OAEjCA,EAAO+3G,OAEP/6H,EAAMguE,cAAehrD,EAAO8qD,OAAO,KAClC9tE,EAAM29F,cAAe39F,EAAMoL,SAASyxB,UAAW,IAGhD7Z,EAAOg4G,QACR,CCxJe,MAAMgB,WAAsB,GAkB1C,WAAAryH,CAAauyB,EAAgB/P,GAC5B/W,MAAO8mB,GAEPt0B,KAAKukB,UAAYA,EACjBvkB,KAAKwzH,QAAU,IAAId,GAAcp+F,EAAOl8B,MAAOk8B,EAAOtL,OAAO5gB,IAAK,oBAGlEpI,KAAK80B,4BAA6B,CACnC,CAKA,UAAW1Z,GACV,OAAOpb,KAAKwzH,OACb,CAagB,OAAAj+F,CAAShxB,EAIrB,CAAC,GACJ,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MACpBif,EAAMjf,EAAMoL,SAElBpL,EAAMguE,cAAepmE,KAAKwzH,QAAQttD,OAAO1rC,IACxCx6B,KAAKwzH,QAAQL,OAEb,MAAMl+F,EAAYuF,EAAOwd,gBAAiBzzC,EAAQ0wB,WAAa5d,EAAI4d,WAGnE,IAAM78B,EAAMi9B,UAAWJ,GACtB,OAGD,MAAMo/F,EAAW9vH,EAAQ8vH,UAAY,EAO/Bp+B,EAA0BhhE,EAAU+V,YAY1C,GATK/V,EAAU+V,aACd5yC,EAAMo+F,gBAAiBvhE,EAAW,CACjC1Q,UAAWvkB,KAAKukB,UAChBoC,KAAMpiB,EAAQoiB,KACds1E,wBAAwB,IAKrBj8F,KAAKs0H,4CAA6CD,GAGtD,YAFAr0H,KAAKu0H,mCAAoC/5F,GAO1C,GAAKx6B,KAAKw0H,sCAAuCv/F,EAAWo/F,GAG3D,YAFAr0H,KAAKs0B,OAAOiB,QAAS,YAAa,CAAEN,cAMrC,GAAKA,EAAU+V,YACd,OAGD,IAAIiiD,EAAc,EAElBh4D,EAAUoY,gBAAiB6sB,uBAAuBt8D,SAASknB,IAC1DmoE,GAAetoF,EACdmgB,EAAM+lB,UAAW,CAAEhC,kBAAkB,EAAME,kBAAkB,EAAMD,SAAS,IAC5E,IAUF1wC,EAAM29F,cAAe9gE,EAAW,CAC/BghE,0BACA1xE,UAAWvkB,KAAKukB,YAGjBvkB,KAAKwzH,QAAQ/2G,MAAOwwE,GAEpBzyD,EAAOgZ,aAAcve,GAErBj1B,KAAKwzH,QAAQJ,QAAQ,GAEvB,CAkBQ,2CAAAkB,CAA6CD,GAEpD,GAAKA,EAAW,EACf,OAAO,EAGR,MAAMj8H,EAAQ4H,KAAKs0B,OAAOl8B,MAEpB68B,EADM78B,EAAMoL,SACIyxB,UAChB4hD,EAAez+E,EAAMsrE,OAAOyO,gBAAiBl9C,GAMnD,KAF4BA,EAAU+V,aAAe/V,EAAUmuC,sBAAuByT,IAGrF,OAAO,EAGR,IAAMz+E,EAAMsrE,OAAOyF,WAAY0N,EAAc,aAC5C,OAAO,EAGR,MAAM49C,EAAyB59C,EAAah7C,SAAU,GAKtD,OAAK44F,IAA0BA,EAAuBvmH,GAAI,UAAW,YAKtE,CAOQ,kCAAAqmH,CAAoC/5F,GAC3C,MAAMpiC,EAAQ4H,KAAKs0B,OAAOl8B,MAEpB68B,EADM78B,EAAMoL,SACIyxB,UAChB4hD,EAAez+E,EAAMsrE,OAAOyO,gBAAiBl9C,GAC7C+4C,EAAYxzC,EAAOl1B,cAAe,aAExCk1B,EAAOv0B,OAAQu0B,EAAOud,cAAe8+B,IACrCr8C,EAAO90B,OAAQsoE,EAAW6I,GAE1Br8C,EAAOgZ,aAAcw6B,EAAW,EACjC,CASQ,qCAAAwmD,CAAuCv/F,EAAsBo/F,GACpE,MAAMj8H,EAAQ4H,KAAKs0B,OAAOl8B,MAG1B,GAAKi8H,EAAW,GAAuB,YAAlBr0H,KAAKukB,UACzB,OAAO,EAGR,IAAM0Q,EAAU+V,YACf,OAAO,EAGR,MAAMvlB,EAAWwP,EAAUE,mBACrB0hD,EAAez+E,EAAMsrE,OAAOyO,gBAAiB1sD,GAC7CgvG,EAAyB59C,EAAah7C,SAAU,GAItD,OAAKpW,EAAS7mB,QAAU61H,MAKlBx/F,EAAUmuC,sBAAuBqxD,OAKjCr8H,EAAMsrE,OAAOyF,WAAY0N,EAAc,cAKT,aAA/B49C,EAAuBzyH,MAK7B,ECzOD,MACM0yH,GAAc,OAEdC,GAAmB,YACnBC,GAAkB,WAClBC,GAAiB,UAOjBC,GAAsD,CAI3D/+B,cAAe,CACdpvE,KAAMguG,GAINpwG,UAAWqwG,IAGZG,sBAAuB,CAetBpuG,KArCwB,YAsCxBpC,UAAWqwG,IAIZI,mBAAoB,CACnBruG,KAAM+tG,GACNnwG,UAAWqwG,IAGZK,uBAAwB,CACvBtuG,KAAMguG,GACNpwG,UAAWqwG,IAGZM,uBAAwB,CACvBvuG,KAAMguG,GACNpwG,UAAWqwG,IAOZO,qBAAsB,CAUrBxuG,KAzEuB,YA0EvBpC,UAAWswG,IAGZO,kBAAmB,CAClBzuG,KAAM+tG,GACNnwG,UAAWswG,IAKZQ,sBAAuB,CACtB1uG,KAAMguG,GACNpwG,UAAWswG,IAIZS,sBAAuB,CACtB3uG,KAAMguG,GACNpwG,UAAWswG,KAOE,MAAMU,WAAuB5qE,GAI3C,WAAA5oD,CAAak4B,GACZzsB,MAAOysB,GAEP,MAAMz2B,EAAWy2B,EAAKz2B,SAQtB,IAAI6wH,EAAW,EAEf7wH,EAAS6L,GAAI,WAAW,KACvBglH,GAAA,IAGD7wH,EAAS6L,GAAI,SAAS,KACrBglH,EAAW,CAAC,IAGb7wH,EAAS6L,GAA4B,eAAe,CAAEgG,EAAKvS,KAC1D,IAAM9C,KAAK00B,UACV,OAGD,MAAM,aAAEg/B,EAAY,SAAEzI,EAAQ,UAAE0I,GAAc7wD,EACxC0yH,EAAkBV,GAAoBnhE,GAE5C,IAAM6hE,EACL,OAGD,MAAM1yE,EAAuC,CAC5Cv+B,UAAWixG,EAAgBjxG,UAC3BoC,KAAM6uG,EAAgB7uG,KACtB0tG,YAGIvxE,EAAWn8B,MAAQguG,KACvB7xE,EAAW2yE,kBAAoBx7F,EAAK+d,gBAAiB0b,EAAc,KAKjD,0BAAdC,IAEC,EAAItqD,YACRy5C,EAAWuxE,SAAW,GAoJ3B,SAAgC3gE,GAG/B,GAA4B,GAAvBA,EAAap7D,QAAeo7D,EAAc,GAAI1oB,YAClD,OAAO,EAGR,MAAM0L,EAASgd,EAAc,GAAI7oB,UAAW,CAC3CtmB,UAAW,WACXskB,kBAAkB,EAClBE,kBAAkB,IAGnB,IAAIpkC,EAAQ,EAEZ,UAAY,aAAEslC,KAAkByM,EAAS,CAExC,GAAMzM,EAAarrC,OAAOsP,GAAI,SAEvB,CACN,MAAMpL,EAAOmnC,EAAarrC,OAAOkE,KAC3B8Y,EAASquB,EAAaruB,OAG5B,GACC8X,GAAuB5wB,EAAM8Y,IAC7BiY,GAAwB/wB,EAAM8Y,IAC9BsY,GAAuBpxB,EAAM8Y,GAE7B,SAGDjX,GACD,MAfCA,IAiBD,GAAKA,EAAQ,EACZ,OAAO,CAET,CAEA,OAAO,CACR,CAzLS+wH,CAAuBhiE,KAC3B5Q,EAAWn8B,KAAOguG,GAClB7xE,EAAW2yE,kBAAoBx7F,EAAK+d,gBAAiB0b,KAIvD,MAAMhjD,EAAY,IAAI2+B,GAAmB7rC,EAAU,SAAUkwD,EAAc,IAE3ElwD,EAASgN,KAAME,EAAW,IAAIs6C,GAAc/wB,EAAMgxB,EAAUnI,IAIvDpyC,EAAUvE,KAAKF,QACnBoJ,EAAIlJ,MACL,IAII,EAAI7C,SA0DX,SAAiCyrD,GAChC,MAAM96B,EAAO86B,EAAS96B,KAChBz2B,EAAWy2B,EAAKz2B,SAEtB,IAAImyH,EAAgC,KAChCC,GAAsB,EAmD1B,SAASC,EAAiBvoG,GACzB,OAAOA,GAAWjB,GAASK,WAAaY,GAAWjB,GAAS1a,MAC7D,CAEA,SAASmkH,EAAoBxoG,GAC5B,OAAOA,GAAWjB,GAASK,UAAYkoG,GAAkBC,EAC1D,CAvDArxH,EAAS6L,GAA8B,WAAW,CAAEgG,GAAOiY,cAC1DqoG,EAAiBroG,EACjBsoG,GAAsB,CAAK,IAG5BpyH,EAAS6L,GAA4B,SAAS,CAAEgG,GAAOiY,UAAS29B,eAC/D,MAAMh2B,EAAYzxB,EAASyxB,UACrB8gG,EAAwBhhE,EAASrgC,WACtCpH,GAAWqoG,GACXE,EAAiBvoG,KAChB2H,EAAU+V,cACV4qF,EAIF,GAFAD,EAAiB,KAEZI,EAAwB,CAC5B,MAAMC,EAAc/gG,EAAUoY,gBACxB38B,EAAY,IAAI2+B,GAAmB7rC,EAAU,SAAUwyH,GACvDlzE,EAAuC,CAC5Cn8B,KAAMguG,GACNpwG,UAAWuxG,EAAoBxoG,GAC/BmoG,kBAAmBxgG,GAGpBzxB,EAASgN,KAAME,EAAW,IAAIs6C,GAAc/wB,EAAMgxB,EAAUnI,GAC7D,KAGDt/C,EAAS6L,GAA4B,eAAe,CAAEgG,GAAOs+C,gBAC5D,MAAM6hE,EAAkBV,GAAoBnhE,GACdkiE,EAAiBF,IAC9CH,GACAA,EAAgBjxG,WAAauxG,EAAoBH,KAGjDC,GAAsB,EACvB,GACE,CAAEhpH,SAAU,SAEfpJ,EAAS6L,GAA4B,eAAe,CAAEgG,GAAOs+C,YAAW7wD,WACvC6yH,GAAkBtpG,GAAS1a,QAC7C,cAAbgiD,GACQ,KAAR7wD,GAGAuS,EAAIlJ,MACL,GACE,CAAES,SAAU,QAShB,CAxHGqpH,CAAwBj2H,KAE1B,CAKO,OAAA+lB,GAAiB,CAKjB,aAAAulC,GAAuB,ECvMhB,MAAM,WAAe,GASnC,qBAAkBh1B,GACjB,MAAO,QACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd2F,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SACpBgvH,EAAgBl+F,EAAOl8B,MAAMoL,SAEnCy2B,EAAKu6B,YAAa+gE,IAElBv1H,KAAKk2H,kBAAmB,EAExB,MAAMC,EAAuB,IAAI/B,GAAe9/F,EAAQ,WAGxDA,EAAO0qE,SAASnxF,IAAK,gBAAiBsoH,GACtC7hG,EAAO0qE,SAASnxF,IAAK,gBAAiBsoH,GAEtC7hG,EAAO0qE,SAASnxF,IAAK,SAAU,IAAIumH,GAAe9/F,EAAQ,aAE1Dt0B,KAAKwP,SAAmCgkD,EAAc,UAAU,CAAEn+C,EAAKvS,KAGhE0wD,EAAa73B,aAClB74B,EAAKowB,iBAGN,MAAM,UAAE3O,EAAS,SAAE8vG,EAAQ,kBAAEoB,EAAiB,KAAE9uG,GAAS7jB,EACnD82B,EAA4B,YAAdrV,EAA0B,gBAAkB,SAC1D6xG,EAA2D,CAAE/B,YAEnE,GAAa,aAAR1tG,EAAsB,CAC1B,MAAMotG,EAAcpzH,MAAMrB,KAAMm2H,EAAmBroF,aAAcpwC,KAAKypD,GAC9DnyB,EAAO6qE,QAAQ3hC,OAAOL,aAAc1W,KAG5C2vE,EAAYnhG,UAAYX,EAAOl8B,MAAM4/C,gBAAiB+7E,EACvD,MACCqC,EAAYzvG,KAAOA,EAGpB2N,EAAOiB,QAASqE,EAAaw8F,GAE7Bn8F,EAAKo7B,sBAAsB,GACzB,CAAEzoD,SAAU,QAEV5M,KAAKs0B,OAAOqC,QAAQ/oB,IAAK,iBAC7B5N,KAAKwP,SAAmCgkD,EAAc,UAAU,CAAEn+C,EAAKvS,KACjE9C,KAAKk2H,kBAAsC,YAAlBpzH,EAAKyhB,WAA4C,GAAjBzhB,EAAKuxH,UAA8B,aAAbvxH,EAAK6jB,OACxF3mB,KAAKk2H,kBAAmB,EAExB5hG,EAAOiB,QAAS,QAEhBzyB,EAAKowB,iBACL7d,EAAIlJ,OACL,GACE,CAAEoB,QAAS,aAEdvN,KAAKwP,SAAUgjH,EAAe,UAAU,KACvCxyH,KAAKk2H,kBAAmB,CAAK,IAGhC,CAOO,sBAAAG,GACDr2H,KAAKs0B,OAAOqC,QAAQ/oB,IAAK,iBAC7B5N,KAAKk2H,kBAAmB,EAE1B,ECrFc,MAAMI,WAAe,GACnC,mBAAkBp/F,GACjB,MAAO,CAAE08F,GAAO,GACjB,CAKA,qBAAkBt9F,GACjB,MAAO,QACR,ECSc,SAASigG,GAAiBzxG,EAAc1sB,GACtD,IAAIsoC,EAAQ5b,EAAM4b,MAalB,MAAO,CAAExG,KAXIv5B,MAAMrB,KAAMwlB,EAAM+lB,UAAW,CAAE9B,kBAAkB,KAAY9a,QAAQ,CAAEgmG,GAAah0H,UAExFA,EAAKiO,GAAI,UAAajO,EAAKiO,GAAI,cAMhC+lH,EAAYh0H,EAAK6C,MALvB49B,EAAQtoC,EAAMu/C,oBAAqB13C,GAE5B,KAIN,IAEY6kB,MAAO1sB,EAAMy/C,YAAanX,EAAO5b,EAAM6b,KACvD,CC1Be,MAAM61F,WAAoBrjH,KA0CxC,WAAApR,CAAa3J,EAAcq+H,GAC1BjpH,QAEAxN,KAAK5H,MAAQA,EACb4H,KAAKy2H,aAAeA,EACpBz2H,KAAK02H,WAAY,EAEjB12H,KAAKyR,IAAK,aAAa,GAGvBzR,KAAKqP,GAA2B,oBAAoB,KAC9CrP,KAAK00B,UACT10B,KAAK22H,mBAEL32H,KAAK2P,cAAevX,EAAMoL,SAASyxB,WACnCj1B,KAAK2P,cAAevX,EAAMoL,UAC3B,IAGDxD,KAAK22H,iBACN,CAKA,YAAWC,GACV,OAAO52H,KAAK02H,SACb,CAKQ,eAAAC,GACP,MACMnzH,EADQxD,KAAK5H,MACIoL,SAEvBxD,KAAKwP,SAAwChM,EAASyxB,UAAW,gBAAgB,CAAE5f,GAAOotD,mBAEnFA,IAKAj/D,EAASyxB,UAAU+V,YASzBhrC,KAAK62H,6BAA8B,aAR7B72H,KAAK42H,WACT52H,KAAKwQ,KAAM,aACXxQ,KAAK02H,WAAY,GAM6B,IAGjD12H,KAAKwP,SAA+BhM,EAAU,eAAe,CAAE6R,EAAK6wD,MAC9DA,EAAM8jB,QAAW9jB,EAAM6jB,SAI5B/pF,KAAK62H,6BAA8B,OAAQ,CAAE3wD,SAAS,GAExD,CAYQ,4BAAA2wD,CAA8BC,EAA8Bh0H,EAA0B,CAAC,GAC9F,MAAM1K,EAAQ4H,KAAK5H,MAEb68B,EADW78B,EAAMoL,SACIyxB,UAErB8hG,EAAuB3+H,EAAMy/C,YAAaz/C,EAAMs/C,iBAAkBziB,EAAUgY,MAAOruC,OAAQ,GAAKq2B,EAAUgY,QAE1G,KAAE/S,EAAI,MAAEpV,GAAUyxG,GAAiBQ,EAAsB3+H,GAEzD4+H,EAAah3H,KAAKy2H,aAAcv8F,GAQtC,IANM88F,GAAch3H,KAAK42H,UACxB52H,KAAKwQ,KAAiC,aAGvCxQ,KAAK02H,YAAcM,EAEdA,EAAa,CACjB,MAAMC,EAAYjgI,OAAO+wB,OAAQjlB,EAAM,CAAEo3B,OAAMpV,UAGrB,iBAAdkyG,GACXhgI,OAAO+wB,OAAQkvG,EAAWD,GAG3Bh3H,KAAKwQ,KAA+B,WAAYsmH,IAAWG,EAC5D,CACD,ECzBc,MAAM,WAA6B,GA8BjD,WAAAl1H,CAAauyB,GACZ9mB,MAAO8mB,GAEPt0B,KAAKuF,WAAa,IAAIqO,IACtB5T,KAAKk3H,aAAe,IACrB,CAZA,qBAAkB5gG,GACjB,MAAO,sBACR,CAeO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf6hC,EAAO3F,EAAO6qE,QAAQllE,KACtBf,EAAS5E,EAAO4E,OAEhBo6C,EAAiBl7E,EAAMoL,SAASyxB,UAGtCj1B,KAAKwP,SAAqCyqB,EAAKz2B,SAAU,YAAY,CAAE6R,EAAKvS,KAE3E,IAAMwwE,EAAetoC,YACpB,OAKD,GAAKloC,EAAK2qB,UAAY3qB,EAAKyqB,QAAUzqB,EAAK0qB,QACzC,OAGD,MAAM2pG,EAAoBr0H,EAAKwqB,SAAWjB,GAASG,WAC7C4qG,EAAmBt0H,EAAKwqB,SAAWjB,GAASC,UAGlD,IAAM6qG,IAAsBC,EAC3B,OAGD,MAAMC,EAAmBn+F,EAAO3K,yBAChC,IAAI+oG,GAAoB,EAGvBA,EAD2B,QAArBD,GAA8BF,GAA8C,QAArBE,GAA8BD,EACvEp3H,KAAKu3H,uBAAwBz0H,GAE7B9C,KAAKw3H,wBAAyB10H,IAKxB,IAAtBw0H,GACJjiH,EAAIlJ,MACL,GACE,CAAEoB,QAAS,QAASX,SAAU,YAEjC5M,KAAKy3H,kCAAmC,EAGxCz3H,KAAKwP,SAAwC8jE,EAAgB,gBAAgB,CAAEj+D,EAAKvS,KAI9E9C,KAAKy3H,iCACTz3H,KAAKy3H,kCAAmC,EAOnCz3H,KAAK03H,wBAOL50H,EAAK2/D,cAAgBk1D,GAA8BrkD,EAAen+C,mBAAqBn1B,KAAKuF,aAIlGvF,KAAKolE,kBAAiB,GAExB,CAOO,iBAAAwyD,CAAmBxhH,GACzBpW,KAAKuF,WAAWsI,IAAKuI,EACtB,CASQ,sBAAAmhH,CAAwBz0H,GAC/B,MAAMyC,EAAavF,KAAKuF,WAElB0vB,EADQj1B,KAAKs0B,OAAOl8B,MACFoL,SAASyxB,UAC3BxP,EAAWwP,EAAUE,mBAW3B,OAAKn1B,KAAK03H,yBAWLjyG,EAASqkB,YAAa+tF,GAAiB5iG,EAAW1vB,QAWlDoyH,GAA8BlyG,EAAUlgB,KAC5CuyH,GAAsBh1H,GACtB9C,KAAKklE,oBAEE,IAIT,CASQ,uBAAAsyD,CAAyB10H,GAChC,MAAMyC,EAAavF,KAAKuF,WAClBnN,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3BxP,EAAWwP,EAAUE,mBAU3B,OAAKn1B,KAAK03H,sBACTI,GAAsBh1H,GACtB9C,KAAKolE,kBACL2yD,GAAyC3/H,EAAOmN,EAAYkgB,IAErD,GAOFA,EAASqkB,YACR+tF,GAAiB5iG,EAAW1vB,KAChCuyH,GAAsBh1H,GACtBi1H,GAAyC3/H,EAAOmN,EAAYkgB,IAErD,KAoIZ,SAA0CA,EAAoBlgB,GAC7D,MAAMyyH,EAAiBvyG,EAAS6kB,cAAe,GAC/C,OAAOqtF,GAA8BK,EAAgBzyH,EACtD,CAxHQ0yH,CAAiCxyG,EAAUlgB,KAO9CkgB,EAAS+jB,UACRquF,GAAiB5iG,EAAW1vB,IAC7BoyH,GAA8BlyG,EAAUlgB,IAExCuyH,GAAsBh1H,GACtBi1H,GAAyC3/H,EAAOmN,EAAYkgB,IAErD,IAKRzlB,KAAKy3H,kCAAmC,EACxCz3H,KAAKklE,oBAKE,GAKV,CAKA,wBAAYwyD,GACX,QAAS13H,KAAKk3H,YACf,CAQQ,gBAAAhyD,GACPllE,KAAKk3H,aAAel3H,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,GACtCA,EAAOw6D,4BAEhB,CAOQ,eAAA5vB,GACPplE,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzBA,EAAOy6D,wBAAyBj1F,KAAKk3H,cACrCl3H,KAAKk3H,aAAe,IAAI,GAE1B,EAMD,SAASW,GAAiB5iG,EAA8B1vB,GACvD,UAAY2yH,KAAqB3yH,EAChC,GAAK0vB,EAAUgR,aAAciyF,GAC5B,OAAO,EAIT,OAAO,CACR,CAOA,SAASH,GAAyC3/H,EAAcmN,EAAyBkgB,GACxF,MAAM0kB,EAAa1kB,EAAS0kB,WAC5B/xC,EAAMmiC,QAAQC,IACb,GAAK2P,EAAa,CACjB,MAAM5kC,EAAuC,GACvC4yH,EAAiB//H,EAAMsrE,OAAOgE,SAAUv9B,IAAgB/xC,EAAMsrE,OAAOkE,SAAUz9B,GAErF,UAAcpzC,EAAKyB,KAAW2xC,EAAWnE,iBAEvC5tC,EAAMsrE,OAAOmE,eAAgB,QAAS9wE,IACnCohI,IAAgF,IAA9D//H,EAAMsrE,OAAOoE,uBAAwB/wE,GAAMgxE,gBAEhExiE,EAAW3F,KAAM,CAAE7I,EAAKyB,IAI1BgiC,EAAOm6D,sBAAuBpvF,EAC/B,MACCi1B,EAAOskD,yBAA0Bv5E,EAClC,GAEF,CAOA,SAASuyH,GAAsBh1H,GAC9BA,EAAKowB,gBACN,CAaA,SAASykG,GAA8BlyG,EAAoBlgB,GAC1D,MAAM,WAAE4kC,EAAU,UAAED,GAAczkB,EAClC,UAAYyyG,KAAqB3yH,EAAa,CAC7C,MAAM6yH,EAAajuF,EAAaA,EAAW9O,aAAc68F,QAAsB,EAG/E,IAFkBhuF,EAAYA,EAAU7O,aAAc68F,QAAsB,KAEzDE,EAClB,OAAO,CAET,CACA,OAAO,CACR,CC1fA,IAAI,GAAe,sBACfC,GAAkB1uH,OAAO,GAAazG,QAwB1C,SAPA,SAAsBzG,GAEpB,OADAA,EAAS,GAASA,KACA47H,GAAgBp2H,KAAKxF,GACnCA,EAAOwc,QAAQ,GAAc,QAC7Bxc,CACN,ECNM67H,GAAiE,CAEtEC,UAAW,CAAEj5H,KAAM,MAAOC,GAAI,KAC9Bi5H,oBAAqB,CAAEl5H,KAAM,MAAOC,GAAI,KACxCk5H,UAAW,CAAEn5H,KAAM,OAAQC,GAAI,KAG/Bm5H,QAAS,CAAEp5H,KAAM,qCAAsCC,GAAI,CAAE,KAAM,IAAK,OACxEo5H,SAAU,CAAEr5H,KAAM,qCAAsCC,GAAI,CAAE,KAAM,IAAK,OACzEq5H,UAAW,CAAEt5H,KAAM,qCAAsCC,GAAI,CAAE,KAAM,IAAK,OAC1Es5H,SAAU,CAAEv5H,KAAM,qCAAsCC,GAAI,CAAE,KAAM,IAAK,OACzEu5H,cAAe,CAAEx5H,KAAM,qCAAsCC,GAAI,CAAE,KAAM,IAAK,OAC9Ew5H,gBAAiB,CAAEz5H,KAAM,KAAMC,GAAI,KACnCy5H,mBAAoB,CAAE15H,KAAM,KAAMC,GAAI,KACtC05H,SAAU,CAAE35H,KAAM,KAAMC,GAAI,KAC5B25H,UAAW,CAAE55H,KAAM,KAAMC,GAAI,KAC7B45H,WAAY,CAAE75H,KAAM,KAAMC,GAAI,KAG9B65H,mBAAoB,CAAE95H,KAAM,MAAOC,GAAI,KACvC85H,OAAQ,CAAE/5H,KAAM,gBAAiBC,GAAI,CAAE,KAAM,IAAK,OAClD+5H,OAAQ,CAAEh6H,KAAM,iBAAkBC,GAAI,CAAE,KAAM,IAAK,OAGnDg6H,cAAe,CAAEj6H,KAAMk6H,GAAmB,KAAOj6H,GAAI,CAAE,KAAM,IAAK,KAAM,MACxEk6H,gBAAiB,CAAEn6H,KAAMk6H,GAAmB,KAAQj6H,GAAI,CAAE,KAAM,IAAK,KAAM,MAG3Em6H,kBAAmB,CAAEp6H,KAAMk6H,GAAmB,KAAQj6H,GAAI,CAAE,KAAM,IAAK,KAAM,MAC7Eo6H,oBAAqB,CAAEr6H,KAAMk6H,GAAmB,KAAOj6H,GAAI,CAAE,KAAM,IAAK,KAAM,MAG9Eq6H,gBAAiB,CAAEt6H,KAAMk6H,GAAmB,KAAOj6H,GAAI,CAAE,KAAM,IAAK,KAAM,MAC1Es6H,kBAAmB,CAAEv6H,KAAMk6H,GAAmB,KAAQj6H,GAAI,CAAE,KAAM,IAAK,KAAM,OAIxEu6H,GAAuD,CAC5DC,QAAS,CAAE,YAAa,sBAAuB,aAC/CC,aAAc,CACb,UAAW,WAAY,YAAa,WAAY,gBAChD,kBAAmB,qBAAsB,WACzC,YAAa,cAEdC,WAAY,CAAE,qBAAsB,SAAU,UAC9CC,OAAQ,CAAE,gBAAiB,oBAItBC,GAA0B,CAC/B,UACA,eACA,aACA,UA0HD,SAASC,GAAe96H,GACvB,MAAoB,iBAARA,EACJ,IAAIqK,OAAQ,IAAK,GAAcrK,QAIhCA,CACR,CAOA,SAAS+6H,GAAa96H,GACrB,MAAkB,iBAANA,EACJ,IAAM,CAAEA,GACJA,aAAcoB,MAClB,IAAMpB,EAIPA,CACR,CAMA,SAAS+6H,GAAgC70G,GAGxC,OAFiBA,EAASgY,SAAWhY,EAASgY,SAAWhY,EAASykB,WAEjDlE,eAClB,CAOA,SAASwzF,GAAmBe,GAC3B,OAAO,IAAI5wH,OAAQ,WAAY4wH,QAAuBA,QAAuBA,MAC9E,CCxNe,SAASC,GACvB/0G,EACAmhB,EACApuC,EACAJ,GAEA,OAAOA,EAAMy/C,YACZ4iF,GAAyBh1G,EAAUmhB,EAAepuC,GAAO,EAAMJ,GAC/DqiI,GAAyBh1G,EAAUmhB,EAAepuC,GAAO,EAAOJ,GAElE,CAYO,SAASqiI,GACfh1G,EACAmhB,EACApuC,EACAkiI,EACAtiI,GAIA,IAAI+G,EAAOsmB,EAASgY,WAAci9F,EAAWj1G,EAAS0kB,WAAa1kB,EAASykB,WAExEywF,EAAW,KAEf,KAAQx7H,GAAQA,EAAKk8B,aAAcuL,IAAmBpuC,GACrDmiI,EAAWx7H,EACXA,EAAOu7H,EAAWv7H,EAAKynB,gBAAkBznB,EAAK88B,YAG/C,OAAO0+F,EAAWviI,EAAMs/C,iBAAkBijF,EAAUD,EAAW,SAAW,SAAYj1G,CACvF,CC7CO,SAAUm1G,GAChBl3D,EACAm3D,GAEA,UAAYzkH,KAAaykH,EACnBzkH,GAAastD,EAAOoE,uBAAwB1xD,EAAW,IAAM0kH,oBAC3D1kH,EAGT,CCRe,MAAM2kH,WAAqB,GAIzB,OAAAxlG,GACfv1B,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzBx6B,KAAKg7H,WAAYxgG,GACjBx6B,KAAKwQ,KAAqC,eAAgB,CAAEgqB,UAAU,GAExE,CAwBO,UAAAwgG,CAAYxgG,GAClB,MAAMpiC,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3ByuC,EAAStrE,EAAMsrE,OACfu3D,EAAmBhmG,EAAU+V,YAC7BlmB,EAAQmQ,EAAUoY,gBAClBhB,EAAevnB,EAAM4b,MAAM9hC,OAC3B0tC,EAAaxnB,EAAM6b,IAAI/hC,OAG7B,GAAK8kE,EAAOI,QAASz3B,IAAkBq3B,EAAOI,QAASx3B,GAStD,OAJM2uF,GAAoB5uF,GAAgBC,GACzCl0C,EAAM29F,cAAe9gE,IAGf,EAGR,GAAKgmG,EAAmB,CACvB,MAAM5/B,EAAmBu/B,GAA0BpgG,EAAOpiC,MAAMsrE,OAAQzuC,EAAU+Q,iBAKlF,OAHAk1F,GAAY1gG,EAAQ1V,EAAM4b,OAC1BlG,EAAOm6D,sBAAuB0G,IAEvB,CACR,CAAO,CACN,MAAMzE,IAAmB9xE,EAAM4b,MAAMoJ,WAAahlB,EAAM6b,IAAI6I,SACtD2xF,EAAgC9uF,GAAgBC,EAItD,GAFAl0C,EAAM29F,cAAe9gE,EAAW,CAAE2hE,kBAE7BA,EAAgB,CAIpB,GAAKukC,EAGJ,OAFAD,GAAY1gG,EAAQvF,EAAUgY,QAEvB,EAMPzS,EAAOgZ,aAAclH,EAAY,EAEnC,CACD,CAEA,OAAO,CACR,EAaD,SAAS4uF,GAAY1gG,EAAgB4gG,GACpC5gG,EAAOz9B,MAAOq+H,GACd5gG,EAAOgZ,aAAc4nF,EAASx8H,OAAOq9B,YAAc,EACpD,CCvGA,MAAMo/F,GAAyD,CAC9DllC,gBAAiB,CAAEmlC,QAAQ,GAC3BC,gBAAiB,CAAED,QAAQ,IAMb,MAAME,WAAsB7wE,GAI1C,WAAA5oD,CAAak4B,GACZzsB,MAAOysB,GAEP,MAAM5iB,EAAMrX,KAAKwD,SACjB,IAAIi4H,GAAe,EAEnBpkH,EAAIhI,GAA8B,WAAW,CAAEgG,EAAKvS,KACnD24H,EAAe34H,EAAK2qB,QAAQ,IAG7BpW,EAAIhI,GAA4B,eAAe,CAAEgG,EAAKvS,KACrD,IAAM9C,KAAK00B,UACV,OAGD,IAAIi/B,EAAY7wD,EAAK6wD,UAGhB,EAAIzqD,UAAYuyH,GAA6B,mBAAb9nE,IACpCA,EAAY,mBAGb,MAAM1I,EAAWnoD,EAAKmoD,SAChBywE,EAAiBL,GAAmB1nE,GAE1C,IAAM+nE,EACL,OAGD,MAAMpsH,EAAQ,IAAI+/B,GAAmBh4B,EAAK,QAASvU,EAAK4wD,aAAc,IAEtEr8C,EAAI7G,KAAMlB,EAAO,IAAI07C,GAAc/wB,EAAMgxB,EAAU,CAClDqwE,OAAQI,EAAeJ,UAKnBhsH,EAAMnD,KAAKF,QACfoJ,EAAIlJ,MACL,GAEF,CAKO,OAAA4Z,GAAiB,CAKjB,aAAAulC,GAAuB,EChEhB,MAAM,WAAc,GAIlC,qBAAkBh1B,GACjB,MAAO,OACR,CAEO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd2F,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SAE1By2B,EAAKu6B,YAAagnE,IAElBlnG,EAAO0qE,SAASnxF,IAAK,QAAS,IAAIktH,GAAczmG,IAEhDt0B,KAAKwP,SAAkCgkD,EAAc,SAAS,CAAEn+C,EAAKvS,KAG9D0wD,EAAa73B,aAClB74B,EAAKowB,iBAIDpwB,EAAKw4H,SAIVhnG,EAAOiB,QAAS,SAEhB0E,EAAKo7B,uBAAsB,GACzB,CAAEzoD,SAAU,OAChB,EC5Bc,MAAM+uH,WAA0B,GAI9B,OAAApmG,GACf,MAAMn9B,EAAQ4H,KAAKs0B,OAAOl8B,MACpBif,EAAMjf,EAAMoL,SAElBpL,EAAMmiC,QAAQC,KA2DhB,SAA0BpiC,EAAcoiC,EAAgBvF,GACvD,MAAMgmG,EAAmBhmG,EAAU+V,YAC7BlmB,EAAQmQ,EAAUoY,gBAClBhB,EAAevnB,EAAM4b,MAAM9hC,OAC3B0tC,EAAaxnB,EAAM6b,IAAI/hC,OACvBu8H,EAAgC9uF,GAAgBC,EAEtD,GAAK2uF,EAAmB,CACvB,MAAM5/B,EAAmBu/B,GAA0BxiI,EAAMsrE,OAAQzuC,EAAU+Q,iBAE3E41F,GAAaxjI,EAAOoiC,EAAQ1V,EAAM6b,KAElCnG,EAAOskD,yBAA0B7pD,EAAUqJ,oBAC3C9D,EAAOm6D,sBAAuB0G,EAC/B,KAAO,CACN,MAAMzE,IAAmB9xE,EAAM4b,MAAMoJ,WAAahlB,EAAM6b,IAAI6I,SAE5DpxC,EAAM29F,cAAe9gE,EAAW,CAAE2hE,kBAK7BukC,EACJS,GAAaxjI,EAAOoiC,EAAQvF,EAAUgY,OAcjC2pD,GACJp8D,EAAOgZ,aAAclH,EAAY,EAGpC,CACD,CApGGuvF,CAAiBzjI,EAAOoiC,EAAQnjB,EAAI4d,WACpCj1B,KAAKwQ,KAA0C,eAAgB,CAAEgqB,UAAU,GAE7E,CAKgB,OAAAzF,GACf,MAAM38B,EAAQ4H,KAAKs0B,OAAOl8B,MACpBif,EAAMjf,EAAMoL,SAElBxD,KAAK00B,UAiBP,SAAoBgvC,EAAgBzuC,GAGnC,GAAKA,EAAUiY,WAAa,EAC3B,OAAO,EAGR,MAAM4uF,EAAY7mG,EAAUyG,OAG5B,IAAMogG,IAAcp4D,EAAOyF,WAAY2yD,EAAW,aACjD,OAAO,EAGR,MAAMh3G,EAAQmQ,EAAUoY,gBAClBhB,EAAevnB,EAAM4b,MAAM9hC,OAC3B0tC,EAAaxnB,EAAM6b,IAAI/hC,OAG7B,IAAOm9H,GAAsB1vF,EAAcq3B,IAAYq4D,GAAsBzvF,EAAYo3B,KAAcr3B,IAAiBC,EACvH,OAAO,EAGR,OAAO,CACR,CAzCmB5X,CAAWt8B,EAAMsrE,OAAQrsD,EAAI4d,UAC/C,EAyFD,SAAS2mG,GAAaxjI,EAAcoiC,EAAgB/U,GACnD,MAAMu2G,EAAmBxhG,EAAOl1B,cAAe,aAE/ClN,EAAMqjG,cAAeugC,EAAkBv2G,GACvC+U,EAAOgZ,aAAcwoF,EAAkB,QACxC,CASA,SAASD,GAAsBtkH,EAAkBisD,GAEhD,OAAKjsD,EAAQvJ,GAAI,iBAIVw1D,EAAOI,QAASrsD,IAAaskH,GAAsBtkH,EAAQ7Y,OAAmB8kE,GACtF,CCzIe,MAAMu4D,WAAmB,GAIvC,qBAAkB3lG,GACjB,MAAO,YACR,CAEO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdovC,EAASpvC,EAAOl8B,MAAMsrE,OACtBrlE,EAAai2B,EAAOj2B,WACpB47B,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SAG1BkgE,EAAOwR,SAAU,YAAa,CAC7B6D,WAAY,QACZnR,UAAU,IAIXvpE,EAAWuhF,IAAK,UACdvX,iBAAkB,CAClBjwE,MAAO,YACP6hC,KAAM,OAGR57B,EAAWuhF,IAAK,YACdvX,iBAAkB,CAClBjwE,MAAO,YACP6hC,KAAM,CAAEkiC,GAAgB3hC,YAAcA,EAAO2Z,mBAAoB,QAGnEla,EAAKu6B,YAAagnE,IAElBlnG,EAAO0qE,SAASnxF,IAAK,aAAc,IAAI8tH,GAAmBrnG,IAE1Dt0B,KAAKwP,SAAkCgkD,EAAc,SAAS,CAAEn+C,EAAKvS,KAG9D0wD,EAAa73B,aAClB74B,EAAKowB,iBAIApwB,EAAKw4H,SAIXhnG,EAAOiB,QAAS,cAChB0E,EAAKo7B,uBAAsB,GACzB,CAAEzoD,SAAU,OAChB,ECjDc,MAAMsvH,WAAuB9sH,KAA5C,kCACC,KAAiB+sH,OAAqC,EAAC,CAOhD,GAAAtuH,CAAKs9D,EAAiC3wC,GAC5C,MAAMnsB,EAAQrO,KAAKm8H,OAGbC,EAAS/tH,EAAO,GACtBrO,KAAKq8H,kBAAmBlxD,GACxB,MAAMmxD,EAASjuH,EAAO,GAGjB+tH,IAAWE,GAAWC,GAAoBH,EAAQE,IACtDt8H,KAAKwQ,KAAiC,aAAc,CACnDgsH,cAAeJ,EACfK,cAAeH,EACf9hG,UAGH,CAQO,MAAAv0B,CAAQpG,EAAY26B,GAC1B,MAAMnsB,EAAQrO,KAAKm8H,OAEbC,EAAS/tH,EAAO,GACtBrO,KAAK08H,kBAAmB78H,GACxB,MAAMy8H,EAASjuH,EAAO,GAGjB+tH,IAAWE,GAAWC,GAAoBH,EAAQE,IACtDt8H,KAAKwQ,KAAiC,aAAc,CACnDgsH,cAAeJ,EACfK,cAAeH,EACf9hG,UAGH,CAMQ,iBAAA6hG,CAAmBlxD,GAC1B,MAAM98D,EAAQrO,KAAKm8H,OACbv3H,EAAQyJ,EAAMygG,WAAW7uG,GAAQA,EAAKJ,KAAOsrE,EAAWtrE,KAG9D,GAAK08H,GAAoBpxD,EAAY98D,EAAOzJ,IAC3C,OAIIA,GAAS,GACbyJ,EAAM7G,OAAQ5C,EAAO,GAKtB,IAAIpK,EAAI,EAER,KAAQ6T,EAAO7T,IAAOmiI,GAAkBtuH,EAAO7T,GAAK2wE,IACnD3wE,IAGD6T,EAAM7G,OAAQhN,EAAG,EAAG2wE,EACrB,CAOQ,iBAAAuxD,CAAmB78H,GAC1B,MAAMwO,EAAQrO,KAAKm8H,OACbv3H,EAAQyJ,EAAMygG,WAAW7uG,GAAQA,EAAKJ,KAAOA,IAG9C+E,GAAS,GACbyJ,EAAM7G,OAAQ5C,EAAO,EAEvB,EAQD,SAAS23H,GAAoB5gI,EAAwBhD,GACpD,OAAOgD,GAAKhD,GAAKgD,EAAEiR,UAAYjU,EAAEiU,UAAYgwH,GAAiBjhI,EAAE+iC,UAAak+F,GAAiBjkI,EAAE+lC,QACjG,CAKA,SAASi+F,GAAkBhhI,EAAwBhD,GAClD,OAAKgD,EAAEiR,SAAYjU,EAAEiU,YAETjR,EAAEiR,SAAYjU,EAAEiU,WAKrBgwH,GAAiBjhI,EAAE+iC,SAAYk+F,GAAiBjkI,EAAE+lC,QAC1D,CAMA,SAASk+F,GAAiBl+F,GACzB,OAAO/9B,MAAMC,QAAS89B,GAAYA,EAAQvd,OAAOjkB,KAAM,KAAQwhC,CAChE,CCnJA,qbC2Cam+F,GAAoB,YAKpBC,GAA6B,qBAKnC,SAAS,GAAU39H,GACzB,QAAMA,EAAK+O,GAAI,cAIN/O,EAAKqnC,kBAAmB,SAClC,CAgDO,SAASu2F,GACftlH,EACA+iB,EACAj2B,EAGI,CAAC,GAEL,IAAMkT,EAAQvJ,GAAI,oBAQjB,MAAM,IAAI,EACT,sCACA,KACA,CAAEuJ,YAsBJ,OAlBA+iB,EAAO/0B,aAAc,kBAAmB,QAASgS,GAEjD+iB,EAAOoB,SAAUihG,GAAmBplH,GACpC+iB,EAAOoa,kBAAmB,UAAU,EAAMn9B,GAC1CA,EAAQuvB,gBAAkB,GAE1BxM,EAAOoa,kBAAmB,cAAe,GAAIn9B,GAExClT,EAAQgqC,OA+EP,SAAmB92B,EAAsBulH,GAC/C,MAAMC,EAAcxlH,EAAQ+uB,kBAAmB,eAE/Cy2F,EAAYr9H,KAAMo9H,EACnB,CAlFEE,CAAUzlH,EAASlT,EAAQgqC,OAGvBhqC,EAAQ44H,oBA+Rd,SAA6BC,EAAqC5iG,GACjE,MAAM6iG,EAAkB7iG,EAAO6Z,gBAAiB,MAAO,CAAE9V,MAAO,mCAAoC,SAAUqT,GAC7G,MAAMG,EAAa/xC,KAAK8xC,aAAcF,GAGhC83D,EAAO,IAAIf,GAQjB,OAPAe,EAAKj4F,IAAK,UAAW,IAGrBi4F,EAAK/3D,SAELI,EAAWpsC,YAAa+jG,EAAKjyF,SAEtBs6B,CACR,IAGAvX,EAAO90B,OAAQ80B,EAAOkd,iBAAkB0lF,EAAe,GAAKC,GAC5D7iG,EAAOoB,SAAU,CAAE,mCAAqCwhG,EACzD,CAjTEE,CAAoB7lH,EAAS+iB,GAG9B+iG,GAAsB9lH,EAAS+iB,GAExB/iB,CACR,CAMA,SAAS+lH,GAAc/lH,EAAsB0zD,EAAiC3wC,GAK7E,GAJK2wC,EAAWzsC,SACflE,EAAOoB,SAAU,GAASuvC,EAAWzsC,SAAWjnB,GAG5C0zD,EAAW5lE,WACf,UAAYxO,KAAOo0E,EAAW5lE,WAC7Bi1B,EAAO/0B,aAAc1O,EAAKo0E,EAAW5lE,WAAYxO,GAAO0gB,EAG3D,CAMA,SAAS,GAAiBA,EAAsB0zD,EAAiC3wC,GAKhF,GAJK2wC,EAAWzsC,SACflE,EAAOO,YAAa,GAASowC,EAAWzsC,SAAWjnB,GAG/C0zD,EAAW5lE,WACf,UAAYxO,KAAOo0E,EAAW5lE,WAC7Bi1B,EAAO9zB,gBAAiB3P,EAAK0gB,EAGhC,CAMO,SAAS8lH,GACf9lH,EACA+iB,EACA3sB,EAAiG2vH,GACjGv3H,EAAoG,IAEpG,MAAMoI,EAAQ,IAAI6tH,GAElB7tH,EAAMgB,GAA+B,cAAc,CAAEgG,EAAKvS,KACpDA,EAAK05H,eACTv2H,EAAQwR,EAAS3U,EAAK05H,cAAe15H,EAAK03B,QAGtC13B,EAAK25H,eACT5uH,EAAK4J,EAAS3U,EAAK25H,cAAe35H,EAAK03B,OACxC,IAMDA,EAAOoa,kBAAmB,gBAHyB,CAAEn9B,EAAS0zD,EAAY3wC,IAAYnsB,EAAMR,IAAKs9D,EAAY3wC,IAG7C/iB,GAChE+iB,EAAOoa,kBAAmB,mBAH+B,CAAEn9B,EAAS5X,EAAI26B,IAAYnsB,EAAMpI,OAAQpG,EAAI26B,IAGhC/iB,EACvE,CAsEO,SAAS,GACf4yB,EACA7P,EACAj2B,EAEI,CAAC,GA4BL,OA1BAi2B,EAAOoB,SAAU,CAAE,sBAAuB,8BAAgCyO,GAE1E7P,EAAO/0B,aAAc,OAAQ,UAAW4kC,GAEnC9lC,EAAQgqC,OACZ/T,EAAO/0B,aAAc,aAAclB,EAAQgqC,MAAOlE,GAInD7P,EAAO/0B,aAAc,kBAAmB4kC,EAAS/U,WAAa,QAAU,OAAQ+U,GAGhFA,EAASh7B,GAAoC,qBAAqB,CAAEgG,EAAKjC,EAAUlF,KAClFssB,EAAO/0B,aAAc,kBAAmByI,EAAK,QAAU,OAAQm8B,EAAU,IAG1EA,EAASh7B,GAAoC,oBAAoB,CAAEgG,EAAKjC,EAAUlF,KAC5EA,EACJssB,EAAOoB,SAAU,qCAAsCyO,GAEvD7P,EAAOO,YAAa,qCAAsCsP,EAC3D,IAGDkzF,GAAsBlzF,EAAU7P,GAEzB6P,CACR,CAiBO,SAAS,GACfpV,EACA78B,GAEA,MAAM83C,EAAkBjb,EAAU+Y,qBAElC,GAAKkC,EAAkB,CACtB,MAAMutF,EAA8BC,GAAgCzoG,GAIpE,GAAKwoG,EACJ,OAAOrlI,EAAMy/C,YAAaz/C,EAAMs/C,iBAAkBxH,EAAiButF,GAErE,CAEA,OAAO,GAAiCxoG,EAAW78B,EACpD,CA0EA,SAAS,KACR,OAAO,IACR,CChZO,MAAMulI,GAAkC,qBAKxC,SAASC,GAAoBrmF,EAAsC4kB,EAAuBuH,GAChG,QAASnsB,GAAe,GAAUA,KAAkBmsB,EAAOkE,SAAUzL,EACtE,CAoCO,SAASuhE,GAAgCzoG,GAC/C,OAAOA,EAAUoG,aAAcsiG,GAChC,C,eClEI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ7jG,OAAvB,MC6DM+jG,GAA+B,CAAE,SAAU,SAG3CC,IAA4B,IAAI/4E,WAAYC,gBC5ElD,mID4E+E,iBAAkBr+C,WAE3Fo3H,GAAqC,kCAY5B,MAAMC,WAAyB,GAA9C,kCAMC,KAAQC,8BAAgD,KAKxD,qBAAkB3nG,GACjB,MAAO,kBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GAAO,GACjB,CAKO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACditF,EAAcjtF,EAAO6qE,QAAQllE,KAInCj6B,KAAKqP,GAAoC,oBAAoB,CAAEgG,EAAKvS,EAAM4xB,KACzE6sF,EAAYhnF,QAAQC,IACnB,UAAYlkC,KAAQirH,EAAY/9G,SAASqtC,MACnCnc,EACJ8F,EAAOO,YAAagjG,GAAoCznI,GAExDkkC,EAAOoB,SAAUmiG,GAAoCznI,EAEvD,IAGKo+B,GACLJ,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAOskD,yBAA0B6+C,GAAiC,GAEpE,IAGD39H,KAAKk+H,+BACLl+H,KAAKm+H,0CACLn+H,KAAKo+H,4CACLp+H,KAAKq+H,8CACLr+H,KAAKs+H,0DACLt+H,KAAKu+H,2BACLv+H,KAAKw+H,kCACLx+H,KAAKy+H,iCACLz+H,KAAK0+H,iCACN,CAKgB,OAAA14G,GACfxY,MAAMwY,UAENhmB,KAAKi+H,8BAAgC,IACtC,CAWQ,gBAAAU,CAAkBC,EAA6Bn5G,GACtD,MAAM6O,EAASt0B,KAAKs0B,OACditF,EAAcjtF,EAAO6qE,QAAQllE,KAE7BohE,EAAmB/mE,EAAOl8B,MAAMsrE,OAAO4T,0BAA2BsnD,EAAoB,iBAAiB,GAE7GtqG,EAAOiB,QAAS,kBAAmB,CAClC9P,SAAU6O,EAAOl8B,MAAMs/C,iBAAkBknF,EAAoBn5G,GAC7DlgB,WAAY81F,IAGbkmB,EAAYt0E,QACZs0E,EAAYlsD,sBACb,CAaQ,kBAAAwpE,CACPjvH,EACAN,EACAC,EACAhL,GAEAvE,KAAKwP,SAAUI,EAASN,GAAO,IAAKpT,KAE9B8D,KAAK00B,WACTnlB,KAAarT,EACd,GACEqI,EACJ,CAaQ,4CAAAu6H,GACP,MAEMxrD,EAFStzE,KAAKs0B,OACCl8B,MACQoL,SAASyxB,UAChCwoG,EAA8BC,GAAgCpqD,GAEpE,IAAMmqD,EACL,OAAO,EASR,MAAMsB,EAAuBzrD,EAAetlC,qBAI5C,OAFAhuC,KAAK2+H,iBAAkBI,EAAsBtB,IAEtC,CACR,CASQ,4BAAAS,GACP,MAAM5pG,EAASt0B,KAAKs0B,OACdovC,EAASpvC,EAAOl8B,MAAMsrE,OACtB1oE,EAAIs5B,EAAO4E,OAAOl+B,EAClBgkI,EAAe,CACpBC,OAAQjkI,EAAG,iCACXkkI,MAAOlkI,EAAG,iCAGXs5B,EAAO6qE,QAAQ9rB,mBAAmBhkE,GAAkC,UAAU,CAAEgG,EAAKvS,EAAMw8D,KAC1F,MAAM/nB,EAAc+nB,EAAc9B,OAAOf,cAAe35D,EAAK7C,MAE7D,GAAMs3C,GAKDqmF,GAAoBrmF,EAAaz0C,EAAK7C,KAAMyjE,GAAW,EAqmB/D,SACC2G,EACA20D,EACAG,GACA,MAAMC,EAAoB/0D,EAAWh2B,gBAAiB,MAAO,CAC5D9V,MAAO,2CACL,SAAUqT,GACZ,MAAMytF,EAAoBr/H,KAAK8xC,aAAcF,GAK7C,OAYF,SAAwBytF,EAAgCL,GACvD,UAAYv5G,KAAYo4G,GAA+B,CACtD,MAAMyB,EAAiB,IAAI,GAAU,CACpCnoH,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,iCACA,kCAAmC9Y,KAEpC6+F,MAAO0a,EAAcv5G,GACrB,cAAe,QAEhBnO,SAAU,CACT+nH,EAAkBjgH,cAAcmgH,WAAYzB,IAA2B,MAIzEuB,EAAkB15H,YAAa25H,EAAe3tF,SAC/C,CACD,CAnCE6tF,CAAeH,EAAmBL,GAqCpC,SAA0BK,GACzB,MAAMI,EAAgB,IAAI,GAAU,CACnCtoH,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,yCAKH8gG,EAAkB15H,YAAa85H,EAAc9tF,SAC9C,CAhDE+tF,CAAiBL,GAEVA,CACR,IAGAh1D,EAAW3kE,OAAQ2kE,EAAW3yB,iBAAkBynF,EAAmB,OAASC,EAC7E,CArnBIO,CAAoBrgE,EAAc9kC,OAAQwkG,EAAcznF,GAEpCA,EAAY/Q,kBAAmB,eAEvC5mC,MAAM,IACVI,KAAK00B,UAAY15B,EAAG,8EAAiF,IAE9G,IACE,CAAE4R,SAAU,OAChB,CA2BQ,uDAAA0xH,GACP,MAAMhqG,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfk7E,EAAiBl7E,EAAMoL,SAASyxB,UAChCyuC,EAAStrE,EAAMsrE,OACf69C,EAAcjtF,EAAO6qE,QAAQllE,KA6FnC,SAAS2lG,EAA0Bn6G,GAClC,MAAO,yCAA0CA,GAClD,CA3FAzlB,KAAK6+H,mBAA+Ctd,EAAY/9G,SAAU,YAAY,CAAE6R,EAAKwqH,KAC5F7/H,KAAK8/H,qBAAsBzqH,EAAKwqH,EAAc,GAC5C,CAAEtyH,QAAS,CAAE,GAAU,SAAWX,SAAU,SAM/C5M,KAAK6+H,mBAA+CvrD,EAAgB,gBAAgB,CAAEj+D,EAAKvS,KAEpFA,EAAK2/D,cAMXnuC,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAOskD,yBAA0B6+C,GAAiC,GAChE,IAKJ39H,KAAK6+H,mBAAyCzmI,EAAMoL,SAAU,eAAe,KAC5E,MAAMu7H,EAAuBzrD,EAAetlC,qBAE5C,GAAK+wF,EAAuB,CAG3B,GAAKnB,GAFuBtpG,EAAO6qE,QAAQ3hC,OAAOf,cAAesiE,GAEnBA,EAAsBr7D,GACnE,MAEF,CAEApvC,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAOskD,yBAA0B6+C,GAAiC,GAChE,IAMJ39H,KAAK6+H,mBAA4CvqG,EAAO6qE,QAAQ9rB,mBAAoB,aAAa,CAAEh+D,EAAKvS,EAAMw8D,KAC7G,MAAM9kC,EAAS8kC,EAAc9kC,OAE7B,GAAKx6B,KAAKi+H,8BAAgC,CACzC,MAAM8B,EAAsBzgE,EAAc9B,OAAOf,cAAez8D,KAAKi+H,+BAEhE8B,IAEJvlG,EAAOO,YAAa8iG,GAA6B7gI,IAAK4iI,GAA4BG,GAElF//H,KAAKi+H,8BAAgC,KAEvC,CAEA,MAAMc,EAAuBj8H,EAAKmyB,UAAU+Y,qBAE5C,IAAM+wF,EACL,OAGD,MAAMgB,EAAsBzgE,EAAc9B,OAAOf,cAAesiE,GAEhE,IAAMnB,GAAoBmC,EAAqBhB,EAAsBr7D,GACpE,OAGD,MAAM+5D,EAA8BC,GAAgC56H,EAAKmyB,WAEnEwoG,IAINjjG,EAAOoB,SAAUgkG,EAA0BnC,GAA+BsC,GAI1E//H,KAAKi+H,8BAAgCc,EAAoB,IAG1D/+H,KAAK6+H,mBAAoDvqG,EAAO6E,GAAGqxE,aAAc,oBAAoB,CAAEn1F,EAAKrT,EAAM8wB,KAC3GA,GACLwB,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAOskD,yBAA0B6+C,GAAiC,GAEpE,GAMF,CAcQ,oBAAAmC,CAAsBzqH,EAAoCwqH,GACjE,MAAMvrG,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfk7E,EAAiBl7E,EAAMoL,SAASyxB,UAChCyuC,EAAStrE,EAAMsrE,OACf69C,EAAcjtF,EAAO6qE,QAAQllE,KAG7B+hE,EvWhND,SACN1uE,EACAiB,GAEA,MAAMyxG,EAA4B,GAAmC1yG,EAASiB,GAE9E,MAAqC,SAA9ByxG,GAAsE,UAA9BA,CAChD,CuWyMoBC,CADFJ,EAAavyG,QACqBgH,EAAO4E,OAAO3K,0BAC1DwxG,EAAsBxe,EAAY/9G,SAASyxB,UAAU+Y,qBAE3D,IAAIkyF,EAGCtC,GAAoBmC,EAJIzrG,EAAO6qE,QAAQ3hC,OAAOnB,eAAgB0jE,GAICr8D,GACnEw8D,EAA8BlgI,KAAKmgI,qCAAsCnkC,GAIhE1oB,EAAetoC,YACxBk1F,EAA8BlgI,KAAKogI,+CAAgDpkC,GAGzE6jC,EAAapyG,WACvByyG,EAA8BlgI,KAAKqgI,8CAA+CrkC,IAG9EkkC,IACJL,EAAa3sG,iBACb7d,EAAIlJ,OAEN,CAYQ,oCAAAg0H,CAAsCnkC,GAC7C,MACM5jG,EADS4H,KAAKs0B,OACCl8B,MAEfqlI,EAA8BC,GADbtlI,EAAMoL,SAASyxB,WAGtC,OAAO78B,EAAMmiC,QAAQC,IAEpB,IAAKijG,EAsBJ,OAFAjjG,EAAOm6D,sBAAuBgpC,GAAiC3hC,EAAY,QAAU,WAE9E,EAXP,KAVwByhC,KAAkCzhC,EAAY,QAAU,WAa/E,OAFAxhE,EAAOskD,yBAA0B6+C,KAE1B,EAWT,OAAO,CAAK,GAEd,CAgBQ,8CAAAyC,CAAgDpkC,GACvD,MAAM1nE,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfsrE,EAAStrE,EAAMsrE,OACf48D,EAAuBhsG,EAAOqC,QAAQvuB,IAAK,UAG3Cm4H,EAA8BD,EAAaE,iCAAkCxkC,GAGnF,QAAK4hC,GAF8BtpG,EAAO6qE,QAAQ3hC,OAAOf,cAAe8jE,GAEnBA,EAA6B78D,KACjFtrE,EAAMmiC,QAAQC,IACb8lG,EAAaG,yBAA0BF,GACvC/lG,EAAOm6D,sBAAuBgpC,GAAiC3hC,EAAY,SAAW,QAAS,KAKzF,EAIT,CAWQ,6CAAAqkC,CAA+CrkC,GACtD,MAAM1nE,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfsrE,EAAStrE,EAAMsrE,OACflG,EAASlpC,EAAO6qE,QAAQ3hC,OACxB8V,EAAiBl7E,EAAMoL,SAASyxB,UAEhCyrG,EAAoB1kC,EACzB1oB,EAAe9lC,kBAAmBrD,WAClCmpC,EAAen+C,mBAAoB+U,UAKpC,QAAK0zF,GAHoBpgE,EAAOf,cAAeikE,GAGJA,EAA0Bh9D,KACpEtrE,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAcktF,EAAoB,MACzClmG,EAAOm6D,sBAAuBgpC,GAAiC3hC,EAAY,QAAU,SAAU,KAGzF,EAIT,CAOQ,uCAAAmiC,GACP,MAAM7pG,EAASt0B,KAAKs0B,OACditF,EAAcjtF,EAAO6qE,QAAQllE,KAEnCj6B,KAAK6+H,mBAAgDtd,EAAY/9G,SAAU,aAAa,CAAE6R,EAAKwqH,KAC9F,MAAMrqB,EAAwCqqB,EAAa3tF,UFhiB3CqrE,QAAS,mCEkiBzB,IAAM/H,EACL,OAGD,MAAMmrB,EF5hBF,SAAsC5uF,GAC5C,OAAOA,EAAW6uF,UAAU18G,SAAU,yCAA4C,SAAW,OAC9F,CE0hB0B28G,CAA6BrrB,GAC9C2pB,EFthBF,SAAsCptF,EAAyBF,GACrE,MAAMivF,EAAmB/uF,EAAWwrE,QAAS,cAE7C,OAAO1rE,EAAasQ,aAAc2+E,EACnC,CEkhB6BC,CAA6BvrB,EAAQ+L,EAAY1vE,cACrE+sF,EAAqBtqG,EAAO6qE,QAAQ3hC,OAAOnB,eAAgB8iE,GAEjEn/H,KAAK2+H,iBAAkBC,EAAqB+B,GAE5Cd,EAAa3sG,iBACb7d,EAAIlJ,MAAM,GAEZ,CAeQ,yCAAAiyH,GACP,MAAM9pG,EAASt0B,KAAKs0B,OACdW,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAClCssF,EAAcjtF,EAAO6qE,QAAQllE,KAEnCj6B,KAAK6+H,mBAA4Ctd,EAAY/9G,SAAU,SAAS,CAAE6R,EAAKwqH,KAGtF,GAAuB,YAAlBxqH,EAAIo6B,WACR,OAGD,MAAMsvF,EAAuB9pG,EAAU+Y,qBACjC+xF,EAAsBzrG,EAAO6qE,QAAQ3hC,OAAOf,cAAesiE,GAE3Dr7D,EAASpvC,EAAOl8B,MAAMsrE,OAC5B,IAAIs9D,EAIChhI,KAAK8+H,+CACTkC,GAAa,EAIJpD,GAAoBmC,EAAqBhB,EAAsBr7D,KACxE1jE,KAAK2+H,iBAAkBI,EAAsBc,EAAavE,OAAS,SAAW,SAE9E0F,GAAa,GAGTA,IACJnB,EAAa3sG,iBACb7d,EAAIlJ,OACL,GACE,CAAEoB,QAAS,IACf,CAeQ,2CAAA8wH,GACP,MACM7qE,EADSxzD,KAAKs0B,OACQ6qE,QAAQllE,KAAKz2B,SAGzCxD,KAAK6+H,mBAAiDrrE,EAAc,cAAc,CAAEn+C,EAAKvS,KACnF9C,KAAK8+H,iDAKTh8H,EAAKmyB,UAAYu+B,EAAav+B,UAC/B,GACE,CAAEroB,SAAU,SAEV,EAAIvD,UAIRrJ,KAAK6+H,mBAA8CrrE,EAAc,WAAW,CAAEn+C,EAAKvS,KAC7D,KAAhBA,EAAKwqB,SACTttB,KAAK8+H,8CACN,IAID9+H,KAAK6+H,mBAAuDrrE,EAAc,oBAAoB,KAC7FxzD,KAAK8+H,8CAA8C,GACjD,CAAElyH,SAAU,QAEjB,CAUQ,wBAAA2xH,GACP,MAAMjqG,EAASt0B,KAAKs0B,OACditF,EAAcjtF,EAAO6qE,QAAQllE,KAC7B7hC,EAAQk8B,EAAOl8B,MACfsrE,EAAStrE,EAAMsrE,OAErB1jE,KAAK6+H,mBAA6Ctd,EAAY/9G,SAAU,UAAU,CAAE6R,EAAKwqH,KAGxF,GAAuB,YAAlBxqH,EAAIo6B,WACR,OAGD,MAAMguF,EAA8BC,GAAgCtlI,EAAMoL,SAASyxB,WAGnF,IAAMwoG,EACL,OAGD,MAAMl5G,EAAYs7G,EAAat7G,UACzB08G,EAAsB7oI,EAAMoL,SAASyxB,UAAU+Y,qBAG/CkzF,EAA+B,WAAb38G,EAGxB,GAJ0D,WAAhCk5G,IAE6ByD,EAGtD5sG,EAAOiB,QAAS,SAAU,CACzBN,UAAW78B,EAAM4/C,gBAAiBipF,EAAsB,YAEnD,CACN,MAAMn8G,EAAQ4+C,EAAOiE,yBACpBvvE,EAAMs/C,iBAAkBupF,EAAsBxD,GAC9Cl5G,GAID,GAAKO,EAEJ,GAAMA,EAAMkmB,YAKL,CACN,MAAMm2F,EAAQ/oI,EAAM4/C,gBAAiBlzB,EAAM4b,OAK3C,GAJAtoC,EAAMo+F,gBAAiB2qC,EAAO,CAAE58G,cAI1B48G,EAAMl0F,MAAOhpB,QAASa,EAAM4b,OAS7B,CACJ,MAAM0gG,EA0Lb,SAAyC19D,EAAgBjsD,GACxD,IAAI4pH,EAAuB5pH,EAE3B,UAAYm/B,KAAYn/B,EAAQ0kB,aAAc,CAAEG,aAAa,IAAW,CACvE,GAAOsa,EAAkBtb,WAAa,GAAKooC,EAAOI,QAASltB,GAC1D,MAGDyqF,EAAuBzqF,CACxB,CAEA,OAAOyqF,CACR,CAtMyCC,CAAgC59D,EAAQ5+C,EAAM4b,MAAM9hC,QAEtFxG,EAAM29F,cAAe39F,EAAM4/C,gBAAiBopF,EAA2B,MAAQ,CAC9E9qC,oBAAoB,GAEtB,MAdCl+F,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAc1uB,GACrBwP,EAAOiB,QAAS2rG,EAAkB,gBAAkB,SAAU,GAajE,MA1BC9oI,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAc1uB,GACrBwP,EAAOiB,QAAS2rG,EAAkB,gBAAkB,SAAU,GA0BlE,CAIArB,EAAa3sG,iBACb7d,EAAIlJ,MAAM,GACR,CAAEoB,QAAS,IACf,CAQQ,+BAAAixH,GACP,MAAMlqG,EAASt0B,KAAKs0B,OACdl8B,EAAQ4H,KAAKs0B,OAAOl8B,MACpBmpI,EAAoBnpI,EAAMoL,SAASyxB,UAEzCj1B,KAAK6+H,mBAA6CvqG,EAAOl8B,MAAO,iBAAiB,CAAEid,GAAOnV,EAAS+tC,MAClG,GAAKA,IAAiBA,EAAoB//B,GAAI,qBAC7C,OAGD,MAAMuvH,EAA8BC,GAAgC6D,GAEpE,OAAM9D,GAINpoH,EAAIlJ,OAEG/T,EAAMmiC,QAAQC,IACpB,MAAM0V,EAAkBqxF,EAAkBvzF,qBACpCvoB,EAAWrtB,EAAMs/C,iBAAkBxH,EAAkButF,GACrDxoG,EAAYuF,EAAOwd,gBAAiBvyB,GAEpCtnB,EAAS/F,EAAMqjG,cAAev7F,EAAS+0B,GAI7C,OAFAuF,EAAOgZ,aAAcve,GAEd92B,CAAM,UAfd,CAgBG,GACD,CAAEyO,SAAU,QAChB,CASQ,8BAAA6xH,GACP,MAAMnqG,EAASt0B,KAAKs0B,OAEditG,EADQvhI,KAAKs0B,OAAOl8B,MACMoL,SAASyxB,UAEzCj1B,KAAK6+H,mBAA4CvqG,EAAOl8B,MAAO,gBAAgB,CAAEid,EAAKnZ,KACrF,MAAQ,CAAE+xC,EAAY1pC,EAAU,CAAC,GAAMrI,EAEvC,GAAK+xC,IAAiBA,EAAoB//B,GAAI,qBAC7C,OAGD,MAAMuvH,EAA8BC,GAAgC6D,GAE9D9D,IAINl5H,EAAQ42F,oBAAsBsiC,EAC9BvhI,EAAM,GAAMqI,EAAO,GACjB,CAAEqI,SAAU,QAChB,CAUQ,+BAAA8xH,GACP,MAAMpqG,EAASt0B,KAAKs0B,OAEditG,EADQvhI,KAAKs0B,OAAOl8B,MACMoL,SAASyxB,UAEzCj1B,KAAK6+H,mBAA6CvqG,EAAOl8B,MAAO,iBAAiB,CAAEid,GAAO4f,MACzF,GAAKA,IAAcA,EAAU/mB,GAAI,qBAChC,OAGmCwvH,GAAgC6D,IAInElsH,EAAIlJ,MACL,GACE,CAAES,SAAU,QAChB,EE10Bc,SAAS40H,GACvBriC,GAEA,MAAM/mG,EAAQ+mG,EAAQ/mG,MAEtB,MAAO,CAAEid,EAAKvS,KACb,MAAM2+H,EAAiB3+H,EAAKwqB,SAAWjB,GAASE,QAC1Cm1G,EAAmB5+H,EAAKwqB,SAAWjB,GAASI,UAC5Ck1G,EAAkB7+H,EAAK2qB,SACvBwH,EAAY78B,EAAMoL,SAASyxB,UAEjC,IAAMwsG,IAAmBC,EACxB,OAGD,MAAM1lC,EAAY0lC,EAIlB,GAAKC,GA2MP,SAA8B1sG,EAA8B+mE,GAC3D,OAAQ/mE,EAAU+V,aAAe/V,EAAUkY,YAAc6uD,CAC1D,CA7M0B4lC,CAAqB3sG,EAAW+mE,GACvD,OAID,MAAMl3E,EAuDR,SAAqCq6E,EAA4BlqE,EAA0C+mE,GAC1G,MAAM5jG,EAAQ+mG,EAAQ/mG,MAEtB,GAAK4jG,EAAY,CAChB,MAAMvzD,EAAgBxT,EAAU+V,YAAc/V,EAAUgY,MAAShY,EAAUuY,kBACrEkL,EAAcmpF,GAA0BzpI,EAAOqwC,EAAe,WAGpE,IAAMiQ,EACL,OAAO,KAGR,MAAM5zB,EAAQ1sB,EAAMy/C,YAAapP,EAAeiQ,GAC1CopF,EAAoBC,GAAwB3pI,EAAMsrE,OAAQ5+C,EAAO,YAEvE,OAAKg9G,EACG1pI,EAAMy/C,YAAapP,EAAeq5F,GAGnC,IACR,CAAO,CACN,MAAMppF,EAAczjB,EAAU+V,YAAc/V,EAAUgY,MAAShY,EAAUE,mBACnEsT,EAAgBo5F,GAA0BzpI,EAAOsgD,EAAa,YAGpE,IAAMjQ,EACL,OAAO,KAGR,MAAM3jB,EAAQ1sB,EAAMy/C,YAAapP,EAAeiQ,GAC1CspF,EAAqBD,GAAwB3pI,EAAMsrE,OAAQ5+C,EAAO,WAExE,OAAKk9G,EACG5pI,EAAMy/C,YAAamqF,EAAoBtpF,GAGxC,IACR,CACD,CA7FgBupF,CAA4B9iC,EAASlqE,EAAW+mE,GAG9D,GAAMl3E,EAAN,CAKA,GAAKA,EAAMkmB,YAAc,CAExB,GAAK/V,EAAU+V,YACd,OACD,GAGU22F,EACT,MAEF,EAMK78G,EAAMkmB,aAkIb,SAA4Bm0D,EAA4B7hC,EAAmB0+B,GAC1E,MAAM5jG,EAAQ+mG,EAAQ/mG,MAChBy5C,EAAestD,EAAQllE,KAAK4X,aAMlC,GAAKmqD,EAAY,CAChB,MAAMmlC,EAAQ/oI,EAAM4/C,gBAAiBslB,EAAW58B,OAEhDtoC,EAAMo+F,gBAAiB2qC,GAKjBA,EAAMl0F,MAAOzD,SAAY8zB,EAAW58B,MAAMzc,QAASk9G,EAAMl0F,SAC9DqwB,EAAallE,EAAMy/C,YAAaspF,EAAMl0F,MAAQqwB,EAAW38B,KAE3D,CAEA,MAAM8lB,EAAY04C,EAAQ3hC,OAAOH,YAAaC,GACxCnc,EAAWtP,EAAa2U,eAAgBC,GACxC1hC,EAAQ,GAAKxC,iBAAkB4+B,GAErC,IAAI+gF,EAEJ,UAAY9+G,KAAQ2B,EACnB,QAAkC,IAA7Bm9G,EAAL,CAMA,GAAKrpI,KAAKuD,MAAOgnB,EAAK3B,MAASygH,EAC9B,OAAO,EAGRA,EAA2BrpI,KAAKC,IAAKopI,EAA0BrpI,KAAKuD,MAAOgnB,EAAKvB,QAPhF,MAFCqgH,EAA2BrpI,KAAKuD,MAAOgnB,EAAKvB,QAY9C,OAAO,CACR,CA5K4BsgH,CAAmBhjC,EAASr6E,EAAOk3E,MAC5D5jG,EAAMmiC,QAAQC,IACb,MAAMsb,EAAckmD,EAAYl3E,EAAM6b,IAAM7b,EAAM4b,MAElD,GAAKihG,EAAkB,CACtB,MAAMl0E,EAAer1D,EAAM4/C,gBAAiB/iB,EAAUyG,QACtD+xB,EAAahf,SAAUqH,GAEvBtb,EAAOgZ,aAAcia,EACtB,MACCjzB,EAAOgZ,aAAcsC,EACtB,IAGDzgC,EAAIlJ,OACJrJ,EAAKowB,iBACLpwB,EAAKqwB,kBAnCN,CAoCA,CAEF,CAyDA,SAAS0uG,GAA0BzpI,EAAcqwC,EAAyBlkB,GACzE,MAAMm/C,EAAStrE,EAAMsrE,OACf5+C,EAAQ1sB,EAAM2/C,cAAetP,EAAcnyC,MAE3C8rI,EAA+B,WAAb79G,EAAyB,eAAiB,aAElE,UAAY,iBAAEglB,EAAgB,KAAEtpC,EAAI,KAAEgL,KAAU6Z,EAAM+lB,UAAW,CAAEpC,gBAAelkB,cAAgB,CACjG,GAAKm/C,EAAOI,QAAS7jE,KAAWyjE,EAAOkE,SAAU3nE,GAChD,OAAOspC,EAIR,GAAKt+B,GAAQm3H,GAAmB1+D,EAAOC,QAAS1jE,GAC/C,OAAO,IAET,CAEA,OAAO,IACR,CAYA,SAAS8hI,GAAwBr+D,EAAgB5+C,EAAcP,GAC9D,MAAMkB,EAAwB,YAAblB,EAA0BO,EAAM6b,IAAM7b,EAAM4b,MAE7D,GAAKgjC,EAAOyF,WAAY1jD,EAAU,SACjC,OAAOA,EAGR,UAAY,aAAEwkB,KAAkBnlB,EAAM+lB,UAAW,CAAEtmB,cAClD,GAAKm/C,EAAOyF,WAAYl/B,EAAc,SACrC,OAAOA,EAIT,OAAO,IACR,C,eCpMI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQnQ,OCwCR,MAAM,WAAe,GAApC,kCAIC,KAAQuoG,oBAAsB,IAAIzuH,GAAiB,CAKnD,qBAAkB0iB,GACjB,MAAO,QACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE8mG,GAAkB,GAC5B,CAKO,IAAAtnG,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd2F,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SAqB1BxD,KAAKs0B,OAAO6qE,QAAQ9rB,mBAAmBhkE,GAA4B,aAAa,CAAEgG,EAAKvS,EAAMw8D,KAC5F,MAAM+K,EAAa/K,EAAc9kC,OAC3B84C,EAAiBxwE,EAAKmyB,UAG5B,GAAKq+C,EAAetoC,YACnB,OAGD,MAAM+zF,EAAuBzrD,EAAetlC,qBAE5C,IAAM+wF,EACL,OAGD,MAAMgB,EAAsBzrG,EAAO6qE,QAAQ3hC,OAAOf,cAAesiE,GPgH7D,IAAmBtnH,EO9GjB,GAAUsoH,KAIVzgE,EAAcwB,WAAWrC,QAAS6U,EAAgB,cAIxDjJ,EAAW72B,aAAc62B,EAAWvyB,cAAeioF,GAAuB,CACzEzxF,MAAM,EACNC,OPoGsB92B,EOpGLsoH,EPqGAtoH,EAAQ+uB,kBAAmB,eAE5BvY,QAAQ,CAAE4nB,EAAc92C,IAClB,mBAAZA,EACJ82C,EAAOA,EAAO,KAAO92C,IAAYA,IAEjC82C,EAAOA,EAAO,KAAO92C,EAAUA,GAErC,OO5GE,IAKJiB,KAAKs0B,OAAO6qE,QAAQ9rB,mBAAmBhkE,GAA4B,aAAa,CAAEgG,EAAKvS,EAAMw8D,KAE5Ft/D,KAAKsiI,gCAAiChjE,EAAc9kC,QAEpD,MAAM6vC,EAAa/K,EAAc9kC,OAC3Bwc,EAAgBqzB,EAAW7mE,SAASyxB,UAE1C,IAAIstG,EAAa,KAEjB,UAAYz9G,KAASkyB,EAAc5J,YAGlC,UAAY50C,KAASssB,EAAQ,CAC5B,MAAM3lB,EAAO3G,EAAMyH,KAEd,GAAUd,KAAWqjI,GAASrjI,EAAMojI,KACxCl4D,EAAWzuC,SAAUkhG,GAA4B39H,GACjDa,KAAKqiI,oBAAoBx0H,IAAK1O,GAC9BojI,EAAapjI,EAEf,CACD,GACE,CAAEyN,SAAU,QAGfqtB,EAAKu6B,YAAa,IAClBx0D,KAAKwP,SAAsCgkD,EAAc,aAAa,IAAKt3D,IAAU8D,KAAKyiI,gBAAiBvmI,KAa3G8D,KAAKwP,SAAqCgkD,EAAc,YAAY,IAAKt3D,KACxE8D,KAAK0iI,yCAA0CxmI,EAAM,GACnD,CAAEqR,QAAS,CAAE,GAAU,WAE1BvN,KAAKwP,SAAqCgkD,EAAc,YAAY,IAAKt3D,KACxE8D,KAAK2iI,kCAAmCzmI,EAAM,GAC5C,CAAEqR,QAAS,UAEdvN,KAAKwP,SACJgkD,EACA,WACAguE,GAA2BxhI,KAAKs0B,OAAO6qE,SACvC,CAAE5xF,QAAS,UAIZvN,KAAKwP,SAAmCgkD,EAAc,UAAU,CAAEn+C,EAAKvS,KACjE9C,KAAK4iI,cAAiC,WAAlB9/H,EAAKyhB,aAC7BzhB,EAAKowB,iBACL7d,EAAIlJ,OACL,GACE,CAAEoB,QAAS,SACf,CAKQ,YAAAk1H,CAAc/xH,EAAsBmvH,GAC3C,MAAMvrG,EAASt0B,KAAKs0B,OACd2F,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SAC1B,IAAIiU,EAA8BooH,EAAaj8H,OAG/C,GAoPF,SAAiC6T,GAChC,IAAIorH,EAA4DprH,EAEhE,KAAQorH,GAAiB,CACxB,GAAKA,EAAe30H,GAAI,qBAAwB20H,EAAe30H,GAAI,eAClE,OAAO,EAIR,GAAK,GAAU20H,GACd,OAAO,EAGRA,EAAiBA,EAAejkI,MACjC,CAEA,OAAO,CACR,CArQOkkI,CAAwBrrH,GAAY,CAIxC,IAAO,EAAIvO,UAAY,EAAID,UAAa42H,EAAa50E,SAASipD,QAAU,EAAI,CAC3E,MAAM12C,EAASlpC,EAAO6qE,QAAQ3hC,OACxBjmB,EAAc9/B,EAAQvJ,GAAI,oBAC/BuJ,EAAQ6uB,cAAc7uB,IAAYA,EAAQvJ,GAAI,sBAA0BuJ,EACnE0kD,EAAeqB,EAAOnB,eAAgB9kB,GAE5CsoF,EAAa3sG,iBAEblzB,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzBA,EAAOgZ,aAAc2oB,EAAc,KAAM,GAE3C,CAEA,MACD,CAGA,IAAM,GAAU1kD,KACfA,EAAUA,EAAQ6uB,aAAc,KAE1B7uB,GACL,OAMG,EAAIpO,WACRw2H,EAAa3sG,iBAIRsgC,EAAa1gC,WAClBmH,EAAKgT,QAIN,MAAMkvB,EAAe7nC,EAAO6qE,QAAQ3hC,OAAOnB,eAAgB5kD,GAE3DzX,KAAKygI,yBAA0BtkE,EAChC,CAWQ,qCAAAumE,CAAuChyH,EAAsBmvH,GACpE,MAAMvyG,EAAUuyG,EAAavyG,QAEvBl1B,EAAQ4H,KAAKs0B,OAAOl8B,MACpBsrE,EAAStrE,EAAMsrE,OACf4P,EAAiBl7E,EAAMoL,SAASyxB,UAChC8tG,EAAgBzvD,EAAetlC,qBAC/BzpB,EAAY,GAAmC+I,EAASttB,KAAKs0B,OAAO4E,OAAO3K,0BAC3EytE,EAAyB,QAAbz3E,GAAoC,SAAbA,EACnCy+G,EAAoC,MAAbz+G,GAAkC,QAAbA,EAGlD,GAAKw+G,GAAiBr/D,EAAOgE,SAAUq7D,GAAkB,CACxD,MAAMt9G,EAAWu2E,EAAY1oB,EAAe9lC,kBAAoB8lC,EAAen+C,mBACzEgiB,EAAWusB,EAAOiE,yBAA0BliD,EAAWu2E,EAAY,UAAY,YAWrF,YATK7kD,IACJ/+C,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAc2D,EAAU,IAGhC0oF,EAAa3sG,iBACbxiB,EAAUvE,QAIZ,CAIA,IAAMmnE,EAAetoC,cAAgB60F,EAAapyG,SAAW,CAC5D,MAAMw1G,EAAgB3vD,EAAen+C,mBAC/B+tG,EAAe5vD,EAAe9lC,kBAE9B21F,EAAoBF,EAAe/4F,UACnCk5F,EAAmBF,EAAc/4F,WAWvC,aATKg5F,GAAqBz/D,EAAOgE,SAAUy7D,IAAuBC,GAAoB1/D,EAAOgE,SAAU07D,MACtGhrI,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAcwoD,EAAYknC,EAAeD,EAAe,IAGhEpD,EAAa3sG,iBACbxiB,EAAUvE,QAIZ,CAGA,IAAMmnE,EAAetoC,YACpB,OAKD,MAAMq4F,EAA+BrjI,KAAKwgI,iCAAkCxkC,GAE5E,GAAKqnC,GAAgC3/D,EAAOgE,SAAU27D,GAAiC,CAEtF,GAAK3/D,EAAOkE,SAAUy7D,IAAkCL,EACvD,OAGDhjI,KAAKygI,yBAA0B4C,GAE/BxD,EAAa3sG,iBACbxiB,EAAUvE,MACX,CACD,CASQ,8BAAAw2H,CAAgCjyH,EAAsBmvH,GAC7D,MAAMznI,EAAQ4H,KAAKs0B,OAAOl8B,MACpBsrE,EAAStrE,EAAMsrE,OACfq/D,EAAgB3qI,EAAMoL,SAASyxB,UAAU+Y,qBAG1C+0F,GAAiBr/D,EAAOgE,SAAUq7D,KACtClD,EAAa3sG,iBACbxiB,EAAUvE,OAEZ,CAQQ,aAAAy2H,CAAe5mC,GACtB,MACM1oB,EADgBtzE,KAAKs0B,OAAOl8B,MAAMoL,SACHyxB,UAGrC,IAAMj1B,KAAKs0B,OAAOl8B,MAAMi9B,UAAWi+C,GAClC,OAID,IAAMA,EAAetoC,YACpB,OAGD,MAAM+3F,EAAgB/iI,KAAKwgI,iCAAkCxkC,GAE7D,OAAK+mC,GACJ/iI,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzB,IAAI8oG,EAAehwD,EAAe53C,OAAQ98B,OAG1C,KAAQ0kI,EAAanhG,SAAU,CAC9B,MAAMohG,EAAeD,EACrBA,EAAeC,EAAa3kI,OAE5B47B,EAAOv0B,OAAQs9H,EAChB,CAEAvjI,KAAKygI,yBAA0BsC,EAAe,KAGxC,QAfR,CAiBD,CAOO,wBAAAtC,CAA0BhpH,GAChCzX,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzBA,EAAOgZ,aAAchZ,EAAOsd,cAAergC,GAAW,GAExD,CAUO,gCAAA+oH,CAAkC1pD,GACxC,MAAM1+E,EAAQ4H,KAAKs0B,OAAOl8B,MACpBsrE,EAAStrE,EAAMsrE,OACf4P,EAAiBl7E,EAAMoL,SAASyxB,UAIhCksG,EAAQ/oI,EAAM4/C,gBAAiBs7B,GAIrC,GAHAl7E,EAAMo+F,gBAAiB2qC,EAAO,CAAE58G,UAAWuyD,EAAU,UAAY,aAG5DqqD,EAAMl9G,QAASqvD,GACnB,OAAO,KAGR,MAAMyvD,EAAgBjsD,EAAUqqD,EAAMl0F,MAAO9C,WAAag3F,EAAMl0F,MAAO/C,UAEvE,OAAO64F,GAAiBr/D,EAAOgE,SAAUq7D,GACjCA,EAGD,IACR,CAKQ,+BAAAT,CAAiC9nG,GACxC,UAAYgpG,KAAUxjI,KAAKqiI,oBAC1B7nG,EAAOO,YAAa+hG,GAA4B0G,GAGjDxjI,KAAKqiI,oBAAoBzwH,OAC1B,EA+BD,SAAS4wH,GAAS/qH,EAAsB7Y,GACvC,QAAMA,GAIC+B,MAAMrB,KAAMmY,EAAQ0kB,gBAAiB5M,SAAU3wB,EACvD,CCnae,MAAM,WAAgC,GAArD,kCAIC,KAAQ6kI,oBAAsB,IAAIlyH,GAA+C,CAOjF,mBAAkB2lB,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,yBACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAGpB,GAAKA,EAAOqC,QAAQ/oB,IAAK,kBAAqB,CAC7C,MAAM81H,EAAiCpvG,EAAOqC,QAAQvuB,IAAK,kBAE3DpI,KAAKwP,SAAmCk0H,EAAgB,QAAQruH,KA2OnE,SAA2B4f,GAC1B,MAAMsiB,EAActiB,EAAU+Y,qBAE9B,SAAWuJ,IAAe,GAAUA,GACrC,EA9OSosF,CAAkBrvG,EAAO6qE,QAAQllE,KAAKz2B,SAASyxB,YACnD5f,EAAIlJ,MACL,GACE,CAAES,SAAU,QAChB,CAEA5M,KAAKonH,SAAWpnH,KAAKs0B,OAAOqC,QAAQvuB,IAAK,qBAEzCpI,KAAKqP,GAA2B,oBAAoB,KACnDrP,KAAK4jI,2BAA2B,IAGjC5jI,KAAKwP,SAA+B8kB,EAAO6E,GAAI,UAAU,KACxDn5B,KAAK4jI,2BAA2B,IAIjC5jI,KAAKwP,SAAiC8kB,EAAO6E,GAAGqxE,aAAc,oBAAoB,KACjFxqG,KAAK4jI,2BAA2B,GAC9B,CAAEh3H,SAAU,OAChB,CAEgB,OAAAoZ,GACfxY,MAAMwY,UAEN,UAAY69G,KAAiB7jI,KAAKyjI,oBAAoBv4H,SACrD24H,EAAc5pG,KAAKjU,SAErB,CAgBO,QAAAkvD,CACN4uD,GACA,UAAEvxB,EAAS,MAAE3hF,EAAK,kBAAEmzG,EAAiB,iBAAErd,EAAmB,yBAQ1D,IAAM91F,EAAMt4B,OAiBX,YAFA,EAAY,0BAA2B,CAAEwrI,cAK1C,MAAMxvG,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACXo0G,EAAc,IAAI,GAAa96E,EAAO4E,QAI5C,GAFAk2E,EAAYmD,UAAYA,GAAav3G,EAAG,kBAEnCgF,KAAKyjI,oBAAoB71H,IAAKk2H,GAOlC,MAAM,IAAI,EAAe,4BAA6B9jI,KAAM,CAAE8jI,cAG/D,MAAME,EAAoB,CACzB/pG,KAAMm1E,EACN20B,oBACArd,mBACAud,YAAarzG,EACbszG,aAAa,GAId5vG,EAAO6E,GAAG8oF,WAAY7S,EAAa,CAClCmU,cAAc,EACdP,YAAa,KACZ,MAAMmhB,EAAiBJ,EAAmBzvG,EAAO6qE,QAAQllE,KAAKz2B,SAASyxB,WAElEkvG,GACJnkI,KAAKokI,aAAcJ,EAAmBG,EACvC,EAEDvhB,UAAW,KACV5iH,KAAKqkI,aAAcL,EAAmB,IAIxChkI,KAAKyjI,oBAAoBhyH,IAAKqyH,EAAWE,EAC1C,CAKQ,yBAAAJ,GACP,IAAIU,EAAyB,EACzBC,EAAwB,KACxBC,EAA2B,KAE/B,UAAYt8H,KAAclI,KAAKyjI,oBAAoBv4H,SAAW,CAC7D,MAAMi5H,EAAiBj8H,EAAW67H,kBAAmB/jI,KAAKs0B,OAAO6qE,QAAQllE,KAAKz2B,SAASyxB,WAEvF,GAAMj1B,KAAK00B,WAAcyvG,EAIlB,GAAMnkI,KAAKs0B,OAAO6E,GAAGqxE,aAAa13E,UAIlC,CACN,MAAM2xG,EAAsBN,EAAehoG,eAAe7jC,OAMrDmsI,EAAsBH,IAC1BA,EAAyBG,EACzBF,EAAwBJ,EACxBK,EAA2Bt8H,EAE7B,MAfMlI,KAAK0kI,kBAAmBx8H,IAC5BlI,KAAKqkI,aAAcn8H,QALflI,KAAK2kI,oBAAqBz8H,IAC9BlI,KAAKqkI,aAAcn8H,EAmBtB,CAEKs8H,GACJxkI,KAAKokI,aAAcI,EAA0BD,EAE/C,CAKQ,YAAAF,CAAcL,GACrBhkI,KAAKonH,SAASnhH,OAAQ+9H,EAAkB/pG,MACxCj6B,KAAK2P,cAAe3P,KAAKonH,SAAU,qBACpC,CASQ,YAAAgd,CAAcJ,EAAsDG,GACtEnkI,KAAK0kI,kBAAmBV,GAC5B,GAA6BhkI,KAAKs0B,OAAQ6vG,GAC9BnkI,KAAK2kI,oBAAqBX,KAChCA,EAAkBE,cACvBF,EAAkBE,aAAc,EAChCF,EAAkB/pG,KAAKm0E,eAAgB41B,EAAkBC,YAAajkI,KAAKs0B,OAAO6E,GAAG61E,mBAGtFhvG,KAAKonH,SAASv5G,IAAK,CAClBosB,KAAM+pG,EAAkB/pG,KACxBxU,SAAUm/G,GAAwB5kI,KAAKs0B,OAAQ6vG,GAC/Czd,iBAAkBsd,EAAkBtd,mBAOrC1mH,KAAKwP,SAAiCxP,KAAKonH,SAAU,sBAAsB,KAC1E,UAAYl/G,KAAclI,KAAKyjI,oBAAoBv4H,SAClD,GAAKlL,KAAK0kI,kBAAmBx8H,GAAe,CAC3C,MAAMi8H,EAAiBj8H,EAAW67H,kBAAmB/jI,KAAKs0B,OAAO6qE,QAAQllE,KAAKz2B,SAASyxB,WACvF,GAA6Bj1B,KAAKs0B,OAAQ6vG,EAC3C,CACD,IAGH,CAEQ,iBAAAO,CAAmBG,GAC1B,OAAO7kI,KAAKonH,SAAS5B,cAAgBqf,EAAQ5qG,IAC9C,CAEQ,mBAAA0qG,CAAqBE,GAC5B,OAAO7kI,KAAKonH,SAASpC,QAAS6f,EAAQ5qG,KACvC,EAGD,SAAS,GAA6B3F,EAAgB6vG,GACrD,MAAMjmB,EAA6B5pF,EAAOqC,QAAQvuB,IAAK,qBACjDqd,EAAWm/G,GAAwBtwG,EAAQ6vG,GAEjDjmB,EAAQ0H,eAAgBngG,EACzB,CAEA,SAASm/G,GAAwBtwG,EAAgB6vG,GAChD,MAAM5iB,EAAcjtF,EAAO6qE,QAAQllE,KAC7Bg+E,EAAmBsB,GAAiBtB,iBAE1C,MAAO,CACNr0G,OAAQ29G,EAAY1vE,aAAayK,aAAc6nF,GAC/C/8G,UAAW,CACV6wF,EAAiBM,gBACjBN,EAAiBS,oBACjBT,EAAiBU,oBACjBV,EAAiBC,gBACjBD,EAAiBI,oBACjBJ,EAAiBK,oBACjBL,EAAiBW,qBAGpB,CCtTyCzlG,I,kLCWJA,I,eCxBjC,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ2mB,O,cCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCuHR,MAAMgrG,WAAiB,GAoCrC,qBAAkBxuG,GACjB,MAAO,UACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GAAmB,GAC7B,CAKO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd2F,EAAO3F,EAAO6qE,QAAQllE,KAE5Bj6B,KAAK+kI,cAAgB,KACrB/kI,KAAKglI,aAAe,GACpBhlI,KAAKilI,kBAAoB,KACzBjlI,KAAKklI,2BAA6B,IAAUlP,GAAeh2H,KAAKmlI,kBAAmBnP,IAAe,IAClGh2H,KAAKolI,yBAA2B,IAAO,IAAMplI,KAAKqlI,qBAAqB,IACvErlI,KAAKslI,iCAAmC,IAAO,IAAMtlI,KAAKulI,6BAA6B,IAElFjxG,EAAOqC,QAAQ/oB,IAAK,wBACxB5N,KAAKw0B,cAAe,yBAKrByF,EAAKu6B,YAAa,IAClBv6B,EAAKu6B,YAAa,IAElBx0D,KAAKwlI,iBACLxlI,KAAKylI,oCACLzlI,KAAK0lI,kCACL1lI,KAAK2lI,mBACL3lI,KAAK4lI,mCAEL5lI,KAAKwP,SAA0C8kB,EAAQ,qBAAqB,CAAEjf,EAAKrT,EAAMszB,KACnFA,EACJt1B,KAAKw0B,cAAe,gBAEpBx0B,KAAK20B,mBAAoB,eAC1B,IAGD30B,KAAKqP,GAAoC,oBAAoB,CAAEgG,EAAKrT,EAAM0yB,KACnEA,GACL10B,KAAK6lI,mBAAmB,EACzB,IAGI,EAAIx8H,WACRrJ,KAAKw0B,cAAe,oBAEtB,CAKgB,OAAAxO,GAUf,OATKhmB,KAAK+kI,gBACT/kI,KAAK+kI,cAAcjkH,SACnB9gB,KAAK+kI,cAAgB,MAGtB/kI,KAAKklI,2BAA2BzxG,SAChCzzB,KAAKolI,yBAAyB3xG,SAC9BzzB,KAAKslI,iCAAiC7xG,SAE/BjmB,MAAMwY,SACd,CAKQ,cAAAw/G,GACP,MAAMlxG,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfo6H,EAAgBp6H,EAAMoL,SACtBy2B,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SAG1BxD,KAAKwP,SAAsCgkD,EAAc,aAAa,CAAEn+C,EAAKvS,KAC5E,MAAMmyB,EAAYu9F,EAAcv9F,UAGhC,GAAKnyB,EAAKc,QAAUd,EAAKc,OAAOsK,GAAI,mBAGnC,YAFApL,EAAKowB,iBAUN,MAAM4yG,EAAkBhjI,EAAKc,OAASmiI,GAAqBjjI,EAAKc,QAAW,KAE3E,GAAKkiI,EAAkB,CACtB,MAAM3pE,EAAe7nC,EAAO6qE,QAAQ3hC,OAAOnB,eAAgBypE,GAK3D,GAHA9lI,KAAK+kI,cAAgB,GAAU5gE,UAAW/rE,EAAM0/C,cAAeqkB,IAG1D7nC,EAAOqC,QAAQ/oB,IAAK,2BAA8B,CACG0mB,EAAOqC,QAAQvuB,IAAK,2BAErDosB,cAAe,WACxC,CACD,UAGWg/B,EAAav+B,UAAU+V,YAAc,CAC/C,MAAMkF,EAAkBsjB,EAAav+B,UAAU+Y,qBAEzCkC,GAAoB,GAAUA,KACnClwC,KAAK+kI,cAAgB,GAAU5gE,UAAWlvC,EAAUoY,iBAEtD,CAEA,IAAMrtC,KAAK+kI,cAGV,YAFAjiI,EAAKowB,iBAKNlzB,KAAKglI,aAAe,IAEpB,MAAMgB,EAAwBhmI,KAAK00B,WAAaJ,EAAOl8B,MAAMi9B,UAAWr1B,KAAK+kI,eAE7EjiI,EAAK2wD,aAAaZ,cAAgBmzE,EAAwB,WAAa,OACvEljI,EAAK2wD,aAAab,QAAS,qCAAsC5yD,KAAKglI,cAEtE,MAAMiB,EAAmB7tI,EAAM4/C,gBAAiBh4C,KAAK+kI,cAAc7gE,WAC7DhkE,EAAUo0B,EAAOxxB,KAAKq6E,OAAQ/kF,EAAMulG,mBAAoBsoC,IAE9DzyE,EAAahjD,KAAwC,kBAAmB,CACvEijD,aAAc3wD,EAAK2wD,aACnBvzD,UACAq4B,OAAQ,cAGHytG,IACLhmI,KAAK+kI,cAAcjkH,SACnB9gB,KAAK+kI,cAAgB,KACrB/kI,KAAKglI,aAAe,GACrB,GACE,CAAEp4H,SAAU,QAKf5M,KAAKwP,SAAoCgkD,EAAc,WAAW,CAAEn+C,EAAKvS,KACxE9C,KAAK6lI,mBAAoB/iI,EAAK2wD,aAAaR,YAA8C,QAAhCnwD,EAAK2wD,aAAaX,WAAsB,GAC/F,CAAElmD,SAAU,QAGf5M,KAAKwP,SAAsCgkD,EAAc,aAAa,KAC/DxzD,KAAK00B,WAIXuF,EAAKgT,OAAO,IAIbjtC,KAAKwP,SAAsCgkD,EAAc,aAAa,KAGrExzD,KAAKolI,0BAA0B,IAIhCplI,KAAKwP,SAAqCgkD,EAAc,YAAY,CAAEn+C,EAAKvS,KAC1E,IAAM9C,KAAK00B,UAGV,YAFA5xB,EAAK2wD,aAAaX,WAAa,QAKhC9yD,KAAKolI,yBAAyB3xG,SAE9B,MAAMuiG,EAAckQ,GAAqB5xG,EAAQxxB,EAAK4wD,aAAc5wD,EAAKc,QAGnE0wB,EAAOl8B,MAAMi9B,UAAW2gG,IAQxBh2H,KAAK+kI,gBACVjiI,EAAK2wD,aAAaX,WAAa,QAI1B,EAAI7pD,UAC+B,QAAnCnG,EAAK2wD,aAAaZ,cACtB/vD,EAAK2wD,aAAaX,WAAa,OACpB,CAAE,MAAO,YAAavjC,SAAUzsB,EAAK2wD,aAAaZ,iBAC7D/vD,EAAK2wD,aAAaX,WAAa,SAK5BkjE,GACJh2H,KAAKklI,2BAA4BlP,IAtBjClzH,EAAK2wD,aAAaX,WAAa,MAuBhC,GACE,CAAElmD,SAAU,OAChB,CAKQ,+BAAA84H,GACP,MAAMpxG,EAASt0B,KAAKs0B,OAEdk/B,EADOl/B,EAAO6qE,QAAQllE,KACFz2B,SAG1BxD,KAAKwP,SAA2CgkD,EAAc,kBAAkB,CAAEn+C,EAAKvS,KACtF,GAAoB,QAAfA,EAAKy1B,OACT,OAGD,MAAMy9F,EAAckQ,GAAqB5xG,EAAQxxB,EAAK4wD,aAAc5wD,EAAKc,QAOzE,GAHA5D,KAAKqlI,qBAGCrP,IAAgB1hG,EAAOl8B,MAAMi9B,UAAW2gG,GAI7C,OAHAh2H,KAAK6lI,mBAAmB,QACxBxwH,EAAIlJ,OAOAnM,KAAK+kI,eAAiB/kI,KAAKglI,cAAgBliI,EAAK2wD,aAAad,QAAS,wCAC1E3yD,KAAK+kI,cAAcjkH,SACnB9gB,KAAK+kI,cAAgB,KACrB/kI,KAAKglI,aAAe,IAMrB,GAF0D,QAA3CmB,GAAoBrjI,EAAK2wD,eAEzBzzD,KAAK+kI,eAAiB/kI,KAAK+kI,cAAcv5F,cAAewqF,GAAa,GAInF,OAHAh2H,KAAK6lI,mBAAmB,QACxBxwH,EAAIlJ,OAMLrJ,EAAK4wD,aAAe,CAAEp/B,EAAO6qE,QAAQ3hC,OAAOH,YAAa24D,GAAe,GACtE,CAAEppH,SAAU,QAChB,CAKQ,iCAAA64H,GACP,MAAMW,EAAoBpmI,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IAEnDg+H,EAAkB/2H,GAAoC,oBAAoB,CAAEgG,EAAKvS,KAChF,IAAM9C,KAAK00B,WAA6B,SAAhB5xB,EAAKy1B,OAC5B,OAKD,MAAMsT,EAAS/oC,EAAK4wD,aAAc12D,KAAKypD,GAAazmD,KAAKs0B,OAAO6qE,QAAQ3hC,OAAOL,aAAc1W,KAE7FzmD,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,GAAUA,EAAOgZ,aAAc3H,IAAU,GACjE,CAAEj/B,SAAU,SAEfw5H,EAAkB/2H,GAAoC,oBAAoB,CAAEgG,EAAKvS,KAChF,IAAM9C,KAAK00B,WAA6B,SAAhB5xB,EAAKy1B,OAC5B,OAID,MAAM8tG,EAAoD,QAA3CF,GAAoBrjI,EAAK2wD,cAKlC6yE,GAAaxjI,EAAKyvH,cAAgBzvH,EAAKyvH,YAAYvnF,YAEzDhrC,KAAK6lI,kBAAmBS,GAAaD,EAAQ,GAC3C,CAAEz5H,SAAU,UAChB,CAKQ,gCAAAg5H,GACP,MAAMtxG,EAASt0B,KAAKs0B,OACd2F,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SAI1BxD,KAAKwP,SAAsCgkD,EAAc,aAAa,CAAEn+C,EAAKvS,KAG5E,GAAK,EAAIuG,YAAcvG,EACtB,OAGD9C,KAAKslI,iCAAiC7xG,SAGtC,IAAI8yG,EAAmBR,GAAqBjjI,EAAKc,QAUjD,GAAK,EAAI0F,UAAYi9H,IAAqB/yE,EAAav+B,UAAU+V,YAAc,CAC9E,MAAMkF,EAAkBsjB,EAAav+B,UAAU+Y,qBAE/C,IAAMkC,IAAoB,GAAUA,GAAoB,CACvD,MAAM/H,EAAkBqrB,EAAav+B,UAAUkT,gBAE1CA,IAAoBA,EAAgB7S,aACxCixG,EAAmBp+F,EAErB,CACD,CAEKo+F,IACJtsG,EAAKM,QAAQC,IACZA,EAAO/0B,aAAc,YAAa,OAAQ8gI,EAAmB,IAI9DvmI,KAAKilI,kBAAoB3wG,EAAO6qE,QAAQ3hC,OAAOnB,eAAgBkqE,GAChE,IAIDvmI,KAAKwP,SAAoCgkD,EAAc,WAAW,KAC3D,EAAInqD,WACTrJ,KAAKslI,kCACN,GAEF,CAKQ,yBAAAC,GACP,MAAMpmC,EAAUn/F,KAAKs0B,OAAO6qE,QAE5BA,EAAQllE,KAAKM,QAAQC,IAEfx6B,KAAKilI,mBAA6D,cAAxCjlI,KAAKilI,kBAAkB3uI,KAAK8+B,UAC1DoF,EAAO9zB,gBAAiB,YAAay4F,EAAQ3hC,OAAOf,cAAez8D,KAAKilI,oBAGzEjlI,KAAKilI,kBAAoB,IAAI,GAE/B,CAKQ,gBAAAU,GACP,MAAMrxG,EAASt0B,KAAKs0B,OAGpBA,EAAOj2B,WAAWuhF,IAAK,mBAAoB3U,kBAAmB,CAC7D7yE,MAAO,cACP6hC,KAAM,CACLyE,QAAS,CAAE,qCAKbpK,EAAOj2B,WAAWuhF,IAAK,mBAAoBhV,gBAAiB,CAC3DxyE,MAAO,cACP6hC,KAAM,CAAEn3B,GAAQ03B,aAGf,GAFelG,EAAOl8B,MAAMsrE,OAAOyF,WAAYrmE,EAAK49D,YAAYhgC,MAAO,SAMvE,OAAOlG,EAAO6Z,gBAAiB,OAAQ,CAAE9V,MAAO,yCAA0C,SAAUqT,GACnG,MAAMG,EAAa/xC,KAAK8xC,aAAcF,GAKtC,OAFAG,EAAW8T,OAAQ,IAAUjU,EAAYtsC,cAAe,QAAU,KAE3DysC,CACR,GAAG,GAGN,CAOQ,iBAAAozF,CAAmBnP,GAC1B,MAAM1hG,EAASt0B,KAAKs0B,OACdsrC,EAAUtrC,EAAOl8B,MAAMwnE,QAE7BtrC,EAAOl8B,MAAMmiC,QAAQC,IACfolC,EAAQhyD,IAAK,eACXgyD,EAAQx3D,IAAK,eAAiBu4D,WAAW18C,QAAS+xG,IACvDx7F,EAAOs4D,aAAc,cAAe,CAAEhuE,MAAOkxG,IAG9Cx7F,EAAOu4D,UAAW,cAAe,CAChCjuE,MAAOkxG,EACPnjC,gBAAgB,EAChB79D,aAAa,GAEf,GAEF,CAKQ,iBAAAqwG,GACP,MAAMjtI,EAAQ4H,KAAKs0B,OAAOl8B,MAE1B4H,KAAKolI,yBAAyB3xG,SAC9BzzB,KAAKklI,2BAA2BzxG,SAE3Br7B,EAAMwnE,QAAQhyD,IAAK,gBACvBxV,EAAMmiC,QAAQC,IACbA,EAAOs2D,aAAc,cAAe,GAGvC,CAOQ,iBAAA+0C,CAAmBW,GAC1B,MAAMlyG,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MAKrB,GAHA4H,KAAKqlI,oBACLrlI,KAAKulI,4BAEAjxG,EAAOqC,QAAQ/oB,IAAK,2BAA8B,CACG0mB,EAAOqC,QAAQvuB,IAAK,2BAErDusB,mBAAoB,WAC7C,CAEA30B,KAAKglI,aAAe,GAEdhlI,KAAK+kI,gBAKNyB,GAASxmI,KAAK00B,WAClBt8B,EAAM29F,cAAe39F,EAAM4/C,gBAAiBh4C,KAAK+kI,eAAiB,CAAEzuC,oBAAoB,IAGzFt2F,KAAK+kI,cAAcjkH,SACnB9gB,KAAK+kI,cAAgB,KACtB,EAMD,SAASmB,GAAqB5xG,EAAgBmyG,EAA2CC,GACxF,MAAMtuI,EAAQk8B,EAAOl8B,MACfolE,EAASlpC,EAAO6qE,QAAQ3hC,OAE9B,IAAI14C,EAAsB,KAE1B,MAAM6hH,EAAqBF,EAAmBA,EAAkB,GAAI/lG,MAAQ,KAU5E,GAPKgmG,EAAkBx4H,GAAI,eAC1Bw4H,EAAoBA,EAAkB9nI,QAIvCkmB,EA4CD,SAAsCwP,EAAgBoyG,GACrD,MAAMtuI,EAAQk8B,EAAOl8B,MACfolE,EAASlpC,EAAO6qE,QAAQ3hC,OAG9B,GAAK,GAAUkpE,GACd,OAAOtuI,EAAM0/C,cAAe0lB,EAAOnB,eAAgBqqE,IAIpD,IAAMA,EAAkBx4H,GAAI,mBAAsB,CAEjD,MAAM0oC,EAAW8vF,EAAkBpgG,cAAcnnC,GAAQ,GAAUA,IAAUA,EAAK+O,GAAI,qBAGtF,GAAK,GAAU0oC,GACd,OAAOx+C,EAAM0/C,cAAe0lB,EAAOnB,eAAgBzlB,GAErD,CAEA,OAAO,IACR,CAjESgwF,CAA6BtyG,EAAQoyG,GAExC5hH,EACJ,OAAOA,EAMR,MAAM+hH,EA0HP,SAAuCvyG,EAAgB7c,GACtD,MAAM+lD,EAASlpC,EAAO6qE,QAAQ3hC,OACxBvjC,EAAO3F,EAAO6qE,QAAQllE,KAEtB4sG,EAAqBrpE,EAAOnB,eAAgB5kD,GAElD,GAAKovH,EACJ,OAAOA,EAIR,MAAMl0F,EAAe1Y,EAAK2d,qBAAsBngC,GAC1C8/B,EAAcimB,EAAOzB,uBAAwBppB,GAEnD,OAAO6qB,EAAOnB,eAAgB9kB,EAC/B,CAzI4BuvF,CAA8BxyG,EAAQoyG,GAC3DK,EAAsBJ,EAAqBnpE,EAAOJ,gBAAiBupE,GAAuB,KAKhG,OAAMI,GAMNjiH,EA6DD,SAA2CwP,EAAgByyG,EAA+BF,GACzF,MAAMzuI,EAAQk8B,EAAOl8B,MAGrB,IAAMA,EAAMsrE,OAAOyF,WAAY09D,EAAoB,UAClD,OAAO,KAIR,MAAMG,EAAyB5uI,EAAMs/C,iBAAkBmvF,EAAoB,GAGrEhpC,EAAakpC,EAAoBtnI,KAAKqC,MAAO,EAAGklI,EAAuBvnI,KAAKnH,QAG5E2uI,EAAwB7uI,EAAMi5F,uBAAwB01C,EAAoBzwI,KAAMunG,GAChF3zD,EAAY+8F,EAAsB/8F,UAIxC,GAAKA,GAAa9xC,EAAMsrE,OAAOgE,SAAUx9B,GACxC,OAAO9xC,EAAM0/C,cAAe5N,GAG7B,OAAO,IACR,CAtFSg9F,CAAkC5yG,EAAQyyG,EAAqBF,GAElE/hH,IAMLA,EAAQ1sB,EAAMsrE,OAAOiE,yBAA0Bo/D,EAAqB,EAAI99H,QAAU,UAAY,YAEzF6b,GAiFN,SAA8CwP,EAAgB7c,GAC7D,MAAMrf,EAAQk8B,EAAOl8B,MACrB,IAAIyqI,EAAiCprH,EAErC,KAAQorH,GAAiB,CACxB,GAAKzqI,EAAMsrE,OAAOgE,SAAUm7D,GAC3B,OAAOzqI,EAAM0/C,cAAe+qF,GAG7BA,EAAiBA,EAAejkI,MACjC,CAGA,OAAO,IACR,CAxFQuoI,CAAqC7yG,EAAQyyG,EAAoBnoI,UAgCzE,SAAuC01B,EAAgBuyG,GACtD,MAAMzuI,EAAQk8B,EAAOl8B,MACfsrE,EAAStrE,EAAMsrE,OAEfsjE,EAAyB5uI,EAAMs/C,iBAAkBmvF,EAAoB,GAE3E,OAAOnjE,EAAOiE,yBAA0Bq/D,EAAwB,UACjE,CA7DSI,CAA8B9yG,EAAQuyG,EAuB/C,CAiHA,SAASV,GAAoB1yE,GAC5B,OAAK,EAAIxqD,QACDwqD,EAAaX,WAGd,CAAE,MAAO,YAAavjC,SAAUkkC,EAAaZ,eAAkB,OAAS,MAChF,CAKA,SAASkzE,GAAqBniI,GAE7B,GAAKA,EAAOsK,GAAI,mBACf,OAAO,KAIR,GAAKtK,EAAOk3B,SAAU,+BACrB,OAAOl3B,EAAO0iC,aAAc,IAI7B,GAAK,GAAU1iC,GACd,OAAOA,EAIR,MAAMgzC,EAAWhzC,EAAO0iC,cAAcnnC,GAAQ,GAAUA,IAAUA,EAAK+O,GAAI,qBAG3E,OAAK,GAAU0oC,GACPA,EAGD,IACR,CCz0Be,MAAMywF,WAAuB,GAI3C,qBAAkB/wG,GACjB,MAAO,gBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GACV,CAKO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf6hC,EAAO3F,EAAO6qE,QAAQllE,KACtBu5B,EAAev5B,EAAKz2B,SACpByxB,EAAY78B,EAAMoL,SAASyxB,UAEjC,IAAIwmG,GAAe,EAEnBxhG,EAAKu6B,YAAa,IAElBx0D,KAAKwP,SAAoCgkD,EAAc,WAAW,CAAEn+C,EAAKvS,KACxE24H,EAAe34H,EAAK2qB,QAAQ,IAG7B6G,EAAOqC,QAAQvuB,IAAK,IAAoBiH,GAAoC,oBAAoB,CAAEgG,EAAKvS,MAGhG24H,GAiCT,SAA8B9/C,EAAoCjY,GACjE,GAAKiY,EAAiBrgD,WAAa,EAClC,OAAO,EAGR,MAAM3jB,EAAQgkE,EAAiB9/C,SAAU,GAEzC,GAAK6nC,EAAOgE,SAAU/vD,GACrB,OAAO,EAGR,OAAwD,GAAjDhX,MAAMrB,KAAMqY,EAAM2mB,oBAAqBhmC,MAC/C,CA7C0BgvI,CAAqBxkI,EAAK5C,QAAS9H,EAAMsrE,UAIhEtrE,EAAMmiC,QAAQC,IAEb,MAAM+sG,EAAiB5mI,MAAMrB,KAAM21B,EAAU+Q,iBAC3CjgC,QAAQ,EAAIhP,KAAWqB,EAAMsrE,OAAOoE,uBAAwB/wE,GAAMywI,eAE9DvyG,EAAU+V,aACf5yC,EAAM29F,cAAe9gE,EAAW,CAAEqhE,oBAAoB,IAMvDixC,EAAe3nI,QAASq1B,EAAU+Q,iBAElC,MAAMlhB,EAAQ0V,EAAOud,cAAej1C,EAAK5C,SAEzC,UAAYD,KAAQ6kB,EAAMonB,WACpBjsC,EAAKiO,GAAI,eACbssB,EAAO+4D,cAAeg0C,EAAgBtnI,EAExC,GACE,GAEL,EC3Dc,MAAM,WAAkB,GAItC,qBAAkBq2B,GACjB,MAAO,WACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GAAmB4tG,GAAUuC,GACvC,ECxBY3gH,GAAQ,M,kLCYN,MAAM+gH,WAAyB,GAI7C,WAAA1lI,CAAauyB,GACZ9mB,MAAO8mB,GAGPt0B,KAAKg1B,aAAc,CACpB,CAKgB,OAAAO,GACf,MAAMn9B,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UACjC,IAAIyyG,EAA+BtvI,EAAMsrE,OAAOyO,gBAAiBl9C,GAIjE,GAAKA,EAAUmuC,sBAAuBskE,KAAmBC,GAAkBvvI,EAAMsrE,OAAQgkE,GACxF,GAIC,GAHAA,EAAeA,EAAa9oI,QAGtB8oI,EACL,cAESC,GAAkBvvI,EAAMsrE,OAAQgkE,IAG5CtvI,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAck0F,EAAe,KAAM,GAE5C,EAUD,SAASC,GAAkBjkE,EAAgBjsD,GAC1C,OAAOisD,EAAOI,QAASrsD,KAAeisD,EAAOyF,WAAY1xD,EAAS,UAAaisD,EAAOyF,WAAY1xD,EAAS,aAC5G,CC3DA,MAAMmwH,GAAuBj6G,GAAgB,UAQ9B,MAAMk6G,WAAyB,GAI7C,qBAAkBvxG,GACjB,MAAO,kBACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAEdk/B,EADOl/B,EAAO6qE,QAAQllE,KACFz2B,SAE1B8wB,EAAO0qE,SAASnxF,IAAK,YAAa,IAAI45H,GAAkBnzG,IAExDt0B,KAAKwP,SAAoCgkD,EAAc,WAAW,CAAE9iD,EAAWmvH,KACzE,GAASA,KAAmB+H,KAChCtzG,EAAOiB,QAAS,aAChBsqG,EAAa3sG,iBACd,GAEF,ECzBc,MAAM40G,WAAoB,GAIxC,qBAAkBxxG,GACjB,MAAO,aACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAEpBA,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,aAAaqrB,IAC5C,MAAMxD,EAAUpB,EAAO0qE,SAAS52F,IAAK,aAC/B6xB,EAAO,IAAI,GAAYf,GACvBl+B,EAAIk+B,EAAOl+B,EAiBjB,OAfAi/B,EAAKxoB,IAAK,CACT88B,MAAOvzC,EAAG,cACV0uG,KC1CJ,6jDD2CI97E,UAAW,SACXi8E,SAAS,IAGV5vE,EAAKjzB,KAAM,aAAczH,GAAIm2B,EAAS,aAGtC11B,KAAKwP,SAAUyqB,EAAM,WAAW,KAC/B3F,EAAOiB,QAAS,aAChBjB,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBhT,CAAI,GAEb,EEpCc,MAAM8tG,WAAkB,GAItC,mBAAkB7wG,GACjB,MAAO,CAAE2wG,GAAkBC,GAC5B,CAKA,qBAAkBxxG,GACjB,MAAO,WACR,E,+NCXc,MAAe0xG,WAAoB,GAmBjD,WAAAjmI,CAAauyB,GACZ9mB,MAAO8mB,GAbR,KAAU6nG,OAA4F,GAOtG,KAAO8L,gBAAkB,IAAIv6H,QAS5B1N,KAAK+0B,UAGL/0B,KAAK80B,4BAA6B,EAIlC90B,KAAKwP,SAAkC8kB,EAAOxxB,KAAM,OAAO,CAAEuS,EAAKvS,KAGjEA,EAAM,G,yHAAM,IAAKA,EAAM,IAEvB,MAAMyB,EAAUzB,EAAM,GAGhByB,EAAQs6E,YACbt6E,EAAQs6E,UAAY,CAAElB,YAAY,GACnC,GACE,CAAE/wE,SAAU,SAGf5M,KAAKwP,SAAkC8kB,EAAOxxB,KAAM,OAAO,CAAEuS,EAAKvS,KAGjDA,EAAM,GAER+7E,UAAWlB,YACxB39E,KAAKkoI,YACN,GAEF,CAKgB,OAAAnzG,GACf/0B,KAAK00B,UAAY10B,KAAKm8H,OAAO7jI,OAAS,CACvC,CAKA,kBAAW6vI,GACV,OAAOnoI,KAAKioI,eACb,CAQO,QAAAG,CAAUliE,GAChB,MAAMmiE,EAAeroI,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAE1CA,EAAY,CACjB4W,OAAQw8F,EAAa3jE,YAAc/jE,MAAMrB,KAAM+oI,EAAaj7F,aAAgB,GAC5ED,WAAYk7F,EAAal7F,YAG1BntC,KAAKm8H,OAAOv8H,KAAM,CAAEsmE,QAAOjxC,cAC3Bj1B,KAAK+0B,SACN,CAKO,UAAAmzG,GACNloI,KAAKm8H,OAAS,GACdn8H,KAAK+0B,SACN,CASU,iBAAAuzG,CACTz8F,EACAsB,EACAktB,GAEA,MAAMjiE,EAAQ4H,KAAKs0B,OAAOl8B,MACpBoL,EAAWpL,EAAMoL,SAGjB+kI,EAAgC,GAGhCC,EAAyB38F,EAAO7uC,KAAK8nB,GAASA,EAAMs1C,2BAA4BC,KAChFouE,EAAYD,EAAuBh/D,OAEzC,UAAYk/D,KAAcF,EAAyB,CAElD,MAAM/uE,EAAcivE,EAClB3iI,QAAQ+e,GAASA,EAAMxuB,MAAQkN,EAASo2D,YACxC7zD,QAAQ+e,IAAU6jH,GAAiC7jH,EAAO2jH,KAGtDhvE,EAAYnhE,SAMlBswI,GAAiBnvE,GAKjB8uE,EAAgB3oI,KAAM65D,EAAa,IACpC,CAKK8uE,EAAgBjwI,QACpBF,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAc+0F,EAAiB,CAAE/5F,SAAUrB,GAAc,GAGnE,CASU,KAAA07F,CAAOC,EAAoBC,GACpC,MAAM3wI,EAAQ4H,KAAKs0B,OAAOl8B,MACpBoL,EAAWpL,EAAMoL,SAGvBxD,KAAKioI,gBAAgBp6H,IAAKk7H,GAE1B,MAAMC,EAAmBF,EAAYzuE,WAAWv4D,QAAQiE,QAAQ0yD,GAAaA,EAAU2L,sBACvF4kE,EAAiB39H,UAIjB,UAAY49H,KAAmBD,EAAmB,CACjD,MAAME,EAAkBD,EAAgB9oD,YAAe,EACjDgpD,EAAoBxoI,MAAMrB,KAAMkE,EAASgiF,QAAQoK,cAAes5C,IAahEE,EAXkBxlD,GACvB,CAAEqlD,EAAgB1nD,eAClB4nD,EACA,CACCllD,cAAc,EACdzgF,SAAUxD,KAAKs0B,OAAOl8B,MAAMoL,SAC5B2hF,cAAc,EACdjB,iBAAiB,IAIwBL,YAG3C,QAAUprB,KAAa2wE,EAAqB,CAE3C,MAAM/nD,EAAqB5oB,EAAU4oB,mBAEhCA,IAAuBjpF,EAAMi9B,UAAWgsD,KAC5C5oB,EAAY,IAAIkqB,GAAalqB,EAAU0nB,cAIxC4oD,EAAa7+C,aAAczxB,GAC3BrgE,EAAMu6F,eAAgBl6B,GAEtBj1D,EAASgiF,QAAQ+K,qBAAsB04C,EAAiBxwE,EACzD,CACD,CACD,EAQD,SAASmwE,GAAiB/8F,GACzBA,EAAO1qB,MAAM,CAAExlB,EAAGhD,IAAOgD,EAAE+kC,MAAMhE,SAAU/jC,EAAE+nC,QAAW,EAAI,IAE5D,QAAUlmC,EAAI,EAAGA,EAAIqxC,EAAOvzC,OAAQkC,IAAM,CACzC,MACM6uI,EADgBx9F,EAAQrxC,EAAI,GACAw/D,UAAWnuB,EAAQrxC,IAAK,GAErD6uI,IAEJ7uI,IACAqxC,EAAOrkC,OAAQhN,EAAG,EAAG6uI,GAEvB,CACD,CAEA,SAASV,GAAiC7jH,EAAc+mB,GACvD,OAAOA,EAAOxX,MAAMiX,GAAcA,IAAexmB,GAASwmB,EAAWE,cAAe1mB,GAAO,IAC5F,CCrOe,MAAMwkH,WAAoBtB,GAUxB,OAAAzyG,CAAS2wC,EAAsB,MAE9C,MAAMqjE,EAAarjE,EAAQlmE,KAAKm8H,OAAOrtB,WAAWnzG,GAAKA,EAAEuqE,OAASA,IAAUlmE,KAAKm8H,OAAO7jI,OAAS,EAE3F2H,EAAOD,KAAKm8H,OAAO30H,OAAQ+hI,EAAY,GAAK,GAC5CR,EAAe/oI,KAAKs0B,OAAOl8B,MAAM+lG,YAAa,CAAEnU,QAAQ,IAI9DhqF,KAAKs0B,OAAOl8B,MAAMguE,cAAe2iE,GAAc,KAC9C/oI,KAAK6oI,MAAO5oI,EAAKimE,MAAO6iE,GAExB,MAAM1uE,EAAar6D,KAAKs0B,OAAOl8B,MAAMoL,SAASgiF,QAAQoK,cAAe3vF,EAAKimE,MAAMia,aAChFngF,KAAKsoI,kBAAmBroI,EAAKg1B,UAAU4W,OAAQ5rC,EAAKg1B,UAAUkY,WAAYktB,EAAY,IAMvFr6D,KAAKwQ,KAA8B,SAAUvQ,EAAKimE,MAAO6iE,GAEzD/oI,KAAK+0B,SACN,EChCc,MAAMy0G,WAAoBxB,GASxB,OAAAzyG,GACf,MAAMt1B,EAAOD,KAAKm8H,OAAOn9H,MACnByqI,EAAezpI,KAAKs0B,OAAOl8B,MAAM+lG,YAAa,CAAEnU,QAAQ,IAI9DhqF,KAAKs0B,OAAOl8B,MAAMguE,cAAeqjE,GAAc,KAC9C,MACMP,EADgBjpI,EAAKimE,MAAM7L,WAAYp6D,EAAKimE,MAAM7L,WAAW/hE,OAAS,GACtC6nF,YAAe,EAC/C9lB,EAAar6D,KAAKs0B,OAAOl8B,MAAMoL,SAASgiF,QAAQoK,cAAes5C,GAErElpI,KAAKsoI,kBAAmBroI,EAAKg1B,UAAU4W,OAAQ5rC,EAAKg1B,UAAUkY,WAAYktB,GAC1Er6D,KAAK6oI,MAAO5oI,EAAKimE,MAAOujE,EAAc,IAGvCzpI,KAAK+0B,SACN,ECpBc,MAAM20G,WAAoB,GAAzC,kCAgBC,KAAQC,eAAiB,IAAIj8H,OAAe,CAK5C,qBAAkB4oB,GACjB,MAAO,aACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAGpBt0B,KAAK4pI,aAAe,IAAIN,GAAah1G,GACrCt0B,KAAK6pI,aAAe,IAAIL,GAAal1G,GAGrCA,EAAO0qE,SAASnxF,IAAK,OAAQ7N,KAAK4pI,cAClCt1G,EAAO0qE,SAASnxF,IAAK,OAAQ7N,KAAK6pI,cAElC7pI,KAAKwP,SAAoC8kB,EAAOl8B,MAAO,kBAAkB,CAAEid,EAAKnZ,KAC/E,MAAMu8D,EAAYv8D,EAAM,GAOxB,IAAMu8D,EAAU2L,oBACf,OAGD,MAAM8B,EAAQzN,EAAUyN,MAElB4jE,EAAc9pI,KAAK6pI,aAAa1B,eAAev6H,IAAKs4D,GACpD6jE,EAAc/pI,KAAK4pI,aAAazB,eAAev6H,IAAKs4D,GACrClmE,KAAK2pI,eAAe/7H,IAAKs4D,KAQ9ClmE,KAAK2pI,eAAe97H,IAAKq4D,GAEnBA,EAAMyX,aAIPmsD,EAEJ9pI,KAAK4pI,aAAaxB,SAAUliE,GAChB6jE,IAGZ/pI,KAAK4pI,aAAaxB,SAAUliE,GAC5BlmE,KAAK6pI,aAAa3B,eACnB,GACE,CAAEt7H,SAAU,YAEf5M,KAAKwP,SAAkCxP,KAAK4pI,aAAc,UAAU,CAAEv0H,EAAK20H,EAAajB,KACvF/oI,KAAK6pI,aAAazB,SAAUW,EAAc,IAG3Cz0G,EAAO8qE,WAAW3tF,IAAK,SAAU,QACjC6iB,EAAO8qE,WAAW3tF,IAAK,SAAU,QACjC6iB,EAAO8qE,WAAW3tF,IAAK,eAAgB,OACxC,EC/GD,sVCAA,gVCkBe,MAAMw4H,WAAe,GAInC,qBAAkB3zG,GACjB,MAAO,QACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd4E,EAAS5E,EAAO4E,OAChBl+B,EAAIs5B,EAAOt5B,EAEXkvI,EAAkD,OAA9BhxG,EAAOvJ,oBAA+Bw6G,GAAWC,GACrEC,EAAkD,OAA9BnxG,EAAOvJ,oBAA+By6G,GAAWD,GAE3EnqI,KAAKsqI,WAAY,OAAQtvI,EAAG,QAAU,SAAUkvI,GAChDlqI,KAAKsqI,WAAY,OAAQtvI,EAAG,QAAU,SAAUqvI,EACjD,CAUQ,UAAAC,CAAYtoI,EAAuBusC,EAAe3gB,EAAmB28G,GAC5E,MAAMj2G,EAASt0B,KAAKs0B,OAEpBA,EAAO6E,GAAG61E,iBAAiBnhG,IAAK7L,GAAMk3B,IACrC,MAAMxD,EAAUpB,EAAO0qE,SAAS52F,IAAKpG,GAC/Bi4B,EAAO,IAAI,GAAYf,GAgB7B,OAdAe,EAAKxoB,IAAK,CACT88B,QACAm7D,KAAM6gC,EACN38G,YACAi8E,SAAS,IAGV5vE,EAAKjzB,KAAM,aAAczH,GAAIm2B,EAAS,aAEtC11B,KAAKwP,SAAUyqB,EAAM,WAAW,KAC/B3F,EAAOiB,QAASvzB,GAChBsyB,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBhT,CAAI,GAEb,ECsCc,MAAM,WAAa,GAIjC,mBAAkB/C,GACjB,MAAO,CAAEwyG,GAAaO,GACvB,CAKA,qBAAkB3zG,GACjB,MAAO,MACR,ECzGc,MAAMk0G,WAAmBr3H,KAyBvC,WAAApR,GACCyL,QAEA,MAAMi9H,EAAS,IAAIlnI,OAAOinI,WAE1BxqI,KAAK0qI,QAAUD,EACfzqI,KAAKo9B,WAAQ,EAEbp9B,KAAKyR,IAAK,SAAU,GAEpBg5H,EAAOE,WAAat1H,IACnBrV,KAAK4qI,OAASv1H,EAAIu1H,MAAM,CAE1B,CAKA,SAAWhhI,GACV,OAAO5J,KAAK0qI,QAAQ9gI,KACrB,CAMA,QAAW9G,GACV,OAAO9C,KAAKo9B,KACb,CASO,IAAAytG,CAAMC,GACZ,MAAML,EAASzqI,KAAK0qI,QAGpB,OAFA1qI,KAAK+qI,MAAQD,EAAKj3H,KAEX,IAAI4kB,SAAS,CAAElc,EAASyuH,KAC9BP,EAAOQ,OAAS,KACf,MAAM9sI,EAASssI,EAAOtsI,OAEtB6B,KAAKo9B,MAAQj/B,EAEboe,EAASpe,EAAQ,EAGlBssI,EAAOS,QAAU,KAChBF,EAAQ,QAAS,EAGlBP,EAAOU,QAAU,KAChBH,EAAQ,UAAW,EAGpBhrI,KAAK0qI,QAAQU,cAAeN,EAAM,GAEpC,CAKO,KAAAO,GACNrrI,KAAK0qI,QAAQW,OACd,ECvEc,MAAM,WAAuB,GAA5C,kCAIC,KAAOC,QAAU,IAAI,GAgBrB,KAAQC,YAAc,IAAIh6H,IAM1B,KAAQi6H,eAAuC,KAiC/C,qBAAkBl1G,GACjB,MAAO,gBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE8oE,GACV,CAKO,IAAAtpE,GAEN12B,KAAKsrI,QAAQj8H,GAA2B,UAAU,IAAMrP,KAAKyrI,yBAE7DzrI,KAAKyR,IAAK,WAAY,GACtBzR,KAAKyR,IAAK,cAAe,MAEzBzR,KAAKgH,KAAM,mBAAoBzH,GAAIS,KAAM,WAAYA,KAAM,eAAe,CAAE0rI,EAAUX,IAC9EA,EAAUW,EAAWX,EAAQ,IAAQ,GAE9C,CASO,SAAAY,CAAWC,GACjB,OAAO5rI,KAAKurI,YAAYnjI,IAAKwjI,IAAmB,IACjD,CASO,YAAAC,CAAcD,GACpB,IAAM5rI,KAAK8rI,oBA0BV,OAFA,EAAY,oCAEL,KAGR,MAAMC,EAAS,IAAIC,GAAYvzG,QAAQlc,QAASqvH,GAAiB5rI,KAAK8rI,qBAuCtE,OArCA9rI,KAAKsrI,QAAQz9H,IAAKk+H,GAClB/rI,KAAKurI,YAAY95H,IAAKm6H,EAAeG,GAGhCH,aAAyBnzG,SAC7BszG,EAAOjB,KACLhzG,MAAMgzG,IACN9qI,KAAKurI,YAAY95H,IAAKq5H,EAAOiB,EAAQ,IAKrCxgB,OAAO,SAGVwgB,EAAO18H,GAA2B,mBAAmB,KACpD,IAAI48H,EAAqB,EAEzB,UAAYF,KAAU/rI,KAAKsrI,QAC1BW,GAAsBF,EAAOL,SAG9B1rI,KAAK0rI,SAAWO,CAAkB,IAGnCF,EAAO18H,GAA2B,sBAAsB,KACvD,IAAI68H,EAAkB,EAEtB,UAAYH,KAAU/rI,KAAKsrI,QACrBS,EAAOI,cACXD,GAAmBH,EAAOI,aAI5BnsI,KAAKmsI,YAAcD,CAAe,IAG5BH,CACR,CAOO,aAAAK,CAAeC,GACrB,MAAMN,EAASM,aAAiCL,GAAaK,EAAwBrsI,KAAK2rI,UAAWU,GAErGN,EAAOzgB,WAEPtrH,KAAKsrI,QAAQrlI,OAAQ8lI,GAErB/rI,KAAKurI,YAAY3tI,SAAS,CAAEpF,EAAOzB,KAC7ByB,IAAUuzI,GACd/rI,KAAKurI,YAAY55H,OAAQ5a,EAC1B,GAEF,CAKQ,oBAAA00I,GACP,MAAMa,EAAiBtsI,KAAKs0B,OAAOqC,QAAQvuB,IAAK43F,IAEhD,GAAKhgG,KAAKsrI,QAAQhzI,QACjB,IAAM0H,KAAKwrI,eAAiB,CAC3B,MAAMxwI,EAAIgF,KAAKs0B,OAAOt5B,EAChBuxI,EAAe/zI,GAAmB,GAAIwC,EAAG,yBAA4BoC,SAAU5E,OAErFwH,KAAKwrI,eAAiBc,EAAez+H,IAAK0+H,EAAYvsI,KAAKwsI,kBAC3DxsI,KAAKwrI,eAAexkI,KAAM,WAAYzH,GAAIS,KAAM,kBAAmBusI,EACpE,OAEAD,EAAermI,OAAQjG,KAAKwrI,gBAC5BxrI,KAAKwrI,eAAiB,IAExB,EAQD,MAAMQ,WAAmB74H,KAuFxB,WAAApR,CAAa0qI,EAA4BC,GACxCl/H,QAEAxN,KAAKH,GAAK,IACVG,KAAK2sI,oBAAsB3sI,KAAK4sI,0BAA2BH,GAC3DzsI,KAAK6sI,SAAWH,EAAsB1sI,MACtCA,KAAK0qI,QAAU,IAAIF,GAEnBxqI,KAAKyR,IAAK,SAAU,QACpBzR,KAAKyR,IAAK,WAAY,GACtBzR,KAAKyR,IAAK,cAAe,MAEzBzR,KAAKgH,KAAM,mBAAoBzH,GAAIS,KAAM,WAAYA,KAAM,eAAe,CAAE0rI,EAAUX,IAC9EA,EAAUW,EAAWX,EAAQ,IAAQ,IAG7C/qI,KAAKyR,IAAK,iBAAkB,KAC7B,CAKA,QAAWq5H,GACV,OAAM9qI,KAAK2sI,oBAYH3sI,KAAK2sI,oBAAoBn0G,QAAQV,MAAMgzG,GAAQ9qI,KAAK2sI,oBAAsB7B,EAAO,OAVjFryG,QAAQlc,QAAS,KAY1B,CAMA,QAAWzZ,GACV,OAAO9C,KAAK0qI,QAAQ5nI,IACrB,CAyBO,IAAA+nI,GACN,GAAoB,QAAf7qI,KAAK8sI,OAMT,MAAM,IAAI,EAAe,mCAAoC9sI,MAK9D,OAFAA,KAAK8sI,OAAS,UAEP9sI,KAAK8qI,KACVhzG,MAAMgzG,GAAQ9qI,KAAK0qI,QAAQG,KAAMC,KACjChzG,MAAMh1B,IAGN,GAAqB,YAAhB9C,KAAK8sI,OACT,MAAM9sI,KAAK8sI,OAKZ,OAFA9sI,KAAK8sI,OAAS,OAEPhqI,CAAI,IAEXyoH,OAAO/pH,IACP,GAAa,YAARA,EAEJ,MADAxB,KAAK8sI,OAAS,UACR,UAIP,MADA9sI,KAAK8sI,OAAS,QACR9sI,KAAK0qI,QAAQ9gI,MAAQ5J,KAAK0qI,QAAQ9gI,MAAQpI,CAAG,GAEtD,CAwBO,MAAAurI,GACN,GAAoB,QAAf/sI,KAAK8sI,OAMT,MAAM,IAAI,EAAe,qCAAsC9sI,MAKhE,OAFAA,KAAK8sI,OAAS,YAEP9sI,KAAK8qI,KACVhzG,MAAM,IAAM93B,KAAK6sI,SAASE,WAC1Bj1G,MAAMh1B,IACN9C,KAAKgtI,eAAiBlqI,EACtB9C,KAAK8sI,OAAS,OAEPhqI,KAEPyoH,OAAO/pH,IACP,GAAqB,YAAhBxB,KAAK8sI,OACT,KAAM,UAIP,MADA9sI,KAAK8sI,OAAS,QACRtrI,CAAG,GAEZ,CAKO,KAAA6pI,GACN,MAAMyB,EAAS9sI,KAAK8sI,OACpB9sI,KAAK8sI,OAAS,UAER9sI,KAAK2sI,oBAAoBM,YAOT,WAAVH,EACX9sI,KAAK0qI,QAAQW,QACQ,aAAVyB,GAAyB9sI,KAAK6sI,SAASxB,OAClDrrI,KAAK6sI,SAASxB,SANdrrI,KAAK2sI,oBAAoBn0G,QAAQ+yF,OAAO,SAExCvrH,KAAK2sI,oBAAoBO,SAAU,YAOpCltI,KAAKsrH,UACN,CAOO,QAAAA,GACNtrH,KAAK2sI,yBAAsB,EAC3B3sI,KAAK0qI,aAAU,EACf1qI,KAAK6sI,cAAW,EAChB7sI,KAAKgtI,oBAAiB,CACvB,CAQQ,yBAAAJ,CAA2BH,GAClC,MAAM/yF,EAAuC,CAAC,EAiB9C,OAfAA,EAAQlhB,QAAU,IAAIC,SAAe,CAAElc,EAASyuH,KAC/CtxF,EAAQwzF,SAAWlC,EACnBtxF,EAAQuzF,aAAc,EAEtBR,EACE30G,MAAMgzG,IACNpxF,EAAQuzF,aAAc,EACtB1wH,EAASuuH,EAAM,IAEfvf,OAAO/pH,IACPk4C,EAAQuzF,aAAc,EACtBjC,EAAQxpI,EAAK,GACX,IAGEk4C,CACR,EC5gBc,MAAMyzF,WAA6B,GA+BjD,WAAAprI,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKmrG,WAAa,IAAI,GAAYjyE,GAElCl5B,KAAKotI,eAAiB,IAAIC,GAAen0G,GACzCl5B,KAAKotI,eAAepmI,KAAM,gBAAiBzH,GAAIS,MAC/CA,KAAKotI,eAAepmI,KAAM,sBAAuBzH,GAAIS,MAErDA,KAAKotI,eAAeh8H,SAAU,QAAS7R,GAAIS,MAE3CA,KAAK4jG,YAAa,CACjBzsF,IAAK,OACL5R,WAAY,CACXg5B,MAAO,yBAERjnB,SAAU,CACTtX,KAAKmrG,WACLnrG,KAAKotI,kBAIPptI,KAAKmrG,WAAW97F,GAAwB,WAAW,KAClDrP,KAAKotI,eAAeE,MAAM,GAE5B,CAKO,KAAArgG,GACNjtC,KAAKmrG,WAAWl+D,OACjB,EAMD,MAAMogG,WAAsB,GAqB3B,WAAAtrI,CAAam3B,GACZ1rB,MAAO0rB,GAEPl5B,KAAKyR,IAAK,oBAAgB,GAC1BzR,KAAKyR,IAAK,sBAAsB,GAEhC,MAAMzK,EAAOhH,KAAK0iG,aAElB1iG,KAAK4jG,YAAa,CACjBzsF,IAAK,QAEL5R,WAAY,CACXg5B,MAAO,CACN,aAEDtzB,KAAM,OACNo+F,SAAU,KACVkkC,OAAQvmI,EAAKzH,GAAI,gBACjBiuI,SAAUxmI,EAAKzH,GAAI,uBAGpB8P,GAAI,CAEHkrB,OAAQvzB,EAAKzH,IAA4C,KACnDS,KAAKyX,SAAWzX,KAAKyX,QAAQi7C,OAAS1yD,KAAKyX,QAAQi7C,MAAMp6D,QAC7D0H,KAAKwQ,KAA8B,OAAQxQ,KAAKyX,QAAQi7C,OAGzD1yD,KAAKyX,QAASjf,MAAQ,QAI1B,CAKO,IAAA80I,GACNttI,KAAKyX,QAAS8xF,OACf,E,kLC9JD,MAAMkkC,GAAoB,cAEpBC,GAAe,uCAOd,SAASC,KACf,IAAIxvB,EAaE,SAAoBn8G,GAC1BA,EAAOA,EAAK4G,cACZ,MAAMiW,EAAQrb,SAASoqI,OAAO7wI,MAAO,KAErC,UAAY+hB,KAAQD,EAAQ,CAC3B,MAAMgvH,EAAO/uH,EAAK/hB,MAAO,KAGzB,GAFY+wI,mBAAoBD,EAAM,GAAIhgH,OAAOjlB,iBAEpC5G,EACZ,OAAO8rI,mBAAoBD,EAAM,GAEnC,CAEA,OAAO,IACR,CA3BaE,CAAWN,IAgCjB,IAAoBzrI,EAAcxJ,EAzBxC,OALM2lH,GAXc,IAWLA,EAAM7lH,SACpB6lH,EAoCF,SAAwB7lH,GACvB,IAAI6F,EAAS,GACb,MAAM6vI,EAAa,IAAItxH,WAAYpkB,GAEnCiL,OAAO0qI,OAAOC,gBAAiBF,GAE/B,QAAU1zE,EAAI,EAAGA,EAAI0zE,EAAW11I,OAAQgiE,IAAM,CAC7C,MAAM3mC,EAAY+5G,GAAargH,OAAQ2gH,EAAY1zE,GAAMozE,GAAap1I,QACtE6F,GAAUtF,KAAK2T,SAAW,GAAMmnB,EAAUh3B,cAAgBg3B,CAC3D,CAEA,OAAOx1B,CACR,CAhDUgwI,CAZW,IAyCMnsI,EA5BdyrI,GA4B4Bj1I,EA5BT2lH,EA6B/B36G,SAASoqI,OAASjrI,mBAAoBX,GAAS,IAAMW,mBAAoBnK,GAAU,WA1B5E2lH,CACR,CCoCA,MAAMiwB,GAqBL,WAAArsI,CAAagqI,EAAoBsC,EAAarzI,GAC7CgF,KAAK+rI,OAASA,EACd/rI,KAAKquI,IAAMA,EACXruI,KAAKhF,EAAIA,CACV,CAOO,MAAA+xI,GACN,OAAO/sI,KAAK+rI,OAAOjB,KAAKhzG,MAAMgzG,GACtB,IAAIryG,SAAyB,CAAElc,EAASyuH,KAC9ChrI,KAAKsuI,eACLtuI,KAAKuuI,eAAgBhyH,EAASyuH,EAAQF,GACtC9qI,KAAKwuI,aAAc1D,EAAO,KAG7B,CAOO,KAAAO,GACDrrI,KAAKyuI,KACTzuI,KAAKyuI,IAAIpD,OAEX,CAKQ,YAAAiD,GACP,MAAMG,EAAMzuI,KAAKyuI,IAAM,IAAIC,eAE3BD,EAAInB,KAAM,OAAQttI,KAAKquI,KAAK,GAC5BI,EAAIE,aAAe,MACpB,CASQ,cAAAJ,CACPhyH,EACAyuH,EACAF,GAEA,MAAM2D,EAAMzuI,KAAKyuI,IACX1C,EAAS/rI,KAAK+rI,OAEd6C,GAAe5zI,EADXgF,KAAKhF,GACS,uBAA0B,IAAK8vI,EAAK9oI,QAE5DysI,EAAIn+H,iBAAkB,SAAS,IAAM06H,EAAQ4D,KAC7CH,EAAIn+H,iBAAkB,SAAS,IAAM06H,MACrCyD,EAAIn+H,iBAAkB,QAAQ,KAC7B,MAAMu+H,EAAWJ,EAAII,SAErB,IAAMA,IAAaA,EAASnD,SAC3B,OAAOV,EAAQ6D,GAAYA,EAASjlI,OAASilI,EAASjlI,MAAMwE,QAAUygI,EAASjlI,MAAMwE,QAAUwgI,GAGhGryH,EAAS,CACRuyH,QAASD,EAASR,KAChB,IAKCI,EAAI1B,QACR0B,EAAI1B,OAAOz8H,iBAAkB,YAAY+E,IACnCA,EAAI05H,mBACRhD,EAAOI,YAAc92H,EAAI01H,MACzBgB,EAAOL,SAAWr2H,EAAIu1H,OACvB,GAGH,CAOQ,YAAA4D,CAAc1D,GAErB,MAAMhoI,EAAO,IAAIksI,SACjBlsI,EAAK+iD,OAAQ,SAAUilF,GACvBhoI,EAAK+iD,OAAQ,cAAe8nF,MAG5B3tI,KAAKyuI,IAAKQ,KAAMnsI,EACjB,ECnHc,SAASosI,GACvB56G,EACAmC,EACAoH,EACAsxG,GAEA,IAAI5/H,EACAmmB,EAA0B,KAEG,mBAArBy5G,EACX5/H,EAAW4/H,GAGXz5G,EAAUpB,EAAO0qE,SAAS52F,IAAK+mI,GAE/B5/H,EAAW,KACV+kB,EAAOiB,QAAS45G,EAAmB,GAIrC76G,EAAOl8B,MAAMoL,SAAS6L,GAAyB,eAAe,CAAEgG,EAAK6wD,KACpE,GAAKxwC,IAAYA,EAAQhB,YAAc+B,EAAO/B,UAC7C,OAGD,MAAM5P,EAAQ,GAAOwP,EAAOl8B,MAAMoL,SAASyxB,UAAUmY,aAErD,IAAMtoB,EAAMkmB,YACX,OAGD,GAAKk7B,EAAM8jB,SAAW9jB,EAAM6jB,QAC3B,OAGD,MAAM9pB,EAAUt/D,MAAMrB,KAAMg1B,EAAOl8B,MAAMoL,SAASm8D,OAAOQ,cACnDznD,EAAQunD,EAAS,GAGvB,GAAuB,GAAlBA,EAAQ3nE,QAA8B,WAAfogB,EAAMzN,MAAmC,SAAdyN,EAAM1W,MAAmC,GAAhB0W,EAAMpgB,OACrF,OAGD,MAAM82I,EAAgB12H,EAAM+M,SAAS7mB,OAGrC,GAAKwwI,EAAclhI,GAAI,UAAW,aACjC,OAID,GAAKkhI,EAAclhI,GAAI,UAAW,aACJ,mBAAtBihI,IACN,CAAE,eAAgB,eAAgB,YAAa5/G,SAAU4/G,GAE1D,OAKD,GAAKz5G,IAA6B,IAAlBA,EAAQl9B,MACvB,OAGD,MAAM62I,EAAYD,EAAcvzG,SAAU,GAEpCyzG,EAAiBh7G,EAAOl8B,MAAM0/C,cAAeu3F,GAGnD,IAAMC,EAAe9jG,cAAe1mB,KAAYA,EAAM6b,IAAI1c,QAASqrH,EAAe3uG,KACjF,OAGD,MAAM9jC,EAAQghC,EAAQhlB,KAAMw2H,EAAUvsI,KAAKmP,OAAQ,EAAG6S,EAAM6b,IAAI/kB,SAG1D/e,GAKNy3B,EAAOl8B,MAAMguE,eAAe5rC,IAE3B,MAAMkG,EAAQlG,EAAOkd,iBAAkB03F,EAAe,GAChDzuG,EAAMnG,EAAOkd,iBAAkB03F,EAAevyI,EAAO,GAAIvE,QACzDwsB,EAAQ,IAAI,GAAW4b,EAAOC,GAKpC,IAAoB,IAHDpxB,EAAU,CAAE1S,UAGH,CAC3B29B,EAAOv0B,OAAQ6e,GAEf,MAAMgiD,EAAiBxyC,EAAOl8B,MAAMoL,SAASyxB,UAAUoY,gBACjDkiG,EAAa/0G,EAAOud,cAAeq3F,IAIpCA,EAAcjtG,SAAYotG,EAAWtrH,QAAS6iD,IAAqByoE,EAAW/jG,cAAes7B,GAAgB,IACjHtsC,EAAOv0B,OAAQmpI,EAEjB,CACAtqH,EAAMhE,SAENwT,EAAOl8B,MAAMguE,eAAe,KACE9xC,EAAOqC,QAAQvuB,IAAK,UAEpCiuH,wBAAwB,GACnC,GACD,GAEL,CCtFe,SAASmZ,GACvBl7G,EACAmC,EACAg5G,EACAC,GAEA,IAAIC,EACAlZ,EAECgZ,aAAgC9lI,OACpCgmI,EAASF,EAEThZ,EAAegZ,EAIhBhZ,EAAeA,GAAgB,CAAEv8F,IAChC,IAAI/7B,EACJ,MAAM8H,EAA+B,GAC/B0tG,EAA+B,GAErC,KAA6C,QAAnCx1G,EAASwxI,EAAO92H,KAAMqhB,OAE1B/7B,GAAUA,EAAO7F,OAAS,IAFoB,CAMnD,IAAI,MACHsM,EACA,EAAKgrI,EACL,EAAK1vI,EACL,EAAK2vI,GACF1xI,EAGJ,MAAMyvC,EAAQgiG,EAAU1vI,EAAU2vI,EAClCjrI,GAASzG,EAAQ,GAAI7F,OAASs1C,EAAMt1C,OAGpC,MAAMw3I,EAAW,CAChBlrI,EACAA,EAAQgrI,EAAQt3I,QAEXy3I,EAAS,CACdnrI,EAAQgrI,EAAQt3I,OAAS4H,EAAQ5H,OACjCsM,EAAQgrI,EAAQt3I,OAAS4H,EAAQ5H,OAASu3I,EAASv3I,QAGpD2N,EAAOrG,KAAMkwI,GACb7pI,EAAOrG,KAAMmwI,GAEbp8B,EAAO/zG,KAAM,CAAEgF,EAAQgrI,EAAQt3I,OAAQsM,EAAQgrI,EAAQt3I,OAAS4H,EAAQ5H,QACzE,CAEA,MAAO,CACN2N,SACA0tG,SAED,GAEDr/E,EAAOl8B,MAAMoL,SAAS6L,GAAyB,eAAe,CAAEgG,EAAK6wD,KACpE,GAAKA,EAAM8jB,SAAW9jB,EAAM6jB,UAAYtzD,EAAO/B,UAC9C,OAGD,MAAMt8B,EAAQk8B,EAAOl8B,MACf68B,EAAY78B,EAAMoL,SAASyxB,UAGjC,IAAMA,EAAU+V,YACf,OAGD,MAAMi1B,EAAUt/D,MAAMrB,KAAMlH,EAAMoL,SAASm8D,OAAOQ,cAC5CznD,EAAQunD,EAAS,GAGvB,GAAuB,GAAlBA,EAAQ3nE,QAA8B,WAAfogB,EAAMzN,MAAmC,SAAdyN,EAAM1W,MAAmC,GAAhB0W,EAAMpgB,OACrF,OAGD,MAAM20C,EAAQhY,EAAUgY,MAClB+1B,EAAQ/1B,EAAOruC,QACf,KAAEs7B,EAAI,MAAEpV,GAkDhB,SAA2BA,EAAc1sB,GACxC,IAAIsoC,EAAQ5b,EAAM4b,MAElB,MAAMxG,EAAOv5B,MAAMrB,KAAMwlB,EAAMonB,YAAaje,QAAQ,CAAEgmG,EAAW90H,KAExDA,EAAK+O,GAAI,WAAa/O,EAAK+O,GAAI,eAAoB/O,EAAKk8B,aAAc,SAC7EqF,EAAQtoC,EAAMu/C,oBAAqBx4C,GAE5B,IAGD80H,EAAY90H,EAAK2D,MACtB,IAEH,MAAO,CAAEo3B,OAAMpV,MAAO1sB,EAAMy/C,YAAanX,EAAO5b,EAAM6b,KACvD,CAjE0BqvG,CAAkB53I,EAAMy/C,YAAaz/C,EAAMs/C,iBAAkBsrB,EAAO,GAAK/1B,GAAU70C,GACrG63I,EAAaxZ,EAAev8F,GAC5Bg2G,EAAiBC,GAAoBrrH,EAAM4b,MAAOuvG,EAAWt8B,OAAQv7G,GACrEg4I,EAAiBD,GAAoBrrH,EAAM4b,MAAOuvG,EAAWhqI,OAAQ7N,GAEnE83I,EAAe53I,QAAU83I,EAAe93I,QAKhDF,EAAMguE,eAAe5rC,IAKpB,IAAoB,IAHDk1G,EAAgBl1G,EAAQ01G,GAG3C,CAKA,UAAYprH,KAASsrH,EAAe/kI,UACnCmvB,EAAOv0B,OAAQ6e,GAGhB1sB,EAAMguE,eAAe,KACS9xC,EAAOqC,QAAQvuB,IAAK,UAEpCiuH,wBAAwB,GAVtC,CAWG,GACD,GAEL,CAMA,SAAS8Z,GAAoBzvG,EAAiB2vG,EAA8Bj4I,GAC3E,OAAOi4I,EACLtqI,QAAQwS,QAA0B,IAAfA,EAAO,SAAoC,IAAfA,EAAO,KACtDvb,KAAKub,GACEngB,EAAMy/C,YAAanX,EAAM4J,aAAc/xB,EAAO,IAAOmoB,EAAM4J,aAAc/xB,EAAO,MAE1F,CCZA,SAAS+3H,GAAwCh8G,EAAgBmwB,GAChE,MAAO,CAAEjqB,EAAgB01G,KAGxB,IAFgB57G,EAAO0qE,SAAS52F,IAAKq8C,GAEvB/vB,UACb,OAAO,EAGR,MAAM67G,EAAcj8G,EAAOl8B,MAAMsrE,OAAO8S,eAAgB05D,EAAgBzrF,GAExE,UAAY3/B,KAASyrH,EACpB/1G,EAAO/0B,aAAcg/C,GAAc,EAAM3/B,GAK1C0V,EAAOskD,yBAA0Br6B,EAAc,CAEjD,CC7Me,MAAM,WAAyB,GAsB7C,WAAA1iD,CAAauyB,EAAgBmwB,GAC5Bj3C,MAAO8mB,GAEPt0B,KAAKykD,aAAeA,CACrB,CAKgB,OAAA1vB,GACf,MAAM38B,EAAQ4H,KAAKs0B,OAAOl8B,MACpBif,EAAMjf,EAAMoL,SAElBxD,KAAKxH,MAAQwH,KAAKwwI,gCAClBxwI,KAAK00B,UAAYt8B,EAAMsrE,OAAO6S,0BAA2Bl/D,EAAI4d,UAAWj1B,KAAKykD,aAC9E,CAsBgB,OAAAlvB,CAAShxB,EAAoC,CAAC,GAC7D,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MAEpB68B,EADM78B,EAAMoL,SACIyxB,UAChBz8B,OAAiC,IAAvB+L,EAAQksI,YAA8BzwI,KAAKxH,MAAQ+L,EAAQksI,WAE3Er4I,EAAMmiC,QAAQC,IACb,GAAKvF,EAAU+V,YACTxyC,EACJgiC,EAAOm6D,sBAAuB30F,KAAKykD,cAAc,GAEjDjqB,EAAOskD,yBAA0B9+E,KAAKykD,kBAEjC,CACN,MAAM5Y,EAASzzC,EAAMsrE,OAAO8S,eAAgBvhD,EAAUmY,YAAaptC,KAAKykD,cAExE,UAAY3/B,KAAS+mB,EACfrzC,EACJgiC,EAAO/0B,aAAczF,KAAKykD,aAAcjsD,EAAOssB,GAE/C0V,EAAO9zB,gBAAiB1G,KAAKykD,aAAc3/B,EAG9C,IAEF,CAQQ,6BAAA0rH,GACP,MAAMp4I,EAAQ4H,KAAKs0B,OAAOl8B,MACpBsrE,EAAStrE,EAAMsrE,OACfzuC,EAAY78B,EAAMoL,SAASyxB,UAEjC,GAAKA,EAAU+V,YACd,OAAO/V,EAAUgR,aAAcjmC,KAAKykD,cAGrC,UAAY3/B,KAASmQ,EAAUmY,YAC9B,UAAYntC,KAAQ6kB,EAAMonB,WACzB,GAAKw3B,EAAOmE,eAAgB5nE,EAAMD,KAAKykD,cACtC,OAAOxkD,EAAKgmC,aAAcjmC,KAAKykD,cAKlC,OAAO,CACR,ECvHD,MAAMisF,GAAO,OAQE,MAAMC,WAAoB,GAIxC,qBAAkBr6G,GACjB,MAAO,aACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAEpBA,EAAOl8B,MAAMsrE,OAAOxwB,OAAQ,QAAS,CAAE0iC,gBAAiB86D,KACxDp8G,EAAOl8B,MAAMsrE,OAAO0S,uBAAwBs6D,GAAM,CACjDlJ,cAAc,EACd1M,aAAa,IAIdxmG,EAAOj2B,WAAW2rE,mBAAoB,CACrC5xE,MAAOs4I,GACPz2G,KAAM,SACN8lD,WAAY,CACX,IACAxoC,IACC,MAAMq5F,EAAar5F,EAAYxY,SAAU,eAEzC,OAAM6xG,IAKa,QAAdA,GAAwBxrH,OAAQwrH,IAAgB,KAC7C,CACN5uI,MAAM,EACN68B,OAAQ,CAAE,gBAPJ,IAWG,KAMdvK,EAAO0qE,SAASnxF,IAAK6iI,GAAM,IAAI,GAAkBp8G,EAAQo8G,KAGzDp8G,EAAO8qE,WAAW3tF,IAAK,SAAUi/H,GAClC,EC1DD,MAAM,GAAO,OAKE,MAAMG,WAAe,GAInC,qBAAkBv6G,GACjB,MAAO,QACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EAGjBs5B,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,IAAMqrB,IACrC,MAAMxD,EAA4BpB,EAAO0qE,SAAS52F,IAAK,IACjD6xB,EAAO,IAAI,GAAYf,GAkB7B,OAhBAe,EAAKxoB,IAAK,CACT88B,MAAOvzC,EAAG,QACV0uG,KAAM,GAAMvJ,KACZvyE,UAAW,SACXi8E,SAAS,EACTP,cAAc,IAGfrvE,EAAKjzB,KAAM,OAAQ,aAAczH,GAAIm2B,EAAS,QAAS,aAGvD11B,KAAKwP,SAAUyqB,EAAM,WAAW,KAC/B3F,EAAOiB,QAAS,IAChBjB,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBhT,CAAI,GAEb,E,eCrDG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQH,OCAvB,MAAMg3G,GAAS,SAQA,MAAMC,WAAsB,GAI1C,qBAAkBz6G,GACjB,MAAO,eACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAGpBA,EAAOl8B,MAAMsrE,OAAOxwB,OAAQ,QAAS,CAAE0iC,gBAAiBk7D,KACxDx8G,EAAOl8B,MAAMsrE,OAAO0S,uBAAwB06D,GAAQ,CACnDtJ,cAAc,EACd1M,aAAa,IAGdxmG,EAAOj2B,WAAW2rE,mBAAoB,CACrC5xE,MAAO04I,GACP72G,KAAM,IACN8lD,WAAY,CACX,KACA,CACClhD,OAAQ,CACP,aAAc,cAOlBvK,EAAO0qE,SAASnxF,IAAKijI,GAAQ,IAAI,GAAkBx8G,EAAQw8G,KAG3Dx8G,EAAO8qE,WAAW3tF,IAAK,SAAUq/H,GAClC,EC3DD,MCeM,GAAS,SAKA,MAAME,WAAiB,GAIrC,qBAAkB16G,GACjB,MAAO,UACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EAGjBs5B,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,IAAQqrB,IACvC,MAAMxD,EAA4BpB,EAAO0qE,SAAS52F,IAAK,IACjD6xB,EAAO,IAAI,GAAYf,GAkB7B,OAhBAe,EAAKxoB,IAAK,CACT88B,MAAOvzC,EAAG,UACV0uG,KD1CJ,ybC2CI97E,UAAW,SACXi8E,SAAS,EACTP,cAAc,IAGfrvE,EAAKjzB,KAAM,OAAQ,aAAczH,GAAIm2B,EAAS,QAAS,aAGvD11B,KAAKwP,SAAUyqB,EAAM,WAAW,KAC/B3F,EAAOiB,QAAS,IAChBjB,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBhT,CAAI,GAEb,ECxCc,MAAMg3G,WAA0B,GAY9B,OAAAl8G,GACf/0B,KAAKxH,MAAQwH,KAAKkxI,YAClBlxI,KAAK00B,UAAY10B,KAAK61B,eACvB,CAYgB,OAAAN,CAAShxB,EAAoC,CAAC,GAC7D,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MACpBsrE,EAAStrE,EAAMsrE,OACfzuC,EAAY78B,EAAMoL,SAASyxB,UAE3Bk8G,EAASxwI,MAAMrB,KAAM21B,EAAU0tC,qBAE/BnqE,OAAiC,IAAvB+L,EAAQksI,YAA8BzwI,KAAKxH,MAAQ+L,EAAQksI,WAE3Er4I,EAAMmiC,QAAQC,IACb,GAAMhiC,EAEC,CACN,MAAM44I,EAAgBD,EAAOprI,QAAQi9D,GAG7BquE,GAAWruE,IAAWsuE,GAAkB5tE,EAAQV,KAGxDhjE,KAAKuxI,YAAa/2G,EAAQ42G,EAC3B,MATCpxI,KAAKwxI,aAAch3G,EAAQ22G,EAAOprI,OAAQsrI,IAS3C,GAEF,CAKQ,SAAAH,GACP,MAEMp2C,EAAa,GAFD96F,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAET0tC,qBAGpC,SAAWm4B,IAAcu2C,GAAWv2C,GACrC,CAOQ,aAAAjlE,GACP,GAAK71B,KAAKxH,MACT,OAAO,EAGR,MAAMy8B,EAAYj1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UACvCyuC,EAAS1jE,KAAKs0B,OAAOl8B,MAAMsrE,OAE3Bo3B,EAAa,GAAO7lE,EAAU0tC,qBAEpC,QAAMm4B,GAICw2C,GAAkB5tE,EAAQo3B,EAClC,CASQ,YAAA02C,CAAch3G,EAAgB22G,GAErCM,GAAwBj3G,EAAQ22G,GAAS9lI,UAAUzN,SAAS8zI,IAC3D,GAAKA,EAAWhxG,MAAMoJ,WAAa4nG,EAAW/wG,IAAI6I,QAGjD,YAFAhP,EAAO0c,OAAQw6F,EAAWhxG,MAAM9hC,QAMjC,GAAK8yI,EAAWhxG,MAAMoJ,UAAY,CACjC,MAAMkuF,EAAiBx9F,EAAOod,qBAAsB85F,EAAWhxG,MAAM9hC,QAIrE,YAFA47B,EAAO8a,KAAMo8F,EAAY1Z,EAG1B,CAIM0Z,EAAW/wG,IAAI6I,SACpBhP,EAAOz9B,MAAO20I,EAAW/wG,KAK1B,MAAMo6D,EAAgBvgE,EAAOmd,oBAAqB+5F,EAAW/wG,IAAI/hC,QAEjE47B,EAAO8a,KAAMo8F,EAAY32C,EAAe,GAE1C,CAKQ,WAAAw2C,CAAa/2G,EAAgB22G,GACpC,MAAMQ,EAAmD,GAGzDF,GAAwBj3G,EAAQ22G,GAAS9lI,UAAUzN,SAAS8zI,IAC3D,IAAIpxG,EAAQ+wG,GAAWK,EAAWhxG,OAE5BJ,IACLA,EAAQ9F,EAAOl1B,cAAe,cAE9Bk1B,EAAOsc,KAAM46F,EAAYpxG,IAG1BqxG,EAAc/xI,KAAM0gC,EAAO,IAO5BqxG,EAActmI,UAAU4iB,QAAQ,CAAE2jH,EAAcC,IAC1CD,EAAa31G,aAAe41G,GAChCr3G,EAAOo5D,MAAOp5D,EAAOmd,oBAAqBi6F,IAEnCA,GAGDC,GAET,EAGD,SAASR,GAAWS,GACnB,MAAyC,cAAlCA,EAAkBlzI,OAAQoD,KAAuB8vI,EAAkBlzI,OAAS,IACpF,CASA,SAAS6yI,GAAwBj3G,EAAgB22G,GAChD,IAAI1oG,EACAjuC,EAAI,EACR,MAAMqxC,EAAS,GAEf,KAAQrxC,EAAI22I,EAAO74I,QAAS,CAC3B,MAAM0qE,EAAQmuE,EAAQ32I,GAChBu3I,EAAYZ,EAAQ32I,EAAI,GAExBiuC,IACLA,EAAgBjO,EAAOod,qBAAsBorB,IAGxC+uE,GAAa/uE,EAAM/mC,aAAe81G,IACvClmG,EAAOjsC,KAAM46B,EAAOqd,YAAapP,EAAejO,EAAOmd,oBAAqBqrB,KAC5Ev6B,EAAgB,MAGjBjuC,GACD,CAEA,OAAOqxC,CACR,CAKA,SAASylG,GAAkB5tE,EAAgBV,GAE1C,MAAMgvE,EAActuE,EAAOyF,WAAYnG,EAAMpkE,OAAmB,cAC1DqzI,EAAqBvuE,EAAOyF,WAAY,CAAE,QAAS,cAAgBnG,GAEzE,OAAOgvE,GAAeC,CACvB,CCvMe,MAAMC,WAA0B,GAI9C,qBAAkB57G,GACjB,MAAO,mBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GAAO,GACjB,CAKO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdovC,EAASpvC,EAAOl8B,MAAMsrE,OAE5BpvC,EAAO0qE,SAASnxF,IAAK,aAAc,IAAIojI,GAAmB38G,IAE1DovC,EAAOwR,SAAU,aAAc,CAC9BsE,eAAgB,eAGjBllD,EAAOj2B,WAAWgqE,iBAAkB,CAAEjwE,MAAO,aAAc6hC,KAAM,eAGjE3F,EAAOl8B,MAAMoL,SAASk3B,mBAAmBF,IACxC,MAAMylC,EAAU3rC,EAAOl8B,MAAMoL,SAASm8D,OAAOQ,aAE7C,UAAYznD,KAASunD,EACpB,GAAmB,UAAdvnD,EAAMzN,KAAmB,CAC7B,MAAMwM,EAAUiB,EAAM+M,SAASykB,UAE/B,IAAMzyB,EAEL,SAGD,GAAKA,EAAQvJ,GAAI,UAAW,eAAkBuJ,EAAQ0qB,QAIrD,OAFA3H,EAAOv0B,OAAQwR,IAER,EACD,GAAKA,EAAQvJ,GAAI,UAAW,gBAAmBw1D,EAAOyF,WAAYzwD,EAAM+M,SAAUhO,GAIxF,OAFA+iB,EAAO0c,OAAQz/B,IAER,EACD,GAAKA,EAAQvJ,GAAI,WAAc,CAErC,MAAM4W,EAAQ0V,EAAOud,cAAetgC,GAEpC,UAAYE,KAASmN,EAAMonB,WAC1B,GACCv0B,EAAMzJ,GAAI,UAAW,gBACpBw1D,EAAOyF,WAAY3uC,EAAOod,qBAAsBjgC,GAASA,GAI1D,OAFA6iB,EAAO0c,OAAQv/B,IAER,CAGV,CACD,MAAO,GAAmB,UAAde,EAAMzN,KAAmB,CACpC,MAAMrM,EAAS8Z,EAAM+M,SAAS7mB,OAE9B,GAAKA,EAAOsP,GAAI,UAAW,eAAkBtP,EAAOujC,QAInD,OAFA3H,EAAOv0B,OAAQrH,IAER,CAET,CAGD,OAAO,CAAK,IAGb,MAAM40D,EAAexzD,KAAKs0B,OAAO6qE,QAAQllE,KAAKz2B,SACxCyxB,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAClCk9G,EAAuC79G,EAAO0qE,SAAS52F,IAAK,cAIlEpI,KAAKwP,SAAkCgkD,EAAc,SAAS,CAAEn+C,EAAKvS,KACpE,IAAMmyB,EAAU+V,cAAgBmnG,EAAkB35I,MACjD,OAGsBy8B,EAAUuY,kBAAmB5uC,OAEhCujC,UACnB7N,EAAOiB,QAAS,cAChBjB,EAAO6qE,QAAQllE,KAAKo7B,uBAEpBvyD,EAAKowB,iBACL7d,EAAIlJ,OACL,GACE,CAAEoB,QAAS,eAIdvN,KAAKwP,SAAmCgkD,EAAc,UAAU,CAAEn+C,EAAKvS,KACtE,GAAuB,YAAlBA,EAAKyhB,YAA4B0Q,EAAU+V,cAAgBmnG,EAAmB35I,MAClF,OAGD,MAAMi9C,EAAiBxgB,EAAUuY,kBAAmB5uC,OAE/C62C,EAAetT,UAAYsT,EAAe7uB,kBAC9C0N,EAAOiB,QAAS,cAChBjB,EAAO6qE,QAAQllE,KAAKo7B,uBAEpBvyD,EAAKowB,iBACL7d,EAAIlJ,OACL,GACE,CAAEoB,QAAS,cACf,E,eC7IG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQusB,OCUR,MAAMs4G,WAAqB,GAIzC,qBAAkB97G,GACjB,MAAO,cACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EAEjBs5B,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,cAAcqrB,IAC7C,MAAMxD,EAA6BpB,EAAO0qE,SAAS52F,IAAK,cAClD+iG,EAAa,IAAI,GAAYjyE,GAkBnC,OAhBAiyE,EAAW15F,IAAK,CACf88B,MAAOvzC,EAAG,eACV0uG,KAAM,GAAMppE,MACZupE,SAAS,EACTP,cAAc,IAIf6B,EAAWnkG,KAAM,OAAQ,aAAczH,GAAIm2B,EAAS,QAAS,aAG7D11B,KAAKwP,SAAU27F,EAAY,WAAW,KACrC72E,EAAOiB,QAAS,cAChBjB,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBk+D,CAAU,GAEnB,ECzCc,MAAMknC,WAAgB,GAIpC,qBAAkB/7G,GACjB,MAAO,SACR,CAKO,SAAAmxF,GACN,MAAMnzF,EAASt0B,KAAKs0B,OAEdoB,EAAoCpB,EAAO0qE,SAAS52F,IAAK,SAG/D,IAAMstB,EACL,OAGD,MAAM16B,EAAIs5B,EAAOt5B,EACQs5B,EAAO6E,GAAG61E,iBAElBnhG,IAAK,SAASqrB,IAC9B,MAAMs8E,EAAS,IAAI,GAAYt8E,GAc/B,OAZAs8E,EAAO/jG,IAAK,CACX88B,MAAOvzC,EAAG,qBACV0uG,KC/CJ,6cDgDIG,SAAS,IAGV2L,EAAOxuG,KAAM,OAAQ,aAAczH,GAAIm2B,EAAS,QAAS,aAEzD8/E,EAAOnmG,GAAI,WAAW,KACrBilB,EAAOiB,QAAS,QAAS,IAGnBigF,CAAM,GAEf,EExCM,SAAS88B,GAAcC,GAQ7B,MAAMC,EAAgC,GACtC,IAAIxkC,EAAW,EAEf,UAAYj3G,KAAOw7I,EAAY,CAC9B,MAAM1vH,EAAQzlB,SAAUrG,EAAK,IAEvB+nH,MAAOj8F,KACPA,EAAQmrF,IACZA,EAAWnrF,GAGZ2vH,EAAe5yI,KAAM,GAAI2yI,EAAWx7I,MAAWA,MAEjD,CAEA,MAAM07I,EAAe,CAAE,CACtBC,OAAQF,EAAet1I,KAAM,KAC7By1I,MAAO,eAAgB3kC,eAAwBA,MAC/C/iG,KAAM,eAGP,MAAO,CACN2nI,iBAAkBL,EAAUzD,QAC5B2D,eAEF,CCAe,MAAMI,WAAqB,GA8BzC,WAAA9wI,CAAauyB,GACZ9mB,MAAO8mB,GAXR,KAAgBw+G,cAAgB,IAAIl/H,IAKpC,KAAQm/H,SAA2B,KAQlC/yI,KAAKuuI,gBACN,CAKgB,OAAAx5G,GACf/0B,KAAKxH,MAAQwH,KAAKkxI,YAClBlxI,KAAK00B,UAAY10B,KAAK61B,eACvB,CAKgB,OAAAN,GACfv1B,KAAKwQ,KAA0B,aAChC,CAQQ,SAAA0gI,GACP,OAAyB,OAAlBlxI,KAAK+yI,QACb,CAKQ,aAAAl9G,GACP,MAAMm9G,EAAehzI,KAAKs0B,OAAO0qE,SAAS52F,IAAK,eACzC6qI,EAAcjzI,KAAKs0B,OAAO0qE,SAAS52F,IAAK,QAE9C,SAAM4qI,EAAat+G,YAAcu+G,EAAYv+G,UAK9C,CAaQ,eAAAw+G,GACP,MACMC,EADSnzI,KAAKs0B,OACOtL,OAAO5gB,IAAK,SAEvC,MAAO,CACNgrI,MAAOD,EAAYC,MACnB1kH,SAAUykH,EAAYzkH,SACtB2kH,SAAUF,EAAYE,SACtBC,cAAeH,EAAYG,cAC3BC,OAAQ,CACPC,QAAS,IAAMxzI,KAAKwQ,KAA2B,gBAEhDijI,OAAQ,CACPC,SAAYD,GAA4CzzI,KAAKwQ,KAA4B,eAAgBijI,IAG5G,CAKQ,cAAAlF,GACP,MAAMj6G,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfu7I,GAAsBr/G,EAAOtL,OAAO5gB,IAAK,sBAG/CpI,KAAKqP,GAAgB,SAAS,KAC7BrP,KAAK+0B,SAAS,GACZ,CAAEnoB,SAAU,QAGf5M,KAAKqP,GAAwB,cAAc,KACpCrP,KAAK00B,YAAa10B,KAAKxH,QAI7BwH,KAAK+yI,SAAW,GAAevvI,SAAU,MAAO,CAAE+6B,MAAO,qBACzD/6B,SAASgiB,KAAK7f,YAAa3F,KAAK+yI,UAEhCxvI,OAAOqwI,MAAMC,MAAO7zI,KAAK+yI,SAAU/yI,KAAKkzI,mBAAmB,IAI5DlzI,KAAKqP,GAAyB,eAAe,KACtCrP,KAAKxH,QAIXwH,KAAK+yI,SAAU9sI,SACfjG,KAAK+yI,SAAW,KAAI,IAIrB/yI,KAAKqP,GAA0B,gBAAgB,CAAEgG,EAAKo+H,KACrD,IAAMzzI,KAAK00B,UACV,OAGD,MAAMs+G,EAAe1+G,EAAO0qE,SAAS52F,IAAK,eACpC6qI,EAAc3+G,EAAO0qE,SAAS52F,IAAK,QAEnC0rI,EAiHT,UACC,OAAEL,EAAM,eAAEM,EAAc,cAAEC,IAM1B,OAAOP,EACLz2I,KAAKi3I,GA6CR,SAAkBA,GACjB,MAAMC,EAAWD,EAAMnxI,KAAKoxI,SAE5B,IAAMA,EACL,OAAO,EAGR,OAAOA,EAASrxH,OAASqxH,EAASpxH,MACnC,CArDiBqxH,CAASF,GACvB,CACCp0I,GAAIo0I,EAAMnxI,KAAKjD,GACfoL,KAAM,QACN1F,WAAY6uI,GAA6BH,IAE1C,CACCp0I,GAAIo0I,EAAMnxI,KAAKjD,GACfoL,KAAM,OACN1F,WAAY8uI,GAA4BJ,MAGzCluI,QAAQkuI,GAAwB,UAAfA,EAAMhpI,KAAmB8oI,EAAiBC,GAC9D,CAtI2BM,CAAe,CACtCb,SACAM,eAAgBf,EAAat+G,UAC7Bs/G,cAAef,EAAYv+G,YAGI,IAA3Bo/G,EAAgBx7I,QAKrBF,EAAMmiC,QAAQC,IACb,UAAYy5G,KAASH,EAAkB,CACtC,MAAMS,EAAcN,IAAUH,EAAiBA,EAAgBx7I,OAAS,GAExE0H,KAAKw0I,aAAcP,EAAOM,EAAa/5G,GAIlCm5G,IACJ5gH,YAAY,IAAM/yB,KAAK8yI,cAAcnhI,OAAQsiI,IAxLd,KA0L/Bj0I,KAAK8yI,cAAcjlI,IAAKomI,GAE1B,IACE,IAIJj0I,KAAKwP,SAAU8kB,EAAQ,WAAW,KACjCt0B,KAAKwQ,KAA2B,eAChCxQ,KAAK8yI,cAAclhI,OAAO,GAE5B,CASQ,YAAA4iI,CACPP,EACAM,EACA/5G,GAEA,MAEMvF,EAFSj1B,KAAKs0B,OACCl8B,MACGoL,SAASyxB,UAGjCuF,EAAOskD,yBAA0B,YAEb,UAAfm1D,EAAMhpI,KACVjL,KAAKy0I,aAAcR,GAEnBj0I,KAAK00I,YAAaT,EAAOz5G,GAKpB+5G,GACL/5G,EAAOgZ,aAAcve,EAAUuY,kBAEjC,CAOQ,YAAAinG,CAAcR,GACrB,MAAM3/G,EAASt0B,KAAKs0B,QACd,iBAAEs+G,EAAgB,aAAEH,EAAY,qBAAEkC,GAAyBV,EAAM1uI,WAEvE+uB,EAAOiB,QAAS,cAAe,CAC9BryB,OAAQ,CACP0xI,IAAKhC,EACL3vI,QAASwvI,EACTvmH,IAAKyoH,IAGR,CAQQ,WAAAD,CAAaT,EAAiCz5G,GACrD,MAAMlG,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf68B,EAAY78B,EAAMoL,SAASyxB,WAC3B,SAAE4/G,EAAQ,SAAEC,GAAab,EAAM1uI,WAGrC,GAAK0vB,EAAU+V,YAAc,CAC5B,MAAM+pG,EAAsB,GAAO9/G,EAAU+Q,iBACvCvI,EAAWjD,EAAOmZ,WAAYkhG,EAAUE,GACxCjwH,EAAQ1sB,EAAMqjG,cAAeh+D,GAEnCjD,EAAOgZ,aAAc1uB,EACtB,CAEAwP,EAAOiB,QAAS,OAAQu/G,EACzB,EAkCD,SAASV,GAA6BH,GACrC,MAAM,iBAAErB,EAAgB,aAAEH,GAAiBH,GAAc2B,EAAMnxI,KAAKyvI,WAEpE,MAAO,CACNK,mBACAH,eACAkC,qBAAsBV,EAAMnxI,KAAKoxI,SAAUj1E,aAAe,GAE5D,CAOA,SAASo1E,GAA4BJ,GACpC,MAAO,CACNY,SAAUZ,EAAMnxI,KAAKd,KACrB8yI,SAAUE,GAAaf,GAEzB,CAoBA,SAASe,GAAaf,GACrB,MAAM5F,EAAM,IAAI4G,IAAKhB,EAAMnxI,KAAKurI,KAIhC,OAFAA,EAAI6G,aAAazjI,IAAK,WAAY,QAE3B48H,EAAI3xI,UACZ,C,gNChWe,MAAMy4I,WAA2B,GAI/C,mBAAkBj+G,GACjB,MAAO,CAAE,qBAAsB,sBAAuB,GAAgBk+G,GACvE,CAKA,qBAAkB9+G,GACjB,MAAO,oBACR,CAKa,SAAAmxF,GAA2B,iCACvC,MAAMnzF,EAASt0B,KAAKs0B,OAEd+gH,IAAqB/gH,EAAOtL,OAAO5gB,IAAK,SACxCktI,IAAoB/xI,OAAOqwI,MAKjC,IAAMyB,IAAqBC,EAC1B,OAGD,MAAMC,EAAiBjhH,EAAOqC,QAAQvuB,IAAK,IACrCotI,EAAelhH,EAAOqC,QAAQvuB,IAAKgtI,IAEzCG,EAAezJ,oBAAsBC,GAC7B,IAAI,GAASA,EAAQyJ,EAAaC,WAAYnhH,GAGtD,MAAMq/G,GAAsBr/G,EAAOtL,OAAO5gB,IAAK,sBACzCstI,EAAyCphH,EAAOqC,QAAQvuB,IAAK,sBAG9DurI,GACJ+B,EAAmBrmI,GAA8B,kBAAkB,CAAEgG,GAAOsgI,eAAc7yI,WACzFwxB,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAO/0B,aAAc,eAAgB3C,EAAK8yI,aAAcD,EAAc,GACpE,GAGN,KAMD,MAAM,GA6BL,WAAA5zI,CAAagqI,EAAoB5tB,EAAyB7pF,GACzDt0B,KAAK+rI,OAASA,EACd/rI,KAAKm+G,MAAQA,EACbn+G,KAAKs0B,OAASA,EACdt0B,KAAK61I,WAAa,IAAIC,gBAEtB91I,KAAKszI,cAAgBh/G,EAAOtL,OAAO5gB,IAAK,sBACzC,CAKO,cAAA2tI,GACN,MACMC,GAAoCh7I,EADhCgF,KAAKs0B,OAAOt5B,GACuB,oCACvCi7I,EAAqBj2I,KAAKs0B,OAAOtL,OAAO5gB,IAAK,kCAC7C8tI,EF5ED,SAAyB/3B,EAAyB83B,GACxD,MAAQ,CAAEE,GAAuBh4B,EAAM3lH,MAAMuE,MAAO,KAC9Cq5I,EAAUxzI,KAAK+7E,MAAOj7E,KAAMyyI,IAC5BE,EAAeD,EAAQE,MAAQF,EAAQE,KAAKC,OAASH,EAAQE,KAAKC,MAAMF,YAAgB,CAAED,EAAQI,KAExG,OAAMP,EAMO,eAFAG,EAAQE,MAAQF,EAAQE,KAAKC,OAASH,EAAQE,KAAKC,MAAMntC,OAEzCitC,EAAW9mH,SAAU0mH,GAC1CA,EAGD,KATCI,EAAY,EAUrB,CE4DsBN,CAAgB/1I,KAAKm+G,MAAO83B,GAEhD,GAAoB,MAAfC,EAQJ,MAFAznI,EAAU,wCAEJunI,EAGP,OAAOE,CACR,CAOa,sBAAAO,CAAwB76H,EAAiB,GAAuC,iCAC5F,MACM86H,EAAc,IAAIzB,IAAK,aAAcj1I,KAAKszI,eAMhD,OAJAoD,EAAYxB,aAAazjI,IAAK,QAHJ,IAG+B/U,YACzDg6I,EAAYxB,aAAazjI,IAAK,SAAUmK,EAAOlf,YAC/Cg6I,EAAYxB,aAAazjI,IAAK,cAAezR,KAAK+1I,kBAE3C/1I,KAAK22I,iBAAkB,CAAEtI,IAAKqI,IACnC5+G,MAAYh1B,GAAQ,0BAGpB,GAFuBA,EAAK8zI,YAAeh7H,EATnB,IAWF,EAAI,CACzB,MAAMi7H,QAAoB72I,KAAKy2I,uBAAwB76H,EAZhC,IAcvB,MAAO,IACH9Y,EAAK8tB,SACLimH,EAEL,CAEA,OAAO/zI,EAAK8tB,KACb,MACC26F,OAAO,KACPvrH,KAAK61I,WAAWiB,OAAOC,iBAOvBtoI,EAAU,kCAAmC,GAEhD,IAKa,oBAAAuoI,CAAsBlM,GAAqC,iCACvE,MAAMmM,EAmLR,SAA2Bz+I,GAC1B,MAAM0+I,EAAkB,WAAC,qBAGzB,OAFc1+I,EAAMqE,MAAOq6I,GAEbjhG,OAAQ4xD,IAAIj/F,aAC3B,CAxLoBuuI,CAAkBrM,EAAK9oI,MACnCo1I,QAAsBp3I,KAAKy2I,yBAGjC,IAAMW,EACL,OAAO,KAIR,MAAMC,EAAoBr3I,KAAKs0B,OAAOtL,OAAO5gB,IAAK,iCAGlD,GAAKivI,EAAoB,CACxB,MAAMC,EAAetgJ,OAAOC,KAAMogJ,GAAoB9lH,MAAMgmH,GACpDF,EAAmBE,GAAWhmH,MAAMrtB,GAAKA,EAAE0E,eAAiBquI,MAIpE,GAAKK,EAAe,CACnB,MAAME,EAAiBJ,EAAc7lH,MAAMgmH,GAAYA,EAAS13I,KAAOy3I,GAAgBC,EAASv1I,OAASs1I,IAEzG,OAAME,EAICA,EAAe33I,GAHd,IAIT,CACD,CAGA,MAAM03I,EAAWH,EAAc7lH,MAAMgmH,GAAYA,EAASE,WAAWlmH,MAAMrtB,GAAKA,EAAE0E,eAAiBquI,MAEnG,OAAMM,EAICA,EAAS13I,GAHR,IAIT,IAOa,MAAAktI,GAAkC,iCAC9C,MAAM/xI,EAAIgF,KAAKs0B,OAAOt5B,EAChB08I,EAA0B18I,EAAG,sDAC7B8vI,QAAe9qI,KAAK+rI,OAAOjB,KAC3ByM,QAAiBv3I,KAAKg3I,qBAAsBlM,GAElD,IAAMyM,EACL,OAAO9+G,QAAQuyG,OAAQ0M,GAGxB,MAAMC,EAAY,IAAI1C,IAAK,SAAUj1I,KAAKszI,eACpCsE,EAAW,IAAI5I,SAErB2I,EAAUzC,aAAazjI,IAAK,cAAezR,KAAK+1I,kBAEhD6B,EAAS/xF,OAAQ,aAAc0xF,GAC/BK,EAAS/xF,OAAQ,OAAQilF,GAEzB,MAAM+M,EAAgB,CACrBt/G,OAAQ,OACR81G,IAAKsJ,EACL70I,KAAM80I,EACNE,iBAAoBziI,IAEdA,EAAI05H,mBACR/uI,KAAK+rI,OAAOI,YAAc92H,EAAI01H,MAC9B/qI,KAAK+rI,OAAOL,SAAWr2H,EAAIu1H,OAC5B,GAIF,OAAO5qI,KAAK22I,iBAAkBkB,GAC5B//G,MAAYh1B,GAAQ,0BACpB,MAAMyvI,EAAYD,GAAcxvI,EAAKyvI,WAErC,MAAO,CACNqD,aAAc9yI,EAAKjD,GACnBivI,QAASyD,EAAUK,iBACnB3vI,QAASsvI,EAAUE,aAErB,MACClnB,OAAO,KACP,MAAMqjB,EAAe5zI,EAAG,uBAA0B,IAAK8vI,EAAK9oI,QAE5D,OAAOy2B,QAAQuyG,OAAQ4D,EAAc,GAExC,IAOO,KAAAvD,GACNrrI,KAAK61I,WAAWxK,OACjB,CAUQ,gBAAAsL,EAAkB,IAAEtI,EAAG,OAAE91G,EAAS,WAAOz1B,EAAI,iBAAEg1I,IAMtD,MAAMhB,EAAS92I,KAAK61I,WAAWiB,OAEzBrI,EAAM,IAAIC,eAChBD,EAAInB,KAAM/0G,EAAQ81G,EAAI3xI,YAAY,GAClC+xI,EAAIsJ,iBAAkB,gBAAiB/3I,KAAKm+G,MAAM3lH,OAClDi2I,EAAIsJ,iBAAkB,gBAAiB,cACvCtJ,EAAIE,aAAe,OAGnB,MAAMqJ,EAAgB,KACrBvJ,EAAIpD,OAAO,EAGZ,OAAO,IAAI5yG,SAAc,CAAElc,EAASyuH,KACnC8L,EAAOxmI,iBAAkB,QAAS0nI,GAElCvJ,EAAIn+H,iBAAkB,aAAa,KAClCwmI,EAAOxmI,iBAAkB,QAAS0nI,EAAe,IAGlDvJ,EAAIn+H,iBAAkB,WAAW,KAChCwmI,EAAOvmI,oBAAqB,QAASynI,EAAe,IAGrDvJ,EAAIn+H,iBAAkB,SAAS,KAC9B06H,GAAQ,IAGTyD,EAAIn+H,iBAAkB,SAAS,KAC9B06H,GAAQ,IAGTyD,EAAIn+H,iBAAkB,QAAQ,IAAY,0BACzC,MAAMu+H,EAAWJ,EAAII,SAErB,OAAMA,GAAYA,EAASoJ,YAAc,IACjCjN,EAAQ6D,GAAYA,EAASzgI,SAG9BmO,EAASsyH,EACjB,MAGKiJ,GACJrJ,EAAI1B,OAAOz8H,iBAAkB,YAAY+E,IACxCyiI,EAAkBziI,EAAK,IAKzBo5H,EAAIQ,KAAMnsI,EAAM,GAElB,ECtUc,MAAMsyI,WAAqB,GASzC,qBAAkB9+G,GACjB,MAAO,cACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,gBAAiB,cAAe,iBAAkBi+G,GAC5D,CAKa,IAAAz+G,GAAsB,O,EAAA,K,EAAA,K,EAAA,YAClC,MAAMpC,EAASt0B,KAAKs0B,OACd+gH,IAAqB/gH,EAAOtL,OAAO5gB,IAAK,SACxCktI,IAAoB/xI,OAAOqwI,MAIjC,IAAMyB,IAAqBC,EAC1B,OAGDt1I,KAAKk4I,cAEL,MAAMC,EAAuC7jH,EAAOqC,QAAQvuB,IAAK,qBAC3DgwI,EAAgB9jH,EAAOtL,OAAO5gB,IAAK,kBAKzC,GAAKgwI,IAJyB9jH,EAAOtL,OAAO5gB,IAAK,0BAIF,CAC9C,MAAMiwI,EAA+B/jH,EAAOqC,QAAQvuB,IAAK,iBAEzDpI,KAAKs4I,OAASD,EAAcl6B,KAC7B,MAGCn+G,KAAKs4I,aAAeH,EAAkBI,YAAaH,GAAgB1hH,OAK9DpC,EAAOtL,OAAO5gB,IAAK,wBACxBpI,KAAKw4I,cACLx4I,KAAKy4I,kBACLz4I,KAAK04I,eAIDpD,GACJhhH,EAAO0qE,SAASnxF,IAAK,QAAS,IAAIglI,GAAcv+G,GAElD,E,yMAAA,CAKO,QAAAmhH,GACN,OAAOz1I,KAAKs4I,MACb,CAKQ,WAAAJ,GACP,MAAM5jH,EAASt0B,KAAKs0B,OAEpBA,EAAOtL,OAAOtyB,OAAQ,QAAS,CAC9B48I,cAAe,uBACfqF,wBAAyB,KACzBC,cAAc,EACdlqH,SAAU4F,EAAO4E,OAAOzJ,WACxB2jH,MAAO,UACPC,SAAU/+G,EAAOtL,OAAO5gB,IAAK,4BAK9B,IAFiBksB,EAAOtL,OAAO5gB,IAAK,kBAoBnC,MAAM,IAAI,EAAe,iCAAkCpI,MAGtDs0B,EAAOqC,QAAQ/oB,IAAK,sBAA0B0mB,EAAOqC,QAAQ/oB,IAAK,uBAavEa,EAAU,qCAAsC6lB,EAElD,CAKQ,WAAAkkH,GACP,MACM90E,EADS1jE,KAAKs0B,OACEl8B,MAAMsrE,OAE5BA,EAAOxwB,OAAQ,QAAS,CAAE0iC,gBAAiB,gBAEtClS,EAAO8R,aAAc,eACzB9R,EAAOxwB,OAAQ,aAAc,CAAE0iC,gBAAiB,CAAE,eAAgB,iBAG9DlS,EAAO8R,aAAc,gBACzB9R,EAAOxwB,OAAQ,cAAe,CAAE0iC,gBAAiB,CAAE,eAAgB,iBAGpElS,EAAOyS,mBAAmB,CAAE5oE,EAASq5B,KAGpC,MAFiBr5B,EAAQmjB,KAAK2K,aAAc,aAEV,gBAAlBuL,EACf,OAAO,CACR,GAEF,CAKQ,eAAA6xG,GACP,MAAMnkH,EAASt0B,KAAKs0B,OAGpBA,EAAOj2B,WAAWuhF,IAAK,YAAa/xE,KAAK2xD,IAExCA,EAAWnwD,GAAqC,oCAAoC,CAAEgG,EAAKvS,EAAMw8D,KAChG,MAAM,OAAE9kC,EAAM,OAAEgjC,EAAM,WAAEsD,GAAexB,EAEvC,IAAMwB,EAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACxC,OAGD,MACM62I,EAAc,IADDr7E,EAAOf,cAAe35D,EAAK7C,MACTu7B,eACnCjK,MAAQ5Z,GAA+B,MAAfA,EAAM3V,OAG1B62I,IAID/1I,EAAK7C,KAAKgmC,aAAc,eAC5BzL,EAAO/0B,aAAc,yBAA0B3C,EAAK7C,KAAKo7B,aAAc,eAAiBw9G,GAExFr+G,EAAO9zB,gBAAiB,yBAA0BmyI,GACnD,GACE,CAAEjsI,SAAU,QAEf4yD,EAAWnwD,GAA4B,yBAAyB,CAAEgG,EAAKvS,EAAMw8D,KAC5E,MAAM,OAAE9kC,EAAM,OAAEgjC,EAAM,WAAEsD,GAAexB,EAEvC,GAAMwB,EAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MAAzC,CAKA,GAAKc,EAAK09D,kBAAoB,CAC7B,MAAMjpB,EAAcuhG,GAAmBt+G,EAAQ13B,EAAK09D,mBAEpDhmC,EAAO0c,OAAQsmB,EAAOH,YAAav6D,EAAKgiB,OAASyyB,EAClD,CAGA,GAAKz0C,EAAK29D,kBAAoB,CAC7B,MAAMlpB,EAAcuhG,GAAmBt+G,EAAQ13B,EAAK29D,mBAEpD,GAAK39D,EAAK7C,KAAKiO,GAAI,aAAgB,CAClC,MAAM8oC,EAAgBxc,EAAOh3B,SAASyxB,UAEtCuF,EAAOsc,KAAME,EAAc3J,gBAAkBkK,EAC9C,MACC/c,EAAOsc,KAAM0mB,EAAOH,YAAav6D,EAAKgiB,OAASyyB,EAEjD,CApBA,CAoBA,GACE,CAAE3qC,SAAU,OAAS,IAOzB0nB,EAAOj2B,WAAWuhF,IAAK,UAAW/xE,KAAK2xD,IACtCA,EAAWnwD,GAAwB,aAAa,CAAEgG,EAAKvS,EAAMw8D,KAC5D,MAAM,OAAE9kC,EAAM,WAAEsmC,GAAexB,EAG/B,IAAMx8D,EAAKgtE,SAASz0C,aAAc,QACjC,OAKD,IAAMylC,EAAWrC,QAAS37D,EAAKgtE,SAFF,CAAEvqE,WAAY,CAAE,4BAG5C,OAGD,MAAMm/C,EAAiB5hD,EAAKgtE,SAASz0C,aAAc,0BAGnD,GAAMqpB,EAIN,GAAK5hD,EAAKw6D,WAGT,QAAUr9D,KAAQ6C,EAAKw6D,WAAWpxB,WAC5BjsC,EAAKiO,GAAI,gBACbjO,EAAOA,EAAKw9B,UAKRs7G,GAA8B94I,IAClCu6B,EAAO/0B,aAAc,cAAei/C,EAAgBzkD,OAGhD,CAEN,MAAMk8D,EAAer5D,EAAK+sE,YAAY1lC,YAAcrnC,EAAK+sE,YAAYjxE,OAErE47B,EAAO/0B,aAAc,cAAei/C,EAAgByX,EACrD,IACE,CAAEvvD,SAAU,OAAS,IAIzB0nB,EAAOj2B,WAAWuhF,IAAK,YAAarV,qBAAsB,CACzDnyE,MAAO,eACP6hC,KAAM,2BAIP3F,EAAOj2B,WAAWuhF,IAAK,UAAWzR,mBAAoB,CACrD/1E,MAAO,CACNrB,IAAK,eACLyB,MAAS++C,GAA8BA,EAAYlc,aAAc,2BAElEpB,KAAM,CACL10B,WAAY,CACX,yBAA0B,aAI9B,CAKQ,WAAAmzI,GACP,MAAMpkH,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf68B,EAAY78B,EAAMoL,SAASyxB,UAGjC78B,EAAMoL,SAASk3B,kBAUjB,SAA8BpG,GAC7B,OAASkG,IACR,IAAIosC,GAAU,EAEd,MAAMxuE,EAAQk8B,EAAOl8B,MACf4gJ,EAA6B1kH,EAAO0qE,SAAS52F,IAAK,SAIxD,IAAM4wI,EACL,OAAOpyE,EAGR,UAAYluD,KAAStgB,EAAMoL,SAASm8D,OAAOQ,aAAe,CACzD,GAAoB,WAAfznD,EAAMzN,MAAoC,cAAfyN,EAAMzN,KACrC,SAGD,MAAM6Z,EAAuB,WAAfpM,EAAMzN,KACnB,IAAI,GAAOyN,EAAM+M,SAAU/M,EAAM+M,SAAS6kB,aAAc5xB,EAAMpgB,SAC9DogB,EAAMoM,MAEDm0H,EAA4C,cAAfvgI,EAAMzN,MACjB,aAAvByN,EAAM+rC,cACsB,OAA5B/rC,EAAM+nD,kBAEP,UAAYxgE,KAAQ6kB,EAAMonB,WAAa,CAEtC,GAAK+sG,GAA8Bh5I,EAAKgmC,aAAc,eAAkB,CACvEzL,EAAO9zB,gBAAiB,cAAezG,GAEvC2mE,GAAU,EAEV,QACD,CAIA,MAAM6sE,EAASyF,GAAmBj5I,EAAM+4I,EAAalG,eAErD,UAAYmB,KAASR,EAAS,CAC7B,MAAM7sG,EAA+B,UAAfqtG,EAAMhpI,KAAmB,eAAiB,cAE3DgpI,EAAMp0I,KAAOI,EAAKo7B,aAAcuL,KAIrCpM,EAAO/0B,aAAcmhC,EAAeqtG,EAAMp0I,GAAII,GAE9C2mE,GAAU,EACX,CACD,CACD,CAEA,OAAOA,CAAO,CAEhB,CAlEoCuyE,CAAqB7kH,IAGvDl8B,EAAMoL,SAASk3B,kBAoEjB,SAAmCzF,GAClC,OAASuF,KAC6BvF,EAAUgR,aAAc,cAAgBhR,EAAUgR,aAAc,kBAGpGzL,EAAOskD,yBAA0B,gBAE1B,EAKV,CAhFoC,CAA0B7pD,GAC7D,EAyFD,SAASikH,GAAmBj5I,EAAYwzI,GACvC,MAAM2F,EAAiBn5I,EAAKiO,GAAI,UAAW,gBAAmBjO,EAAKiO,GAAI,UAAW,cAC5EmrI,EAAgBp5I,EAAKgmC,aAAc,YAEzC,MAAO,IAAKwtG,GAAS1tI,QAAQkuI,GACR,UAAfA,EAAMhpI,MAAoBmuI,EACvBnF,EAAM1uI,WAAWqtI,mBAAqB3yI,EAAKo7B,aAAc,OAG7C,SAAf44G,EAAMhpI,MAAmBouI,EACtBpF,EAAM1uI,WAAWuvI,WAAa70I,EAAKo7B,aAAc,iBADzD,GAIF,CAKA,SAASy9G,GAAmBt+G,EAAwB36B,GAEnD,MAAM03C,EAAc/c,EAAOoZ,uBAAwB,IAAK,CAAE,yBAA0B/zC,GAAM,CAAE+M,SAAU,IAItG,OAFA4tB,EAAOoa,kBAAmB,QAAQ,EAAM2C,GAEjCA,CACR,CAKA,SAASwhG,GAA8B55I,GACtC,QAAKA,EAAK+O,GAAI,aAIT/O,EAAK+O,GAAI,UAAW,iBAAmB/O,EAAK+O,GAAI,UAAW,cAKjE,CC3be,MAAMorI,WAAmB,GAIvC,qBAAkBhjH,GACjB,MAAO,YACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd06E,EAAmB16E,EAAO6E,GAAG61E,iBAC7Bh0G,EAAIs5B,EAAOt5B,EAEjBg0G,EAAiBnhG,IAAK,YAAYqrB,IACjC,MAAMxD,EAA2BpB,EAAO0qE,SAAS52F,IAAK,YAEhDotG,EAAS,IAAI,GAAYt8E,GAe/B,OAbAs8E,EAAO/jG,IAAK,CACX88B,MAAOvzC,EAAG,wBACV0uG,KC1CJ,6cD2CIG,SAAS,IAGV2L,EAAOxuG,KAAM,aAAczH,GAAIm2B,GAE/B8/E,EAAOnmG,GAAI,WAAW,KACrBilB,EAAOiB,QAAS,YAChBjB,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBuoE,CAAM,GAEf,EE1Bc,MAAM+jC,WAAwB,GAI5C,WAAAx3I,CAAauyB,GACZ9mB,MAAO8mB,GAGPt0B,KAAKg1B,aAAc,EAGnBh1B,KAAK2P,cAAe3P,KAAKs0B,OAAOl8B,MAAMoL,SAAU,UAGhDxD,KAAKwP,SAAUxP,KAAKs0B,OAAOl8B,MAAMoL,SAAU,UAAU,IAAMxD,KAAK+0B,WAAW,CAAEnoB,SAAU,OACxF,CAKgB,OAAAmoB,GACf,MAAMi+G,EAAehzI,KAAKs0B,OAAO0qE,SAAS52F,IAAK,eACzC6qI,EAAcjzI,KAAKs0B,OAAO0qE,SAAS52F,IAAK,QAG9CpI,KAAK00B,UAAYs+G,EAAat+G,WAAau+G,EAAYv+G,SACxD,CAKgB,OAAAa,GACf,MAAMjB,EAASt0B,KAAKs0B,OAEdklH,EAAex5I,KAAKs0B,OAAOtL,OAAO5gB,IAAK,0BAA6B,QAE1E,GAAqB,SAAhBoxI,GAA2C,SAAhBA,EAM/B,MAAM,IAAI,EAAe,gCAAiCllH,GAG3D,MAAM/vB,EAAUvE,KAAKs0B,OAAOtL,OAAO5gB,IAAK,qBAAwB,CAAC,EAEjE7D,EAAQk1I,aAAc,EAGtB,MAAMC,EAAiBn1I,EAAQo1I,OAGzBp1I,EAAQmqB,WACbnqB,EAAQmqB,SAAW4F,EAAO4E,OAAOzJ,YAIlClrB,EAAQo1I,OAASC,IAEXF,GACJA,EAAgBE,GAGjBA,EAAOvqI,GAAI,gBAAkBgG,IAC5B,MAAMq9C,EAAQr9C,EAAIvS,KAAK4vD,MAAMmnF,UAGvBC,EAAQpnF,EAAM3sD,QAAU+kI,IAAgBA,EAAKqJ,YAC7C4F,EAASrnF,EAAM3sD,QAAU+kI,GAAeA,EAAKqJ,YAEnD,UAAY6F,KAAYF,EACvBxlH,EAAOiB,QAAS,OAAQykH,EAASC,UAGlC,MAAMC,EAAa,GAEnB,UAAYlnF,KAAS+mF,EAAS,CAC7B,MAAM1L,EAAMr7E,EAAMinF,SAElBC,EAAWt6I,KAAMyuI,GAAYuL,EAAOO,QAAS,mBAAoB,CAAErP,KAAM93E,IAC1E,CAEKknF,EAAW5hJ,QACf8hJ,GAAc9lH,EAAQ4lH,EACvB,IAGDN,EAAOvqI,GAAI,4BAA8BgG,IACxC,MAAMglI,EAAahlI,EAAIvS,KAAKu3I,WAE5B,GAAMA,EAYND,GAAc9lH,EAAQ,CAAE+lH,QAZxB,CACC,MAAMC,EAA6BhmH,EAAOqC,QAAQvuB,IAAK,gBACjDpN,EAAIs5B,EAAO4E,OAAOl+B,EAExBs/I,EAAa91B,YAAaxpH,EAAG,uCAAyC,CACrEspH,MAAOtpH,EAAG,kCACVuc,UAAW,YAIb,CAEsC,GACpC,EAGFhU,OAAgBg3I,SAAUf,GAAgBj1I,EAC7C,EAGD,SAAS61I,GAAc9lH,EAAgBkmH,GAItC,GAHqBlmH,EAAO0qE,SAAS52F,IAAK,eAGvBssB,UAYnBJ,EAAOiB,QAAS,cAAe,CAAEryB,OAAQs3I,QAZzC,CACC,MAAMF,EAA6BhmH,EAAOqC,QAAQvuB,IAAK,gBACjDpN,EAAIs5B,EAAO4E,OAAOl+B,EAExBs/I,EAAa91B,YAAaxpH,EAAG,mDAAqD,CACjFspH,MAAOtpH,EAAG,0BACVuc,UAAW,YAIb,CAGD,CC3Ie,MAAMkjI,WAAwB,GAI5C,qBAAkBnkH,GACjB,MAAO,iBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAEgtF,GAAc,cACxB,CAKO,IAAAxtF,GACN,MAAMpC,EAASt0B,KAAKs0B,OAEpB,IAAMA,EAAOqC,QAAQ/oB,IAAK,uBAA0B0mB,EAAOqC,QAAQ/oB,IAAK,sBAavE,MAAM,IAAI,EAAe,gCAAiC0mB,GAG3DA,EAAO0qE,SAASnxF,IAAK,WAAY,IAAI0rI,GAAiBjlH,GACvD,ECnCc,MAAMomH,WAAmC,GAMvD,qBAAkBpkH,GACjB,MAAO,4BACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,gBAAiB,GAC3B,CAKO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OAEd+jH,EAA+B/jH,EAAOqC,QAAQvuB,IAAK,iBAEnD+1G,EAAQk6B,EAAcl6B,MACtBw5B,EAAYU,EAAcV,UAEhC,IAAMx5B,EACL,OAGD,MAAMg6B,EAAuC7jH,EAAOqC,QAAQvuB,IAAK,qBACjEpI,KAAK26I,eAAiBxC,EAAkByC,oBAAqBz8B,EAAOw5B,GAEpErjH,EAAOqC,QAAQvuB,IAAK,IAAiB0jI,oBAAsBC,GACnD,IAAI,GAAS/rI,KAAK26I,eAAiB5O,EAE5C,EAGD,MAAM,GAKL,WAAAhqI,CAAa84I,EAA8B9O,GAC1C/rI,KAAK66I,cAAgBA,EAErB76I,KAAK+rI,OAASA,CACf,CAEO,MAAAgB,GACN,OAAO/sI,KAAK+rI,OAAOjB,KAAKhzG,MAAMgzG,IAC7B9qI,KAAK86I,aAAe96I,KAAK66I,cAAc9N,OAAQjC,GAE/C9qI,KAAK86I,aAAazrI,GAAI,YAAY,CAAEgG,EAAKvS,KACxC9C,KAAK+rI,OAAOI,YAAcrpI,EAAKioI,MAC/B/qI,KAAK+rI,OAAOL,SAAW5oI,EAAK4oI,QAAQ,IAG9B1rI,KAAK86I,aAAa7L,SAE3B,CAEO,KAAA5D,GACNrrI,KAAK86I,aAAczP,OACpB,ECvEc,MAAM0P,WAAyB,GACtC,WAAAh5I,CAAauyB,GACnB9mB,MAAO8mB,GAGPt0B,KAAK80B,4BAA6B,CACnC,CAagB,OAAAC,GACf,MAAM38B,EAAQ4H,KAAKs0B,OAAOl8B,MAEpB4qE,EAAQ,GADG5qE,EAAMoL,SACOyxB,UAAU0tC,qBAExC3iE,KAAKxH,QAAUwqE,GAASA,EAAM90D,GAAI,UAAW,aAC7ClO,KAAK00B,YAAcsuC,GAASg4E,GAAyBh4E,EAAO5qE,EAAMsrE,OACnE,CAWgB,OAAAnuC,CAAShxB,EAErB,CAAC,GACJ,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MACpBoL,EAAWpL,EAAMoL,SAEjByxB,EAAY1wB,EAAQ0wB,WAAazxB,EAASyxB,UAG1C78B,EAAMi9B,UAAWJ,IAIvB78B,EAAMmiC,QAAQC,IACb,MAAM22G,EAASl8G,EAAU0tC,oBAEzB,UAAYK,KAASmuE,GACdnuE,EAAM90D,GAAI,UAAW,cAAiB8sI,GAAyBh4E,EAAO5qE,EAAMsrE,SACjFlpC,EAAO6c,OAAQ2rB,EAAO,YAExB,GAEF,EASD,SAASg4E,GAAyBh4E,EAAgBU,GACjD,OAAOA,EAAOyF,WAAYnG,EAAMpkE,OAAmB,eAAkB8kE,EAAOgE,SAAU1E,EACvF,CC1De,MAAMi4E,WAA+B,GAC5C,WAAAl5I,CAAauyB,GACnB9mB,MAAO8mB,GAGPt0B,KAAK80B,4BAA6B,CACnC,CAUgB,OAAAS,CAAShxB,GAIxB,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MACpBmN,EAAahB,EAAQgB,WAE3B,IAAIkgB,EAA4BlhB,EAAQkhB,SAGlCrtB,EAAMi9B,UAAW5P,IAIvBrtB,EAAMmiC,QAAQC,IAGb,GAFA/U,EAAWzlB,KAAKk7I,+BAAgCz1H,EAAW+U,IAErD/U,EACL,OAGD,MAAMuoD,EAAYxzC,EAAOl1B,cAAe,aAEnCC,GACJnN,EAAMsrE,OAAOwT,qBAAsBlJ,EAAWzoE,EAAYi1B,GAG3DpiC,EAAMqjG,cAAeztB,EAAWvoD,GAChC+U,EAAOgZ,aAAcw6B,EAAW,KAAM,GAExC,CAKQ,8BAAAktE,CAAgCz1H,EAAoB+U,GAC3D,MAAMpiC,EAAQ4H,KAAKs0B,OAAOl8B,MAE1B,GAAKA,EAAMsrE,OAAOyF,WAAY1jD,EAAU,aACvC,OAAOA,EAGR,MAAM62D,EAAgBlkF,EAAMsrE,OAAOuT,kBAAmBxxD,EAAU,aAIhE,IAAM62D,EACL,OAAO,KAGR,MAAM7mC,EAAiBhwB,EAAS7mB,OAC1B24F,EAAgBn/F,EAAMsrE,OAAOyF,WAAY1zB,EAAgB,SAK/D,OAAKA,EAAetT,SAAWo1D,GAAiB9xE,EAAS+jB,QACjDpxC,EAAMu/C,oBAAqBlC,IAK7BA,EAAetT,SAAWo1D,GAAiB9xE,EAASqkB,UAClD1xC,EAAMw/C,qBAAsBnC,GAG7Bjb,EAAOz9B,MAAO0oB,EAAU62D,GAAgB72D,QAChD,ECtFD,MAAqB,GAArB,cAAuC,GAItC,qBAAkB6Q,GACjB,MAAO,WACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MAErBk8B,EAAO0qE,SAASnxF,IAAK,YAAa,IAAIktI,GAAkBzmH,IACxDA,EAAO0qE,SAASnxF,IAAK,kBAAmB,IAAIotI,GAAwB3mH,IAGpEl8B,EAAMsrE,OAAOwR,SAAU,YAAa,CAAEsE,eAAgB,WAEtDllD,EAAOj2B,WAAWgqE,iBAAkB,CAAEjwE,MAAO,YAAa6hC,KAAM,MAGhE3F,EAAOj2B,WAAWuhF,IAAK,UAAWvX,iBAAkB,CACnDjwE,MAAO,CAAEm/C,GAAe/c,YACjB,GAAU2gH,sBAAsBvtI,IAAK2pC,EAAYv1C,MAKlDu1C,EAAYpV,QACT,KAGD3H,EAAOl1B,cAAe,aARrB,KAUT20B,KAAM,KACN8uC,kBAAmB,OAErB,GAxCD,IAAqB,GAArB,GAAqB,GA2ENoyE,sBAAwB,IAAIvnI,IAAK,CAC9C,aACA,KACA,MACA,KACA,KACA,KACA,KACA,KACA,KACA,KACA,KACA,IACA,KACA,OClGa,MAAMwnI,WAAuB,GAuB3C,WAAAr5I,CAAauyB,EAAgB+mH,GAC5B7tI,MAAO8mB,GAEPt0B,KAAKq7I,cAAgBA,CACtB,CAKgB,OAAAtmH,GACf,MAAMiuC,EAAQ,GAAOhjE,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAAU0tC,qBAE1D3iE,KAAKxH,QAAUwqE,GAAShjE,KAAKq7I,cAAc9rH,SAAUyzC,EAAMhhE,OAAUghE,EAAMhhE,KAC3EhC,KAAK00B,YAAcsuC,GAAShjE,KAAKq7I,cAAchnH,MAAMinH,GAAWC,GAAuBv4E,EAAOs4E,EAASt7I,KAAKs0B,OAAOl8B,MAAMsrE,SAC1H,CASgB,OAAAnuC,CAAShxB,GACxB,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MACpBoL,EAAWpL,EAAMoL,SAEjB24D,EAAe53D,EAAQ/L,MAE7BJ,EAAMmiC,QAAQC,IACb,MAAM22G,EAASxwI,MAAMrB,KAAMkE,EAASyxB,UAAU0tC,qBAC5C58D,QAAQi9D,GACDu4E,GAAuBv4E,EAAO7G,EAAc/jE,EAAMsrE,UAG3D,UAAYV,KAASmuE,EACdnuE,EAAM90D,GAAI,UAAWiuD,IAC1B3hC,EAAO6c,OAAQ2rB,EAAO7G,EAExB,GAEF,EAUD,SAASo/E,GAAuBv4E,EAAgBs4E,EAAiB53E,GAChE,OAAOA,EAAOyF,WAAYnG,EAAMpkE,OAAmB08I,KAAc53E,EAAOgE,SAAU1E,EACnF,CC5EA,MAAMw4E,GAAsB,YAOb,MAAMC,WAAuB,GAI3C,qBAAkBnlH,GACjB,MAAO,gBACR,CAKA,WAAAv0B,CAAauyB,GACZ9mB,MAAO8mB,GAEPA,EAAOtL,OAAOtyB,OAAQ,UAAW,CAChC6N,QAAS,CACR,CAAEnM,MAAO,YAAaksH,MAAO,YAAa/lF,MAAO,wBACjD,CAAEnmC,MAAO,WAAY6hC,KAAM,KAAMqqF,MAAO,YAAa/lF,MAAO,uBAC5D,CAAEnmC,MAAO,WAAY6hC,KAAM,KAAMqqF,MAAO,YAAa/lF,MAAO,uBAC5D,CAAEnmC,MAAO,WAAY6hC,KAAM,KAAMqqF,MAAO,YAAa/lF,MAAO,yBAG/D,CAKA,mBAAkBrH,GACjB,MAAO,CAAE,GACV,CAKO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd/vB,EAAgC+vB,EAAOtL,OAAO5gB,IAAK,mBAEnDizI,EAAgB,GAEtB,UAAYn6H,KAAU3c,EAEC,cAAjB2c,EAAO9oB,QAKZk8B,EAAOl8B,MAAMsrE,OAAOwR,SAAUh0D,EAAO9oB,MAAO,CAC3CohF,eAAgB,WAGjBllD,EAAOj2B,WAAWgqE,iBAAkBnnD,GAEpCm6H,EAAcz7I,KAAMshB,EAAO9oB,QAG5B4H,KAAK07I,wBAAyBpnH,GAG9BA,EAAO0qE,SAASnxF,IAAK,UAAW,IAAIutI,GAAgB9mH,EAAQ+mH,GAC7D,CAKO,SAAA5zB,GAGN,MAAMnzF,EAASt0B,KAAKs0B,OACdqnH,EAAernH,EAAO0qE,SAAS52F,IAAK,SACpC7D,EAAgC+vB,EAAOtL,OAAO5gB,IAAK,mBAEpDuzI,GACJ37I,KAAKwP,SAAyCmsI,EAAc,gBAAgB,CAAEtmI,EAAKvS,KAClF,MAAM2yC,EAAiBnhB,EAAOl8B,MAAMoL,SAASyxB,UAAUE,mBAAoBv2B,OACzD2F,EAAQ8vB,MAAMnT,GAAUu0B,EAAevnC,GAAI,UAAWgT,EAAO9oB,WAE5Dq9C,EAAevnC,GAAI,UAAWstI,KAAuD,IAA9B/lG,EAAena,YACxFx4B,EAAK03B,OAAO6c,OAAQ5B,EAAgB+lG,GACrC,GAGH,CAOQ,uBAAAE,CAAyBpnH,GAChCA,EAAOj2B,WAAWuhF,IAAK,UAAWvX,iBAAkB,CACnDjwE,MAAO,WACP6hC,KAAM,KAGN8uC,kBAAmB,EAAW/7D,IAAM,GAEtC,E,eCtHG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ8sB,OCgBR,MAAM8hH,WAAkB,GAItC,qBAAkBtlH,GACjB,MAAO,WACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACXuJ,ECrBD,SAA8B+vB,GACpC,MAAMt5B,EAAIs5B,EAAOt5B,EACX6gJ,EAA0C,CAC/C,UAAa7gJ,EAAG,aAChB,YAAaA,EAAG,aAChB,YAAaA,EAAG,aAChB,YAAaA,EAAG,aAChB,YAAaA,EAAG,aAChB,YAAaA,EAAG,aAChB,YAAaA,EAAG,cAGjB,OAAOs5B,EAAOtL,OAAO5gB,IAAK,mBAAqBpL,KAAKkkB,IACnD,MAAMojG,EAAQu3B,EAAiB36H,EAAOojG,OAMtC,OAJKA,GAASA,GAASpjG,EAAOojG,QAC7BpjG,EAAOojG,MAAQA,GAGTpjG,CAAM,GAEf,CDAkB,CAAqBoT,GAC/BwnH,EAAe9gJ,EAAG,kBAClB+gJ,EAAkB/gJ,EAAG,WAG3Bs5B,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,WAAWqrB,IAC1C,MAAM8iH,EAAiC,CAAC,EAClCC,EAA0D,IAAI,GAE9DC,EAAiC5nH,EAAO0qE,SAAS52F,IAAK,WACtD+zI,EAAqC7nH,EAAO0qE,SAAS52F,IAAK,aAE1D42F,EAA2B,CAAEk9C,GAEnC,UAAYh7H,KAAU3c,EAAU,CAC/B,MAAMkxE,EAAkC,CACvCxqE,KAAM,SACN7S,MAAO,IAAI,GAAO,CACjBm2C,MAAOrtB,EAAOojG,MACd/lF,MAAOrd,EAAOqd,MACd6qE,KAAM,gBACN6F,UAAU,KAIU,cAAjB/tF,EAAO9oB,OACXq9E,EAAIr9E,MAAM4O,KAAM,QAASzH,GAAI48I,EAAkB,SAC/C1mE,EAAIr9E,MAAMqZ,IAAK,cAAe,aAC9ButF,EAASp/F,KAAMu8I,KAEf1mE,EAAIr9E,MAAM4O,KAAM,QAASzH,GAAI28I,EAAgB,SAAS1jJ,GAASA,IAAU0oB,EAAO9oB,QAChFq9E,EAAIr9E,MAAMqZ,IAAK,CACdmoB,YAAa,UACbwiH,aAAcl7H,EAAO9oB,SAKvB6jJ,EAAgBpuI,IAAK4nE,GAErBumE,EAAQ96H,EAAO9oB,OAAU8oB,EAAOojG,KACjC,CAEA,MAAMpV,EAAe,GAAgBh2E,GAgDrC,OA/CA,GAAmBg2E,EAAc+sC,EAAiB,CACjD1pC,UAAWwpC,EACX3yC,KAAM,SAGP8F,EAAa/D,WAAW15F,IAAK,CAC5B8gG,UAAWwpC,EACXnyC,oBAAgB,EAChByI,MAAM,EACNpD,UAAU,EACVpF,QAASkyC,IAGV7sC,EAAazM,eAAgB,CAC5Bl9F,WAAY,CACXg5B,MAAO,CACN,0BAKH2wE,EAAaloG,KAAM,aAAcmN,OAAQ6qF,EAAU,aAAa,IAAKq9C,IAC7DA,EAAWhoH,MAAMK,GAAaA,MAGtCw6E,EAAa/D,WAAWnkG,KAAM,SAAUzH,GAAI28I,EAAgB,QAASC,EAAkB,SAAS,CAAE3jJ,EAAO8jJ,KACxG,MAAMC,EAAa/jJ,GAAS8jJ,GAAQ,YAEpC,MAA2B,kBAAfC,EACJT,EAIFE,EAAQO,GAIPP,EAAQO,GAHPT,CAGmB,IAI5B97I,KAAKwP,SAA8B0/F,EAAc,WAAW75F,IAC3D,MAAM,YAAEukB,EAAW,aAAEwiH,GAAiB/mI,EAAInS,OAC1CoxB,EAAOiB,QAASqE,EAAawiH,EAAe,CAAE5jJ,MAAO4jJ,QAAiB,GACtE9nH,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBiiE,CAAY,GAErB,EEpFM,SAASstC,GAA6BhiH,GAC5C,OAAOA,EAAOuZ,uBAAwB,SAAU,CAAExV,MAAO,SAAW,CACnE/D,EAAO2Z,mBAAoB,OAC3B3Z,EAAOyd,WAAY,aAErB,CAQO,SAASwkG,GAA0BnoH,EAAgBooH,GACzD,MAAMC,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cw0I,EAA4BtoH,EAAOqC,QAAQ/oB,IAAK,uBAA0B0mB,EAAOqC,QAAQ/oB,IAAK,qBAEpG,OAAO6J,IAEN,IAAMklI,EAAWE,kBAAmBplI,GACnC,OAAO,KAIR,IAAMmlI,EACL,OAAOE,EAAyBrlI,GASjC,OAJmD,SAAjCA,EAAQsnB,SAAU,YAA0BtnB,EAAQ6uB,aAAcq2G,EAAWI,kBAC9F,aACA,iBAEkBL,EACX,KAGDI,EAAyBrlI,EAAS,EAG1C,SAASqlI,EAAyBrlI,GACjC,MAAMomB,EAAmC,CACxC77B,MAAM,GAQP,OAJKyV,EAAQwuB,aAAc,SAC1BpI,EAAQt4B,WAAa,CAAE,QAGjBs4B,CACR,CACD,CAaO,SAASm/G,GACft5E,EACAzuC,GAEA,MAAM6lE,EAAa,GAAO7lE,EAAU0tC,qBAGpC,OAAMm4B,GAAcp3B,EAAOgE,SAAUozB,IAMhCA,EAAW34D,SAA8B,YAAnB24D,EAAW94F,KAL9B,aAUD,aACR,C,0VC1Ge,MAAM,WAAmB,GAIvC,qBAAkBs0B,GACjB,MAAO,YACR,CAKO,OAAA69G,CAASh4E,GACf,OAAOn8D,KAAKi9I,cAAe9gF,IAAkBn8D,KAAKk9I,aAAc/gF,EACjE,CAOO,iBAAA0gF,CAAmBplI,GACzB,QAASA,GAAWA,EAAQvJ,GAAI,UAAW,MAC5C,CAOO,gBAAA6uI,CAAkBtlI,GACxB,QAASA,GAAWA,EAAQvJ,GAAI,UAAW,WAAcuJ,EAAQqjB,SAAU,QAC5E,CAyBO,WAAAqiH,CACN53I,EAAsC,CAAC,EACvC0oC,EAA0C,KAC1CmvG,EAAqD,MAErD,MAAM9oH,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf68B,EAAY78B,EAAMoL,SAASyxB,UAEjCmoH,EAAYC,GAAgC/oH,EAAQ2Z,GAAchZ,EAAWmoH,GAK7E73I,EAAa,SACTvO,OAAOo2B,YAAa6H,EAAU+Q,kBAC9BzgC,GAGJ,UAAYqhC,KAAiBrhC,EACtBnN,EAAMsrE,OAAOmE,eAAgBu1E,EAAWx2G,WACtCrhC,EAAYqhC,GAIrB,OAAOxuC,EAAMmiC,QAAQC,IACpB,MAAMm7G,EAAen7G,EAAOl1B,cAAe83I,EAAY73I,GAUvD,OARAnN,EAAM4iG,aAAc26C,EAAc1nG,EAAY,KAAM,CACnDuF,aAAc,KAGd2nD,oBAAsBltD,GAA2B,eAAbmvG,OAAsC,EAAT,SAI7DzH,EAAa/2I,OACV+2I,EAGD,IAAI,GAEb,CAKO,6BAAA2H,CAA+BroH,GACrC,MAAMspB,EAAoBtpB,EAAUE,mBAEpC,IAAMopB,EACL,OAAO,KAGR,MAAMhH,EAActiB,EAAU+Y,qBAE9B,GAAKuJ,GAAev3C,KAAKu9I,cAAehmG,GACvC,OAAOA,EAGR,IAAI34C,EAAiD2/C,EAAkB3/C,OAEvE,KAAQA,GAAS,CAChB,GAAKA,EAAOsP,GAAI,YAAelO,KAAKu9I,cAAe3+I,GAClD,OAAOA,EAGRA,EAASA,EAAOA,MACjB,CAEA,OAAO,IACR,CAKO,8BAAA4+I,CAAgCvoH,GACtC,MAAMib,EAAkBjb,EAAU+Y,qBAElC,OAAOhuC,KAAKm0I,QAASjkG,GAAoBA,EAAkBjb,EAAUE,mBAAoBmR,aAAc,aACxG,CAOO,cAAAytG,GACN,MACM9+G,EADQj1B,KAAKs0B,OAAOl8B,MACFoL,SAASyxB,UAEjC,OAqEF,SAAiCX,EAAgBW,GAChD,MAAMmoH,EAAYC,GAAgC/oH,EAAQW,EAAW,MAErE,GAAkB,cAAbmoH,EAA4B,CAChC,MAAMx+I,EAsBR,SAA+Bq2B,EAA0C78B,GACxE,MAAMqlJ,EAAiB,GAA2BxoH,EAAW78B,GACvDwG,EAAS6+I,EAAe/8G,MAAM9hC,OAEpC,GAAKA,EAAOujC,UAAYvjC,EAAOsP,GAAI,UAAW,SAC7C,OAAOtP,EAAOA,OAGf,OAAOA,CACR,CA/BiB8+I,CAAsBzoH,EAAWX,EAAOl8B,OAEvD,GAAKk8B,EAAOl8B,MAAMsrE,OAAOyF,WAAYvqE,EAAmB,cACvD,OAAO,CAET,MAAO,GAAK01B,EAAOl8B,MAAMsrE,OAAOyF,WAAYl0C,EAAUgY,MAAQ,eAC7D,OAAO,EAGR,OAAO,CACR,CAnFS0wG,CAAwB39I,KAAKs0B,OAAQW,IAwF9C,SAA2BA,GAC1B,MAAO,IAAKA,EAAUgY,MAAO9Q,gBAAiB5lB,OAAOqgC,IAAaA,EAAS1oC,GAAI,UAAW,eAC3F,CA1F6D0vI,CAAkB3oH,EAC9E,CAWO,aAAA4oH,CAAetmG,EAA0B/c,EAAwB+T,GACvE/T,EAAOoa,kBAAmB,SAAS,EAAM2C,GASzC,OAAOwlF,GAAUxlF,EAAa/c,EAAQ,CAAE+T,MAPnB,KACpB,MACMuvG,EADa99I,KAAK+9I,mBAAoBxmG,GACjBlc,aAAc,OAEzC,OAAOyiH,EAAU,GAAIA,KAAavvG,IAAWA,CAAK,GAIpD,CAKU,aAAAgvG,CAAehmG,GACxB,QAASA,EAAY/Q,kBAAmB,UAAa,GAAU+Q,EAChE,CAKO,YAAA2lG,CAAc/gF,GACpB,QAASA,GAAgBA,EAAajuD,GAAI,UAAW,aACtD,CAKO,aAAA+uI,CAAe9gF,GACrB,QAASA,GAAgBA,EAAajuD,GAAI,UAAW,cACtD,CAOO,kBAAA6vI,CAAoBC,GAC1B,GAAKh+I,KAAK68I,kBAAmBmB,GAC5B,OAAOA,EAGR,MAAMz8B,EAAcvhH,KAAKs0B,OAAO6qE,QAAQllE,KAExC,UAAY,KAAEh6B,KAAUshH,EAAYxpE,cAAeimG,GAClD,GAAKh+I,KAAK68I,kBAAmB58I,GAC5B,OAAOA,CAGV,EAiDD,SAASo9I,GACR/oH,EACA2Z,EACAmvG,GAEA,MAAM15E,EAASpvC,EAAOl8B,MAAMsrE,OACtBu6E,EAAwB3pH,EAAOtL,OAAO5gB,IAAK,qBAEjD,OAAMksB,EAAOqC,QAAQ/oB,IAAK,qBAIpB0mB,EAAOqC,QAAQ/oB,IAAK,sBAIrBwvI,IAI0B,WAA1Ba,EACG,cAGuB,UAA1BA,EACG,aAIHhwG,EAAW//B,GAAI,aACZ8uI,GAA2Ct5E,EAAQz1B,GAGpDy1B,EAAOyF,WAAYl7B,EAAY,eAAkB,cAAgB,cApBhE,aAJA,aAyBT,C,kLChTyB,IAAItkC,OAAQsR,OAAQ,sDAAsD/X,OAClG,0DAA0DA,OAC1D,kCAAkCA,OAClC,kCAAkCA,SCPpB,MAAMg7I,WAAoC,GAYxC,OAAAnpH,GACf,MAEMtd,EAFSzX,KAAKs0B,OACkBqC,QAAQvuB,IAAK,cACxBo1I,+BAAgCx9I,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WAEtFj1B,KAAK00B,YAAcjd,EAEdzX,KAAK00B,WAAajd,EAAQwuB,aAAc,OAC5CjmC,KAAKxH,MAAQif,EAAQ4jB,aAAc,OAEnCr7B,KAAKxH,OAAQ,CAEf,CASgB,OAAA+8B,CAAShxB,GACxB,MAAM+vB,EAASt0B,KAAKs0B,OACdqoH,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7ChQ,EAAQk8B,EAAOl8B,MACfu9I,EAAegH,EAAWa,+BAAgCplJ,EAAMoL,SAASyxB,WAE/E78B,EAAMmiC,QAAQC,IACbA,EAAO/0B,aAAc,MAAOlB,EAAQkP,SAAUkiI,EAAe,GAE/D,ECvCc,MAAMwI,WAAoC,GAIxD,mBAAkBjnH,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,6BACR,CAKO,IAAAI,GACN12B,KAAKs0B,OAAO0qE,SAASnxF,IAAK,uBAAwB,IAAIqwI,GAA6Bl+I,KAAKs0B,QACzF,E,eCnCG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQwF,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCmBR,MAAMskH,WAAgC,GAuCpD,WAAAr8I,CAAam3B,GACZ1rB,MAAO0rB,GACP,MAAMl+B,EAAIgF,KAAKk5B,OAAQl+B,EAEvBgF,KAAKwqG,aAAe,IAAI,GAExBxqG,KAAKo/F,WAAa,IAAI,GAEtBp/F,KAAKq+I,aAAer+I,KAAKs+I,0BAEzBt+I,KAAKu+I,eAAiBv+I,KAAKw+I,cAAexjJ,EAAG,QAAU,GAAMqlG,MAAO,kBACpErgG,KAAKu+I,eAAetzI,KAAO,SAE3BjL,KAAKy+I,iBAAmBz+I,KAAKw+I,cAAexjJ,EAAG,UAAY,GAAMy4B,OAAQ,mBAAoB,UAE7FzzB,KAAK0+I,YAAc,IAAI,GAEvB1+I,KAAK6tG,aAAe,IAAI,GAAa,CACpCnB,WAAY1sG,KAAK0+I,YACjBl0C,aAAcxqG,KAAKwqG,aACnBmC,iBAAkB3sG,KAAKo/F,WACvB1hD,QAAS,CAERyvD,cAAe,cAGfD,UAAW,SAIbltG,KAAK4jG,YAAa,CACjBzsF,IAAK,OAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,2BACA,sBAID8qE,SAAU,MAGX/xF,SAAU,CACTtX,KAAKq+I,aACLr+I,KAAKu+I,eACLv+I,KAAKy+I,mBAGR,CAKgB,MAAA9sG,GACfnkC,MAAMmkC,SAEN3xC,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,SAE/B,GAAe,CAAEwiB,KAAMj6B,OAEvB,CAAEA,KAAKq+I,aAAcr+I,KAAKu+I,eAAgBv+I,KAAKy+I,kBAC7C7gJ,SAAStE,IAET0G,KAAK0+I,YAAY7wI,IAAKvU,GAGtB0G,KAAKwqG,aAAa38F,IAAKvU,EAAEme,QAAU,GAEtC,CAKgB,OAAAuO,GACfxY,MAAMwY,UAENhmB,KAAKwqG,aAAaxkF,UAClBhmB,KAAKo/F,WAAWp5E,SACjB,CAWQ,aAAAw4H,CAAejwG,EAAem7D,EAAcvjE,EAAmB30B,GACtE,MAAMgkG,EAAS,IAAI,GAAYx1G,KAAKk5B,QAkBpC,OAhBAs8E,EAAO/jG,IAAK,CACX88B,QACAm7D,OACAG,SAAS,IAGV2L,EAAO/S,eAAgB,CACtBl9F,WAAY,CACXg5B,MAAO4H,KAIJ30B,GACJgkG,EAAOpkG,SAAU,WAAY7R,GAAIS,KAAMwR,GAGjCgkG,CACR,CAOQ,uBAAA8oC,GACP,MAAMtjJ,EAAIgF,KAAKk5B,OAAQl+B,EACjBqjJ,EAAe,IAAI,GAA6Br+I,KAAKk5B,OAAQ,IAInE,OAFAmlH,EAAa9vG,MAAQvzC,EAAG,oBAEjBqjJ,CACR,EC3JM,SAAS,GAAwB/pH,GACvC,MAAMitF,EAAcjtF,EAAO6qE,QAAQllE,KAC7Bg+E,EAAmBsB,GAAiBtB,iBACpC0kC,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAEnD,MAAO,CACNxE,OAAQ29G,EAAY1vE,aAAayK,aAChCqgG,EAAWW,8BAA+B/7B,EAAY/9G,SAASyxB,YAEhE7N,UAAW,CACV6wF,EAAiBM,gBACjBN,EAAiBS,oBACjBT,EAAiBU,oBACjBV,EAAiBC,gBACjBD,EAAiBI,oBACjBJ,EAAiBK,oBACjBL,EAAiBW,qBAGpB,CC9Be,MAAM+lC,WAA+B,GAcnD,mBAAkBznH,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,wBACR,CAKO,IAAAI,GACN12B,KAAKw+I,eACN,CAKgB,OAAAx4H,GACfxY,MAAMwY,UAGDhmB,KAAK4+I,OACT5+I,KAAK4+I,MAAM54H,SAEb,CAMQ,aAAAw4H,GACP,MAAMlqH,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EAEjBs5B,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,wBAAwBqrB,IACvD,MAAMxD,EAAuCpB,EAAO0qE,SAAS52F,IAAK,wBAC5D6xB,EAAO,IAAI,GAAYf,GAe7B,OAbAe,EAAKxoB,IAAK,CACT88B,MAAOvzC,EAAG,iCACV0uG,KAAM,GAAMlJ,UACZqJ,SAAS,IAGV5vE,EAAKjzB,KAAM,aAAczH,GAAIm2B,EAAS,aACtCuE,EAAKjzB,KAAM,QAASzH,GAAIm2B,EAAS,SAASl9B,KAAWA,IAErDwH,KAAKwP,SAAUyqB,EAAM,WAAW,KAC/Bj6B,KAAK6+I,WAAW,IAGV5kH,CAAI,GAEb,CAMQ,WAAA6kH,GACP,MAAMxqH,EAASt0B,KAAKs0B,OAEdk/B,EADOl/B,EAAO6qE,QAAQllE,KACFz2B,SACpBm5I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAEnDpI,KAAKonH,SAAWpnH,KAAKs0B,OAAOqC,QAAQvuB,IAAK,qBAEzCpI,KAAK4+I,MAAQ,IAAMx8C,GAA4Bg8C,IAAlC,CAA+D9pH,EAAO4E,QAGnFl5B,KAAK4+I,MAAMjtG,SAEX3xC,KAAKwP,SAAuBxP,KAAK4+I,MAAO,UAAU,KACjDtqH,EAAOiB,QAAS,uBAAwB,CACvC9hB,SAAUzT,KAAK4+I,MAAOP,aAAan0C,UAAUzyF,QAASjf,QAGvDwH,KAAK++I,WAAW,EAAM,IAGvB/+I,KAAKwP,SAAuBxP,KAAK4+I,MAAO,UAAU,KACjD5+I,KAAK++I,WAAW,EAAM,IAIvB/+I,KAAK4+I,MAAMx/C,WAAW3tF,IAAK,OAAO,CAAE3O,EAAM2wB,KACzCzzB,KAAK++I,WAAW,GAChBtrH,GAAQ,IAITzzB,KAAKwP,SAAU8kB,EAAO6E,GAAI,UAAU,KAC7BwjH,EAAWW,8BAA+B9pF,EAAav+B,WAEjDj1B,KAAKg/I,YDvHb,SAAsC1qH,GAC5C,MAAM4pF,EAA6B5pF,EAAOqC,QAAQvuB,IAAK,qBAGvD,GAF+BksB,EAAOqC,QAAQvuB,IAAK,cAEnCk1I,8BAA+BhpH,EAAO6qE,QAAQllE,KAAKz2B,SAASyxB,WAAc,CACzF,MAAMxP,EAAW,GAAwB6O,GAEzC4pF,EAAQ0H,eAAgBngG,EACzB,CACD,CC+GI,CAA6B6O,GAF7Bt0B,KAAK++I,WAAW,EAGjB,IAID,GAAqB,CACpBnvI,QAAS5P,KAAK4+I,MACd58C,UAAW,IAAMhiG,KAAKg/I,WACtB/8C,gBAAiB,IAAM,CAAEjiG,KAAKonH,SAAUntF,KAAKxiB,SAC7ClI,SAAU,IAAMvP,KAAK++I,aAEvB,CAKQ,SAAAF,GACP,GAAK7+I,KAAKg/I,WACT,OAGKh/I,KAAK4+I,OACV5+I,KAAK8+I,cAGN,MAAMxqH,EAASt0B,KAAKs0B,OACdoB,EAAuCpB,EAAO0qE,SAAS52F,IAAK,wBAC5Di2I,EAAer+I,KAAK4+I,MAAOP,aAEjCr+I,KAAK4+I,MAAOv8C,wBAENriG,KAAKi/I,cACVj/I,KAAKonH,SAAUv5G,IAAK,CACnBosB,KAAMj6B,KAAK4+I,MACXn5H,SAAU,GAAwB6O,KASpC+pH,EAAan0C,UAAU1xG,MAAQ6lJ,EAAan0C,UAAUzyF,QAASjf,MAAQk9B,EAAQl9B,OAAS,GAExFwH,KAAK4+I,MAAOP,aAAan0C,UAAUW,SAEnC7qG,KAAK4+I,MAAOr8C,sBACb,CAOQ,SAAAw8C,CAAWG,GAAyB,GACrCl/I,KAAKi/I,eAMNj/I,KAAK4+I,MAAOp0C,aAAa13E,WAC7B9yB,KAAK4+I,MAAOL,eAAetxG,QAG5BjtC,KAAKonH,SAAUnhH,OAAQjG,KAAK4+I,OAEvBM,GACJl/I,KAAKs0B,OAAO6qE,QAAQllE,KAAKgT,QAE3B,CAKA,cAAY+xG,GACX,QAASh/I,KAAKonH,UAAYpnH,KAAKonH,SAAS5B,cAAgBxlH,KAAK4+I,KAC9D,CAKA,gBAAYK,GACX,QAASj/I,KAAKonH,UAAYpnH,KAAKonH,SAASpC,QAAShlH,KAAK4+I,MACvD,EC7Mc,MAAMO,WAA6B,GAIjD,mBAAkBjoH,GACjB,MAAO,CAAEinH,GAA6BQ,GACvC,CAKA,qBAAkBroH,GACjB,MAAO,sBACR,ECsJM,SAAS8oH,GACfzC,EACAS,GAEA,MAAM/uE,EAA0D,CAAEh5D,EAAKvS,EAAMw8D,KAC5E,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAGD,MAAMw4B,EAAS8kC,EAAc9kC,OACvB/iB,EAAU6nD,EAAc9B,OAAOf,cAAe35D,EAAK7C,MACnDo/I,EAAM1C,EAAWoB,mBAAoBtmI,GAE3C,GAAgC,OAA3B3U,EAAK29D,kBAA6B,CACtC,MAAMiyE,EAAS5vI,EAAK09D,kBAEfkyE,GAAUA,EAAO5vI,OACrB03B,EAAO9zB,gBAAiB,SAAU24I,GAClC7kH,EAAO9zB,gBAAiB,QAAS24I,GAE5B3M,EAAO7vH,OACX2X,EAAO9zB,gBAAiB,QAAS24I,GAGpC,KAAO,CACN,MAAM3M,EAAS5vI,EAAK29D,kBAEfiyE,GAAUA,EAAO5vI,OACrB03B,EAAO/0B,aAAc,SAAUitI,EAAO5vI,KAAMu8I,GAE5C7kH,EAAO/0B,aAAc,QAAS,QAAS45I,GAElC3M,EAAO7vH,OACX2X,EAAO/0B,aAAc,QAASitI,EAAO7vH,MAAOw8H,GAG/C,GAGD,OAAO7/E,IACNA,EAAWnwD,GAAqC,oBAAqB+tI,IAAc/uE,EAAW,CAEhG,CAwEO,SAASixE,GACf3C,EACAS,EACA34F,GAEA,MAAM4pB,EAA0D,CAAEh5D,EAAKvS,EAAMw8D,KAC5E,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAGD,MAAMqoE,EAAa/K,EAAc9kC,OAC3B/iB,EAAU6nD,EAAc9B,OAAOf,cAAe35D,EAAK7C,MACnDo/I,EAAM1C,EAAWoB,mBAAoBtmI,GAE3C4yD,EAAW5kE,aAAc3C,EAAK2hD,aAAc3hD,EAAK29D,mBAAqB,GAAI4+E,EAAK,EAGhF,OAAO7/E,IACNA,EAAWnwD,GAAqC,aAAco1C,KAAkB24F,IAAc/uE,EAAW,CAE3G,CC5Se,MAAM,WAA0B1jB,GAIvC,OAAA5kC,CAAS26B,GACf1gD,KAAKwP,SAAUkxC,EAAS,QAAQ,CAAEpxC,EAAO27C,KACxC,MAAMlZ,EAAakZ,EAASrnD,OAEvB5D,KAAK+qD,iCAAkChZ,IAIjB,OAAtBA,EAAW4P,SACf3hD,KAAKu/I,YAAat0F,EACnB,GAEE,CAAEvrC,YAAY,GAClB,CAKgB,aAAA4rC,CAAe5K,GAC9B1gD,KAAK2P,cAAe+wC,EACrB,CASQ,WAAA6+F,CAAat0F,GACfjrD,KAAK00B,YACT10B,KAAKwD,SAASgN,KAAM,iBACpBxQ,KAAKwD,SAASgN,KAAwB,cAAey6C,GAEvD,E,0VCRc,MAAMu0F,WAA2B,GAI/C,WAAAz9I,CAAauyB,GACZ9mB,MAAO8mB,GAEP,MAAM2pH,EAAwB3pH,EAAOtL,OAAO5gB,IAAK,qBAE3CksB,EAAOqC,QAAQ/oB,IAAK,sBACM,UAA1BqwI,GAOJ,EAAY,+BAIR3pH,EAAOqC,QAAQ/oB,IAAK,uBACM,WAA1BqwI,GAOJ,EAAY,+BAGf,CAKgB,OAAAlpH,GACf,MAAM4nH,EAAyB38I,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAExDpI,KAAK00B,UAAYioH,EAAW5I,gBAC7B,CAUgB,OAAAx+G,CAAShxB,GACxB,MAAMk7I,EAAoB,GAA2Cl7I,EAAQrB,QACvE+xB,EAAYj1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UACvC0nH,EAAyB38I,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAUlD2sI,EAAsB/9I,OAAOo2B,YAAa6H,EAAU+Q,iBAE1Dy5G,EAAkB7hJ,SAAS,CAAE8hJ,EAAkB96I,KAC9C,MAAMsrC,EAAkBjb,EAAU+Y,qBAQlC,GANiC,iBAArB0xG,IACXA,EAAmB,CAAE9K,IAAK8K,IAKtB96I,GAASsrC,GAAmBysG,EAAWxI,QAASjkG,GAAoB,CACxE,MAAMzqB,EAAWzlB,KAAKs0B,OAAOl8B,MAAMu/C,oBAAqBzH,GAExDysG,EAAWQ,YAAa,SAAKuC,GAAqB3K,GAAuBtvH,EAC1E,MACCk3H,EAAWQ,YAAa,SAAKuC,GAAqB3K,GACnD,GAEF,EC/Gc,MAAM4K,WAAkC,GAMtC,OAAA5qH,GACf,MACM4nH,EADS38I,KAAKs0B,OACkBqC,QAAQvuB,IAAK,cAC7CqP,EAAUzX,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAAU+Y,qBAErDhuC,KAAK00B,UAAYioH,EAAWxI,QAAS18H,GACrCzX,KAAKxH,MAAQwH,KAAK00B,UAAYjd,EAAQ4jB,aAAc,OAAoB,IACzE,CASgB,OAAA9F,CAAShxB,GACxB,MAAMyuD,EAAQhzD,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAAU+Y,qBACnDhuC,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzBA,EAAO/0B,aAAc,MAAOlB,EAAQrB,OAAQ8vD,GAC5Cx4B,EAAO9zB,gBAAiB,SAAUssD,GAClCx4B,EAAO9zB,gBAAiB,QAASssD,EAAO,GAE1C,EC3Bc,MAAM4sF,WAAqB,GAIzC,mBAAkB1oH,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,cACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdj2B,EAAai2B,EAAOj2B,WAG1Bi2B,EAAO6qE,QAAQllE,KAAKu6B,YAAa,IAEjCn2D,EAAWuhF,IAAK,UACdrV,qBAAsB,CACtBtwC,KAAM,CACLj4B,KAAM,MACNjL,IAAK,OAENqB,MAAO,QAEPmyE,qBAAsB,CACtBtwC,KAAM,CACLj4B,KAAM,MACNjL,IAAK,UAENqB,MAAO,CACNrB,IAAK,SACLyB,MAASqnJ,IACR,MAAMrnJ,EAAgC,CACrCsK,KAAM+8I,EAAUxkH,aAAc,WAO/B,OAJKwkH,EAAU55G,aAAc,WAC5BztC,EAAMqqB,MAAQg9H,EAAUxkH,aAAc,UAGhC7iC,CAAK,KAKhB,MAAMsnJ,EAAqB,IAAIN,GAAoBlrH,GAC7CyrH,EAA4B,IAAIJ,GAA2BrrH,GAEjEA,EAAO0qE,SAASnxF,IAAK,cAAeiyI,GACpCxrH,EAAO0qE,SAASnxF,IAAK,qBAAsBkyI,GAG3CzrH,EAAO0qE,SAASnxF,IAAK,cAAeiyI,EACrC,ECrEc,MAAME,WAAyB,GAW7C,WAAAj+I,CAAauyB,EAAgB2rH,GAC5BzyI,MAAO8mB,GAEPt0B,KAAKkgJ,kBAAoBD,CAC1B,CAKgB,OAAAlrH,GACf,MACM4nH,EADS38I,KAAKs0B,OACkBqC,QAAQvuB,IAAK,cAC7CqP,EAAUklI,EAAWa,+BAAgCx9I,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WAEtD,eAA3Bj1B,KAAKkgJ,kBACTlgJ,KAAK00B,UAAYioH,EAAWM,cAAexlI,GAE3CzX,KAAK00B,UAAYioH,EAAWO,aAAczlI,EAE5C,CAUgB,OAAA8d,GACf,MAAMjB,EAASt0B,KAAKs0B,OACdl8B,EAAQ4H,KAAKs0B,OAAOl8B,MACpBukJ,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cu2F,EAAag+C,EAAWa,+BAAgCplJ,EAAMoL,SAASyxB,WACvE1vB,EAAavO,OAAOo2B,YAAauxE,EAAW34D,iBAKlD,OAAMzgC,EAAWqvI,KAAQrvI,EAAW46I,SAI7B/nJ,EAAMmiC,QAAQC,IAEpB,MAAMolC,EAAUj/D,MAAMrB,KAAMlH,EAAMwnE,SAChC75D,QAAQq7D,GAAUA,EAAOT,WAAW5G,aAAc4kC,KAE9CxpD,EAAawnG,EAAWQ,YAAa53I,EAAYnN,EAAM4/C,gBAAiB2mD,EAAY,MAAQ3+F,KAAKkgJ,mBAEvG,IAAM/qG,EACL,OAAO,KAGR,MAAMirG,EAAkB5lH,EAAOsd,cAAe3C,GAG9C,UAAYisB,KAAUxB,EAAU,CAC/B,MAAMc,EAAcU,EAAOT,WAIrB77C,EAAqC,cAA7B47C,EAAYpqE,KAAK8+B,SAC9BsrC,EAAY1G,UAAWomF,GAAiB,GAAUA,EAEnD5lH,EAAOs4D,aAAc1xB,EAAQ,CAAEt8C,SAChC,CAEA,MAAO,CACN65E,aACAxpD,aACA,IA/BM,IAiCT,EC9Dc,MAAMkrG,WAA0B,GAI9C,mBAAkBnpH,GACjB,MAAO,CAAE0oH,GAAc,GAAY,GACpC,CAKA,qBAAkBtpH,GACjB,MAAO,mBACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACLA,EAAOl8B,MAAMsrE,OAGrBwR,SAAU,aAAc,CAC9BsE,eAAgB,eAChB5D,gBAAiB,CAAE,MAAO,MAAO,YAGlC51E,KAAKsgJ,mBAEAhsH,EAAOqC,QAAQ/oB,IAAK,wBACxB0mB,EAAO0qE,SAASnxF,IAAK,iBAAkB,IAAImyI,GAAkBhgJ,KAAKs0B,OAAQ,eAE1Et0B,KAAKugJ,6BAEP,CAMQ,gBAAAD,GACP,MAAMhsH,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACXqD,EAAai2B,EAAOj2B,WACpBs+I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAEnD/J,EAAWuhF,IAAK,gBACd1W,mBAAoB,CACpB9wE,MAAO,aACP6hC,KAAM,CAAEkiC,GAAgB3hC,YAAcgiH,GAA6BhiH,KAGrEn8B,EAAWuhF,IAAK,mBACd1W,mBAAoB,CACpB9wE,MAAO,aACP6hC,KAAM,CAAEkiC,GAAgB3hC,YAAcmiH,EAAWkB,cAChDrB,GAA6BhiH,GAAUA,EAAQx/B,EAAG,mBAIrDqD,EAAWuhF,IAAK,YACd/xE,IAAKyxI,GAAwB3C,EAAY,aAAc,QACvD9uI,IAAKyxI,GAAwB3C,EAAY,aAAc,QACvD9uI,IAAKuxI,GAAyBzC,EAAY,eAG5Ct+I,EAAWuhF,IAAK,UACdvX,iBAAkB,CAClBpuC,KAAMwiH,GAA0BnoH,EAAQ,cACxCl8B,MAAO,CAAEynJ,GAAarlH,YAAcA,EAAOl1B,cAC1C,aACAu6I,EAAU55G,aAAc,OAAU,CAAE2uG,IAAKiL,EAAUxkH,aAAc,aAAY,KAG9ExtB,INxEG,SAA4B8uI,GAClC,MAAMtuE,EAA6C,CAAEh5D,EAAKvS,EAAMw8D,KAE/D,IAAMA,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAE9tE,MAAM,EAAM08B,QAAS,UAC1E,OAID,MAAMmhH,EAAYlD,EAAWoB,mBAAoBj7I,EAAKgtE,UAGtD,IAAM+vE,IAAcvgF,EAAcwB,WAAW7+D,KAAM49I,EAAW,CAAE79I,MAAM,IACrE,OAIDs9D,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAE9tE,MAAM,EAAM08B,QAAS,UAGxE,MAGM8hH,EAAa,GAHMlhF,EAAc4C,YAAa29E,EAAW/8I,EAAK+sE,aAGzBvS,WAAYpxB,YAGjDs0G,GAQNlhF,EAAc6C,gBAAiBr/D,EAAKgtE,SAAU0wE,GAE9ClhF,EAAc+Q,uBAAwBmwE,EAAY19I,IARjDw8D,EAAcwB,WAAWnC,OAAQ77D,EAAKgtE,SAAU,CAAE9tE,MAAM,EAAM08B,QAAS,SAQhB,EAGzD,OAAO8gC,IACNA,EAAWnwD,GAAwB,iBAAkBg/D,EAAW,CAElE,CM+BSoyE,CAAmB9D,GAC3B,CAgBQ,0BAAA4D,GACP,MAAMjsH,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfmpH,EAAcjtF,EAAO6qE,QAAQllE,KAC7B0iH,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cg+H,EAAuC9xG,EAAOqC,QAAQvuB,IAAK,qBAEjEpI,KAAKwP,SACJ42H,EACA,uBACA,CAAE/wH,EAAKvS,KACN,MAAM49I,EAAsB//I,MAAMrB,KAAMwD,EAAK5C,QAAQs7B,eACrD,IAAI8hC,EAIJ,IAAMojF,EAAoBnqI,MAAOomI,EAAWE,mBAC3C,OAOAv/E,EADIx6D,EAAK4wD,aACIp/B,EAAO6qE,QAAQ3hC,OAAOL,aAAcr6D,EAAK4wD,aAAc,IAIvDt7D,EAAMoL,SAASyxB,UAAUoY,gBAGvC,MAAMpY,EAAY78B,EAAM4/C,gBAAiBslB,GAIzC,GAA8E,eAAzE0/E,GAA2C5kJ,EAAMsrE,OAAQzuC,GAA+B,CAC5F,MAAMuF,EAAS,IAAI,GAAc+mF,EAAY/9G,UAGvCm9I,EAAkBD,EAAoB1jJ,KAC3C4jJ,GAAmBpmH,EAAOl1B,cAAe,SAAU,CAAEi5B,MAAO,SAAWqiH,KAGxE99I,EAAK5C,QAAUs6B,EAAOkZ,uBAAwBitG,EAC/C,IAEH,E,eC7KG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ7mH,OCgBR,MAAM+mH,WAAmB,GAIvC,mBAAkB3pH,GACjB,MAAO,CAAEmpH,GAAmB,GAAQlB,GACrC,CAKA,qBAAkB7oH,GACjB,MAAO,YACR,ECJc,MAAMwqH,WAA2B,GAI/C,mBAAkB5pH,GACjB,MAAO,CAAE0oH,GAAc,GAAY,GACpC,CAKA,qBAAkBtpH,GACjB,MAAO,oBACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdovC,EAASpvC,EAAOl8B,MAAMsrE,OAG5BA,EAAOwR,SAAU,cAAe,CAC/BsE,eAAgB,gBAChB5D,gBAAiB,CAAE,MAAO,MAAO,YAMlClS,EAAOsS,eAAe,CAAEzoE,EAASmvF,KAChC,GAAKnvF,EAAQwgB,SAAU,YAAwC,gBAAzB2uE,EAAgB16F,KACrD,OAAO,CACR,IAGDhC,KAAKsgJ,mBAEAhsH,EAAOqC,QAAQ/oB,IAAK,uBACxB0mB,EAAO0qE,SAASnxF,IAAK,kBAAmB,IAAImyI,GAAkBhgJ,KAAKs0B,OAAQ,gBAE3Et0B,KAAKugJ,6BAEP,CAMQ,gBAAAD,GACP,MAAMhsH,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACXqD,EAAai2B,EAAOj2B,WACpBs+I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAEnD/J,EAAWuhF,IAAK,gBACdvX,iBAAkB,CAClBjwE,MAAO,cACP6hC,KAAM,CAAEkiC,GAAgB3hC,YAAcA,EAAO2Z,mBAAoB,SAGnE91C,EAAWuhF,IAAK,mBACd1W,mBAAoB,CACpB9wE,MAAO,cACP6hC,KAAM,CAAEkiC,GAAgB3hC,YAAcmiH,EAAWkB,cpBpE9C,SAAuCrjH,GAC7C,OAAOA,EAAOuZ,uBAAwB,OAAQ,CAAExV,MAAO,gBACtD/D,EAAO2Z,mBAAoB,OAE7B,CoBiEK4sG,CAA8BvmH,GAAUA,EAAQx/B,EAAG,mBAItDqD,EAAWuhF,IAAK,YACd/xE,IAAKyxI,GAAwB3C,EAAY,cAAe,QACxD9uI,IAAKyxI,GAAwB3C,EAAY,cAAe,QACxD9uI,IAAKuxI,GAAyBzC,EAAY,gBAG5Ct+I,EAAWuhF,IAAK,UACdvX,iBAAkB,CAClBpuC,KAAMwiH,GAA0BnoH,EAAQ,eACxCl8B,MAAO,CAAEynJ,GAAarlH,YAAcA,EAAOl1B,cAC1C,cACAu6I,EAAU55G,aAAc,OAAU,CAAE2uG,IAAKiL,EAAUxkH,aAAc,aAAY,IAGjF,CAiBQ,0BAAAklH,GACP,MAAMjsH,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfmpH,EAAcjtF,EAAO6qE,QAAQllE,KAC7B0iH,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cg+H,EAAuC9xG,EAAOqC,QAAQvuB,IAAK,qBAEjEpI,KAAKwP,SACJ42H,EACA,uBACA,CAAE/wH,EAAKvS,KACN,MAAM49I,EAAsB//I,MAAMrB,KAAMwD,EAAK5C,QAAQs7B,eACrD,IAAI8hC,EAIJ,IAAMojF,EAAoBnqI,MAAOomI,EAAWI,kBAC3C,OAOAz/E,EADIx6D,EAAK4wD,aACIp/B,EAAO6qE,QAAQ3hC,OAAOL,aAAcr6D,EAAK4wD,aAAc,IAIvDt7D,EAAMoL,SAASyxB,UAAUoY,gBAGvC,MAAMpY,EAAY78B,EAAM4/C,gBAAiBslB,GAIzC,GAA8E,gBAAzE0/E,GAA2C5kJ,EAAMsrE,OAAQzuC,GAAgC,CAC7F,MAAMuF,EAAS,IAAI,GAAc+mF,EAAY/9G,UAIvCw9I,EAAmBN,EAAoB1jJ,KAAKikJ,GAKd,IAA9BA,EAAe3lH,YAGnB36B,MAAMrB,KAAM2hJ,EAAej7G,iBACzBpoC,SAASwY,GAAaokB,EAAO/0B,gBAC1B2Q,EACHumI,EAAWoB,mBAAoBkD,MAG1BA,EAAeplH,SAAU,IAEzBolH,IAITn+I,EAAK5C,QAAUs6B,EAAOkZ,uBAAwBstG,EAC/C,IAEH,EC7Kc,MAAME,WAAoB,GAIxC,mBAAkBhqH,GACjB,MAAO,CAAE4pH,GAAoB,GAAQ3B,GACtC,CAKA,qBAAkB7oH,GACjB,MAAO,aACR,ECxBc,MAAM6qH,WAA0B,GAI9C,qBAAkB7qH,GACjB,MAAO,mBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GACV,CAKO,+BAAAkqH,CAAiCC,GACvC,UAAYliJ,KAAQkiJ,EAAkB7lH,cACrC,GAAOr8B,GAAQA,EAAK+O,GAAI,UAAW,WAClC,OAAO/O,EAIT,OAAO,IACR,CAKO,4BAAAmiJ,CAA8BrsH,GACpC,MAAM0nH,EAAyB38I,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDm5I,EAAiBtsH,EAAUE,mBAAoBmR,aAAc,WAEnE,OAAMi7G,GAID5E,EAAWO,aAAcqE,EAAe3iJ,QACrC2iJ,EAJA,IAQT,CAQO,4BAAAC,CAA8B/pI,GACpC,MAAMklI,EAAyB38I,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAGxD,MAAqB,cAAhBqP,EAAQzV,MAAwB26I,EAAWI,iBAAkBtlI,EAAQ7Y,QAClE,CAAEoD,MAAM,GAGT,IACR,ECpCc,MAAMy/I,WAAkC,GAMtC,OAAA1sH,GACf,MAAMT,EAASt0B,KAAKs0B,OACdotH,EAAuCptH,EAAOqC,QAAQvuB,IAAK,qBAC3Du0I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAGnD,IAAMksB,EAAOqC,QAAQ/oB,IAAKyyI,IAIzB,OAHArgJ,KAAK00B,WAAY,OACjB10B,KAAKxH,OAAQ,GAKd,MAAMy8B,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAClCib,EAAkBjb,EAAU+Y,qBAElC,IAAMkC,EAAkB,CACvB,MAAMyxG,EAAyBD,EAAkBJ,6BAA8BrsH,GAK/E,OAHAj1B,KAAK00B,YAAcitH,OACnB3hJ,KAAKxH,QAAUmpJ,EAGhB,CAIA3hJ,KAAK00B,UAAYioH,EAAWxI,QAASjkG,GAE/BlwC,KAAK00B,UAGV10B,KAAKxH,QAAUkpJ,EAAkBN,gCAAiClxG,GAFlElwC,KAAKxH,OAAQ,CAIf,CAagB,OAAA+8B,CAAShxB,EAA4C,CAAC,GACrE,MAAM,mBAAEq9I,GAAuBr9I,EAE/BvE,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACpBx6B,KAAKxH,MACTwH,KAAK6hJ,kBAAmBrnH,GAExBx6B,KAAK8hJ,kBAAmBtnH,EAAQonH,EACjC,GAEF,CASQ,iBAAAE,CAAmBtnH,EAAgBonH,GAC1C,MACM3sH,EADQj1B,KAAKs0B,OAAOl8B,MACFoL,SAASyxB,UAC3B8sH,EAA2C/hJ,KAAKs0B,OAAOqC,QAAQvuB,IAAK,uBACpEu0I,EAAyB38I,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAExD,IAAI45I,EAAgB/sH,EAAU+Y,qBAE9B,MAAMi0G,EAAeF,EAAoBG,iBAAkBF,GAGtDrF,EAAWM,cAAe+E,KAC9BhiJ,KAAKs0B,OAAOiB,QAAS,kBAGrBysH,EAAgB/sH,EAAU+Y,sBAI3B,MAAMm0G,EAAoBF,GAAgBznH,EAAOl1B,cAAe,WAEhEk1B,EAAOqrB,OAAQs8F,EAAmBH,GAE7BJ,GACJpnH,EAAOgZ,aAAc2uG,EAAmB,KAE1C,CAQQ,iBAAAN,CAAmBrnH,GAC1B,MAAMlG,EAASt0B,KAAKs0B,OACdW,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAClC8sH,EAA2CztH,EAAOqC,QAAQvuB,IAAK,uBAC/Ds5I,EAAuCptH,EAAOqC,QAAQvuB,IAAK,qBACjE,IACIm5I,EADAS,EAAgB/sH,EAAU+Y,qBAGzBg0G,EACJT,EAAiBG,EAAkBN,gCAAiCY,IAEpET,EAAiBG,EAAkBJ,6BAA8BrsH,GACjE+sH,EAAgBT,EAAgB3iJ,QAIjCmjJ,EAAoBK,aAAcJ,EAAeT,GAEjD/mH,EAAOgZ,aAAcwuG,EAAe,MACpCxnH,EAAOv0B,OAAQs7I,EAChB,EC/Ic,MAAMc,WAA4B,GA0BhD,WAAAtgJ,CAAauyB,GACZ9mB,MAAO8mB,GAEPt0B,KAAKsiJ,kBAAoB,IAAIjyH,OAC9B,CA1BA,mBAAkB6G,GACjB,MAAO,CAAE,GAAYiqH,GACtB,CAKA,qBAAkB7qH,GACjB,MAAO,qBACR,CAsBO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdovC,EAASpvC,EAAOl8B,MAAMsrE,OAGtBA,EAAO8R,aAAc,WAO1B9R,EAAOxwB,OAAQ,UAAW,CACzBqlC,QAAS,eAPV7U,EAAOwR,SAAU,UAAW,CAC3BqD,QAAS,aACTO,eAAgB,SAChBhV,SAAS,IAQXxvC,EAAO0qE,SAASnxF,IAAK,qBAAsB,IAAI4zI,GAA2BzhJ,KAAKs0B,SAE/Et0B,KAAKsgJ,mBACLtgJ,KAAKuiJ,qCACLviJ,KAAKwiJ,8BACN,CAMQ,gBAAAlC,GACP,MAAMhsH,EAASt0B,KAAKs0B,OACd2F,EAAO3F,EAAO6qE,QAAQllE,KACtB0iH,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cs5I,EAAuCptH,EAAOqC,QAAQvuB,IAAK,qBAC3DpN,EAAIs5B,EAAOt5B,EAGjBs5B,EAAOj2B,WAAWuhF,IAAK,UAAWvX,iBAAkB,CACnDpuC,KAAMxiB,GAAWiqI,EAAkBF,6BAA8B/pI,GACjErf,MAAO,YAIRk8B,EAAOj2B,WAAWuhF,IAAK,gBAAiBvX,iBAAkB,CACzDjwE,MAAO,UACP6hC,KAAM,CAAEkiC,GAAgB3hC,YACjBmiH,EAAWO,aAAc/gF,EAAav9D,QAIrC47B,EAAOuZ,uBAAwB,cAH9B,OAQVzf,EAAOj2B,WAAWuhF,IAAK,mBAAoBvX,iBAAkB,CAC5DjwE,MAAO,UACP6hC,KAAM,CAAEkiC,GAAgB3hC,aACvB,IAAMmiH,EAAWO,aAAc/gF,EAAav9D,QAC3C,OAAO,KAGR,MAAM6jJ,EAAoBjoH,EAAO0Z,sBAAuB,cACxD1Z,EAAOoa,kBAAmB,gBAAgB,EAAM6tG,GAEhDA,EAAkB7nH,YAAc5/B,EAAG,uBACnC,GAAmB,CAClBi/B,OACAxiB,QAASgrI,EACTroH,aAAa,IAGd,MAAMsoH,EAAavmF,EAAav9D,OAAoBy8B,aAAc,OAGlE,OAAO,GAAkBonH,EAAmBjoH,EAAQ,CAAE+T,MAFxCm0G,EAAW1nJ,EAAG,wBAAyB,CAAE0nJ,IAAe1nJ,EAAG,0BAEV,GAGlE,CAOQ,kCAAAunJ,GACP,MAAMjuH,EAASt0B,KAAKs0B,OACdqoH,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cs5I,EAAuCptH,EAAOqC,QAAQvuB,IAAK,qBAC3Du6I,EAAyBruH,EAAO0qE,SAAS52F,IAAK,mBAC9Cw6I,EAAwBtuH,EAAO0qE,SAAS52F,IAAK,kBAE7Cy6I,EAA0DxtI,IAE/D,IAAMA,EAAIlE,OACT,OAGD,MAAM,WAAEwtF,EAAU,WAAExpD,GAAe9/B,EAAIlE,OAGvC,IAAMwtF,EACL,OAGD,GAAKg+C,EAAWO,aAAcv+C,GAAe,CAC5C,MAAMmkD,EAAoBpB,EAAkBN,gCAAiCziD,GAI7E,GAAKmkD,EAGJ,YAFA9iJ,KAAKoiJ,aAAcjtG,EAAY2tG,EAIjC,CAEA,MAAMC,EAAyB/iJ,KAAKkiJ,iBAAkBvjD,GAUjDokD,GAGJ/iJ,KAAKoiJ,aAAcjtG,EAAY4tG,EAChC,EAIIJ,GACJ3iJ,KAAKwP,SAA+BmzI,EAAwB,UAAWE,EAAuB,CAAEj2I,SAAU,QAGtGg2I,GACJ5iJ,KAAKwP,SAA+BozI,EAAuB,UAAWC,EAAuB,CAAEj2I,SAAU,OAE3G,CAYO,gBAAAs1I,CAAkBb,GACxB,MAAM2B,EAAahjJ,KAAKsiJ,kBAAkBl6I,IAAKi5I,GAE/C,OAAO2B,EAAa,GAAQ/rF,SAAU+rF,GAAe,IACtD,CAoBO,YAAAZ,CAAcf,EAA4BjhD,GAChDpgG,KAAKsiJ,kBAAkB7wI,IAAK4vI,EAAmBjhD,EAAQpjE,SACxD,CAMQ,4BAAAwlH,GACP,MAAMluH,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfukJ,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cs5I,EAAuCptH,EAAOqC,QAAQvuB,IAAK,qBAEjEhQ,EAAMoL,SAAS6L,GAAyB,eAAe,KACtD,MAAM4wD,EAAU7nE,EAAMoL,SAASm8D,OAAOQ,aAEtC,UAAY5lC,KAAU0lC,EAAsC,CAC3D,GAA6B,QAAxB1lC,EAAOkqB,aACX,SAGD,MAAMuO,EAAQz4B,EAAOzV,MAAM4b,MAAMwJ,UAEjC,GAAKyyG,EAAWO,aAAclqF,GAAU,CACvC,MAAMotC,EAAUshD,EAAkBN,gCAAiCpuF,GAEnE,IAAMotC,EACL,OAGD9rE,EAAO6qE,QAAQrrB,cAAessB,EAC/B,CACD,IAEF,EC/Pc,MAAM6iD,WAAuB,GAI3C,mBAAkB/rH,GACjB,MAAO,CAAEiqH,GACV,CAKA,qBAAkB7qH,GACjB,MAAO,gBACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACditF,EAAcjtF,EAAO6qE,QAAQllE,KAC7BynH,EAAuCptH,EAAOqC,QAAQvuB,IAAK,qBAC3DpN,EAAIs5B,EAAOt5B,EAEjBs5B,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,sBAAsBqrB,IACrD,MAAMxD,EAAqCpB,EAAO0qE,SAAS52F,IAAK,sBAC1D6xB,EAAO,IAAI,GAAYf,GA8B7B,OA5BAe,EAAKxoB,IAAK,CACTi4F,KAAM,GAAMtJ,QACZyJ,SAAS,EACTP,cAAc,IAGfrvE,EAAKjzB,KAAM,OAAQ,aAAczH,GAAIm2B,EAAS,QAAS,aACvDuE,EAAKjzB,KAAM,SAAUzH,GAAIm2B,EAAS,SAASl9B,GAAiBwC,EAARxC,EAAW,qBAA4B,uBAE3FwH,KAAKwP,SAAUyqB,EAAM,WAAW,KAC/B3F,EAAOiB,QAAS,qBAAsB,CAAEqsH,oBAAoB,IAG5D,MAAMsB,EAAsBxB,EAAkBJ,6BAA8BhtH,EAAOl8B,MAAMoL,SAASyxB,WAElG,GAAKiuH,EAAsB,CAC1B,MAAMT,EAAoBnuH,EAAO6qE,QAAQ3hC,OAAOf,cAAeymF,GAE/D3hC,EAAYlsD,uBAEZksD,EAAYhnF,QAAQC,IACnBA,EAAOoB,SAAU,6BAA8B6mH,EAAoB,GAErE,CAEAnuH,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBhT,CAAI,GAEb,E,eCxEG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQH,OCYhB,SAASqpH,GAAuB5oI,GAEtC,MAAM6oI,EAAkB7oI,EAAMvd,KAAKiO,GAAQA,EAAKgO,QAAS,IAAK,SAE9D,OAAO,IAAItP,OAAQ,aAAcy5I,EAAgBlmJ,KAAM,SACxD,CASO,SAASmmJ,GAAiBrwF,GAChC,OAAO,IAAIv6B,SAAS,CAAElc,EAASyuH,KAC9B,MAAMsY,EAAWtwF,EAAM33B,aAAc,OAGrCkoH,MAAOD,GACLxrH,MAAM0rH,GAAYA,EAASC,SAC3B3rH,MAAM2rH,IACN,MAAMC,EAAWC,GAAkBF,EAAMH,GACnCz7C,EAAM67C,EAASzqI,QAAS,SAAU,IAElC6xH,EAAO,IAAI8Y,KAAM,CAAEH,GADR,SAAU57C,IACgB,CAAE58F,KAAMy4I,IAEnDnnI,EAASuuH,EAAM,IAEfvf,OAAO/pH,GAIAA,GAAoB,cAAbA,EAAIQ,KA4CtB,SAAoCshJ,GACnC,OAaD,SAA4BA,GAC3B,OAAO,IAAI7qH,SAAS,CAAElc,EAASyuH,KAC9B,MAAMh4E,EAAQ,GAAOxvD,SAAS8B,cAAe,OAE7C0tD,EAAM1iD,iBAAkB,QAAQ,KAC/B,MAAMuzI,EAAS,GAAOrgJ,SAAS8B,cAAe,UAE9Cu+I,EAAOhhI,MAAQmwC,EAAMnwC,MACrBghI,EAAO/gI,OAASkwC,EAAMlwC,OAEV+gI,EAAO9+D,WAAY,MAE3B++D,UAAW9wF,EAAO,EAAG,GAEzB6wF,EAAOE,QAAQN,GAAQA,EAAOlnI,EAASknI,GAASzY,KAAU,IAG3Dh4E,EAAM1iD,iBAAkB,SAAS,IAAM06H,MAEvCh4E,EAAM4hF,IAAM0O,CAAQ,GAEtB,CAlCQU,CAAmBV,GAAWxrH,MAAM2rH,IAC1C,MAAMC,EAAWC,GAAkBF,EAAMH,GACnCz7C,EAAM67C,EAASzqI,QAAS,SAAU,IAGxC,OAAO,IAAI2qI,KAAM,CAAEH,GAFF,SAAU57C,IAEU,CAAE58F,KAAMy4I,GAAY,GAE3D,CAnDKO,CAA2BX,GAAWxrH,KAAMvb,GAAUgvG,MAAOyf,GAC7DA,EAAQxpI,IACP,GAEN,CAqBA,SAASmiJ,GAAkBF,EAAY7O,GACtC,OAAK6O,EAAKx4I,KACFw4I,EAAKx4I,KACD2pI,EAAI/3I,MAAO,4BACf+3I,EAAI/3I,MAAO,4BAA+B,GAAI+L,cAG9C,YAET,CCrEe,MAAMs7I,WAAsB,GAI1C,qBAAkB5tH,GACjB,MAAO,eACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACXmpJ,EAAqBjrH,IAC1B,MAAMe,EAAO,IAAIkzG,GAAsBj0G,GACjCxD,EAA8BpB,EAAO0qE,SAAS52F,IAAK,eACnDg8I,EAAa9vH,EAAOtL,OAAO5gB,IAAK,sBAChCi8I,EAAmBlB,GAAuBiB,GAyBhD,OAvBAnqH,EAAKxoB,IAAK,CACT6yI,aAAcF,EAAWpnJ,KAAKiO,GAAQ,SAAUA,MAAU/N,KAAM,KAChEqnJ,oBAAoB,IAGrBtqH,EAAKkxE,WAAW15F,IAAK,CACpB88B,MAAOvzC,EAAG,gBACV0uG,KAAM,GAAM12C,MACZ62C,SAAS,IAGV5vE,EAAKkxE,WAAWnkG,KAAM,aAAczH,GAAIm2B,GAExCuE,EAAK5qB,GAAI,QAAQ,CAAEgG,EAAKq9C,KACvB,MAAM8xF,EAAiB7jJ,MAAMrB,KAAMozD,GAAQ3sD,QAAQ+kI,GAAQuZ,EAAiBpiJ,KAAM6oI,EAAK7/H,QAElFu5I,EAAelsJ,SACnBg8B,EAAOiB,QAAS,cAAe,CAAEu1G,KAAM0Z,IAEvClwH,EAAO6qE,QAAQllE,KAAKgT,QACrB,IAGMhT,CAAI,EAIZ3F,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,cAAes2I,GAC/C7vH,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,cAAes2I,EAChD,E,eCrEG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQrqH,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCqBR,MAAM2qH,WAA4B,GAkBhD,WAAA1iJ,CAAauyB,GACZ9mB,MAAO8mB,GAiCR,KAAQowH,mBAA0D,CAAErvI,EAAKvS,EAAMw8D,KAC9E,MAAMhrC,EAASt0B,KAAKs0B,OACdksH,EAAa19I,EAAK7C,KAClBkgJ,EAAWK,EAAWnlH,aAAc,YAE1C,IAAMikC,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAGD,MAAM26I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7CmtI,EAAiBjhH,EAAOqC,QAAQvuB,IAAK,IACrC0kI,EAASqT,EAAWr9I,EAAK29D,kBAAoB,KAC7C7lC,EAAc56B,KAAK46B,YACnB+pH,EAAarwH,EAAO6qE,QAAQ3hC,OAAOf,cAAe+jF,GAClDn2E,EAAa/K,EAAc9kC,OAEjC,GAAe,WAAVsyG,EAMJ,OAHA8X,GAAoBD,EAAYt6E,QAChCw6E,GAAkBlI,EAAY/hH,EAAa+pH,EAAYt6E,GAMxD,GAAe,aAAVyiE,EAAwB,CAC5B,MAAMf,EAASwJ,EAAejK,QAAQljI,IAAK+3I,GAiB3C,OAdAyE,GAAoBD,EAAYt6E,QAE1B0hE,GAOL+Y,GAAkBH,EAAYt6E,GAqElC,SAA2Bs6E,EAAkCnqH,EAAwBuxG,EAAoB9xG,GACxG,MAAM8qH,EAkCP,SAA6BvqH,GAC5B,MAAMuqH,EAAcvqH,EAAO6Z,gBAAiB,MAAO,CAAE9V,MAAO,oBAI5D,OAFA/D,EAAOoa,kBAAmB,eAAe,EAAMmwG,GAExCA,CACR,CAxCqBC,CAAoBxqH,GACxCA,EAAO90B,OAAQ80B,EAAOkd,iBAAkBitG,EAAY,OAASI,GAG7DhZ,EAAO18H,GAAI,0BAA0B,CAAEgG,EAAKrT,EAAMxJ,KACjDyhC,EAAKM,QAAQC,IACZA,EAAOka,SAAU,QAASl8C,EAAQ,IAAKusJ,EAAa,GAClD,GAEL,CA9EIE,CAAkBN,EAAYt6E,EAAY0hE,EAAQz3G,EAAO6qE,QAAQllE,MAoJrE,SAA6B0iH,EAAwBgI,EAAyBnqH,EAAwBuxG,GACrG,GAAKA,EAAOjpI,KAAO,CAClB,MAAMoiJ,EAAUvI,EAAWoB,mBAAoB4G,GAE/CnqH,EAAO/0B,aAAc,MAAOsmI,EAAOjpI,KAAMoiJ,EAC1C,CACD,CAzJIC,CAAoBxI,EAAYgI,EAAat6E,EAAY0hE,IALzD8Y,GAAkBlI,EAAY/hH,EAAa+pH,EAAYt6E,GASzD,CAEe,YAAVyiE,GAAwByI,EAAejK,QAAQljI,IAAK+3I,IAmF3D,SAA4BwE,EAAkCnqH,EAAwBP,GACrF,MAAMmrH,EAAe5qH,EAAO6Z,gBAAiB,MAAO,CAAE9V,MAAO,kCAE7D/D,EAAO90B,OAAQ80B,EAAOkd,iBAAkBitG,EAAY,OAASS,GAE7DryH,YAAY,KACXkH,EAAKM,QAAQC,GAAUA,EAAOv0B,OAAQu0B,EAAOsd,cAAestG,KAAkB,GAC5E,IACJ,CA1FGC,CAAmBV,EAAYt6E,EAAY/1C,EAAO6qE,QAAQllE,MA2E7D,SAA2B0qH,EAAkCnqH,GAC5D8qH,GAAkBX,EAAYnqH,EAAQ,cACvC,CAzEE+qH,CAAkBZ,EAAYt6E,GAC9By6E,GAAkBH,EAAYt6E,GAiBhC,SAA4Bs6E,EAAkCnqH,GAC7DA,EAAOO,YAAa,YAAa4pH,EAClC,CAlBEa,CAAmBb,EAAYt6E,EAAY,EArF3CrqE,KAAK46B,YAAc,4EACpB,CAlBA,qBAAkBtE,GACjB,MAAO,qBACR,CAqBO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAGfA,EAAOqC,QAAQ/oB,IAAK,sBACxB0mB,EAAO6qE,QAAQ9rB,mBAAmBhkE,GACjC,oCACArP,KAAK0kJ,oBAIFpwH,EAAOqC,QAAQ/oB,IAAK,uBACxB0mB,EAAO6qE,QAAQ9rB,mBAAmBhkE,GACjC,qCACArP,KAAK0kJ,mBAGR,EAqED,SAASE,GAAoBD,EAAkCnqH,GACxDmqH,EAAW7pH,SAAU,cAC1BN,EAAOoB,SAAU,YAAa+oH,EAEhC,CAYA,SAASE,GAAkBlI,EAAwB/hH,EAAqB+pH,EAAkCnqH,GACnGmqH,EAAW7pH,SAAU,gCAC1BN,EAAOoB,SAAU,8BAA+B+oH,GAGjD,MAAMO,EAAUvI,EAAWoB,mBAAoB4G,GAE1CO,EAAQ7pH,aAAc,SAAYT,GACtCJ,EAAO/0B,aAAc,MAAOm1B,EAAasqH,GAGpCO,GAAed,EAAY,gBAChCnqH,EAAO90B,OAAQ80B,EAAOmd,oBAAqButG,GAiE7C,SAA6B1qH,GAC5B,MAAMI,EAAcJ,EAAO6Z,gBAAiB,MAAO,CAAE9V,MAAO,iCAI5D,OAFA/D,EAAOoa,kBAAmB,eAAe,EAAMha,GAExCA,CACR,CAvEwD8qH,CAAoBlrH,GAE5E,CAKA,SAASsqH,GAAkBH,EAAkCnqH,GACvDmqH,EAAW7pH,SAAU,gCACzBN,EAAOO,YAAa,8BAA+B4pH,GAGpDW,GAAkBX,EAAYnqH,EAAQ,cACvC,CAgEA,SAASirH,GAAeE,EAA0BC,GACjD,UAAYjuI,KAASguI,EAAYnqH,cAChC,GAAO7jB,EAAuB6uB,kBAAmBo/G,GAChD,OAAOjuI,CAGV,CAKA,SAAS2tI,GAAkBX,EAAkCnqH,EAAwBorH,GACpF,MAAMnuI,EAAUguI,GAAed,EAAYiB,GAEtCnuI,GACJ+iB,EAAOv0B,OAAQu0B,EAAOsd,cAAergC,GAEvC,C,8RC7Ne,MAAMouI,WAA2B,GAI/B,OAAA9wH,GACf,MAAMT,EAASt0B,KAAKs0B,OACdqoH,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7C8nC,EAAkB5b,EAAOl8B,MAAMoL,SAASyxB,UAAU+Y,qBAGxDhuC,KAAK00B,UAAYioH,EAAW5I,kBAAoB4I,EAAWxI,QAASjkG,EACrE,CASgB,OAAA3a,CAAShxB,GACxB,MAAMmuD,EAAQ,GAASnuD,EAAQumI,MACzB71G,EAAYj1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UACvC0nH,EAAyB38I,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAUlD2sI,EAAsB/9I,OAAOo2B,YAAa6H,EAAU+Q,iBAE1D0sB,EAAM90D,SAAS,CAAEktI,EAAMlmI,KACtB,MAAMsrC,EAAkBjb,EAAU+Y,qBAIlC,GAAKppC,GAASsrC,GAAmBysG,EAAWxI,QAASjkG,GAAoB,CACxE,MAAMzqB,EAAWzlB,KAAKs0B,OAAOl8B,MAAMu/C,oBAAqBzH,GAExDlwC,KAAK8lJ,aAAchb,EAAMiK,EAAqBtvH,EAC/C,MACCzlB,KAAK8lJ,aAAchb,EAAMiK,EAC1B,GAEF,CAKQ,YAAA+Q,CAAchb,EAAYvlI,EAAoBkgB,GACrD,MAAM6O,EAASt0B,KAAKs0B,OAEdy3G,EADiBz3G,EAAOqC,QAAQvuB,IAAK,IACbyjI,aAAcf,GACtC6R,EAAyBroH,EAAOqC,QAAQvuB,IAAK,c,QAG7C2jI,GAIN4Q,EAAWQ,a,2HAAa,IAAK53I,G,EAAL,CAAiB46I,SAAUpU,EAAOlsI,I,aAAM4lB,EACjE,EClFc,MAAMsgI,WAA2B,GA0B/C,WAAAhkJ,CAAauyB,GACZ9mB,MAAO8mB,GAEPA,EAAOtL,OAAOtyB,OAAQ,QAAS,CAC9Bq2I,OAAQ,CACPxyH,MAAO,CAAE,OAAQ,MAAO,MAAO,MAAO,OAAQ,WAIhDva,KAAKgmJ,qBAAuB,IAAIz0I,GACjC,CAhCA,mBAAkB2lB,GACjB,MAAO,CAAE,GAAgBgtF,GAAc,GAAmB,GAC3D,CAEA,qBAAkB5tF,GACjB,MAAO,oBACR,CA+BO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdjd,EAAMid,EAAOl8B,MAAMoL,SACnBnF,EAAai2B,EAAOj2B,WACpBk3I,EAAiBjhH,EAAOqC,QAAQvuB,IAAK,IACrCu0I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cg+H,EAAuC9xG,EAAOqC,QAAQvuB,IAAK,qBAC3Dg8I,EAAajB,GAAuB7uH,EAAOtL,OAAO5gB,IAAK,uBACvD69I,EAAqB,IAAIJ,GAAoBvxH,GAGnDA,EAAO0qE,SAASnxF,IAAK,cAAeo4I,GACpC3xH,EAAO0qE,SAASnxF,IAAK,cAAeo4I,GAGpC5nJ,EAAWuhF,IAAK,UACdrV,qBAAsB,CACtBtwC,KAAM,CACLj4B,KAAM,MACNjL,IAAK,YAENqB,MAAO,aAOT4H,KAAKwP,SAA2C8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,kBAAkB,CAAE6R,EAAKvS,KAGtG,GAmT6B2wD,EAnTR3wD,EAAK2wD,aAoTrB9yD,MAAMrB,KAAMm0D,EAAal5C,OAAQgV,SAAU,cAAyD,KAAxCkkC,EAAad,QAAS,aAnTtF,OAkTG,IAAyBc,EA/S7B,MAAMsmF,EAASp5I,MAAMrB,KAAMwD,EAAK2wD,aAAaf,OAAQ3sD,QAAQ+kI,KAEtDA,GAICsZ,EAAWniJ,KAAM6oI,EAAK7/H,QAGxB8uI,EAAOzhJ,SAIb+c,EAAIlJ,OAEJmoB,EAAOl8B,MAAMmiC,QAAQC,IAEf13B,EAAK4wD,cACTl5B,EAAOgZ,aAAc1wC,EAAK4wD,aAAa12D,KAAKypD,GAAanyB,EAAO6qE,QAAQ3hC,OAAOL,aAAc1W,MAI9FnyB,EAAOl8B,MAAMguE,eAAe,KAC3B9xC,EAAOiB,QAAS,cAAe,CAAEu1G,KAAMiP,GAAU,GAC/C,IACD,IAOJ/5I,KAAKwP,SAAU42H,EAAmB,uBAAuB,CAAE/wH,EAAKvS,KAC/D,MAAMojJ,EAAkBvlJ,MAAMrB,KAAMg1B,EAAO6qE,QAAQllE,KAAK8d,cAAej1C,EAAK5C,UAC1ElD,KAAKxE,GAASA,EAAMyH,OACpB8F,QAAQwxC,GPxEN,SAAuBolG,EAAwBx9I,GACrD,SAAMw9I,EAAWE,kBAAmB19I,KAAWA,EAAKk8B,aAAc,SAIzDl8B,EAAKk8B,aAAc,OAASx+B,MAAO,+BACzCsC,EAAKk8B,aAAc,OAASx+B,MAAO,WACvC,COkEKspJ,CAAcxJ,EAAYplG,KACzBA,EAAYlc,aAAc,qBAC3Br+B,KAAKu6C,IAAwB,CAAE/e,QAAS6qH,GAAiB9rG,GAAeo+F,aAAcp+F,MAExF,IAAM2uG,EAAgB5tJ,OACrB,OAGD,MAAMkiC,EAAS,IAAI,GAAclG,EAAO6qE,QAAQllE,KAAKz2B,UAErD,UAAY4iJ,KAAkBF,EAAkB,CAE/C1rH,EAAO/0B,aAAc,mBAAmB,EAAM2gJ,EAAezQ,cAE7D,MAAM5J,EAASwJ,EAAe1J,aAAcua,EAAe5tH,SAEtDuzG,IACJvxG,EAAO/0B,aAAc,MAAO,GAAI2gJ,EAAezQ,cAC/Cn7G,EAAO/0B,aAAc,WAAYsmI,EAAOlsI,GAAIumJ,EAAezQ,cAE7D,KAIDrhH,EAAO6qE,QAAQllE,KAAKz2B,SAAS6L,GAAI,YAAY,CAAEgG,EAAKvS,KACnDA,EAAKowB,gBAAgB,IAItB7b,EAAIhI,GAAI,UAAU,KAIjB,MAAM4wD,EAAU5oD,EAAIsoD,OAAOQ,WAAY,CAAEmsB,2BAA2B,IAASjhF,UACvEg7I,EAAoB,IAAIzyI,IAE9B,UAAY8E,KAASunD,EACpB,GAAmB,UAAdvnD,EAAMzN,MAAkC,SAAdyN,EAAM1W,KAAkB,CACtD,MAAM/B,EAAOyY,EAAM+M,SAASykB,UACtBo8G,EAAwD,cAAhC5tI,EAAM+M,SAASnvB,KAAK8+B,SAElD,UAAYugH,KAAgB4Q,GAAyBjyH,EAAQr0B,GAAS,CAErE,MAAMkgJ,EAAWxK,EAAat6G,aAAc,YAE5C,IAAM8kH,EACL,SAID,MAAMpU,EAASwJ,EAAejK,QAAQljI,IAAK+3I,GAErCpU,IAIDua,EAGED,EAAkBz4I,IAAKuyI,IAC5BpU,EAAOV,SAMRgb,EAAkBx4I,IAAKsyI,GAMvBngJ,KAAKgmJ,qBAAqBv0I,IAAK0uI,EAAUxK,GAEnB,QAAjB5J,EAAOe,QAEX9sI,KAAKwmJ,eAAgBza,IAGxB,CACD,CACD,IAID/rI,KAAKqP,GAA8B,kBAAkB,CAAEgG,GAAOsgI,eAAc7yI,WAC3E,MAAM03I,EAAO13I,EAAK03I,KAAO13I,EAAK03I,KAAkC13I,EAEhE9C,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzBA,EAAO/0B,aAAc,MAAO+0I,EAAK1L,QAAS6G,GAC1C31I,KAAKymJ,mCAAoCjM,EAAM7E,EAAcn7G,EAAQ,GACnE,GACD,CAAE5tB,SAAU,OAChB,CAKO,SAAA66G,GACN,MAAM/jD,EAAS1jE,KAAKs0B,OAAOl8B,MAAMsrE,OAK5B1jE,KAAKs0B,OAAOqC,QAAQ/oB,IAAK,sBAC7B81D,EAAOxwB,OAAQ,aAAc,CAC5B0iC,gBAAiB,CAAE,WAAY,kBAI5B51E,KAAKs0B,OAAOqC,QAAQ/oB,IAAK,uBAC7B81D,EAAOxwB,OAAQ,cAAe,CAC7B0iC,gBAAiB,CAAE,WAAY,iBAGlC,CASU,cAAA4wE,CAAgBza,GACzB,MAAMz3G,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf4C,EAAIs5B,EAAO4E,OAAOl+B,EAClBu6I,EAAiBjhH,EAAOqC,QAAQvuB,IAAK,IACrCkyI,EAAehmH,EAAOqC,QAAQvuB,IAAK87G,IACnCy4B,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAC7Cs+I,EAAsB1mJ,KAAKgmJ,qBAMjC,OAJA5tJ,EAAMguE,cAAe,CAAEuX,YAAY,IAASnjD,IAC3CA,EAAO/0B,aAAc,eAAgB,UAAWihJ,EAAoBt+I,IAAK2jI,EAAOlsI,IAAO,IAGjFksI,EAAOlB,OACZ/yG,MAAM,KACN,MAAMU,EAAUuzG,EAAOgB,SACjB4I,EAAe+Q,EAAoBt+I,IAAK2jI,EAAOlsI,IAKrD,GAAK,EAAIqJ,SAAW,CACnB,MAAMy7I,EAAarwH,EAAO6qE,QAAQ3hC,OAAOf,cAAek5E,GAClDuP,EAAUvI,EAAWoB,mBAAoB4G,GAE/CrwH,EAAO6qE,QAAQllE,KAAKxqB,KAAM,UAAU,KAGnC,IAAMy1I,EAAQtmJ,OACb,OAGD,MAAM+nJ,EAAYryH,EAAO6qE,QAAQllE,KAAK4X,aAAayK,aAAc4oG,EAAQtmJ,QAEzE,IAAM+nJ,EACL,OAGD,MAAMC,EAAkBD,EAAUthJ,MAAMwhJ,QAExCF,EAAUthJ,MAAMwhJ,QAAU,OAGxBF,EAAmBG,QAAUH,EAAU9hI,aAEzC8hI,EAAUthJ,MAAMwhJ,QAAUD,CAAe,GAE3C,CAMA,OAJAxuJ,EAAMguE,cAAe,CAAEuX,YAAY,IAASnjD,IAC3CA,EAAO/0B,aAAc,eAAgB,YAAakwI,EAAc,IAG1Dn9G,CAAO,IAEdV,MAAMh1B,IACN1K,EAAMguE,cAAe,CAAEuX,YAAY,IAASnjD,IAC3C,MAAMm7G,EAAe+Q,EAAoBt+I,IAAK2jI,EAAOlsI,IAErD26B,EAAO/0B,aAAc,eAAgB,WAAYkwI,GAEjD31I,KAAKwQ,KAAgC,iBAAkB,CAAE1N,OAAM6yI,gBAAgB,IAGhFoR,GAAO,IAEPx7B,OAAO3hH,IAGP,GAAuB,UAAlBmiI,EAAOe,QAAwC,YAAlBf,EAAOe,OACxC,MAAMljI,EAIe,SAAjBmiI,EAAOe,QAAqBljI,GAChC0wI,EAAa91B,YAAa56G,EAAO,CAChC06G,MAAOtpH,EAAG,iBACVuc,UAAW,WAKbnf,EAAMguE,cAAe,CAAEuX,YAAY,IAASnjD,IAC3CA,EAAOv0B,OAAQygJ,EAAoBt+I,IAAK2jI,EAAOlsI,IAAO,IAGvDknJ,GAAO,IAGT,SAASA,IACR3uJ,EAAMguE,cAAe,CAAEuX,YAAY,IAASnjD,IAC3C,MAAMm7G,EAAe+Q,EAAoBt+I,IAAK2jI,EAAOlsI,IAErD26B,EAAO9zB,gBAAiB,WAAYivI,GACpCn7G,EAAO9zB,gBAAiB,eAAgBivI,GAExC+Q,EAAoB/0I,OAAQo6H,EAAOlsI,GAAI,IAGxC01I,EAAenJ,cAAeL,EAC/B,CACD,CAQU,kCAAA0a,CAAoC3jJ,EAA+BkwD,EAAgBx4B,GAE5F,IAAIwzE,EAAW,EAEf,MAAMg5C,EAAkBhwJ,OAAOC,KAAM6L,GAEnCiD,QAAQhP,IACR,MAAM8rB,EAAQzlB,SAAUrG,EAAK,IAE7B,IAAM+nH,MAAOj8F,GAGZ,OAFAmrF,EAAWn1G,KAAKC,IAAKk1G,EAAUnrF,IAExB,CACR,IAIA7lB,KAAKjG,GAAO,GAAI+L,EAAM/L,MAAWA,OAGjCmG,KAAM,MAEgB,IAAnB8pJ,GACJxsH,EAAO/0B,aAAc,SAAU,CAC9B3C,KAAMkkJ,EACNnkI,MAAOmrF,GACLh7C,EAEL,EAUD,SAASuzF,GAAyBjyH,EAAgBr0B,GACjD,MAAM08I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAEnD,OAAOzH,MAAMrB,KAAMg1B,EAAOl8B,MAAM0/C,cAAe73C,IAC7C8F,QAAQvN,GAASmkJ,EAAWxI,QAAS37I,EAAMyH,QAC3CjD,KAAKxE,GAASA,EAAMyH,MACvB,C,eChaI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ65B,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCSR,MAAMmtH,WAA0B,GAoB9C,WAAAllJ,CAAauyB,EAAgBuK,GAC5BrxB,MAAO8mB,GAEPt0B,KAAKknJ,eAAiB,CACrBC,YAAY,EACZC,aAAa,GAGdpnJ,KAAKiiC,QAAU,IAAI1wB,IAAKstB,EAAO7hC,KAAKqI,IACnC,GAAKA,EAAMgiJ,UACV,UAAYpH,KAAoB56I,EAAMg2I,cACrCr7I,KAAKknJ,eAAgBjH,GAAqB56I,EAAMrD,KAIlD,MAAO,CAAEqD,EAAMrD,KAAMqD,EAAO,IAE9B,CAKgB,OAAA0vB,GACf,MAEMtd,EAFSzX,KAAKs0B,OACkBqC,QAAQvuB,IAAK,cACxBo1I,+BAAgCx9I,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WAEtFj1B,KAAK00B,YAAcjd,EAEbzX,KAAK00B,UAECjd,EAAQwuB,aAAc,cACjCjmC,KAAKxH,MAAQif,EAAQ4jB,aAAc,cAEnCr7B,KAAKxH,MAAQwH,KAAKknJ,eAAgBzvI,EAAQzV,MAJ1ChC,KAAKxH,OAAQ,CAMf,CAgBgB,OAAA+8B,CAAShxB,EAA8B,CAAC,GACvD,MAAM+vB,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfukJ,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAEnDhQ,EAAMmiC,QAAQC,IACb,MAAM8sH,EAAiB/iJ,EAAQ/L,MAE/B,IAAIm9I,EAAegH,EAAWa,+BAAgCplJ,EAAMoL,SAASyxB,WAGxEqyH,GAAkBtnJ,KAAKunJ,uBAAwBD,EAAgB3R,KACnE31I,KAAKs0B,OAAOiB,QAASonH,EAAWO,aAAcvH,GAAiB,kBAAoB,kBAGnFA,EAAegH,EAAWa,+BAAgCplJ,EAAMoL,SAASyxB,aAKpEqyH,GAAkBtnJ,KAAKiiC,QAAQ75B,IAAKk/I,GAAkBD,UAC3D7sH,EAAO9zB,gBAAiB,aAAcivI,GAEtCn7G,EAAO/0B,aAAc,aAAc6hJ,EAAgB3R,EACpD,GAEF,CAQO,sBAAA4R,CAAwBD,EAAwB3R,GAGtD,OAFuB31I,KAAKiiC,QAAQ75B,IAAKk/I,GAAkBjM,cAEpC9rH,SAAUomH,EAAa3zI,KAC/C,E,0VCtHD,MAAM,gBACLq/F,GAAe,aACfC,GAAY,WACZJ,GAAU,YAAEE,GAAW,aAAED,GAAY,gBACrCI,GAAe,iBAAEC,IACd,GAoBSgmD,GAA8D,CAE1E,UAAIC,GACH,MAAO,CACNzlJ,KAAM,SACNsiH,MAAO,UACP5a,KAAMpI,GACN+5C,cAAe,CAAE,eACjBgM,WAAW,EAEb,EAGA,aAAIvmD,GACH,MAAO,CACN9+F,KAAM,YACNsiH,MAAO,qBACP5a,KAAMxI,GACNm6C,cAAe,CAAE,aAAc,eAC/Bl1G,UAAW,yBAEb,EAGA,kBAAIuhH,GACH,MAAO,CACN1lJ,KAAM,iBACNsiH,MAAO,qBACP5a,KAAMnI,GACN85C,cAAe,CAAE,cACjBl1G,UAAW,+BAEb,EAGA,eAAI46D,GACH,MAAO,CACN/+F,KAAM,cACNsiH,MAAO,iBACP5a,KAAMvI,GACNk6C,cAAe,CAAE,cACjBl1G,UAAW,2BAEb,EAGA,cAAI66D,GACH,MAAO,CACNh/F,KAAM,aACNsiH,MAAO,sBACP5a,KAAMtI,GACNi6C,cAAe,CAAE,aAAc,eAC/Bl1G,UAAW,0BAEb,EAGA,mBAAIwhH,GACH,MAAO,CACN3lJ,KAAM,kBACNsiH,MAAO,sBACP5a,KAAMlI,GACN65C,cAAe,CAAE,cACjBl1G,UAAW,gCAEb,EAGA,SAAI68B,GACH,MAAO,CACNhhE,KAAM,QACNsiH,MAAO,iBACP5a,KAAMvI,GACNk6C,cAAe,CAAE,cACjBgM,WAAW,EAEb,EAGA,QAAIphE,GACH,MAAO,CACNjkF,KAAM,OACNsiH,MAAO,aACP5a,KAAMtI,GACNi6C,cAAe,CAAE,cACjBl1G,UAAW,mBAEb,GAWYyhH,GAAwC,CACpDC,KAAMxmD,GACNt/E,KAAMw/E,GACN5/E,MAAO6/E,GACPsmD,OAAQ3mD,GACR4mD,WAAY7mD,GACZ8mD,YAAa5mD,GACbqmD,OAAQnmD,IAcI2mD,GAAoE,CAAE,CAClFjmJ,KAAM,sBACNsiH,MAAO,YACP4jC,YAAa,uBACbt3H,MAAO,CAAE,uBAAwB,0BAC/B,CACF5uB,KAAM,uBACNsiH,MAAO,aACP4jC,YAAa,mBACbt3H,MAAO,CAAE,4BAA6B,mBAAoB,gCAoL3D,SAASu3H,GAAkBC,GAa1B,EAAY,+CAAgDA,EAC7D,CAEA,UACCC,gBAnLD,SAA0Br/H,GAYzB,OANyBA,EAAOs/H,iBAAiB/jJ,SAAW,IAG1DvH,KAAKurJ,GAwDR,SAA8BrgJ,GAU3BA,EATyB,iBAAfA,EAELs/I,GAAiBt/I,GAOT,MAAKs/I,GAAiBt/I,IALtB,CAAElG,KAAMkG,GA6ExB,SAAsBhF,EAAoCmC,GACzD,MAAMmjJ,EAAqC,MAAKnjJ,GAEhD,UAAYkD,KAAQrF,EACblM,OAAO6K,UAAU2G,eAAenH,KAAMgE,EAAOkD,KAClDigJ,EAAejgJ,GAASrF,EAAQqF,IAIlC,OAAOigJ,CACR,CA5EeC,CAAajB,GAAiBt/I,EAAWlG,MAAQkG,GAK/B,iBAApBA,EAAWwhG,OACtBxhG,EAAWwhG,KAAOk+C,GAAe1/I,EAAWwhG,OAAUxhG,EAAWwhG,MAGlE,OAAOxhG,CACR,CAlFuBwgJ,CAAqBH,KACzCxiJ,QAAQwiJ,GA2FX,SACCrnI,GACA,oBAAEynI,EAAmB,qBAAEC,IAEvB,MAAM,cAAEvN,EAAa,KAAEr5I,GAASkf,EAEhC,KAAMm6H,GAAkBA,EAAc/iJ,QAAW0J,GAGhD,OAFAmmJ,GAAkB,CAAE9iJ,MAAO6b,KAEpB,EACD,CACN,MAAM2nI,EAAoB,CAAEF,EAAsB,aAAe,KAAMC,EAAuB,cAAgB,MAG9G,IAAMvN,EAAchnH,MAAMswB,GAAekkG,EAAkBt5H,SAAUo1B,KAqBpE,OALA,EAAY,iCAAkC,CAC7Ct/C,MAAO6b,EACP4nI,eAAgBzN,EAAcr+I,KAAKgF,GAAiB,eAATA,EAAwB,oBAAsB,0BAGnF,CAET,CAEA,OAAO,CACR,CAnI0B+mJ,CAAeR,EAAav/H,IAGtD,EAuKCggI,8BAxJD,SAAwCL,EAA8BC,GACrE,OAAKD,GAAuBC,EACpB,CACNrkJ,QAAS,CACR,SAAU,YAAa,aACvB,cAAe,iBAAkB,kBACjC,QAAS,SAGAokJ,EACJ,CACNpkJ,QAAS,CAAE,QAAS,SAEVqkJ,EACJ,CACNrkJ,QAAS,CAAE,SAAU,YAAa,eAI7B,CAAC,CACT,EAqIC0kJ,8BAhID,SAAwCC,GACvC,OAAKA,EAAiBt7I,IAAK,sBAAyBs7I,EAAiBt7I,IAAK,sBAClE,IAAKq6I,IAEL,EAET,EA2HCE,oBACAX,mBACAI,iBACAK,iCCzRD,SAASkB,GAA0BnnJ,EAAc68B,GAChD,UAAYx5B,KAASw5B,EACpB,GAAKx5B,EAAMrD,OAASA,EACnB,OAAOqD,CAGV,CCxEe,MAAM+jJ,WAA0B,GAI9C,qBAAkB9yH,GACjB,MAAO,mBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GACV,CAkBO,IAAAR,GACN,MAAM,gBAAE2xH,EAAe,8BAAEW,GAAkCK,GACrD/0H,EAASt0B,KAAKs0B,OACdq0H,EAAsBr0H,EAAOqC,QAAQ/oB,IAAK,qBAC1Cg7I,EAAuBt0H,EAAOqC,QAAQ/oB,IAAK,sBAEjD0mB,EAAOtL,OAAOtyB,OAAQ,eAAgBsyJ,EAA+BL,EAAqBC,IAE1F5oJ,KAAKspJ,iBAAmBjB,EAAiB,CACxCC,iBAAkBh0H,EAAOtL,OAAO5gB,IAAK,gBACrCugJ,sBACAC,yBAGD5oJ,KAAKsgJ,iBAAkBqI,EAAqBC,GAC5C5oJ,KAAKupJ,kBAGLj1H,EAAO0qE,SAASnxF,IAAK,aAAc,IAAIo5I,GAAmB3yH,EAAQt0B,KAAKspJ,kBACxE,CAOQ,gBAAAhJ,CAAkBqI,EAA8BC,GACvD,MAAMt0H,EAASt0B,KAAKs0B,OACdovC,EAASpvC,EAAOl8B,MAAMsrE,OAEtB8lF,GDhEmC3qH,ECgEe7+B,KAAKspJ,iBD/DvD,CAAEj0I,EAAKvS,EAAMw8D,KACnB,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAID,MAAMynJ,EAAWN,GAA0BrmJ,EAAK29D,kBAA6B5hC,GACvE6qH,EAAWP,GAA0BrmJ,EAAK09D,kBAA6B3hC,GAEvE0Y,EAAc+nB,EAAc9B,OAAOf,cAAe35D,EAAK7C,MACvDoqE,EAAa/K,EAAc9kC,OAE5BkvH,GACJr/E,EAAWtvC,YAAa2uH,EAASvjH,UAAYoR,GAGzCkyG,GACJp/E,EAAWzuC,SAAU6tH,EAAStjH,UAAYoR,EAC3C,GAnBK,IAAoC1Y,ECiEzC,MAAM8qH,EDpCD,SAAoC9qH,GAE1C,MAAM+qH,EAAsE,CAC3ExC,YAAavoH,EAAO94B,QAAQV,IAAUA,EAAMgiJ,WAAahiJ,EAAMg2I,cAAc9rH,SAAU,iBACvF43H,WAAYtoH,EAAO94B,QAAQV,IAAUA,EAAMgiJ,WAAahiJ,EAAMg2I,cAAc9rH,SAAU,iBAGvF,MAAO,CAAEla,EAAKvS,EAAMw8D,KACnB,IAAMx8D,EAAKw6D,WACV,OAGD,MAAM/lB,EAAcz0C,EAAKgtE,SACnB+5E,EAAoB,GAAO/mJ,EAAKw6D,WAAWpxB,YAIjD,GAAM29G,GAKAvqF,EAAcoE,OAAOmE,eAAgBgiF,EAAmB,cAK9D,UAAYxkJ,KAASukJ,EAAoBC,EAA+B7nJ,MAElEs9D,EAAcwB,WAAWrC,QAASlnB,EAAa,CAAE7Y,QAASr5B,EAAM8gC,aAEpEm5B,EAAc9kC,OAAO/0B,aAAc,aAAcJ,EAAMrD,KAAM6nJ,EAE/D,CAEF,CCC+BC,CAA2B9pJ,KAAKspJ,kBAE7Dh1H,EAAO6qE,QAAQ9rB,mBAAmBhkE,GAAI,uBAAwBm6I,GAC9Dl1H,EAAOxxB,KAAKuwE,mBAAmBhkE,GAAI,uBAAwBm6I,GAItDb,IACJjlF,EAAOxwB,OAAQ,aAAc,CAAE0iC,gBAAiB,eAGhDthD,EAAOxxB,KAAKy6E,iBAAiBluE,GAAwB,iBAAkBs6I,EAAsB,CAAE/8I,SAAU,SAGrGg8I,IACJllF,EAAOxwB,OAAQ,cAAe,CAAE0iC,gBAAiB,eAGjDthD,EAAOxxB,KAAKy6E,iBAAiBluE,GAAI,cAAes6I,EAAsB,CAAE/8I,SAAU,QAEpF,CAKQ,eAAA28I,GACP,MAAMj1H,EAASt0B,KAAKs0B,OACd9wB,EAAW8wB,EAAOl8B,MAAMoL,SAExBm5I,EAAaroH,EAAOqC,QAAQvuB,IAAK,IACjCu6B,EAAY,IAAIpxB,IAAKvR,KAAKspJ,iBAAkBtsJ,KAAKqI,GAAS,CAAEA,EAAMrD,KAAMqD,MAG9E7B,EAASk3B,mBAAmBF,IAC3B,IAAIosC,GAAU,EAEd,UAAYrsC,KAAU/2B,EAASm8D,OAAOQ,aACrC,GAAoB,UAAf5lC,EAAOtvB,MAAmC,aAAfsvB,EAAOtvB,MAA8C,cAAvBsvB,EAAOkqB,aAA+B,CACnG,IAAIhtC,EAAyB,UAAf8iB,EAAOtvB,KAAmBsvB,EAAO9U,SAASykB,UAAa3P,EAAOzV,MAAM4b,MAAMwJ,UAMxF,GAJKzyB,GAAWA,EAAQvJ,GAAI,UAAW,cAAiBuJ,EAAQ6jB,WAAa,IAC5E7jB,EAAUA,EAAQokB,SAAU,KAGvB8gH,EAAWxI,QAAS18H,GACzB,SAGD,MAAMsyI,EAAatyI,EAAQ4jB,aAAc,cAEzC,IAAM0uH,EACL,SAGD,MAAMC,EAAuBrnH,EAAUv6B,IAAK2hJ,GAEtCC,GAAyBA,EAAqB3O,cAAc9rH,SAAY9X,EAAqBzV,QAClGw4B,EAAO9zB,gBAAiB,aAAc+Q,GACtCmvD,GAAU,EAEZ,CAGD,OAAOA,CAAO,GAEhB,E,eClJG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ9sC,OCcR,MAAMmwH,WAAqB,GAIzC,mBAAkB/yH,GACjB,MAAO,CAAEkyH,GACV,CAKA,qBAAkB9yH,GACjB,MAAO,cACR,CAiBA,gCAAW4zH,GACV,MAAMlvJ,EAAIgF,KAAKs0B,OAAOt5B,EAEtB,MAAO,CACN,YAAaA,EAAG,aAChB,aAAcA,EAAG,cACjB,UAAWA,EAAG,WACd,kBAAmBA,EAAG,mBACtB,aAAcA,EAAG,cACjB,qBAAsBA,EAAG,sBACzB,iBAAkBA,EAAG,kBACrB,sBAAuBA,EAAG,uBAE5B,CAKO,IAAA07B,GACN,MAAMC,EAAU32B,KAAKs0B,OAAOqC,QACtBktG,EAAgB7jI,KAAKs0B,OAAOtL,OAAO5gB,IAAK,kBAAqB,GAG7D+hJ,EAAgBC,GAFuBzzH,EAAQvuB,IAAK,qBAGvCkhJ,iBAClBtpJ,KAAKkqJ,8BAGN,UAAYG,KAAeF,EAC1BnqJ,KAAKw+I,cAAe6L,GAGrB,MAAMC,EAAmBF,GACxB,IACIvmB,EAAc99H,OAAQ,MACtBsjJ,GAAMJ,8BAA+BtyH,IAEzC32B,KAAKkqJ,8BAGN,UAAYK,KAAkBD,EAC7BtqJ,KAAKwqJ,gBAAiBD,EAAgBJ,EAExC,CAKQ,eAAAK,CAAiBD,EAA8CJ,GACtE,MAAM5zJ,EAAUyJ,KAAKs0B,OAAO6E,GAAG61E,iBAE/Bz4G,EAAQsX,IAAK08I,EAAevoJ,MAAMk3B,IACjC,IAAIuxH,EAEJ,MAAM,YAAEvC,EAAW,MAAEt3H,EAAK,MAAE0zF,GAAUimC,EAChCG,EAAc95H,EAClB7qB,QAAQovE,GAAYg1E,EAAc54H,MAAM,EAAIvvB,UAAY2oJ,GAAoB3oJ,KAAWmzE,MACvFn4E,KAAK4tJ,IACL,MAAMp1C,EAASj/G,EAAQ+mB,OAAQstI,GAM/B,OAJKA,IAAe1C,IACnBuC,EAAgBj1C,GAGVA,CAAM,IAGV5kF,EAAMt4B,SAAWoyJ,EAAYpyJ,QACjC+wJ,GAAMlB,iBAAkB,CAAEh3C,SAAUo5C,IAGrC,MAAMr7C,EAAe,GAAgBh2E,EAAQ,IACvC2xH,EAAkB37C,EAAa/D,WAC/B2/C,EAAuBD,EAAgBr+C,UAiD7C,OA/CA2C,GAAsBD,EAAcw7C,EAAa,CAAEt4C,qCAAqC,IAExFy4C,EAAgBp5I,IAAK,CACpB88B,MAAOw8G,GAAwBzmC,EAAOmmC,EAAel8G,OACrDhQ,MAAO,KACPsrE,SAAS,IAGVihD,EAAqBr2I,OAAQ,SAC7Bq2I,EAAqBr5I,IAAK,CACzB88B,MAAO+1E,IAGRumC,EAAgB7jJ,KAAM,QAASmN,OAAQu2I,EAAa,QAAQ,IAAKM,KAChE,MAAMpmJ,EAAQomJ,EAAMl8C,UAAW,IAE/B,OAASlqG,EAAQ,EAAM6lJ,EAAe/gD,KAAOghD,EAAa9lJ,GAAQ8kG,IAAI,IAGvEmhD,EAAgB7jJ,KAAM,SAAUmN,OAAQu2I,EAAa,QAAQ,IAAKM,KACjE,MAAMpmJ,EAAQomJ,EAAMl8C,UAAW,IAE/B,OAAOi8C,GAAwBzmC,EAAS1/G,EAAQ,EAAM6lJ,EAAel8G,MAASm8G,EAAa9lJ,GAAQ2pC,MAAQ,IAG5Gs8G,EAAgB7jJ,KAAM,QAASmN,OAAQu2I,EAAa,QAAQ,IAAKM,IAAWA,EAAM32H,KAAM,MAExFw2H,EAAgB7jJ,KAAM,SACpBmN,OAAQu2I,EAAa,QAAQ,IAAKM,IAAWA,EAAM32H,KAAM,IAAa,8BAA2B,IAEnGw2H,EAAgBx7I,GAAI,WAAW,KACxBq7I,EAAYr2H,MAAM,EAAIg+E,UAAYA,IAGvCnD,EAAa7D,QAAU6D,EAAa7D,OAFpCo/C,EAAej6I,KAAM,UAGtB,IAGD0+F,EAAaloG,KAAM,aACjBmN,OAAQu2I,EAAa,aAAa,IAAKrO,IAAgBA,EAAWhoH,KAAM,MAI1Er0B,KAAKwP,SAAU0/F,EAAc,WAAW,KACvClvG,KAAKs0B,OAAO6qE,QAAQllE,KAAKgT,OAAO,IAG1BiiE,CAAY,GAErB,CAKQ,aAAAsvC,CAAeyM,GACtB,MAAML,EAAaK,EAAajpJ,KAEhChC,KAAKs0B,OAAO6E,GAAG61E,iBAAiBnhG,IAAK88I,GAAoBC,IAAc1xH,IACtE,MAAMxD,EAA6B11B,KAAKs0B,OAAO0qE,SAAS52F,IAAK,cACvD6xB,EAAO,IAAI,GAAYf,GAa7B,OAXAe,EAAKxoB,IAAK,CACT88B,MAAO08G,EAAa3mC,MACpB5a,KAAMuhD,EAAavhD,KACnBG,SAAS,EACTP,cAAc,IAGfrvE,EAAKjzB,KAAM,aAAczH,GAAIm2B,EAAS,aACtCuE,EAAKjzB,KAAM,QAASzH,GAAIm2B,EAAS,SAASl9B,GAASA,IAAUoyJ,IAC7D3wH,EAAK5qB,GAAI,UAAWrP,KAAKkrJ,gBAAgBlkJ,KAAMhH,KAAM4qJ,IAE9C3wH,CAAI,GAEb,CAEQ,eAAAixH,CAAiBlpJ,GACxBhC,KAAKs0B,OAAOiB,QAAS,aAAc,CAAE/8B,MAAOwJ,IAC5ChC,KAAKs0B,OAAO6qE,QAAQllE,KAAKgT,OAC1B,EAMD,SAASm9G,GACRvrH,EACAm9G,GAEA,UAAY32I,KAASw5B,EAGfm9G,EAAQ32I,EAAMi/G,SAClBj/G,EAAMi/G,MAAQ03B,EAAQ32I,EAAMi/G,QAI9B,OAAOzlF,CACR,CAKA,SAAS8rH,GAAoB3oJ,GAC5B,MAAO,cAAeA,GACvB,CAKA,SAAS+oJ,GAAwBI,EAAmCC,GACnE,OAASD,EAAgBA,EAAgB,KAAO,IAAOC,CACxD,CChOe,MAAMC,WAAsB,GAI1C,qBAAkB/0H,GACjB,MAAO,eACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAEpBA,EAAO0qE,SAASnxF,IAAK,SAAU,IAAI2nB,GAAclB,IACjDA,EAAO0qE,SAASnxF,IAAK,UAAW,IAAI2nB,GAAclB,GACnD,ECnCD,iZCAA,yYCuBe,MAAMg3H,WAAiB,GAIrC,qBAAkBh1H,GACjB,MAAO,UACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd4E,EAAS5E,EAAO4E,OAChBl+B,EAAIs5B,EAAOt5B,EAEXuwJ,EAAoD,OAA9BryH,EAAOvJ,oBAA+B67H,GAAaC,GACzEC,EAAqD,OAA9BxyH,EAAOvJ,oBAA+B87H,GAAcD,GAEjFxrJ,KAAK2rJ,cAAe,SAAU3wJ,EAAG,mBAAqBuwJ,GACtDvrJ,KAAK2rJ,cAAe,UAAW3wJ,EAAG,mBAAqB0wJ,EACxD,CAKQ,aAAAC,CAAe/xH,EAAmC2U,EAAem7D,GACxE,MAAMp1E,EAASt0B,KAAKs0B,OAEpBA,EAAO6E,GAAG61E,iBAAiBnhG,IAAK+rB,GAAaV,IAC5C,MAAMxD,EAAUpB,EAAO0qE,SAAS52F,IAAKwxB,GAC/BK,EAAO,IAAI,GAAYf,GAe7B,OAbAe,EAAKxoB,IAAK,CACT88B,QACAm7D,OACAG,SAAS,IAGV5vE,EAAKjzB,KAAM,aAAczH,GAAIm2B,EAAS,aAEtC11B,KAAKwP,SAAUyqB,EAAM,WAAW,KAC/B3F,EAAOiB,QAASqE,GAChBtF,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBhT,CAAI,GAEb,ECtDc,MAAM2xH,GAArB,cAKC,KAAQC,aAAe,IAAIj4I,GAAgD,CAM3E,UAAWtb,GACV,OAAO0H,KAAK6rJ,aAAah4I,IAC1B,CAOO,GAAAhG,CAAK5N,GACNU,MAAMC,QAASX,GACnBA,EAAKrC,SAASqC,GAAQD,KAAK6rJ,aAAah+I,IAAK5N,KAE7CD,KAAK6rJ,aAAah+I,IAAK5N,EAEzB,CAOO,aAAA6rJ,GACN,OAAOtsF,IACNA,EAAWnwD,GAA4B,sBAAsB,CAAEgG,EAAKvS,EAAMw8D,KAKzE,IAAMA,EAAcwB,WAAW7+D,KAAMa,EAAK7C,KAAM,sBAC/C,OAID,IAAQ6C,EAAK7C,KAAKiO,GAAI,eAAiBoxD,EAAcoE,OAAOkE,SAAU9kE,EAAK7C,MAC1E,OAGD,MAAMoqE,EAAa/K,EAAc9kC,OAC3Bwc,EAAgBqzB,EAAW7mE,SAASyxB,UAE1C,UAAYh1B,KAAQD,KAAK6rJ,aAAe,CACvC,MAAMt0G,EAAc8yB,EAAWz2B,uBAAwB,IAAK3zC,EAAKsF,WAAY,CAC5EqH,SAAU,IAGN3M,EAAKy+B,SACT2rC,EAAWzuC,SAAU37B,EAAKy+B,QAAS6Y,GAGpC,UAAYxgD,KAAOkJ,EAAK4+B,OACvBwrC,EAAW31B,SAAU39C,EAAKkJ,EAAK4+B,OAAQ9nC,GAAOwgD,GAG/C8yB,EAAWz1B,kBAAmB,QAAQ,EAAM2C,GAEvCt3C,EAAKsP,SAAUzM,EAAK29D,mBACnB39D,EAAK7C,KAAKiO,GAAI,aAClBm8D,EAAWvzB,KAAME,EAAc3J,gBAAkBkK,GAEjD8yB,EAAWvzB,KAAMwoB,EAAc9B,OAAOH,YAAav6D,EAAKgiB,OAASyyB,GAGlE8yB,EAAWnzB,OAAQooB,EAAc9B,OAAOH,YAAav6D,EAAKgiB,OAASyyB,EAErE,IACE,CAAE3qC,SAAU,QAAU,CAE3B,CAQO,2BAAAm/I,GACN,OAAOvsF,IACNA,EAAWnwD,GAAqC,iCAAiC,CAAEgG,EAAKvS,GAAQ03B,SAAQgjC,aACvG,MAAMmnF,EAAannF,EAAOf,cAAe35D,EAAK7C,MACxC44I,EAAcl4I,MAAMrB,KAAMqlJ,EAAWnpH,eACzCjK,MAAQ5Z,GAAiCA,EAAMzJ,GAAI,UAAW,OAEhE,UAAYjO,KAAQD,KAAK6rJ,aAAe,CACvC,MAAMtmJ,EAAa,GAAOtF,EAAKsF,YAE/B,GAAKtF,EAAKsP,SAAUzM,EAAK29D,mBAAuC,CAC/D,UAAc1pE,EAAKuD,KAASiL,EAGd,UAARxO,EACJyjC,EAAOoB,SAAUthC,EAAKu+I,GAEtBr+G,EAAO/0B,aAAc1O,EAAKuD,EAAKu+I,GAI5B54I,EAAKy+B,SACTlE,EAAOoB,SAAU37B,EAAKy+B,QAASm6G,GAGhC,UAAY9hJ,KAAOkJ,EAAK4+B,OACvBrE,EAAOka,SAAU39C,EAAKkJ,EAAK4+B,OAAQ9nC,GAAO8hJ,EAE5C,KAAO,CACN,UAAc9hJ,EAAKuD,KAASiL,EACd,UAARxO,EACJyjC,EAAOO,YAAazgC,EAAKu+I,GAEzBr+G,EAAO9zB,gBAAiB3P,EAAK8hJ,GAI1B54I,EAAKy+B,SACTlE,EAAOO,YAAa96B,EAAKy+B,QAASm6G,GAGnC,UAAY9hJ,KAAOkJ,EAAK4+B,OACvBrE,EAAOma,YAAa59C,EAAK8hJ,EAE3B,CACD,IACE,CAEL,ECvID,SANA,SAAmBtgI,EAAOmoB,EAAOC,GAC/B,IAAIroC,EAASigB,EAAMjgB,OAEnB,OADAqoC,OAAc94B,IAAR84B,EAAoBroC,EAASqoC,GAC1BD,GAASC,GAAOroC,EAAUigB,EAAQ,GAAUA,EAAOmoB,EAAOC,EACrE,ECdA,IAWIqrH,GAAeriJ,OAAO,uFAa1B,SAJA,SAAoBlN,GAClB,OAAOuvJ,GAAa/pJ,KAAKxF,EAC3B,ECZA,SAJA,SAAsBA,GACpB,OAAOA,EAAOM,MAAM,GACtB,ECRA,IAAI,GAAgB,kBAQhBkvJ,GAAW,IAAM,GAAgB,IACjCC,GAAU,kDACVC,GAAS,2BAETC,GAAc,KAAO,GAAgB,IACrCC,GAAa,kCACbC,GAAa,qCAIbC,GAPa,MAAQL,GAAU,IAAMC,GAAS,IAOtB,IACxBK,GAAW,oBAEXC,GAAQD,GAAWD,IADP,gBAAwB,CAACH,GAAaC,GAAYC,IAAYpvJ,KAAK,KAAO,IAAMsvJ,GAAWD,GAAW,MAElHG,GAAW,MAAQ,CAACN,GAAcF,GAAU,IAAKA,GAASG,GAAYC,GAAYL,IAAU/uJ,KAAK,KAAO,IAGxGyvJ,GAAYhjJ,OAAOwiJ,GAAS,MAAQA,GAAS,KAAOO,GAAWD,GAAO,KAa1E,SAJA,SAAwBhwJ,GACtB,OAAOA,EAAOI,MAAM8vJ,KAAc,EACpC,ECpBA,SANA,SAAuBlwJ,GACrB,OAAO,GAAWA,GACd,GAAeA,GACf,GAAaA,EACnB,ECMA,SCTA,SAAyByY,GACvB,OAAO,SAASzY,GACdA,EAAS,GAASA,GAElB,IAAImwJ,EAAa,GAAWnwJ,GACxB,GAAcA,QACdoL,EAEAglJ,EAAMD,EACNA,EAAW,GACXnwJ,EAAO4wB,OAAO,GAEdo/B,EAAWmgG,EACX,GAAUA,EAAY,GAAG1vJ,KAAK,IAC9BT,EAAOqF,MAAM,GAEjB,OAAO+qJ,EAAI33I,KAAgBu3C,CAC7B,CACF,CDXiB,CAAgB,eEU3BqgG,GAAwB,8DACxBC,GAAW,kEAGXC,GAAgB,oFAIhBC,GAAmB,2BAKZ,GAAiB,SAYvB,SAAS,GAAmB7sC,GAAc,OAAE5lF,IAElD,MAAM0yH,EAAc1yH,EAAOoZ,uBAAwB,IAAK,CAAEwsE,QAAQ,CAAExzG,SAAU,IAI9E,OAFA4tB,EAAOoa,kBAAmB,QAAQ,EAAMs4G,GAEjCA,CACR,CAWO,SAASC,GAAe9e,GAC9B,MAAM+e,EAAYnyI,OAAQozH,GAE1B,OAMD,SAAoBA,GACnB,MAAMgf,EAAgBhf,EAAIp1H,QAAS6zI,GAAuB,IAE1D,QAASO,EAAcxwJ,MAAOkwJ,GAC/B,CAVQO,CAAWF,GAAcA,EAAY,GAC7C,CAmEO,SAASG,GAAmB91I,EAAyBisD,GAC3D,QAAMjsD,GAICisD,EAAOmE,eAAgBpwD,EAAQzV,KAAM,WAC7C,CAgBO,SAASwrJ,GAA6BnuJ,EAAcouJ,GAC1D,MAAMC,GAZkBl1J,EAYE6G,EAXnB2tJ,GAAc/qJ,KAAMzJ,GAWQ,UAAYi1J,GAZzC,IAAkBj1J,EAaxB,MAAMm1J,IAAqBD,IAAaE,GAAiBvuJ,GAEzD,OAAOA,GAAQsuJ,EAAmBD,EAAWruJ,EAAOA,CACrD,CAKO,SAASuuJ,GAAiBvuJ,GAChC,OAAO4tJ,GAAiBhrJ,KAAM5C,EAC/B,CAKO,SAASwuJ,GAAUxuJ,GACzBkE,OAAO+pI,KAAMjuI,EAAM,SAAU,WAC9B,CCnKe,MAAMyuJ,WAAoB,GAAzC,kCAeC,KAAgBC,iBAAmB,IAAI,GAMvC,KAAgBC,oBAAsB,IAAIpC,EAAoB,CAKvD,4BAAAqC,GACN,UAAYC,KAAmBluJ,KAAK+tJ,iBACnCG,EAAgB11J,MAAQwH,KAAKmuJ,4BAA6BD,EAAgBruJ,GAE5E,CAKgB,OAAAk1B,GACf,MAAM38B,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3Bib,EAAkBjb,EAAU+Y,sBAAwB,GAAO/Y,EAAU0tC,qBAItE4qF,GAAmBr9G,EAAiB93C,EAAMsrE,SAC9C1jE,KAAKxH,MAAQ03C,EAAgB7U,aAAc,YAC3Cr7B,KAAK00B,UAAYt8B,EAAMsrE,OAAOmE,eAAgB33B,EAAiB,cAE/DlwC,KAAKxH,MAAQy8B,EAAUoG,aAAc,YACrCr7B,KAAK00B,UAAYt8B,EAAMsrE,OAAO6S,0BAA2BthD,EAAW,aAGrE,UAAYi5H,KAAmBluJ,KAAK+tJ,iBACnCG,EAAgB11J,MAAQwH,KAAKmuJ,4BAA6BD,EAAgBruJ,GAE5E,CAmEgB,OAAA01B,CAAS6qF,EAAcguC,EAA8C,CAAC,GACrF,MAAMh2J,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAE3Bo5H,EAAwC,GACxCC,EAAuC,GAE7C,UAAYtsJ,KAAQosJ,EACdA,EAAoBpsJ,GACxBqsJ,EAAuBzuJ,KAAMoC,GAE7BssJ,EAAsB1uJ,KAAMoC,GAI9B5J,EAAMmiC,QAAQC,IAEb,GAAKvF,EAAU+V,YAAc,CAC5B,MAAMvlB,EAAWwP,EAAUE,mBAG3B,GAAKF,EAAUgR,aAAc,YAAe,CAC3C,MAAMsoH,EAAWC,GAA0Bv5H,GAE3C,IAAIw5H,EAAYj0B,GAAoB/0G,EAAU,WAAYwP,EAAUoG,aAAc,YAAcjjC,GAE3F68B,EAAUoG,aAAc,cAAiBkzH,IAC7CE,EAAYzuJ,KAAK0uJ,mBAAoBt2J,EAAOoiC,EAAQi0H,EAAWruC,IAGhE5lF,EAAO/0B,aAAc,WAAY26G,EAAMquC,GAEvCJ,EAAuBzwJ,SAASqC,IAC/Bu6B,EAAO/0B,aAAcxF,GAAM,EAAMwuJ,EAAW,IAG7CH,EAAsB1wJ,SAASqC,IAC9Bu6B,EAAO9zB,gBAAiBzG,EAAMwuJ,EAAW,IAI1Cj0H,EAAOgZ,aAAchZ,EAAOmd,oBAAqB82G,EAAU9tH,IAAIwJ,YAChE,SAImB,KAATi2E,EAAc,CACvB,MAAM76G,EAAa,GAAO0vB,EAAU+Q,iBAEpCzgC,EAAWkM,IAAK,WAAY2uG,GAE5BiuC,EAAuBzwJ,SAASqC,IAC/BsF,EAAWkM,IAAKxR,GAAM,EAAM,IAG7B,MAAQ0gC,IAAKo6D,GAAkB3iG,EAAMqjG,cAAejhE,EAAOmZ,WAAYysE,EAAM76G,GAAckgB,GAI3F+U,EAAOgZ,aAAcunD,EACtB,CAIA,CAAE,cAAeszD,KAA2BC,GAAwB1wJ,SAASqC,IAC5Eu6B,EAAOskD,yBAA0B7+E,EAAM,GAEzC,KAAO,CAGN,MAAM4rC,EAASzzC,EAAMsrE,OAAO8S,eAAgBvhD,EAAUmY,YAAa,YAG7DuhH,EAAgB,GAEtB,UAAYl3I,KAAWwd,EAAU0tC,oBAC3BvqE,EAAMsrE,OAAOmE,eAAgBpwD,EAAS,aAC1Ck3I,EAAc/uJ,KAAM46B,EAAOsd,cAAergC,IAK5C,MAAMm3I,EAAiBD,EAAc7sJ,QAIrC,UAAYgjB,KAAS+mB,EACf7rC,KAAK6uJ,iBAAkB/pI,EAAO6pI,IAClCC,EAAehvJ,KAAMklB,GAIvB,UAAYA,KAAS8pI,EAAiB,CACrC,IAAIH,EAAY3pI,EAEhB,GAA+B,IAA1B8pI,EAAet2J,OAAe,CAElC,MAAMi2J,EAAWC,GAA0Bv5H,GAEtCA,EAAUoG,aAAc,cAAiBkzH,IAC7CE,EAAYzuJ,KAAK0uJ,mBAAoBt2J,EAAOoiC,EAAQ1V,EAAOs7F,GAC3D5lF,EAAOgZ,aAAchZ,EAAOwd,gBAAiBy2G,IAE/C,CAEAj0H,EAAO/0B,aAAc,WAAY26G,EAAMquC,GAEvCJ,EAAuBzwJ,SAASqC,IAC/Bu6B,EAAO/0B,aAAcxF,GAAM,EAAMwuJ,EAAW,IAG7CH,EAAsB1wJ,SAASqC,IAC9Bu6B,EAAO9zB,gBAAiBzG,EAAMwuJ,EAAW,GAE3C,CACD,IAEF,CAQQ,2BAAAN,CAA6BW,GACpC,MAAM12J,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3Bib,EAAkBjb,EAAU+Y,qBAIlC,OAAKu/G,GAAmBr9G,EAAiB93C,EAAMsrE,QACvCxzB,EAAgB7U,aAAcyzH,GAG/B75H,EAAUoG,aAAcyzH,EAChC,CAQQ,gBAAAD,CAAkB/pI,EAAc6pI,GACvC,UAAYI,KAAgBJ,EAE3B,GAAKI,EAAavjH,cAAe1mB,GAChC,OAAO,EAIT,OAAO,CACR,CAUQ,kBAAA4pI,CAAoBt2J,EAAcoiC,EAAgB1V,EAAcs7F,GACvE,MAAMlmF,EAAOM,EAAOmZ,WAAYysE,EAAM,CAAE00B,SAAU10B,IAElD,OAAOhoH,EAAMqjG,cAAevhE,EAAMpV,EACnC,EAID,SAAS0pI,GAA0Bv5H,GAClC,GAAKA,EAAU+V,YAAc,CAC5B,MAAMi4F,EAAgBhuG,EAAUE,mBAEhC,OAAO8tG,EAAexlG,UAAYwlG,EAAexlG,SAAS36B,IAC3D,CAAO,CACN,MAAMksJ,EAAaruJ,MAAMrB,KAAM21B,EAAUoY,gBAAiBnB,YAE1D,GAAK8iH,EAAW12J,OAAS,EACxB,OAAO,KAGR,MAAM+2I,EAAY2f,EAAY,GAE9B,OAAK3f,EAAUnhI,GAAI,UAAamhI,EAAUnhI,GAAI,cACtCmhI,EAAUvsI,KAGX,IACR,CACD,CC3Te,MAAMmsJ,WAAsB,GAI1B,OAAAl6H,GACf,MAAM38B,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3Bib,EAAkBjb,EAAU+Y,qBAI7Bu/G,GAAmBr9G,EAAiB93C,EAAMsrE,QAC9C1jE,KAAK00B,UAAYt8B,EAAMsrE,OAAOmE,eAAgB33B,EAAiB,YAE/DlwC,KAAK00B,UAAYt8B,EAAMsrE,OAAO6S,0BAA2BthD,EAAW,WAEtE,CAegB,OAAAM,GACf,MAAMjB,EAASt0B,KAAKs0B,OACdl8B,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3Bg+G,EAAuC3+G,EAAO0qE,SAAS52F,IAAK,QAElEhQ,EAAMmiC,QAAQC,IAEb,MAAM00H,EAAiBj6H,EAAU+V,YAChC,CAAEwvF,GACDvlG,EAAUE,mBACV,WACAF,EAAUoG,aAAc,YACxBjjC,IAEDA,EAAMsrE,OAAO8S,eAAgBvhD,EAAUmY,YAAa,YAGrD,UAAYtoB,KAASoqI,EAGpB,GAFA10H,EAAO9zB,gBAAiB,WAAYoe,GAE/BmuH,EACJ,UAAYib,KAAmBjb,EAAY8a,iBAC1CvzH,EAAO9zB,gBAAiBwnJ,EAAgBruJ,GAAIilB,EAG/C,GAEF,EC3Dc,MAAMqqI,WAAwBh8I,KAmD5C,WAAApR,EAAa,GAAElC,EAAE,MAAE0uC,EAAK,WAAEhpC,EAAU,QAAEm5B,EAAO,OAAEG,EAAM,aAAE+B,IACtDpzB,QAEAxN,KAAKH,GAAKA,EACVG,KAAKyR,IAAK,aAAS,GACnBzR,KAAK4gC,aAAeA,EACpB5gC,KAAKuuC,MAAQA,EACbvuC,KAAKuF,WAAaA,EAClBvF,KAAK0+B,QAAUA,EACf1+B,KAAK6+B,OAASA,CACf,CAOO,cAAAuwH,GACN,MAAO,CACN7pJ,WAAYvF,KAAKuF,WACjBm5B,QAAS1+B,KAAK0+B,QACdG,OAAQ7+B,KAAK6+B,OAEf,E,eCzFG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ/E,O,0VC2CvB,MACMu1H,GAAsB,YAEtBC,GAAwB,kBAQf,MAAM,WAAoB,GAIxC,qBAAkBh5H,GACjB,MAAO,aACR,CAKA,mBAAkBY,GAEjB,MAAO,CAAE,GAAsB08F,GAAO,GACvC,CAKA,WAAA7xH,CAAauyB,GACZ9mB,MAAO8mB,GAEPA,EAAOtL,OAAOtyB,OAAQ,OAAQ,CAC7B64J,0BAA0B,GAE5B,CAKO,IAAA74H,GACN,MAAMpC,EAASt0B,KAAKs0B,OAGpBA,EAAOl8B,MAAMsrE,OAAOxwB,OAAQ,QAAS,CAAE0iC,gBAAiB,aAExDthD,EAAOj2B,WAAWuhF,IAAK,gBACrB5V,mBAAoB,CAAE5xE,MAAO,WAAY6hC,KAAM,KAEjD3F,EAAOj2B,WAAWuhF,IAAK,mBACrB5V,mBAAoB,CAAE5xE,MAAO,WAAY6hC,KAAM,CAAEmmF,EAAM9gD,IAChD,GAAmB6tF,GAAe/sC,GAAQ9gD,KAGnDhrC,EAAOj2B,WAAWuhF,IAAK,UACrBzR,mBAAoB,CACpBl0C,KAAM,CACLj4B,KAAM,IACNuD,WAAY,CACX66G,MAAM,IAGRhoH,MAAO,CACNrB,IAAK,WACLyB,MAAS++C,GAA8BA,EAAYlc,aAAc,WAKpE/G,EAAO0qE,SAASnxF,IAAK,OAAQ,IAAIigJ,GAAax5H,IAC9CA,EAAO0qE,SAASnxF,IAAK,SAAU,IAAIohJ,GAAe36H,IAElD,MAAMk7H,EL9BD,SACNx0J,EACAy0J,GAEA,MAAMC,EAAoD,CACzD,oBAAqB10J,EAAG,qBACxB,aAAgBA,EAAG,iBAWpB,OARAy0J,EAAW7xJ,SAAS+xJ,IACd,UAAWA,GAAaD,EAA2BC,EAAUphH,SACjEohH,EAAUphH,MAAQmhH,EAA2BC,EAAUphH,QAGjDohH,KAGDF,CACR,CKYyBG,CAAwBt7H,EAAOt5B,ELNjD,SAA8By0J,GACpC,MAAMI,EAAqD,GAE3D,GAAKJ,EACJ,UAAc14J,EAAKyB,KAAWxB,OAAOyhB,QAASg3I,GAAe,CAC5D,MAAME,EAAY34J,OAAO+wB,OACxB,CAAC,EACDvvB,EACA,CAAEqH,GAAI,OAAQ,GAAY9I,OAG3B84J,EAASjwJ,KAAM+vJ,EAChB,CAGD,OAAOE,CACR,CKV2DC,CAAqBx7H,EAAOtL,OAAO5gB,IAAK,qBAEjGpI,KAAK+vJ,2BAA4BP,EAC/BzpJ,QAAU9F,GAA8DA,EAAK02G,OAAS04C,MACxFrvJ,KAAKgwJ,wBAAyBR,EAC5BzpJ,QAAU9F,GA5EW,WA4EgDA,EAAK02G,QAGzCriF,EAAOqC,QAAQvuB,IAAK,IAC5BwvH,kBAAmB,YCjGjC,SACdtjG,EACAsS,EACA+a,EACAxb,GAEA,MAAMlM,EAAO3F,EAAO6qE,QAAQllE,KACtBg2H,EAAsB,IAAIr8I,IAGhCqmB,EAAKz2B,SAASk3B,mBAAmBF,IAChC,MAAMvF,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UACxC,IAAI2xC,GAAU,EAEd,GAAK3xC,EAAUgR,aAAcW,GAAkB,CAC9C,MAAM02B,EAAak9D,GAClBvlG,EAAUE,mBACVyR,EACA3R,EAAUoG,aAAcuL,GACxBtS,EAAOl8B,OAEFquD,EAAYnyB,EAAO6qE,QAAQ3hC,OAAOH,YAAaC,GAIrD,UAAYr9D,KAAQwmD,EAAUva,WACxBjsC,EAAKiO,GAAI,UAAWyzC,KAAc1hD,EAAK66B,SAAUqL,KACrD3L,EAAOoB,SAAUuK,EAAWlmC,GAC5BgwJ,EAAoBpiJ,IAAK5N,GACzB2mE,GAAU,EAGb,CAEA,OAAOA,CAAO,IAIftyC,EAAOj2B,WAAWuhF,IAAK,mBAAoB/xE,KAAK2xD,IAO/C,SAASmM,IACR1xC,EAAKM,QAAQC,IACZ,UAAYv6B,KAAQgwJ,EAAoB/kJ,SACvCsvB,EAAOO,YAAaoL,EAAWlmC,GAC/BgwJ,EAAoBt+I,OAAQ1R,EAC7B,GAEF,CAZAu/D,EAAWnwD,GAAI,SAAUs8D,EAAiB,CAAE/+D,SAAU,YACtD4yD,EAAWnwD,GAAI,SAAUs8D,EAAiB,CAAE/+D,SAAU,YACtD4yD,EAAWnwD,GAAI,YAAas8D,EAAiB,CAAE/+D,SAAU,YACzD4yD,EAAWnwD,GAAI,YAAas8D,EAAiB,CAAE/+D,SAAU,WASzD,GAEF,CD8CE,CAAiB0nB,EAAQ,WAAY,IArFf,oBAwFtBt0B,KAAKkwJ,kBAGLlwJ,KAAKmwJ,+CAGLnwJ,KAAKowJ,2BAGLpwJ,KAAKqwJ,wBAGLrwJ,KAAKswJ,gCAGLtwJ,KAAKuwJ,6BACN,CAWQ,0BAAAR,CAA4BS,GACnC,MAAMl8H,EAASt0B,KAAKs0B,OAId05H,EADuB15H,EAAO0qE,SAAS52F,IAAK,QACd4lJ,oBAG/B15H,EAAOtL,OAAO5gB,IAAK,kCACvB4lJ,EAAoBngJ,IAAK,CACxBhO,GAAI,iBACJ82G,KAAM04C,GACN9/I,SAAU8+H,KAASA,GAAOihB,GAAsBrtJ,KAAMosI,GACtD9oI,WAAY,CACX3B,OAAQ,SACR6sJ,IAAK,yBAKRzC,EAAoBngJ,IAAK2iJ,GAEpBxC,EAAoB11J,QACxBg8B,EAAOj2B,WAAWuhF,IAAK,YAAa/xE,IAAKmgJ,EAAoBlC,gBAE/D,CAWQ,uBAAAkE,CAAyBU,GAChC,IAAMA,EAA2Bp4J,OAChC,OAGD,MAAMg8B,EAASt0B,KAAKs0B,OAEdy5H,EADuBz5H,EAAO0qE,SAAS52F,IAAK,QACjB2lJ,iBAEjC2C,EAA2B9yJ,SAAS+yJ,IACnCr8H,EAAOl8B,MAAMsrE,OAAOxwB,OAAQ,QAAS,CAAE0iC,gBAAiB+6E,EAAoB9wJ,KAG5E,MAAM8vJ,EAAY,IAAIR,GAAiBwB,GAEvC5C,EAAiBlgJ,IAAK8hJ,GAEtBr7H,EAAOj2B,WAAWuhF,IAAK,YAAa5V,mBAAoB,CACvD5xE,MAAOu3J,EAAU9vJ,GACjBo6B,KAAM,CAAE22H,GAAwBp2H,SAAQkpC,WAAYzjE,WAEnD,IAAQA,EAAKiO,GAAI,cAAiBw1D,EAAOkE,SAAU3nE,KAI9C2wJ,EAAuB,CAC3B,MAAMn5I,EAAU+iB,EAAOoZ,uBAAwB,IAAK+7G,EAAUpqJ,WAAY,CAAEqH,SAAU,IAEjF+iJ,EAAUjxH,SACdlE,EAAOoB,SAAU+zH,EAAUjxH,QAASjnB,GAGrC,UAAY1gB,KAAO44J,EAAU9wH,OAC5BrE,EAAOka,SAAU39C,EAAK44J,EAAU9wH,OAAQ9nC,GAAO0gB,GAKhD,OAFA+iB,EAAOoa,kBAAmB,QAAQ,EAAMn9B,GAEjCA,CACR,KAIF6c,EAAOj2B,WAAWuhF,IAAK,UAAWzR,mBAAoB,CACrDl0C,KAAM,IACLj4B,KAAM,KACH2tJ,EAAUP,kBAEdh3J,MAAO,CACNrB,IAAK44J,EAAU9vJ,KAEd,GAEL,CAMQ,eAAAqwJ,GACP,MAAM57H,EAASt0B,KAAKs0B,OAEdk/B,EADOl/B,EAAO6qE,QAAQllE,KACFz2B,SAE1BxD,KAAKwP,SAAkCgkD,EAAc,SAAS,CAAEn+C,EAAKvS,KAGpE,KAFmB,EAAIgG,MAAQhG,EAAKmoD,SAASv9B,QAAU5qB,EAAKmoD,SAASz9B,SAGpE,OAGD,IAAIqjI,EAAiC/tJ,EAAKovC,UAM1C,GAJ6C,KAAxC2+G,EAAelvG,QAAQ/4C,gBAC3BioJ,EAAiBA,EAAetzC,QAAS,OAGpCszC,EACL,OAGD,MAAMxiB,EAAMwiB,EAAex1H,aAAc,QAEnCgzG,IAINh5H,EAAIlJ,OACJrJ,EAAKowB,iBAEL26H,GAAUxf,GAAK,GACb,CAAE9gI,QAAS,aAGdvN,KAAKwP,SAAoCgkD,EAAc,WAAW,CAAEn+C,EAAKvS,KACxE,MACMurI,EAD2B/5G,EAAO0qE,SAAS52F,IAAK,QAC7B5P,QACJ61I,GAAOvrI,EAAKwqB,UAAYjB,GAASM,OAAS7pB,EAAKyqB,SAMpElY,EAAIlJ,OAEJ0hJ,GAAUxf,GAAK,GAEjB,CAWQ,4CAAA8hB,GACP,MACM/3J,EADS4H,KAAKs0B,OACCl8B,MACf68B,EAAY78B,EAAMoL,SAASyxB,UAEjCj1B,KAAKwP,SAAmCpX,EAAO,iBAAiB,KAC/D,MAAM+xC,EAAalV,EAAUyG,OAAQyO,WAC/BD,EAAYjV,EAAUyG,OAAQwO,UAW9BjV,EAAUgR,aAAc,aAexBkE,GAiBAA,EAAWlE,aAAc,cAkB1BiE,GAAaA,EAAUjE,aAAc,aAI1C7tC,EAAMmiC,QAAQC,IACbs2H,GAAmCt2H,EAAQu2H,GAAgC34J,EAAMsrE,QAAU,IACzF,GACD,CAAE92D,SAAU,OAChB,CAWQ,wBAAAwjJ,GACP,MAAM97H,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MAErBk8B,EAAO6qE,QAAQllE,KAAKu6B,YAAa,IAEjC,IAAIw8F,GAAU,EAGdhxJ,KAAKwP,SAAsC8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,aAAa,KACrFwtJ,GAAU,CAAI,IAIfhxJ,KAAKwP,SAA4C8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,mBAAmB,KACjG,IAAMwtJ,EACL,OAIDA,GAAU,EAEV,MAAM/7H,EAAY78B,EAAMoL,SAASyxB,UAGjC,IAAMA,EAAU+V,YACf,OAID,IAAM/V,EAAUgR,aAAc,YAC7B,OAGD,MAAMxgB,EAAWwP,EAAUE,mBACrBs5H,EAAYj0B,GAAoB/0G,EAAU,WAAYwP,EAAUoG,aAAc,YAAcjjC,IAI7FqtB,EAASyyC,WAAYu2F,EAAU/tH,QAAWjb,EAASyyC,WAAYu2F,EAAU9tH,OAC7EvoC,EAAMmiC,QAAQC,IACbs2H,GAAmCt2H,EAAQu2H,GAAgC34J,EAAMsrE,QAAU,GAE7F,GAEF,CAUQ,qBAAA2sF,GACP,MAAM/7H,EAASt0B,KAAKs0B,OACd2F,EAAO3F,EAAO6qE,QAAQllE,KAG5B,IAAI86G,EAAoE,KAGpEkc,GAAiB,EAGrBjxJ,KAAKwP,SAAmCyqB,EAAKz2B,SAAU,UAAU,KAChEytJ,GAAiB,CAAI,GACnB,CAAErkJ,SAAU,SAIf5M,KAAKwP,SAAmC8kB,EAAOl8B,MAAO,iBAAiB,KACtE,MAAM68B,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAGnCA,EAAU+V,cAKVimH,EACJA,GAAiB,EAMZhnE,GAAU31D,IAmJnB,SAA+Bl8B,GAC9B,MAAM68B,EAAY78B,EAAMoL,SAASyxB,UAC3BguG,EAAgBhuG,EAAUE,mBAC1B+tG,EAAejuG,EAAUuY,kBACzB0jH,EAAsBjuB,EAAc/4F,UAG1C,IAAMgnH,EACL,OAAO,EAIR,IAAMA,EAAoBhjJ,GAAI,SAC7B,OAAO,EAIR,IAAMgjJ,EAAoBjrH,aAAc,YACvC,OAAO,EAKR,MAAMkrH,EAAqBjuB,EAAazlG,UAAYylG,EAAa/4F,WAGjE,GAAK+mH,IAAwBC,EAC5B,OAAO,EAQR,OAHkB32B,GAAoByI,EAAe,WAAYiuB,EAAoB71H,aAAc,YAAcjjC,GAGhGozC,cAAepzC,EAAMy/C,YAAaorF,EAAeC,IAAgB,EACnF,CAnLQkuB,CAAsB98H,EAAOl8B,SACjC28I,EAAsB9/G,EAAU+Q,iBACjC,GACE,CAAEp5B,SAAU,SAIf5M,KAAKwP,SAAU8kB,EAAOl8B,MAAO,iBAAiB,CAAEid,GAAOoC,MACtDw5I,GAAiB,EAGXhnE,GAAU31D,IAIVygH,IAINzgH,EAAOl8B,MAAMmiC,QAAQC,IACpB,UAAcpkB,EAAW5d,KAAWu8I,EACnCv6G,EAAO/0B,aAAc2Q,EAAW5d,EAAOif,EACxC,IAGDs9H,EAAsB,KAAI,GACxB,CAAEnoI,SAAU,QAChB,CAcQ,6BAAA0jJ,GACP,MAAMh8H,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf68B,EAAY78B,EAAMoL,SAASyxB,UAC3BgF,EAAO3F,EAAO6qE,QAAQllE,KAG5B,IAAIo3H,GAA2B,EAG3BC,GAAsB,EAG1BtxJ,KAAKwP,SAAmCyqB,EAAKz2B,SAAU,UAAU,CAAE6R,EAAKvS,KACvEwuJ,EAAyC,aAAnBxuJ,EAAKyhB,SAAc,GACvC,CAAE3X,SAAU,SAIf5M,KAAKwP,SAAmCpX,EAAO,iBAAiB,KAE/Di5J,GAA2B,EAE3B,MAAM5rI,EAAWwP,EAAUE,mBACrB2/G,EAAW7/G,EAAUoG,aAAc,YAEzC,IAAMy5G,EACL,OAGD,MAAM2Z,EAAYj0B,GAAoB/0G,EAAU,WAAYqvH,EAAU18I,GAItEi5J,EAA2B5C,EAAUljH,iBAAkB9lB,IAAcgpI,EAAU9tH,IAAI1c,QAASwB,EAAU,GACpG,CAAE7Y,SAAU,SAGf5M,KAAKwP,SAAmCpX,EAAO,iBAAiB,KAEzDk5J,IAINA,GAAsB,EAGjBD,GAKL/8H,EAAOl8B,MAAMguE,eAAe5rC,IAC3Bs2H,GAAmCt2H,EAAQu2H,GAAgC34J,EAAMsrE,QAAU,IACzF,GACD,CAAE92D,SAAU,OAChB,CAKQ,2BAAA2jJ,GACP,MAAMj8H,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfq1J,EAAkBztJ,KAAKs0B,OAAOtL,OAAO5gB,IAAK,wBAE1CqlJ,GAINztJ,KAAKwP,SAA0C8kB,EAAOqC,QAAQvuB,IAAK,qBAAuB,oBAAoB,CAAEiN,EAAKvS,KACpH1K,EAAMmiC,QAAQC,IACb,MAAM1V,EAAQ0V,EAAOud,cAAej1C,EAAK5C,SAEzC,UAAYD,KAAQ6kB,EAAMonB,WACzB,GAAKjsC,EAAKgmC,aAAc,YAAe,CACtC,MAAMsrH,EAAU/D,GAA6BvtJ,EAAKo7B,aAAc,YAAwBoyH,GAExFjzH,EAAO/0B,aAAc,WAAY8rJ,EAAStxJ,EAC3C,CACD,GACE,GAEL,EAQD,SAAS6wJ,GAAmCt2H,EAAgBg3H,GAC3Dh3H,EAAOskD,yBAA0B,YAEjC,UAAY1oE,KAAao7I,EACxBh3H,EAAOskD,yBAA0B1oE,EAEnC,CA8CA,SAAS6zE,GAAU31D,GAGlB,OAFqBA,EAAOl8B,MAAMmiC,QAAQC,GAAUA,EAAO0rC,QAEvC+jB,QACrB,CAKA,SAAS8mE,GAAgCrtF,GAGxC,OAFuBA,EAAOuR,cAAe,SAAWW,gBAElC7vE,QAAQqQ,GAAaA,EAAUwuC,WAAY,SAClE,C,eEzrBI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ9qB,OC6BR,MAAM23H,WAAqB,GAwDzC,WAAA1vJ,CAAam3B,EAAgB+5G,GAC5BzlI,MAAO0rB,GArDR,KAAgBsxE,aAAe,IAAI,GAKnC,KAAgBpL,WAAa,IAAI,GAgCjC,KAAiBs/C,YAAc,IAAI,GAkBlC,MAAM1jJ,EAAIk+B,EAAOl+B,EAEjBgF,KAAK0xJ,aAAe1xJ,KAAK2xJ,kBACzB3xJ,KAAKu+I,eAAiBv+I,KAAKw+I,cAAexjJ,EAAG,QAAU,GAAMqlG,MAAO,kBACpErgG,KAAKu+I,eAAetzI,KAAO,SAC3BjL,KAAKy+I,iBAAmBz+I,KAAKw+I,cAAexjJ,EAAG,UAAY,GAAMy4B,OAAQ,mBAAoB,UAC7FzzB,KAAK4xJ,yBAA2B5xJ,KAAK6xJ,+BAAgC5e,GACrEjzI,KAAKsX,SAAWtX,KAAK8xJ,oBAAqB7e,EAAY8a,kBAEtD/tJ,KAAK6tG,aAAe,IAAI,GAAa,CACpCnB,WAAY1sG,KAAK0+I,YACjBl0C,aAAcxqG,KAAKwqG,aACnBmC,iBAAkB3sG,KAAKo/F,WACvB1hD,QAAS,CAERyvD,cAAe,cAGfD,UAAW,SAIb,MAAM0zB,EAAY,CAAE,KAAM,eAAgB,sBAErCqS,EAAY8a,iBAAiBz1J,QACjCsoI,EAAUhhI,KAAM,+BAAgC,oBAGjDI,KAAK4jG,YAAa,CACjBzsF,IAAK,OAEL5R,WAAY,CACXg5B,MAAOqiG,EAGPv3B,SAAU,MAGX/xF,SAAUtX,KAAKsX,UAEjB,CASO,yBAAAy6I,GACN,OAAOpxJ,MACLrB,KAAMU,KAAK4xJ,0BACX3jI,QAAQ,CAAE+jI,EAAaC,KACvBD,EAAaC,EAAajwJ,MAASiwJ,EAAa5/C,KACzC2/C,IACL,CAAC,EACN,CAKgB,MAAArgH,GACfnkC,MAAMmkC,SAEN,GAAe,CACd1X,KAAMj6B,OAGY,CAClBA,KAAK0xJ,gBACF1xJ,KAAK4xJ,yBACR5xJ,KAAKu+I,eACLv+I,KAAKy+I,kBAGK7gJ,SAAStE,IAEnB0G,KAAK0+I,YAAY7wI,IAAKvU,GAGtB0G,KAAKwqG,aAAa38F,IAAKvU,EAAEme,QAAU,IAIpCzX,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,QAChC,CAKgB,OAAAuO,GACfxY,MAAMwY,UAENhmB,KAAKwqG,aAAaxkF,UAClBhmB,KAAKo/F,WAAWp5E,SACjB,CAKO,KAAAinB,GACNjtC,KAAK6tG,aAAaZ,YACnB,CAOQ,eAAA0kD,GACP,MAAM32J,EAAIgF,KAAKk5B,OAAQl+B,EACjBqjJ,EAAe,IAAI,GAAkBr+I,KAAKk5B,OAAQ,IAIxD,OAFAmlH,EAAa9vG,MAAQvzC,EAAG,YAEjBqjJ,CACR,CAWQ,aAAAG,CAAejwG,EAAem7D,EAAcvjE,EAAmB30B,GACtE,MAAMgkG,EAAS,IAAI,GAAYx1G,KAAKk5B,QAkBpC,OAhBAs8E,EAAO/jG,IAAK,CACX88B,QACAm7D,OACAG,SAAS,IAGV2L,EAAO/S,eAAgB,CACtBl9F,WAAY,CACXg5B,MAAO4H,KAIJ30B,GACJgkG,EAAOpkG,SAAU,WAAY7R,GAAIS,KAAMwR,GAGjCgkG,CACR,CASQ,8BAAAq8C,CAAgC5e,GACvC,MAAMif,EAAWlyJ,KAAKsjG,mBAEtB,UAAY4qD,KAAmBjb,EAAY8a,iBAAmB,CAC7D,MAAMkE,EAAqD,IAAI,GAAkBjyJ,KAAKk5B,QAEtF+4H,EAAaxgJ,IAAK,CACjBzP,KAAMksJ,EAAgBruJ,GACtB0uC,MAAO2/G,EAAgB3/G,MACvB0gE,UAAU,IAGXgjD,EAAajrJ,KAAM,QAASmN,OAAQ,CAAE+5I,EAAiBjb,GAAe,SAAS,CAAEkf,EAAgB/V,SACxE,IAAjBA,QAAiD,IAAnB+V,IAAiCjE,EAAgBttH,eAAiBuxH,IAGxGF,EAAa5iJ,GAAI,WAAW,KAC3B6+I,EAAgBz8I,IAAK,SAAUwgJ,EAAa5/C,KAAM,IAGnD6/C,EAASrkJ,IAAKokJ,EACf,CAEA,OAAOC,CACR,CAaQ,mBAAAJ,CAAqB/D,GAC5B,MAAMz2I,EAAWtX,KAAKsjG,mBAItB,GAFAhsF,EAASzJ,IAAK7N,KAAK0xJ,cAEd3D,EAAiBz1J,OAAS,CAC9B,MAAM85J,EAAwB,IAAI,GAElCA,EAAsBxuD,YAAa,CAClCzsF,IAAK,KACLG,SAAUtX,KAAK4xJ,yBAAyB50J,KAAKi1J,IAAA,CAC5C96I,IAAK,KACLG,SAAU,CAAE26I,GACZ1sJ,WAAY,CACXg5B,MAAO,CACN,KACA,sBAIHh5B,WAAY,CACXg5B,MAAO,CACN,KACA,WACA,cAIHjnB,EAASzJ,IAAKukJ,EACf,CAKA,OAHA96I,EAASzJ,IAAK7N,KAAKu+I,gBACnBjnI,EAASzJ,IAAK7N,KAAKy+I,kBAEZnnI,CACR,E,eCrUG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQwiB,OCcR,MAAMu4H,WAAwB,GAgD5C,WAAAtwJ,CAAam3B,GACZ1rB,MAAO0rB,GA7CR,KAAgBsxE,aAAe,IAAI,GAKnC,KAAgBpL,WAAa,IAAI,GA2BjC,KAAiBs/C,YAAc,IAAI,GAelC,MAAM1jJ,EAAIk+B,EAAOl+B,EAEjBgF,KAAKsyJ,kBAAoBtyJ,KAAKuyJ,uBAC9BvyJ,KAAKwyJ,iBAAmBxyJ,KAAKw+I,cAAexjJ,EAAG,UChFjD,+zBDgFyE,UACvEgF,KAAKyyJ,eAAiBzyJ,KAAKw+I,cAAexjJ,EAAG,aAAe,GAAM6mG,OAAQ,QAE1E7hG,KAAKyR,IAAK,YAAQ,GAElBzR,KAAK6tG,aAAe,IAAI,GAAa,CACpCnB,WAAY1sG,KAAK0+I,YACjBl0C,aAAcxqG,KAAKwqG,aACnBmC,iBAAkB3sG,KAAKo/F,WACvB1hD,QAAS,CAERyvD,cAAe,cAGfD,UAAW,SAIbltG,KAAK4jG,YAAa,CACjBzsF,IAAK,MAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,kBACA,sBAID8qE,SAAU,MAGX/xF,SAAU,CACTtX,KAAKsyJ,kBACLtyJ,KAAKyyJ,eACLzyJ,KAAKwyJ,mBAGR,CAKgB,MAAA7gH,GACfnkC,MAAMmkC,SAEa,CAClB3xC,KAAKsyJ,kBACLtyJ,KAAKyyJ,eACLzyJ,KAAKwyJ,kBAGK50J,SAAStE,IAEnB0G,KAAK0+I,YAAY7wI,IAAKvU,GAGtB0G,KAAKwqG,aAAa38F,IAAKvU,EAAEme,QAAU,IAIpCzX,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,QAChC,CAKgB,OAAAuO,GACfxY,MAAMwY,UAENhmB,KAAKwqG,aAAaxkF,UAClBhmB,KAAKo/F,WAAWp5E,SACjB,CAKO,KAAAinB,GACNjtC,KAAK6tG,aAAaZ,YACnB,CAUQ,aAAAuxC,CAAejwG,EAAem7D,EAAcl4F,GACnD,MAAMgkG,EAAS,IAAI,GAAYx1G,KAAKk5B,QAUpC,OARAs8E,EAAO/jG,IAAK,CACX88B,QACAm7D,OACAG,SAAS,IAGV2L,EAAOpkG,SAAU,WAAY7R,GAAIS,KAAMwR,GAEhCgkG,CACR,CAOQ,oBAAA+8C,GACP,MAAM/8C,EAAS,IAAI,GAAYx1G,KAAKk5B,QAC9BlyB,EAAOhH,KAAK0iG,aACZ1nG,EAAIgF,KAAKhF,EA4Bf,OA1BAw6G,EAAO/jG,IAAK,CACXw9F,UAAU,EACVpF,QAAS7uG,EAAG,0BAGbw6G,EAAO/S,eAAgB,CACtBl9F,WAAY,CACXg5B,MAAO,CACN,KACA,4BAED6hF,KAAMp5G,EAAKzH,GAAI,QAAQ6gH,GAAQA,GAAQ+sC,GAAe/sC,KACtDx8G,OAAQ,SACR6sJ,IAAK,yBAIPj7C,EAAOxuG,KAAM,SAAUzH,GAAIS,KAAM,QAAQogH,GACjCA,GAAQplH,EAAG,0BAGnBw6G,EAAOxuG,KAAM,aAAczH,GAAIS,KAAM,QAAQogH,KAAUA,IAEvD5K,EAAO3R,SAAU1sF,IAAM,IACvBq+F,EAAO3R,SAAU2C,eAAiB,CAAC,EAE5BgP,CACR,EE5ND,MCmCMk9C,GAA+B,UAQtB,MAAM,WAAe,GAApC,kCAIC,KAAOC,YAAsC,KAK7C,KAAOC,SAAgE,KAUvE,mBAAkB17H,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,QACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAEpBA,EAAO6qE,QAAQllE,KAAKu6B,YAAagqC,IAEjCx+F,KAAKonH,SAAW9yF,EAAOqC,QAAQvuB,IAAK,IAGpCpI,KAAK6yJ,2BACL7yJ,KAAK8yJ,2BAGLx+H,EAAOj2B,WAAWuhF,IAAK,mBAAoB3U,kBAAmB,CAC7D7yE,MAAOs6J,GACPz4H,KAAM,CACLyE,QAAS,CAAE,6BAKbpK,EAAOj2B,WAAWuhF,IAAK,mBAAoBhV,gBAAiB,CAC3DxyE,MAAOs6J,GACPz4H,KAAM,CACLj4B,KAAM,OACN08B,QAAS,CAAE,yBAA0B,sCAGxC,CAKgB,OAAA1Y,GACfxY,MAAMwY,UAGDhmB,KAAK4yJ,UACT5yJ,KAAK4yJ,SAAS5sI,UAGVhmB,KAAK2yJ,aACT3yJ,KAAK2yJ,YAAY3sI,SAEnB,CAKQ,YAAA+sI,GACP/yJ,KAAK2yJ,YAAc3yJ,KAAKgzJ,qBACxBhzJ,KAAK4yJ,SAAW5yJ,KAAKizJ,kBAGrBjzJ,KAAKkzJ,gCACN,CAKQ,kBAAAF,GACP,MAAM1+H,EAASt0B,KAAKs0B,OACdq+H,EAAc,IAAIN,GAAiB/9H,EAAO4E,QAC1C+5G,EAA2B3+G,EAAO0qE,SAAS52F,IAAK,QAChD+qJ,EAA+B7+H,EAAO0qE,SAAS52F,IAAK,UA6B1D,OA3BAuqJ,EAAY3rJ,KAAM,QAASzH,GAAI0zI,EAAa,SAC5C0f,EAAYF,eAAezrJ,KAAM,aAAczH,GAAI0zI,GACnD0f,EAAYH,iBAAiBxrJ,KAAM,aAAczH,GAAI4zJ,GAGrDnzJ,KAAKwP,SAAUmjJ,EAAa,QAAQ,KACnC3yJ,KAAKozJ,cAAc,IAIpBpzJ,KAAKwP,SAAUmjJ,EAAa,UAAU,KACrCr+H,EAAOiB,QAAS,UAChBv1B,KAAKqzJ,SAAS,IAIfV,EAAYvzD,WAAW3tF,IAAK,OAAO,CAAE3O,EAAM2wB,KAC1CzzB,KAAKqzJ,UACL5/H,GAAQ,IAITk/H,EAAYvzD,WAAW3tF,IAAK,IAAgB,CAAE3O,EAAM2wB,KACnDzzB,KAAKozJ,eACL3/H,GAAQ,IAGFk/H,CACR,CAKQ,eAAAM,GACP,MAAM3+H,EAASt0B,KAAKs0B,OACd2+G,EAA2B3+G,EAAO0qE,SAAS52F,IAAK,QAChDqlJ,EAAkBn5H,EAAOtL,OAAO5gB,IAAK,wBAErCwqJ,EAAW,IAAMxwD,GAA4BqvD,IAAlC,CAAoDn9H,EAAO4E,OAAQ+5G,GA2BpF,OAzBA2f,EAASlB,aAAaxnD,UAAUljG,KAAM,SAAUzH,GAAI0zI,EAAa,SAGjE2f,EAASlB,aAAa1qJ,KAAM,aAAczH,GAAI0zI,EAAa,aAC3D2f,EAASrU,eAAev3I,KAAM,aAAczH,GAAI0zI,GAGhDjzI,KAAKwP,SAAUojJ,EAAU,UAAU,KAClC,MAAM,MAAEp6J,GAAUo6J,EAASlB,aAAaxnD,UAAUzyF,QAC5C67I,EAAY9F,GAA6Bh1J,EAAOi1J,GACtDn5H,EAAOiB,QAAS,OAAQ+9H,EAAWV,EAASb,6BAC5C/xJ,KAAKuzJ,gBAAgB,IAItBvzJ,KAAKwP,SAAUojJ,EAAU,UAAU,KAClC5yJ,KAAKuzJ,gBAAgB,IAItBX,EAASxzD,WAAW3tF,IAAK,OAAO,CAAE3O,EAAM2wB,KACvCzzB,KAAKuzJ,iBACL9/H,GAAQ,IAGFm/H,CACR,CAMQ,wBAAAC,GACP,MAAMv+H,EAASt0B,KAAKs0B,OACd2+G,EAA2B3+G,EAAO0qE,SAAS52F,IAAK,QAChDpN,EAAIs5B,EAAOt5B,EAEjBs5B,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,QAAQqrB,IACvC,MAAMs8E,EAAS,IAAI,GAAYt8E,GAgB/B,OAdAs8E,EAAO9gF,WAAY,EACnB8gF,EAAOjnE,MAAQvzC,EAAG,QAClBw6G,EAAO9L,KD/NV,gnBCgOG8L,EAAO5nF,UAAY,GACnB4nF,EAAO3L,SAAU,EACjB2L,EAAOlM,cAAe,EAGtBkM,EAAOxuG,KAAM,aAAczH,GAAI0zI,EAAa,aAC5Cz9B,EAAOxuG,KAAM,QAASzH,GAAI0zI,EAAa,SAASz6I,KAAWA,IAG3DwH,KAAKwP,SAAUgmG,EAAQ,WAAW,IAAMx1G,KAAKwzJ,SAAS,KAE/Ch+C,CAAM,GAEf,CAMQ,wBAAAs9C,GACP,MAAMx+H,EAASt0B,KAAKs0B,OACdk/B,EAAel/B,EAAO6qE,QAAQllE,KAAKz2B,SAIzCxD,KAAKwP,SAAkCgkD,EAAc,SAAS,KAC1CxzD,KAAKyzJ,2BAIvBzzJ,KAAKwzJ,SACN,IAIDl/H,EAAO8qE,WAAW3tF,IAAK,IAAgB,CAAEwhB,EAAYQ,KAEpDA,IAEKa,EAAO0qE,SAAS52F,IAAK,QAAUssB,WACnC10B,KAAKwzJ,SAAS,EACf,GAEF,CAMQ,8BAAAN,GAEPlzJ,KAAKs0B,OAAO8qE,WAAW3tF,IAAK,OAAO,CAAE3O,EAAM2wB,KACrCzzB,KAAK0zJ,qBAAuB1zJ,KAAK2yJ,YAAanoD,aAAa13E,YAC/D9yB,KAAK2yJ,YAAa1lH,QAClBxZ,IACD,GACE,CAIF7mB,SAAU,SAIX5M,KAAKs0B,OAAO8qE,WAAW3tF,IAAK,OAAO,CAAE3O,EAAM2wB,KACrCzzB,KAAK2zJ,eACT3zJ,KAAKqzJ,UACL5/H,IACD,IAID,GAAqB,CACpB7jB,QAAS5P,KAAK4yJ,SACd5wD,UAAW,IAAMhiG,KAAK4zJ,aACtB3xD,gBAAiB,IAAM,CAAEjiG,KAAKonH,SAASntF,KAAKxiB,SAC5ClI,SAAU,IAAMvP,KAAKqzJ,WAEvB,CAOO,eAAAQ,GACA7zJ,KAAK2yJ,aACV3yJ,KAAK+yJ,eAGD/yJ,KAAK8zJ,oBAIV9zJ,KAAKonH,SAASv5G,IAAK,CAClBosB,KAAMj6B,KAAK2yJ,YACXltI,SAAUzlB,KAAK4nH,2BAEjB,CAKQ,YAAAwrC,GAKP,GAJMpzJ,KAAK4yJ,UACV5yJ,KAAK+yJ,eAGD/yJ,KAAK+zJ,eACT,OAGD,MACM9gB,EADSjzI,KAAKs0B,OACoB0qE,SAAS52F,IAAK,QAEtDpI,KAAK4yJ,SAAUvwD,wBAEfriG,KAAKonH,SAASv5G,IAAK,CAClBosB,KAAMj6B,KAAK4yJ,SACXntI,SAAUzlB,KAAK4nH,4BAIX5nH,KAAKonH,SAAS5B,cAAgBxlH,KAAK4yJ,UACvC5yJ,KAAK4yJ,SAAUlB,aAAaxnD,UAAUW,SAGvC7qG,KAAK4yJ,SAAUrwD,uBAQfviG,KAAK4yJ,SAAUlB,aAAaxnD,UAAUzyF,QAASjf,MAAQy6I,EAAYz6I,OAAS,EAC7E,CASQ,cAAA+6J,GACP,MAAMtgB,EAA2BjzI,KAAKs0B,OAAO0qE,SAAS52F,IAAK,QAI3D6qI,EAAYgb,oCAEe,IAAtBhb,EAAYz6I,MAChBwH,KAAKg0J,kBAELh0J,KAAKqzJ,SAEP,CAKQ,eAAAW,GACFh0J,KAAK+zJ,iBAGT/zJ,KAAK4yJ,SAAUrU,eAAetxG,QAE9BjtC,KAAKonH,SAASnhH,OAAQjG,KAAK4yJ,UAI3B5yJ,KAAKs0B,OAAO6qE,QAAQllE,KAAKgT,QAEzBjtC,KAAKi0J,2BAEP,CAOO,OAAAT,CAASU,GAAwB,GACjCl0J,KAAK4yJ,UACV5yJ,KAAK+yJ,eAIA/yJ,KAAKyzJ,2BAiBLzzJ,KAAK0zJ,mBACT1zJ,KAAKozJ,eAILpzJ,KAAK6zJ,kBAIDK,GACJl0J,KAAKonH,SAAS/B,UAAW,UAxB1BrlH,KAAKm0J,2BAELn0J,KAAK6zJ,kBAGAK,GACJl0J,KAAKonH,SAAS/B,UAAW,QAG1BrlH,KAAKozJ,gBAoBNpzJ,KAAKo0J,kBACN,CAOQ,OAAAf,GACP,IAAMrzJ,KAAK4zJ,aACV,OAGD,MAAMt/H,EAASt0B,KAAKs0B,OAEpBt0B,KAAK2P,cAAe2kB,EAAO6E,GAAI,UAC/Bn5B,KAAK2P,cAAe3P,KAAKonH,SAAU,sBAInC9yF,EAAO6qE,QAAQllE,KAAKgT,QAGpBjtC,KAAKg0J,kBAGLh0J,KAAKonH,SAASnhH,OAAQjG,KAAK2yJ,aAE3B3yJ,KAAKi0J,0BACN,CAQQ,gBAAAG,GACP,MAAM9/H,EAASt0B,KAAKs0B,OACdk/B,EAAel/B,EAAO6qE,QAAQllE,KAAKz2B,SAEzC,IAAI6wJ,EAAmBr0J,KAAKyzJ,0BACxBa,EAAsBC,IAE1B,MAAMztJ,EAAS,KACd,MAAM0tJ,EAAex0J,KAAKyzJ,0BACpBh1G,EAAkB81G,IAYjBF,IAAqBG,IACxBH,GAAoB51G,IAAoB61G,EAC3Ct0J,KAAKqzJ,UAMIrzJ,KAAK2zJ,cAId3zJ,KAAKonH,SAASxB,eAAgB5lH,KAAK4nH,2BAGpCysC,EAAmBG,EACnBF,EAAsB71G,CAAe,EAGtC,SAAS81G,IACR,OAAO/gG,EAAav+B,UAAUgY,MAAO9Q,eACnC9wB,UACAkmB,MAAQpyB,GAA+BA,EAAK+O,GAAI,YACnD,CAEAlO,KAAKwP,SAAU8kB,EAAO6E,GAAI,SAAUryB,GACpC9G,KAAKwP,SAAUxP,KAAKonH,SAAU,qBAAsBtgH,EACrD,CAKA,kBAAYitJ,GACX,QAAS/zJ,KAAK4yJ,UAAY5yJ,KAAKonH,SAASpC,QAAShlH,KAAK4yJ,SACvD,CAKA,sBAAYkB,GACX,QAAS9zJ,KAAK2yJ,aAAe3yJ,KAAKonH,SAASpC,QAAShlH,KAAK2yJ,YAC1D,CAMA,sBAAYe,GACX,QAAS1zJ,KAAK2yJ,aAAe3yJ,KAAKonH,SAAS5B,cAAgBxlH,KAAK2yJ,WACjE,CAKA,gBAAYiB,GACX,OAAO5zJ,KAAK+zJ,gBAAkB/zJ,KAAK8zJ,kBACpC,CAMA,gBAAYH,GACX,MAAMnuC,EAAcxlH,KAAKonH,SAAS5B,YAElC,QAASxlH,KAAK4yJ,UAAYptC,GAAexlH,KAAK4yJ,UAAY5yJ,KAAK0zJ,kBAChE,CASQ,uBAAA9rC,GACP,MAAM3tF,EAAOj6B,KAAKs0B,OAAO6qE,QAAQllE,KAC3B7hC,EAAQ4H,KAAKs0B,OAAOl8B,MACpBo7D,EAAev5B,EAAKz2B,SAC1B,IAAII,EAEJ,GAAKxL,EAAMwnE,QAAQhyD,IAAK8kJ,IAAiC,CAExD,MAAM+B,EAAqB9zJ,MAAMrB,KAAMU,KAAKs0B,OAAO6qE,QAAQ3hC,OAAOE,qBAAsBg1F,KAClFv7G,EAAWld,EAAK4d,YACrB5d,EAAK2d,qBAAsB68G,EAAoB,IAC/Cx6H,EAAK0d,oBAAqB88G,EAAoBA,EAAmBn8J,OAAS,KAG3EsL,EAASq2B,EAAK4X,aAAa2U,eAAgBrP,EAC5C,MAKCvzC,EAAS,KACR,MAAM8wJ,EAAa10J,KAAKyzJ,0BAExB,OAAOiB,EAENz6H,EAAK4X,aAAayK,aAAco4G,GAEhCz6H,EAAK4X,aAAa2U,eAAgBgN,EAAav+B,UAAUoY,gBAAiB,EAI7E,MAAO,CAAEzpC,SACV,CAWQ,uBAAA6vJ,GACP,MAAMx5H,EAAOj6B,KAAKs0B,OAAO6qE,QAAQllE,KAC3BhF,EAAYgF,EAAKz2B,SAASyxB,UAC1Bib,EAAkBjb,EAAU+Y,qBAGlC,GAAK/Y,EAAU+V,aAAekF,GAAmB,GAAUA,GAC1D,OAAOykH,GAAyB1/H,EAAUE,oBACpC,CAGN,MAAMrQ,EAAQmQ,EAAUoY,gBAAiBlC,aACnCypH,EAAYD,GAAyB7vI,EAAM4b,OAC3Cm0H,EAAUF,GAAyB7vI,EAAM6b,KAE/C,OAAMi0H,GAAaA,GAAaC,GAK3B56H,EAAK8d,cAAe68G,GAAYzpH,aAAalnB,QAASa,GACnD8vI,EALA,IAST,CACD,CAOQ,wBAAAT,GACP,MAAM/7J,EAAQ4H,KAAKs0B,OAAOl8B,MAE1BA,EAAMmiC,QAAQC,IACb,MAAM1V,EAAQ1sB,EAAMoL,SAASyxB,UAAUoY,gBAEvC,GAAKj1C,EAAMwnE,QAAQhyD,IAAK8kJ,IACvBl4H,EAAOs4D,aAAc4/D,GAA8B,CAAE5tI,eAErD,GAAKA,EAAM4b,MAAM8I,QAAU,CAC1B,MAAMf,EAAgB3jB,EAAM4b,MAAM8J,yBACjC,EAAIvqC,WAAa7H,EAAMsrE,OAAOgS,UAAWz1E,IACzC,CAAEuoC,WAAY1jB,IAGf0V,EAAOu4D,UAAW2/D,GAA8B,CAC/C7/D,gBAAgB,EAChB79D,aAAa,EACblQ,MAAO0V,EAAOqd,YAAapP,EAAe3jB,EAAM6b,MAElD,MACCnG,EAAOu4D,UAAW2/D,GAA8B,CAC/C7/D,gBAAgB,EAChB79D,aAAa,EACblQ,SAGH,GAEF,CAKQ,wBAAAmvI,GACP,MAAM77J,EAAQ4H,KAAKs0B,OAAOl8B,MAErBA,EAAMwnE,QAAQhyD,IAAK8kJ,KACvBt6J,EAAMmiC,QAAQC,IACbA,EAAOs2D,aAAc4hE,GAA8B,GAGtD,EASD,SAASiC,GAAyBlvI,GACjC,OAAOA,EAAS0W,eAAe5K,MAAQqlB,IAAgDyiG,ObnpBzDl6I,EampBwEy3C,GblpB1F1oC,GAAI,uBAA0B/O,EAAKqnC,kBAAmB,QAD5D,IAAwBrnC,CampBmF,KAAK,IACvH,CCnrBA,MAGM21J,GAAc,IAAInrJ,OAEvB,oaA0CM,KAOQ,MAAMorJ,WAAiB,GAIrC,mBAAkB79H,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,UACR,CAKO,IAAAI,GACN,MACMzB,EADSj1B,KAAKs0B,OACKl8B,MAAMoL,SAASyxB,UAExCA,EAAU5lB,GAAkC,gBAAgB,KAE3DrP,KAAK00B,WAAaO,EAAUyG,OAAQ98B,OAAOsP,GAAI,UAAW,YAAa,IAGxElO,KAAKg1J,uBACN,CAKO,SAAAvtC,GACNznH,KAAKi1J,uBACLj1J,KAAKk1J,2BACN,CAKQ,qBAAAF,GACP,MAAM1gI,EAASt0B,KAAKs0B,OAEd6gI,EAAU,IAAI3+B,GAAaliG,EAAOl8B,OAAO8hC,IAE9C,IA4IH,SAAgCA,GAC/B,OAAOA,EAAK5hC,OAjP6B,GAiP6C,MAA5B4hC,EAAMA,EAAK5hC,OAAS,IAA2C,MAA5B4hC,EAAMA,EAAK5hC,OAAS,EAClH,CA9IS88J,CAAuBl7H,GAC5B,OAID,MAAMm0G,EAAMgnB,GAAiBn7H,EAAKjoB,OAAQ,EAAGioB,EAAK5hC,OAAS,IAE3D,OAAK+1I,EACG,CAAEA,YADV,CAEA,IAGD8mB,EAAQ9lJ,GAAkD,gBAAgB,CAAEgG,EAAKvS,KAChF,MAAM,MAAEojE,EAAK,MAAEphD,EAAK,IAAEupH,GAAQvrI,EAE9B,IAAMojE,EAAM+jB,SACX,OAGD,MAAMqrE,EAAUxwI,EAAM6b,IAAI2J,cAAe,GACnCirH,EAAYD,EAAQhrH,cAAe+jG,EAAI/1I,QAEvCm2J,EAAYn6H,EAAOl8B,MAAMy/C,YAAa09G,EAAWD,GAEvDt1J,KAAKw1J,eAAgBnnB,EAAKogB,EAAW,IAGtC0G,EAAQnuJ,KAAM,aAAczH,GAAIS,KACjC,CAKQ,oBAAAi1J,GACP,MAAM3gI,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfujJ,EAAyCrnH,EAAO0qE,SAAS52F,IAAK,SAE9DuzI,GAINA,EAAatsI,GAAI,WAAW,KAC3B,MAAMoW,EAAWrtB,EAAMoL,SAASyxB,UAAUE,mBAE1C,IAAM1P,EAAS7mB,OAAOgoB,gBACrB,OAGD,MAAMmxE,EAAe3/F,EAAM2/C,cAAetyB,EAAS7mB,OAAOgoB,iBAE1D5mB,KAAKy1J,8BAA+B19D,EAAc,GAEpD,CAKQ,yBAAAm9D,GACP,MAAM5gI,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MAEfs9J,EAAmDphI,EAAO0qE,SAAS52F,IAAK,cAExEstJ,GAINA,EAAkBrmJ,GAAI,WAAW,KAChC,MAAMoW,EAAWrtB,EAAMoL,SAASyxB,UAAUE,mBAEpC4iE,EAAe3/F,EAAMy/C,YAC1Bz/C,EAAMs/C,iBAAkBjyB,EAAS7mB,OAAQ,GACzC6mB,EAAS6kB,cAAe,IAGzBtqC,KAAKy1J,8BAA+B19D,EAAc,GAEpD,CAKQ,6BAAA09D,CAA+B19D,GACtC,MAAM3/F,EAAQ4H,KAAKs0B,OAAOl8B,OACpB,KAAE8hC,EAAI,MAAEpV,GAAUyxG,GAAiBx+B,EAAc3/F,GAEjDi2I,EAAMgnB,GAAiBn7H,GAE7B,GAAKm0G,EAAM,CACV,MAAMogB,EAAYr2J,EAAMy/C,YACvB/yB,EAAM6b,IAAI2J,cAAe+jG,EAAI/1I,QAC7BwsB,EAAM6b,KAGP3gC,KAAKw1J,eAAgBnnB,EAAKogB,EAC3B,CACD,CAQQ,cAAA+G,CAAgBnnB,EAAavpH,GACpC,MAAM1sB,EAAQ4H,KAAKs0B,OAAOl8B,MAGpBu9J,EAAUnI,GAA6Bnf,EADrBruI,KAAKs0B,OAAOtL,OAAO5gB,IAAK,yBAG1CpI,KAAK00B,WAuCb,SAA+B5P,EAAc1sB,GAC5C,OAAOA,EAAMsrE,OAAO6S,0BAA2Bn+E,EAAM4/C,gBAAiBlzB,GAAS,WAChF,CAzC2B8wI,CAAsB9wI,EAAO1sB,IAAYw1J,GAAiB+H,KA2CrF,SAA2B7wI,GAC1B,MAAM7kB,EAAO6kB,EAAM4b,MAAMwJ,UACzB,QAASjqC,GAAQA,EAAKgmC,aAAc,WACrC,CA9CkG4vH,CAAkB/wI,IAIlH9kB,KAAK81J,iBAAkBH,EAAS7wI,EACjC,CAQQ,gBAAAgxI,CAAkBznB,EAAavpH,GACtC,MAAM1sB,EAAQ4H,KAAKs0B,OAAOl8B,MACpB29J,EAAe/1J,KAAKs0B,OAAOqC,QAAQvuB,IAAK,UAG9ChQ,EAAMguE,eAAe5rC,IACpBA,EAAO/0B,aAAc,WAAY4oI,EAAKvpH,GAEtC1sB,EAAMguE,eAAe,KACpB2vF,EAAa1/B,wBAAwB,GACnC,GAEL,EAQD,SAASg/B,GAAiBn7H,GACzB,MAAMr9B,EAAQi4J,GAAYj8I,KAAMqhB,GAEhC,OAAOr9B,EAAQA,EAtMW,GAsMmB,IAC9C,C,cCrQI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQi9B,OCmHZh5B,OAAOC,SAyHRD,OAAOC,S,iMCrPb,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ+4B,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCsChB,SAAS,GAAgB+hD,EAAiBvc,GAChD,MAAM9B,EAAS8B,EAAc9B,OACvB6M,EAAa/K,EAAc9kC,OAC3Bw7H,EAAmD,YAAxCn6E,EAAUxgD,aAAc,YAA6B,KAAO,KACvEy0C,EArBA,SAAoCt1C,GAC1C,MAAMs1C,EAAWt1C,EAAOuZ,uBAAwB,MAIhD,OAFA+7B,EAAS9oC,gBAAkBivH,GAEpBnmF,CACR,CAekBomF,CAA2B7rF,GAEtC8rF,EAAW9rF,EAAWt2B,uBAAwBiiH,EAAU,MAM9D,OAJA3rF,EAAW3kE,OAAQ2kE,EAAW3yB,iBAAkBy+G,EAAU,GAAKrmF,GAE/DtS,EAAOnf,aAAcw9B,EAAkB/L,GAEhCA,CACR,CAaO,SAAS,GACf+L,EACAu6E,EACA92F,EACAlnE,GAEA,MAAMi+J,EAAeD,EAAax3J,OAC5B4+D,EAAS8B,EAAc9B,OACvB6M,EAAa/K,EAAc9kC,OAGjC,IAAIk/B,EAAiB8D,EAAOD,eAAgBnlE,EAAMw/C,qBAAsBikC,IAKxE,MAAMy6E,EAAU,GAAoBz6E,EAAUj1D,gBAAiB,CAC9D2vI,YAAY,EACZC,eAAe,EACfC,WAAY56E,EAAUxgD,aAAc,gBAE/Bq7H,EAAW76E,EAAUj1D,gBAE3B,GAAK0vI,GAAWA,EAAQj7H,aAAc,eAAkBwgD,EAAUxgD,aAAc,cAAiB,CAGhG,MAAMy0C,EAAWtS,EAAOf,cAAe65F,GACvC58F,EAAiB2Q,EAAWn1B,eAAgBm1B,EAAW1yB,oBAAqBm4B,GAC7E,MAEC,GAAK4mF,GAA6B,YAAjBA,EAAS10J,KAAqB,CAG9C03D,EAAiB8D,EAAOD,eAAgBnlE,EAAMs/C,iBAAkBg/G,EAAU,QAI1E,MAAMC,EAAqBn5F,EAAOzB,uBAAwBrC,GACpDk9F,EA6MF,SAAyBr/G,GAC/B,UAAYp4C,KAAUo4C,EAAY/b,cACjC,GAAkB,MAAbr8B,EAAK6C,MAA6B,MAAb7C,EAAK6C,KAC9B,OAAO7C,EAIT,OAAO,IACR,CArNsB,CAAgBw3J,GAIlCj9F,EADIk9F,EACavsF,EAAWzyB,qBAAsBg/G,GAGjCvsF,EAAW3yB,iBAAkBi/G,EAAoB,MAEpE,MAGCj9F,EAAiB8D,EAAOD,eAAgBnlE,EAAMw/C,qBAAsBikC,IAUtE,GANAniB,EAAiB,GAAyBA,GAG1C2Q,EAAW3kE,OAAQg0D,EAAgB28F,GAG9BK,GAA6B,YAAjBA,EAAS10J,KAAqB,CAC9C,MAAM60J,EAAWr5F,EAAOf,cAAei6F,GAGjChgH,EADmB2zB,EAAWxyB,YAAawyB,EAAW3yB,iBAAkBm/G,EAAU,GAAKn9F,GAC7D7uB,UAAW,CAAE9B,kBAAkB,IAE/D,UAAYvwC,KAASk+C,EACpB,GAAKl+C,EAAMyH,KAAKiO,GAAI,UAAW,MAAS,CACvC,MAAM4oJ,EAAgBzsF,EAAWn1B,eAAgBm1B,EAAWzyB,qBAAsBp/C,EAAMyH,OAClFk2J,EAAW39J,EAAMyH,KAAKrB,OAEtBy2C,EAAiBg1B,EAAW3yB,iBAAkB0+G,EAAc,OAClEW,GAAgB1sF,EAAYh1B,EAAelL,WAAakL,EAAenL,WACvEmgC,EAAW/0B,KAAM+0B,EAAWvyB,cAAeq+G,GAAY9gH,GAGrDqB,EAAgBhO,UAAYouH,CAC/B,CAEF,KAAO,CACN,MAAME,EAAeX,EAAap6H,YAElC,GAAK+6H,IAAkBA,EAAa9oJ,GAAI,UAAW,OAAU8oJ,EAAa9oJ,GAAI,UAAW,OAAW,CACnG,IAAI+oJ,EAAe,KAEnB,UAAYt/I,KAASq/I,EAAax7H,cAAyC,CAC1E,MAAM07H,EAAa15F,EAAOnB,eAAgB1kD,GAE1C,KACCu/I,GACEA,EAAW77H,aAAc,cAA+BwgD,EAAUxgD,aAAc,eAIlF,MAFA47H,EAAet/I,CAIjB,CAEKs/I,IACJ5sF,EAAWn1B,eAAgBm1B,EAAW1yB,oBAAqBs/G,IAC3D5sF,EAAW/0B,KACV+0B,EAAWvyB,cAAem/G,EAAar4J,QACvCyrE,EAAW3yB,iBAAkB0+G,EAAc,QAG9C,CACD,CAGAW,GAAgB1sF,EAAYgsF,EAAcA,EAAap6H,aACvD86H,GAAgB1sF,EAAYgsF,EAAazvI,gBAAkByvI,EAC5D,CAiBO,SAASU,GACf1sF,EACA8sF,EACAC,GAGA,OAAMD,IAAcC,GAAkC,MAAlBD,EAAUn1J,MAAkC,MAAlBm1J,EAAUn1J,MAKnEm1J,EAAUn1J,MAAQo1J,EAAWp1J,MAAQm1J,EAAU97H,aAAc,WAAc+7H,EAAW/7H,aAAc,SAJjG,KAQDgvC,EAAWz0B,gBAAiBy0B,EAAW1yB,oBAAqBw/G,GACpE,CAWO,SAAS,GAAyBxkH,GACxC,OAAOA,EAAanI,yBAAyBhyC,GAASA,EAAMyH,KAAKiO,GAAI,cACtE,CAYO,SAAS,GACf2tE,EACAt3E,GAOA,MAAMgyJ,IAAehyJ,EAAQgyJ,WACvBC,IAAkBjyJ,EAAQiyJ,cAC1BhL,EAASjnJ,EAAQkyJ,WAEvB,IAAIx2J,EAAY47E,EAEhB,KAAQ57E,GAAqB,YAAbA,EAAK+B,MAAqB,CACzC,MAAMq1J,EAAap3J,EAAKo7B,aAAc,cAEtC,GAAOk7H,GAAc/K,GAAU6L,GAAkBb,GAAiBhL,EAAmB6L,EACpF,OAAOp3J,EAIPA,EAD0B,YAAtBsE,EAAQggB,UACLtkB,EAAKg8B,YAELh8B,EAAK2mB,eAEd,CAEA,OAAO,IACR,CAWO,SAAS,GACf0N,EACAsF,EACA2U,EACAm7D,GAEAp1E,EAAO6E,GAAG61E,iBAAiBnhG,IAAK+rB,GAAaV,IAC5C,MAAMxD,EAAUpB,EAAO0qE,SAAS52F,IAAKwxB,GAC/BuxE,EAAa,IAAI,GAAYjyE,GAkBnC,OAhBAiyE,EAAW15F,IAAK,CACf88B,QACAm7D,OACAG,SAAS,EACTP,cAAc,IAIf6B,EAAWnkG,KAAM,OAAQ,aAAczH,GAAIm2B,EAAS,QAAS,aAG7Dy1E,EAAW97F,GAAwB,WAAW,KAC7CilB,EAAOiB,QAASqE,GAChBtF,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBk+D,CAAU,GAEnB,CAyBO,SAAS,GAAiB1lF,EAAoBlB,GACpD,MAAMqM,EAAwB,GACxB0mI,EAAW7xI,EAAS7mB,OACpB24J,EAAgB,CACrBxuH,kBAAkB,EAClBN,cAAehjB,EACfqjB,SAAS,EACTvkB,aAEKizI,EAAcF,EAASj8H,aAAc,cACrC6L,EAAQ,IAAK,IAAI,GAAYqwH,IACjCxxJ,QAAQvN,GAASA,EAAMyH,KAAKiO,GAAI,aAChClR,KAAKxE,GAASA,EAAMyH,OAEtB,UAAYwX,KAAWyvB,EAAQ,CAE9B,IAAMzvB,EAAQvJ,GAAI,UAAW,YAC5B,MAYD,GAAOuJ,EAAQ4jB,aAAc,cAA6Bm8H,EACzD,MASD,KAAO//I,EAAQ4jB,aAAc,cAA6Bm8H,GAA1D,CASA,GAAK//I,EAAQ4jB,aAAc,cAAiBi8H,EAASj8H,aAAc,YAClE,MASD,GAAK5jB,EAAQ4jB,aAAc,eAAkBi8H,EAASj8H,aAAc,aACnE,MAID,GAAK5jB,EAAQ4jB,aAAc,kBAAqBi8H,EAASj8H,aAAc,gBACtE,MAID,GAAK5jB,EAAQ4jB,aAAc,eAAkBi8H,EAASj8H,aAAc,aACnE,MAGkB,aAAd9W,EACJqM,EAAMxxB,QAASqY,GAEfmZ,EAAMhxB,KAAM6X,EAlCb,CAoCD,CAEA,OAAOmZ,CACR,CAmCA,MAAM6mI,GAA4B,CAAE,OAAQ,SAAU,UAIhDC,GAA4B,CACjC,UACA,uBACA,cACA,cACA,cACA,eAuBD,SAASzB,KACR,MAAM0B,GAAgB33J,KAAKmiC,UAAwC,MAA3BniC,KAAK67B,SAAU,GAAI75B,MAA2C,MAA3BhC,KAAK67B,SAAU,GAAI75B,MAE9F,OAAKhC,KAAKmiC,SAAWw1H,EACb,EAGD3wH,GAAgB3lC,KAAMrB,KAC9B,CC3de,MAAM,WAAe,GAInC,qBAAkBs2B,GACjB,MAAO,QACR,CAKO,IAAAI,GACN,MAAM17B,EAAIgF,KAAKs0B,OAAOt5B,EAGtB,GAAmBgF,KAAKs0B,OAAQ,eAAgBt5B,EAAG,iBCnCrD,8ZDoCE,GAAmBgF,KAAKs0B,OAAQ,eAAgBt5B,EAAG,iBEpCrD,gcFqCC,EG5BD,MAAM48J,GAA+E,CAAC,EAChFC,GAA0E,CAAC,EAC3EC,GAAmE,CAAC,EAEpEC,GAAkH,CACvH,CAAEC,UAAW,OAAQC,cAAe,OAAQjC,SAAU,YACtD,CAAEgC,UAAW,SAAUC,cAAe,SAAUjC,SAAU,YAC1D,CAAEgC,UAAW,SAAUC,cAAe,SAAUjC,SAAU,YAC1D,CAAEgC,UAAW,UAAWC,cAAe,IAAKjC,SAAU,YACtD,CAAEgC,UAAW,uBAAwBC,cAAe,KAAMjC,SAAU,YACpE,CAAEgC,UAAW,cAAeC,cAAe,IAAKjC,SAAU,YAC1D,CAAEgC,UAAW,cAAeC,cAAe,IAAKjC,SAAU,YAC1D,CAAEgC,UAAW,cAAeC,cAAe,IAAKjC,SAAU,YAC1D,CAAEgC,UAAW,cAAeC,cAAe,IAAKjC,SAAU,YAC1D,CAAEgC,UAAW,cAAeC,cAAe,IAAKjC,SAAU,YAC1D,CAAEgC,UAAW,cAAeC,cAAe,IAAKjC,SAAU,aAG3D,IAAK,MAAO,UAAEgC,EAAS,cAAEC,EAAa,SAAEjC,KAAc+B,GACrDH,GAAyBI,GAAchC,EACvC6B,GAA8BG,GAAcC,EAEvCA,IACJH,GAA8BG,GAAkBD,G,eC7B9C,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQl+H,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCIR,MAAMo+H,WAAsB,GAY1C,WAAAn2J,CAAauyB,EAAgB6jI,GAC5B3qJ,MAAO8mB,GAEPt0B,KAAKo4J,UAA+B,WAAnBD,EAA+B,GAAK,CACtD,CAKgB,OAAApjI,GACf/0B,KAAK00B,UAAY10B,KAAK61B,eACvB,CAOgB,OAAAN,GACf,MAAMn9B,EAAQ4H,KAAKs0B,OAAOl8B,MACpBif,EAAMjf,EAAMoL,SAClB,IAAI60J,EAAgB13J,MAAMrB,KAAM+X,EAAI4d,UAAU0tC,qBAE9CvqE,EAAMmiC,QAAQC,IACb,MAAM89H,EAAWD,EAAeA,EAAc//J,OAAS,GAGvD,IAAIgJ,EAAOg3J,EAASr8H,YAGpB,KACC36B,GAAqB,YAAbA,EAAKU,MACXV,EAAK+5B,aAAc,cAA+Bi9H,EAASj9H,aAAc,eAE3Eg9H,EAAcz4J,KAAM0B,GAEpBA,EAAOA,EAAK26B,YAORj8B,KAAKo4J,UAAY,IACrBC,EAAgBA,EAAchtJ,WAG/B,UAAYpL,KAAQo4J,EAAgB,CACnC,MAAM7M,EAAWvrJ,EAAKo7B,aAAc,cAA6Br7B,KAAKo4J,UAIjE5M,EAAS,EAIbhxH,EAAO6c,OAAQp3C,EAAM,aAIrBu6B,EAAO/0B,aAAc,aAAc+lJ,EAAQvrJ,EAE7C,CAIAD,KAAKwQ,KAAM,kBAAmB6nJ,EAAe,GAE/C,CAOQ,aAAAxiI,GAEP,MAAMyhI,EAAW,GAAOt3J,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAAU0tC,qBAG7D,IAAM20F,IAAaA,EAASppJ,GAAI,UAAW,YAC1C,OAAO,EAGR,GAAKlO,KAAKo4J,UAAY,EAAI,CAGzB,MAAM5M,EAAS8L,EAASj8H,aAAc,cAChCpwB,EAAOqsJ,EAASj8H,aAAc,YAEpC,IAAIwa,EAAOyhH,EAAS1wI,gBAEpB,KAAQivB,GAAQA,EAAK3nC,GAAI,UAAW,aAAkB2nC,EAAKxa,aAAc,eAA8BmwH,GAAS,CAC/G,GAAK31G,EAAKxa,aAAc,eAAkBmwH,EAKzC,OAAO31G,EAAKxa,aAAc,aAAgBpwB,EAG3C4qC,EAAOA,EAAKjvB,eACb,CAGA,OAAO,CACR,CAGA,OAAO,CACR,ECzHc,MAAM,WAAoB,GAmBxC,WAAA7kB,CAAauyB,EAAgBrpB,GAC5BuC,MAAO8mB,GAEPt0B,KAAKiL,KAAOA,CACb,CAKgB,OAAA8pB,GACf/0B,KAAKxH,MAAQwH,KAAKkxI,YAClBlxI,KAAK00B,UAAY10B,KAAK61B,eACvB,CAWgB,OAAAN,CAAShxB,EAAoC,CAAC,GAC7D,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MACpBoL,EAAWpL,EAAMoL,SACjB2tI,EAASxwI,MAAMrB,KAAMkE,EAASyxB,UAAU0tC,qBAC5C58D,QAAQi9D,GAASu1F,GAAwBv1F,EAAO5qE,EAAMsrE,UAGlD80F,OAAiC,IAAvBj0J,EAAQksI,YAA4BlsI,EAAQksI,WAAazwI,KAAKxH,MAI9EJ,EAAMmiC,QAAQC,IAGb,GAAKg+H,EAAU,CAEd,IAAIl3J,EAAO6vI,EAAQA,EAAO74I,OAAS,GAAI2jC,YACnCw8H,EAAgBrzI,OAAOC,kBACvB46C,EAAU,GAkDd,KAAQ3+D,GAAqB,YAAbA,EAAKU,MAA4D,IAAtCV,EAAK+5B,aAAc,eAAuB,CAGpF,MAAMmwH,EAASlqJ,EAAK+5B,aAAc,cAG7BmwH,EAASiN,IAEbA,EAAgBjN,GAKjB,MAAMkN,EAAYlN,EAASiN,EAK3Bx4F,EAAQrgE,KAAM,CAAE6X,QAASnW,EAAMm1J,WAAYiC,IAG3Cp3J,EAAOA,EAAK26B,WACb,CAEAgkC,EAAUA,EAAQ50D,UAElB,UAAYpL,KAAQggE,EACnBzlC,EAAO/0B,aAAc,aAAcxF,EAAKw2J,WAAYx2J,EAAKwX,QAE3D,CAmBA,IAAM+gJ,EAAU,CAGf,IAAIG,EAAevzI,OAAOC,kBAE1B,UAAYplB,KAAQkxI,EACdlxI,EAAKiO,GAAI,UAAW,aAAkBjO,EAAKo7B,aAAc,cAA6Bs9H,IAC1FA,EAAe14J,EAAKo7B,aAAc,eAKpCs9H,EAAgC,IAAjBA,EAAqB,EAAIA,EAGxCC,GAAUznB,GAAQ,EAAMwnB,GAGxBC,GAAUznB,GAAQ,EAAOwnB,EAC1B,CAMA,UAAYlhJ,KAAW05H,EAAO9lI,UACxBmtJ,GAA2B,YAAhB/gJ,EAAQzV,KAGvBw4B,EAAO6c,OAAQ5/B,EAAS,aACZ+gJ,GAA2B,YAAhB/gJ,EAAQzV,KAKnBw2J,GAA2B,YAAhB/gJ,EAAQzV,MAAsByV,EAAQ4jB,aAAc,aAAgBr7B,KAAKiL,MAGhGuvB,EAAO/0B,aAAc,WAAYzF,KAAKiL,KAAMwM,IAL5C+iB,EAAO+4D,cAAe,CAAEyiE,SAAUh2J,KAAKiL,KAAMwrJ,WAAY,GAAKh/I,GAC9D+iB,EAAO6c,OAAQ5/B,EAAS,aAiB1BzX,KAAKwQ,KAAM,kBAAmB2gI,EAAQ,GAExC,CAOQ,SAAAD,GAEP,MAAMomB,EAAW,GAAOt3J,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAAU0tC,qBAE7D,QAAS20F,GAAYA,EAASppJ,GAAI,UAAW,aAAgBopJ,EAASj8H,aAAc,aAAgBr7B,KAAKiL,IAC1G,CAOQ,aAAA4qB,GAEP,GAAK71B,KAAKxH,MACT,OAAO,EAGR,MAAMy8B,EAAYj1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UACvCyuC,EAAS1jE,KAAKs0B,OAAOl8B,MAAMsrE,OAE3Bo3B,EAAa,GAAO7lE,EAAU0tC,qBAEpC,QAAMm4B,GAKCy9D,GAAwBz9D,EAAYp3B,EAC5C,EAaD,SAASk1F,GACRznB,EACAhkG,EACAwrH,GAGA,MAAME,EAAe1rH,EAAagkG,EAAQ,GAAMA,EAAQA,EAAO74I,OAAS,GAExE,GAAKugK,EAAa3qJ,GAAI,UAAW,YAAe,CAC/C,IAAIjO,EAAO44J,EAAc1rH,EAAa,kBAAoB,eActDsrH,EAAgBI,EAAax9H,aAAc,cAI/C,KAAQp7B,GAAQA,EAAKiO,GAAI,UAAW,aAAkBjO,EAAKo7B,aAAc,eAA8Bs9H,GACjGF,EAAkBx4J,EAAKo7B,aAAc,gBACzCo9H,EAAgBx4J,EAAKo7B,aAAc,eAI/Bp7B,EAAKo7B,aAAc,eAAkBo9H,GAEzCtnB,EAAQhkG,EAAa,UAAY,QAAUltC,GAG5CA,EAAOA,EAAMktC,EAAa,kBAAoB,cAEhD,CACD,CAQA,SAASorH,GAAwBv1F,EAAgBU,GAChD,OAAOA,EAAOyF,WAAYnG,EAAMpkE,OAAe,cAAiB8kE,EAAOgE,SAAU1E,EAClF,CCnTe,MAAM81F,WAAkB,GAItC,qBAAkBxiI,GACjB,MAAO,WACR,CAKO,4BAAAyiI,CAA8BC,GACpC,OVqbK,SAAuCA,GAC7C,OAAKvB,GAA0BloI,SAAUypI,GACjC,WAGHtB,GAA0BnoI,SAAUypI,GACjC,WAGD,IACR,CU/bS,CAA8BA,EACtC,CASO,oBAAAC,CAAsB7gK,GAC5B,OVkYK,SAA+BA,GAKrC,IAAI8gK,EAAY,IAJC9gK,EAAMoL,SAIOyxB,UAAU0tC,qBACtC58D,QAAQ0R,GAAWA,EAAQvJ,GAAI,UAAW,cAC1ClR,KAAKya,IACL,MAAMgO,EAAWrtB,EAAMmiC,QAAQC,GAAUA,EAAOkd,iBAAkBjgC,EAAS,KAE3E,MAAO,IACH,GAAiBgO,EAAU,eAC3B,GAAiBA,EAAU,WAC9B,IAED+jD,OAMF,OAFA0vF,EAAY,IAAK,IAAItlJ,IAAKslJ,IAEnBA,CACR,CUxZS,CAAsB9gK,EAC9B,CAYO,eAAA+gK,CAAiB1zI,EAAoBlB,GAC3C,OAAO,GAAiBkB,EAAUlB,EACnC,ECPM,SAAS,GAAoBnsB,GACnC,MAAO,CAAEid,EAAKvS,EAAMw8D,KACnB,MAAMwB,EAAaxB,EAAcwB,WAEjC,IAAMA,EAAW7+D,KAAMa,EAAK7C,KAAM,YAChC6gE,EAAW7+D,KAAMa,EAAK7C,KAAM,wBAC5B6gE,EAAW7+D,KAAMa,EAAK7C,KAAM,wBAE7B,OAGD6gE,EAAWrC,QAAS37D,EAAK7C,KAAM,UAC/B6gE,EAAWrC,QAAS37D,EAAK7C,KAAM,sBAC/B6gE,EAAWrC,QAAS37D,EAAK7C,KAAM,wBAE/B,MAAM47E,EAAY/4E,EAAK7C,KAGvB,GAAgB47E,EAFC,GAAgBA,EAAWvc,GAEPA,EAAelnE,EAAO,CAE7D,CAiEO,MAAM,GAAoE,CAAEid,EAAKvS,EAAMw8D,KAC7F,IAAMA,EAAcwB,WAAW7+D,KAAMa,EAAK7C,KAAMoV,EAAIrT,MACnD,OAGD,MAAM8tE,EAAWxQ,EAAc9B,OAAOf,cAAe35D,EAAK7C,MACpDoqE,EAAa/K,EAAc9kC,OAIjC6vC,EAAWn1B,eAAgBm1B,EAAWzyB,qBAAsBk4B,IAC5DzF,EAAWn1B,eAAgBm1B,EAAW1yB,oBAAqBm4B,IAI3D,MAAMqmF,EAAWrmF,EAASlxE,OACpBw6J,EAAqC,YAA1Bt2J,EAAK29D,kBAAkC,KAAO,KAE/D4J,EAAWhzB,OAAQ+hH,EAAUjD,EAAU,EAQ3BkD,GAA8E,CAAEhkJ,EAAKvS,EAAMw8D,KACvGA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MAEjD,MACMm0J,EADW72F,EAAc9B,OAAOf,cAAe35D,EAAK7C,MAChCrB,OACpByrE,EAAa/K,EAAc9kC,OAGjCu8H,GAAgB1sF,EAAY8rF,EAAUA,EAASl6H,aAC/C86H,GAAgB1sF,EAAY8rF,EAASvvI,gBAAkBuvI,EAAU,EA6E3D,MAAMmD,GAAoE,CAAEjkJ,EAAKvS,EAAMw8D,KAC7F,GAAMA,EAAcwB,WAAW7+D,KAAMa,EAAK7C,KAAMoV,EAAIrT,OAI7B,YAAlBc,EAAK7C,KAAK+B,KAAqB,CACnC,IAAI2wC,EAAe2sB,EAAc9B,OAAOD,eAAgBz6D,EAAKgiB,MAAM4b,OAEnE,MAAM2pC,EAAa/K,EAAc9kC,OAC3BpoB,EAAQ,GAgDd,MAA+C,MAArCugC,EAAa/zC,OAAgBoD,MAAuD,MAArC2wC,EAAa/zC,OAAgBoD,QACrF2wC,EAAe03B,EAAWn1B,eAAgBvC,GAEE,MAArCA,EAAa/zC,OAAgBoD,OAHgE,CASpG,MAAMu3J,EAAc5mH,EACd6mH,EAAYnvF,EAAW3yB,iBAAkB/E,EAAa/zC,OAAe,OAG3E,IAAM26J,EAAYt1I,QAASu1I,GAAc,CACxC,MAAMxoI,EAAUq5C,EAAWpkE,OAAQokE,EAAWxyB,YAAa0hH,EAAaC,IACxEpnJ,EAAMxS,KAAMoxB,EACb,CAEA2hB,EAAe03B,EAAW1yB,oBAAqBhF,EAAa/zC,OAC7D,CAGA,GAAKwT,EAAM9Z,OAAS,EAAI,CACvB,QAAUkC,EAAI,EAAGA,EAAI4X,EAAM9Z,OAAQkC,IAAM,CACxC,MAAMi/J,EAAe9mH,EAAaxI,WAKlC,GAHAwI,EADsB03B,EAAW3kE,OAAQitC,EAAcvgC,EAAO5X,IACjCmmC,IAGxBnmC,EAAI,EAAI,CACZ,MAAMk/J,EAAW3C,GAAgB1sF,EAAYovF,EAAeA,EAAcx9H,aAIrEy9H,GAAYA,EAAS96J,QAAU66J,GACnC9mH,EAAa/2B,QAEf,CACD,CAGAm7I,GAAgB1sF,EAAY13B,EAAaxI,WAAawI,EAAazI,UACpE,CACD,GAwBYyvH,GAAwD,CAAEtkJ,EAAKvS,EAAMw8D,KACjF,MAAM3sB,EAAe2sB,EAAc9B,OAAOD,eAAgBz6D,EAAK2iB,UACzDm0I,EAAejnH,EAAaxI,WAC5B0vH,EAAelnH,EAAazI,UAKlC6sH,GAAgBz3F,EAAc9kC,OAAQo/H,EAAeC,EAAe,EAYxDC,GAAsD,CAAEzkJ,EAAKvS,EAAMw8D,KAC/E,GAAKA,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAE9tE,MAAM,IAAW,CACxE,MAAMw4B,EAAS8kC,EAAc9kC,OAGvB88H,EAAW98H,EAAOl1B,cAAe,YAGjCkmJ,EAgqBR,SAAoB8L,GACnB,IAAI9L,EAAS,EAET5sJ,EAAS04J,EAAS14J,OAEtB,KAAQA,GAAS,CAEhB,GAAKA,EAAOsP,GAAI,UAAW,MAC1Bs9I,QACM,CAEN,MAAM5kI,EAAkBhoB,EAAOgoB,gBAQ1BA,GAAmBA,EAAgB1Y,GAAI,UAAW,OACtDs9I,GAEF,CAEA5sJ,EAASA,EAAOA,MACjB,CAEA,OAAO4sJ,CACR,CA5rBiB,CAAW1oJ,EAAKgtE,UAE/Bt1C,EAAO/0B,aAAc,aAAc+lJ,EAAQ8L,GAG3C,MAAMrsJ,EAAOnI,EAAKgtE,SAASlxE,QAAkD,MAAtCkE,EAAKgtE,SAASlxE,OAAgBoD,KAAe,WAAa,WAGjG,GAFAw4B,EAAO/0B,aAAc,WAAYwF,EAAMqsJ,IAEjCh4F,EAAc8Q,WAAYknF,EAAUx0J,EAAK+sE,aAC9C,OAGD,MAAM5lC,EAibR,SACC8vH,EACAvpG,EACA8O,GAEA,MAAM,OAAE9kC,EAAM,OAAEkpC,GAAWpE,EAG3B,IAAIr1B,EAAezP,EAAOmd,oBAAqBoiH,GAI/C,UAAYpiJ,KAAS64C,EACpB,GAA8B,MAAvB74C,EAAe3V,MAAyC,MAAvB2V,EAAe3V,KAOtDioC,EAAeq1B,EAAc4C,YAAavqD,EAAOsyB,GAAe4lC,gBAC1D,CAEN,MAAM1xE,EAASmhE,EAAc4C,YAAavqD,EAAO6iB,EAAOkd,iBAAkBqiH,EAAe,QAUnFC,EAAiB77J,EAAOm/D,WAAY58B,MAAMwJ,UAC/B8vH,GAAkBA,EAAe9rJ,GAAI,aAAgBw1D,EAAOyF,WAAY4wF,EAAeC,EAAeh4J,QAsBrH+3J,EAFI57J,EAAO0xE,YAAYjxE,OAAOsP,GAAI,UAAW,YAE7B/P,EAAO0xE,YAAYjxE,OAGnBq7J,GAAkB97J,EAAO0xE,aAG1C5lC,EAAezP,EAAOmd,oBAAqBoiH,GAE7C,CAGD,OAAO9vH,CACR,CArfuBiwH,CAAsC5C,EAAUx0J,EAAKgtE,SAASt0C,cAAe8jC,GAGlGx8D,EAAKw6D,WAAa9iC,EAAOqd,YAAa/0C,EAAK+sE,YAAa5lC,GAExDq1B,EAAc+Q,uBAAwBinF,EAAUx0J,EACjD,GAUYq3J,GAA6C,CAAE9kJ,EAAKvS,EAAMw8D,KACtE,GAAKA,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAE9tE,MAAM,IAAW,CAErE,MAAMsV,EAAW3W,MAAMrB,KAAMwD,EAAKgtE,SAASt0C,eAE3C,UAAY7jB,KAASL,EAAW,GACLK,EAAMzJ,GAAI,UAAW,OAAUksJ,GAAQziJ,KAGhEA,EAAM0Z,SAER,CACD,GAQYgpI,GAAiD,CAAEhlJ,EAAKvS,EAAMw8D,KAC1E,GAAKA,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAE9tE,MAAM,IAAW,CACrE,GAAkC,IAA7Bc,EAAKgtE,SAASx0C,WAClB,OAGD,MAAMhkB,EAAW,IAAKxU,EAAKgtE,SAASt0C,eAEpC,IAAI8+H,GAAY,EAEhB,UAAY3iJ,KAASL,EACfgjJ,IAAcF,GAAQziJ,IAC1BA,EAAM0Z,UAGF+oI,GAAQziJ,KAEZ2iJ,GAAY,EAGf,GAQM,SAASC,GAAqBtgI,GACpC,MAAO,CAAE5kB,EAAKvS,KACb,GAAKA,EAAK26D,UACT,OAGD,MAAMoe,EAAY/4E,EAAK84D,cAAczxB,WAErC,GAAK0xC,GAAaA,EAAU3tE,GAAI,UAAW,YAAe,CACzD,MAAM4hE,EAAWhtE,EAAK06D,OAAOf,cAAeof,GACtC2+E,EAAkB1qF,EAAS3zC,eAAe5K,KAAM6oI,IAChD1jH,EAASzc,EAAKyd,iBAAkBo4B,EAAU,GAAIjlC,YAEpD,UAAYryC,KAASk+C,EAAS,CAC7B,GAAmB,gBAAdl+C,EAAMyS,MAA0BzS,EAAMyH,KAAKiO,GAAI,UAAW,MAAS,CACvEpL,EAAK6vC,aAAen6C,EAAM+wC,iBAE1B,KACD,CAAO,GAAmB,cAAd/wC,EAAMyS,MAAwBzS,EAAMyH,MAAQu6J,EAAkB,CACzE13J,EAAK6vC,aAAen6C,EAAMyxC,aAE1B,KACD,CACD,CACD,EAEF,CA4RO,MAAMwwH,GAA8D,SAAUplJ,GAAOnV,EAAS+tC,IACpG,MAAM71C,EAAQ4H,KAOd,IAEIi1B,EAFAh1B,EAAOC,EAAQgO,GAAI,oBAAuBhO,EAAQ27B,SAAU,GAAM37B,EAUtE,GAHC+0B,EAHKgZ,EAGO71C,EAAM4/C,gBAAiB/J,GAFvB71C,EAAMoL,SAASyxB,UAKvBh1B,GAAQA,EAAKiO,GAAI,UAAW,YAAe,CAE/C,MAAM6tB,EAAM9G,EAAUE,mBACtB,IAAImhI,EAAU,KASd,GAPKv6H,EAAIn9B,OAAOsP,GAAI,UAAW,YAC9BooJ,EAAUv6H,EAAIn9B,OACHm9B,EAAIoO,YAAcpO,EAAIoO,WAAWj8B,GAAI,UAAW,cAC3DooJ,EAAUv6H,EAAIoO,YAIVmsH,EAAU,CAId,MAAMoE,EAAepE,EAAQj7H,aAAc,cAG3C,GAAKq/H,EAAe,EAEnB,KAAQz6J,GAAQA,EAAKiO,GAAI,UAAW,aACnCjO,EAAKmnC,cAAe,aAAcnnC,EAAKo7B,aAAc,cAA2Bq/H,GAEhFz6J,EAAOA,EAAKg8B,WAGf,CACD,CACD,EAqFA,SAASg+H,GAAkBxxH,GAC1B,MAAMgC,EAAa,IAAI,GAAY,CAAEhC,kBAErC,IAAIjwC,EAEJ,GACCA,EAAQiyC,EAAWnpC,cACT9I,EAAMA,MAAMyH,KAAKiO,GAAI,UAAW,aAE3C,OAAO1V,EAAMA,MAAMyH,IACpB,CAMA,SAAS06J,GACRC,EACAC,EACAC,EACAC,EACAz7F,EACAlnE,GAMA,MAAM4iK,EAAgB,GAAoBH,EAAyB1wH,WAAY,CAC9EosH,YAAY,EACZC,eAAe,EACfC,WAAYmE,IAGPp9F,EAAS8B,EAAc9B,OACvB6M,EAAa/K,EAAc9kC,OAG3BygI,EAAaD,EAAgBA,EAAc3/H,aAAc,cAAiB,KAEhF,IAAIq+B,EAEJ,GAAMshG,EAkBC,GAAKC,GAAcL,EAAa,CAkBtC,MAAMM,EAAe19F,EAAOf,cAAeu+F,GAAiBp8J,OAC5D86D,EAAiB2Q,EAAW1yB,oBAAqBujH,EAClD,KAAO,CAmBN,MAAMt/F,EAAgBxjE,EAAMs/C,iBAAkBsjH,EAAe,OAC7DthG,EAAiB8D,EAAOD,eAAgB3B,EACzC,MA1CClC,EAAiBohG,EA4ClBphG,EAAiB,GAAyBA,GAI1C,UAAY/hD,IAAS,IAAKojJ,EAAgBv/H,eACpC4+H,GAAQziJ,KACZ+hD,EAAiB2Q,EAAW/0B,KAAM+0B,EAAWvyB,cAAengC,GAAS+hD,GAAiB/4B,IAEtFo2H,GAAgB1sF,EAAY1yD,EAAOA,EAAMskB,aACzC86H,GAAgB1sF,EAAY1yD,EAAMiP,gBAAkBjP,GAGvD,CAKA,SAASyiJ,GAAQ7iH,GAChB,OAAOA,EAAYrpC,GAAI,UAAW,OAAUqpC,EAAYrpC,GAAI,UAAW,KACxE,CC19Be,MAAM,WAAoB,GAIxC,qBAAkBooB,GACjB,MAAO,aACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GAAO,GAAQ4hI,GACzB,CAKO,IAAApiI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAMpBA,EAAOl8B,MAAMsrE,OAAOwR,SAAU,WAAY,CACzCsE,eAAgB,SAChB5D,gBAAiB,CAAE,WAAY,gBAIhC,MAAM9yE,EAAOwxB,EAAOxxB,KACdq8F,EAAU7qE,EAAO6qE,QD2alB,IAA8B/mG,ECzanCk8B,EAAOl8B,MAAMoL,SAASk3B,mBAAmBF,GD6fpC,SAA+BpiC,EAAcoiC,GACnD,MAAMylC,EAAU7nE,EAAMoL,SAASm8D,OAAOQ,aAChCg7F,EAAiB,IAAI5pJ,IAE3B,IAAI6pJ,GAAU,EAEd,UAAY1iJ,KAASunD,EACpB,GAAmB,UAAdvnD,EAAMzN,MAAkC,YAAdyN,EAAM1W,KACpCq5J,EAAe3iJ,EAAM+M,eACf,GAAmB,UAAd/M,EAAMzN,MAAkC,YAAdyN,EAAM1W,KAAqB,CAChE,GAAmB,SAAd0W,EAAM1W,KAAkB,CAE5B,MAAM/B,EAAOyY,EAAM+M,SAASykB,UAEvBjqC,EAAKgmC,aAAc,gBACvBzL,EAAO9zB,gBAAiB,aAAczG,GAEtCm7J,GAAU,GAGNn7J,EAAKgmC,aAAc,cACvBzL,EAAO9zB,gBAAiB,WAAYzG,GAEpCm7J,GAAU,GAGNn7J,EAAKgmC,aAAc,eACvBzL,EAAO9zB,gBAAiB,YAAazG,GAErCm7J,GAAU,GAGNn7J,EAAKgmC,aAAc,kBACvBzL,EAAO9zB,gBAAiB,eAAgBzG,GAExCm7J,GAAU,GAGNn7J,EAAKgmC,aAAc,eACvBzL,EAAO9zB,gBAAiB,YAAazG,GAErCm7J,GAAU,GAGX,UAAYE,KAAa36J,MAAMrB,KAAMlH,EAAM2/C,cAAe93C,IAAS8F,QAAQ7B,GAAKA,EAAEjE,KAAKiO,GAAI,UAAW,cACrGmtJ,EAAeC,EAAU/xH,iBAE3B,CAIA8xH,EAFiB3iJ,EAAM+M,SAAS6kB,aAAc5xB,EAAMpgB,QAGrD,KAA0B,UAAdogB,EAAMzN,MAAkC,YAAdyN,EAAM1W,KAC3Cq5J,EAAe3iJ,EAAM+M,WACI,aAAd/M,EAAMzN,MAA6C,cAAtByN,EAAM+rC,cAErB,aAAd/rC,EAAMzN,MAA6C,YAAtByN,EAAM+rC,eAD9C42G,EAAe3iJ,EAAMoM,MAAM4b,OAM7B,UAAY66H,KAAYJ,EAAejwJ,SACtCswJ,EAAiBD,GACjBE,EAAeF,GAGhB,OAAOH,EAEP,SAASC,EAAe51I,GACvB,MAAM69G,EAAe79G,EAAS0kB,WAE9B,GAAMm5F,GAAiBA,EAAap1H,GAAI,UAAW,YAM5C,CACN,IAAIqtJ,EAAWj4B,EAEf,GAAK63B,EAAevtJ,IAAK2tJ,GACxB,OAGD,QAEK30I,EAAkB20I,EAAS30I,gBAC/BA,GAAmBA,EAAgB1Y,GAAI,UAAW,YAClD0Y,EAAkB20I,EAAS30I,gBAI3B,GAFA20I,EAAW30I,EAENu0I,EAAevtJ,IAAK2tJ,GACxB,OAIFJ,EAAe1pJ,IAAK6xH,EAAci4B,EACnC,KA3BkE,CACjE,MAAMt7J,EAAOwlB,EAASykB,UAEjBjqC,GAAQA,EAAKiO,GAAI,UAAW,aAChCitJ,EAAe1pJ,IAAKxR,EAAMA,EAE5B,CAsBD,CAEA,SAASu7J,EAAiBv7J,GACzB,IAAIy7J,EAAY,EACZC,EAAQ,KAEZ,KAAQ17J,GAAQA,EAAKiO,GAAI,UAAW,aAAe,CAClD,MAAMmpJ,EAAap3J,EAAKo7B,aAAc,cAEtC,GAAKg8H,EAAaqE,EAAY,CAC7B,IAAIhD,EAEW,OAAViD,GACJA,EAAQtE,EAAaqE,EACrBhD,EAAYgD,IAEPC,EAAQtE,IACZsE,EAAQtE,GAGTqB,EAAYrB,EAAasE,GAG1BnhI,EAAO/0B,aAAc,aAAcizJ,EAAWz4J,GAE9Cm7J,GAAU,CACX,MACCO,EAAQ,KACRD,EAAYz7J,EAAKo7B,aAAc,cAA2B,EAG3Dp7B,EAAOA,EAAKg8B,WACb,CACD,CAEA,SAASw/H,EAAex7J,GACvB,IAAI27J,EAA4B,GAC5B/lH,EAAO,KAEX,KAAQ51C,GAAQA,EAAKiO,GAAI,UAAW,aAAe,CAClD,MAAMmpJ,EAAap3J,EAAKo7B,aAAc,cAMtC,GAJKwa,GAAQA,EAAKxa,aAAc,cAA2Bg8H,IAC1DuE,EAAaA,EAAW95J,MAAO,EAAGu1J,EAAa,IAG7B,GAAdA,EACJ,GAAKuE,EAAYvE,GAAe,CAC/B,MAAMpsJ,EAAO2wJ,EAAYvE,GAEpBp3J,EAAKo7B,aAAc,aAAgBpwB,IACvCuvB,EAAO/0B,aAAc,WAAYwF,EAAMhL,GAEvCm7J,GAAU,EAEZ,MACCQ,EAAYvE,GAAep3J,EAAKo7B,aAAc,YAIhDwa,EAAO51C,EACPA,EAAOA,EAAKg8B,WACb,CACD,CACD,CChqBqD,CAAsB3H,EAAOl8B,MAAOoiC,KAEvF2kE,EAAQ3hC,OAAOI,0BAA2B,KAAMi+F,IAChD/4J,EAAK06D,OAAOI,0BAA2B,KAAMi+F,IAE7C18D,EAAQ3hC,OAAOnuD,GAAoC,sBAAuBkrJ,GAAqBp7D,EAAQllE,OACvGklE,EAAQ3hC,OAAOnuD,GAAoC,uBDmahBjX,ECna4Dk8B,EAAOl8B,MDoahG,CAAEid,EAAKvS,KACb,MAAMg5J,EAAUh5J,EAAK6vC,aACfmU,EAAag1G,EAAQl9J,OACrB4+D,EAAS16D,EAAK06D,OAEpB,GAAwB,MAAnB1W,EAAW9kD,MAAmC,MAAnB8kD,EAAW9kD,KAAe,CAEzD,GAAM85J,EAAQtyH,QAMP,CAKN,MAAMgkC,EAAYhQ,EAAOnB,eAAgBy/F,EAAQ3xH,YAC3C4xH,EAAcv+F,EAAOQ,eAAgB89F,EAAQ3xH,YAGnDrnC,EAAK84D,cAAgBxjE,EAAMw/C,qBAAsB41B,GAAaljC,aAAcyxH,EAC7E,KAhBwB,CAGvB,MAAMvuF,EAAYhQ,EAAOnB,eAAgBy/F,EAAQ5xH,WAEjDpnC,EAAK84D,cAAgBxjE,EAAMw/C,qBAAsB41B,EAClD,CAYAn4D,EAAIlJ,MACL,MAAO,GACa,MAAnB26C,EAAW9kD,MACX85J,EAAQ3xH,aACgC,MAApC2xH,EAAQ3xH,WAAoBnoC,MAAsD,MAApC85J,EAAQ3xH,WAAoBnoC,MAC7E,CAGD,MAAMwrE,EAAYhQ,EAAOnB,eAAgBvV,GAIzC,IAAIi1G,EAAc,EACd5F,EAA4B2F,EAAQ3xH,WAExC,KAAQgsH,GAAYiE,GAAQjE,IAC3B4F,GAAev+F,EAAOQ,eAAgBm4F,GAEtCA,EAAWA,EAASvvI,gBAGrB9jB,EAAK84D,cAAgBxjE,EAAMw/C,qBAAsB41B,GAAaljC,aAAcyxH,GAE5E1mJ,EAAIlJ,MACL,KCpdArJ,EAAK06D,OAAOnuD,GAAoC,sBAAuBkrJ,GAAqBp7D,EAAQllE,OAEpG3F,EAAOj2B,WAAWuhF,IAAK,mBACrB/xE,KAAK2xD,IACLA,EAAWnwD,GAAkC,SAAUiqJ,GAAwB,CAAE1sJ,SAAU,SAC3F4yD,EAAWnwD,GAAkC,kBAAmB,GAAoBilB,EAAOl8B,QAC3FonE,EAAWnwD,GAAqC,8BAA+B,GAAqB,CAAEzC,SAAU,SAChH4yD,EAAWnwD,GACV,8BAA+BgqJ,GAA+B,CAAEzsJ,SAAU,QAC3E4yD,EAAWnwD,GAAqC,gCD6E7C,SAAgCjX,GACtC,MAAO,CAAEid,EAAKvS,EAAMw8D,KACnB,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAM,wBAClD,OAGD,MAAM6vE,EAAWxQ,EAAc9B,OAAOf,cAAe35D,EAAK7C,MACpDoqE,EAAa/K,EAAc9kC,OAIjC6vC,EAAWn1B,eAAgBm1B,EAAWzyB,qBAAsBk4B,IAC5DzF,EAAWn1B,eAAgBm1B,EAAW1yB,oBAAqBm4B,IAG3D,MAAMqmF,EAAWrmF,EAASlxE,OACpBo9J,EAAe7F,EAASvvI,gBACxByxE,EAAchuB,EAAWvyB,cAAeq+G,GAC9C9rF,EAAWpkE,OAAQoyF,GAEd2jE,GAAgBA,EAAa//H,aACjC86H,GAAgB1sF,EAAY2xF,EAAcA,EAAa//H,aAIxD0+H,GACC73J,EAAK09D,kBAA8B,EACnC19D,EAAKgiB,MAAM4b,MACX23D,EAAY33D,MACZovC,EACAxQ,EACAlnE,GAID,GAAgB0K,EAAK7C,KAAM6vE,EAAUxQ,EAAelnE,GAGpD,UAAYuf,KAAS7U,EAAK7C,KAAKu7B,cAC9B8jC,EAAcwB,WAAWrC,QAAS9mD,EAAO,SAC1C,CAEF,CCvHqFskJ,CAAuB3nI,EAAOl8B,QAC/GonE,EAAWnwD,GAAyB,kBDzBjC,SAA0BjX,GAChC,MAAO,CAAEid,EAAKvS,EAAMw8D,KACnB,MAEMwQ,EAFexQ,EAAc9B,OAAOD,eAAgBz6D,EAAK2iB,UAChC+kB,yBAAyBhyC,IAAUA,EAAMyH,KAAKiO,GAAI,UAAW,QACjEg8B,UACrBmgC,EAAa/K,EAAc9kC,OAIjC6vC,EAAWn1B,eAAgBm1B,EAAWzyB,qBAAsBk4B,IAC5DzF,EAAWn1B,eAAgBm1B,EAAW1yB,oBAAqBm4B,IAG3D,MAAMqmF,EAAWrmF,EAASlxE,OACpBo9J,EAAe7F,EAASvvI,gBACxByxE,EAAchuB,EAAWvyB,cAAeq+G,GACxCnlI,EAAUq5C,EAAWpkE,OAAQoyF,GAG9B2jE,GAAgBA,EAAa//H,aACjC86H,GAAgB1sF,EAAY2xF,EAAcA,EAAa//H,aAMxD0+H,GAFkBr7F,EAAc9B,OAAOnB,eAAgByT,GAG5Cz0C,aAAc,cAA2B,EACnDv4B,EAAK2iB,SACL4yE,EAAY33D,MACZovC,EACAxQ,EACAlnE,GAID,UAAYuf,KAAS0yD,EAAWtyB,cAAe/mB,GAAUkb,WACxDozB,EAAc9B,OAAOpB,kBAAmBzkD,GAGzCtC,EAAIlJ,MAAM,CAEZ,CCjB2D+vJ,CAAiB5nI,EAAOl8B,QAC/EonE,EAAWnwD,GAAyB,SAAUsqJ,GAAqB,CAAE/sJ,SAAU,OAAS,IAG1F0nB,EAAOj2B,WAAWuhF,IAAK,gBACrB/xE,KAAK2xD,IACLA,EAAWnwD,GAAkC,SAAUiqJ,GAAwB,CAAE1sJ,SAAU,SAC3F4yD,EAAWnwD,GAAkC,kBAAmB,GAAoBilB,EAAOl8B,OAAS,IAGtGk8B,EAAOj2B,WAAWuhF,IAAK,UACrB/xE,KAAK2xD,IACLA,EAAWnwD,GAAwB,aAAc8qJ,GAAW,CAAEvtJ,SAAU,SACxE4yD,EAAWnwD,GAAwB,aAAc8qJ,GAAW,CAAEvtJ,SAAU,SACxE4yD,EAAWnwD,GAAwB,aAAcgrJ,GAAe,CAAEztJ,SAAU,SAC5E4yD,EAAWnwD,GAAwB,aAAcyqJ,GAAoB,IAIvExlI,EAAOl8B,MAAMiX,GAA6B,gBAAiBorJ,GAAuB,CAAE7tJ,SAAU,SAG9F0nB,EAAO0qE,SAASnxF,IAAK,eAAgB,IAAI,GAAaymB,EAAQ,aAC9DA,EAAO0qE,SAASnxF,IAAK,eAAgB,IAAI,GAAaymB,EAAQ,aAG9DA,EAAO0qE,SAASnxF,IAAK,aAAc,IAAIqqJ,GAAe5jI,EAAQ,YAC9DA,EAAO0qE,SAASnxF,IAAK,cAAe,IAAIqqJ,GAAe5jI,EAAQ,aAE/D,MAAMk/B,EAAe2rC,EAAQllE,KAAKz2B,SAIlCxD,KAAKwP,SAAkCgkD,EAAc,SAAS,CAAEn+C,EAAKvS,KACpE,MAAMuU,EAAMrX,KAAKs0B,OAAOl8B,MAAMoL,SACxBiyC,EAAiBp+B,EAAI4d,UAAUuY,kBAAmB5uC,OAEnDyY,EAAI4d,UAAU+V,aAAsC,YAAvByK,EAAezzC,MAAsByzC,EAAetT,UACrFniC,KAAKs0B,OAAOiB,QAAS,eAErBzyB,EAAKowB,iBACL7d,EAAIlJ,OACL,GACE,CAAEoB,QAAS,OAIdvN,KAAKwP,SAAmCgkD,EAAc,UAAU,CAAEn+C,EAAKvS,KAEtE,GAAwB,aAAnBA,EAAKyhB,UACT,OAGD,MAAM0Q,EAAYj1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAE7C,IAAMA,EAAU+V,YACf,OAGD,MAAMi4F,EAAgBhuG,EAAUE,mBAEhC,IAAM8tG,EAAcn5F,UACnB,OAGD,MAAM2L,EAAiBwtF,EAAcrkI,OAErC,GAA6B,aAAxB62C,EAAezzC,KACnB,OAG2ByzC,EAAe7uB,iBAAsE,aAAjD6uB,EAAe7uB,gBAAyB5kB,OAMxGhC,KAAKs0B,OAAOiB,QAAS,eAErBzyB,EAAKowB,iBACL7d,EAAIlJ,OAAM,GACR,CAAEoB,QAAS,OAEdvN,KAAKwP,SAAgC8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,OAAO,CAAE6R,EAAKvS,KAChF,MAAM82B,EAAc92B,EAAK2qB,SAAW,cAAgB,aACpCztB,KAAKs0B,OAAO0qE,SAAS52F,IAAKwxB,GAE7BlF,YACZJ,EAAOiB,QAASqE,GAEhB92B,EAAKqwB,kBACLrwB,EAAKowB,iBACL7d,EAAIlJ,OACL,GACE,CAAEoB,QAAS,MACf,CAKO,SAAAk6G,GACN,MAAMzoB,EAAWh/F,KAAKs0B,OAAO0qE,SAEvBwsD,EAASxsD,EAAS52F,IAAK,UACvBqjJ,EAAUzsD,EAAS52F,IAAK,WAEzBojJ,GACJA,EAAO51H,qBAAsBopE,EAAS52F,IAAK,eAGvCqjJ,GACJA,EAAQ71H,qBAAsBopE,EAAS52F,IAAK,eAE9C,EAGD,SAASyzJ,GAAuBpkJ,GAC/B,IAAInf,EAAS,EAEb,UAAYqf,KAASF,EAAQ+jB,cAC5B,GAAmB,MAAd7jB,EAAM3V,MAA8B,MAAd2V,EAAM3V,KAChC,UAAY/B,KAAQ0X,EAAM6jB,cACzBljC,GAAUujK,GAAuB57J,GAKpC,OAAO3H,CACR,CC5L8Bq1B,GAAgB,c,eC3C1C,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQmM,OCgChB,SAASqiI,GACfC,EACA73J,GAEA,MAAM8pE,EAAiD,CAAEh5D,EAAKvS,EAAMw8D,KACnE,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAGD,MAAMqsI,EAAMvrI,EAAK29D,kBACX4J,EAAa/K,EAAc9kC,OAC3B6hI,EAAS/8F,EAAc9B,OAAOf,cAAe35D,EAAK7C,MAClDq8J,EAAsB,IAAKD,EAAO7gI,eACtCjK,MAAM5Z,GAAWA,EAAuB6uB,kBAAmB,mBAG7D6jC,EAAWpkE,OAAQq2J,GAEnB,MAAMC,EAAmBH,EAASI,oBAAqBnyF,EAAYgkE,EAAK9pI,GAExE8lE,EAAW3kE,OAAQ2kE,EAAW3yB,iBAAkB2kH,EAAQ,GAAKE,EAAkB,EAGhF,OAAO/8F,IACNA,EAAWnwD,GAA4B,sBAAuBg/D,EAAW,CAE3E,CCIO,SAASouF,GACfjiI,EACA4hI,EACA/tB,EACA9pI,GAEA,OAAOi2B,EAAOuZ,uBAAwB,SAAU,CAAExV,MAAO,SAAW,CACnE69H,EAASI,oBAAqBhiI,EAAQ6zG,EAAK9pI,GAC3Ci2B,EAAOyd,cAET,CAKO,SAASykH,GAA6BznI,GAC5C,MAAMib,EAAkBjb,EAAU+Y,qBAElC,OAAKkC,GAAmBA,EAAgBhiC,GAAI,UAAW,SAC/CgiC,EAGD,IACR,CAYO,SAASysH,GAAavkK,EAAci2I,EAAapgG,EAAwBktD,GAC/E/iG,EAAMmiC,QAAQC,IACb,MAAMoiI,EAAepiI,EAAOl1B,cAAe,QAAS,CAAE+oI,QAEtDj2I,EAAM4iG,aAAc4hE,EAAc3uH,EAAY,KAAM,CACnDuF,aAAc,KACd2nD,oBAAqBA,EAAsB,YAAS,GAClD,GAEL,CC5Fe,MAAM0hE,WAA0B,GAS9B,OAAA9nI,GACf,MAAM38B,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3B6nI,EAAgBJ,GAA6BznI,GAEnDj1B,KAAKxH,MAAQskK,EAAgBA,EAAczhI,aAAc,YAAoB,EAE7Er7B,KAAK00B,UA6CP,SAA0BO,GACzB,MAAMxd,EAAUwd,EAAU+Y,qBAC1B,QAASv2B,GAA4B,UAAjBA,EAAQzV,IAC7B,CAhDmB+6J,CAAiB9nI,IA8BpC,SAA4BA,EAA0C78B,GACrE,MAAMqlJ,EAAiB,GAA2BxoH,EAAW78B,GAC7D,IAAIwG,EAAS6+I,EAAe/8G,MAAM9hC,OAG7BA,EAAOujC,UAAY/pC,EAAMsrE,OAAOI,QAASllE,KAC7CA,EAASA,EAAOA,QAGjB,OAAOxG,EAAMsrE,OAAOyF,WAAYvqE,EAAQ,QACzC,CAxCmDo+J,CAAmB/nI,EAAW78B,EAChF,CAWgB,OAAAm9B,CAAS84G,GACxB,MAAMj2I,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3B6nI,EAAgBJ,GAA6BznI,GAE9C6nI,EACJ1kK,EAAMmiC,QAAQC,IACbA,EAAO/0B,aAAc,MAAO4oI,EAAKyuB,EAAe,IAGjDH,GAAavkK,EAAOi2I,EAAKp5G,GAAW,EAEtC,ECxCc,MAAMgoI,GAkBpB,WAAAl7J,CAAam3B,EAAgBlQ,GAC5B,MAAMk0I,EAAYl0I,EAAOk0I,UACnBC,EAAiBn0I,EAAOm0I,gBAAkB,GAC1CC,EAAmB,IAAIxpJ,IAAKoV,EAAOq0I,iBACnCC,EAAsBJ,EAC1B/8J,OAAQg9J,GACRp3J,QAAQw3J,IACR,MAAMv7J,EAAOu7J,EAASv7J,KAEtB,OAAMA,GAaEo7J,EAAiBxvJ,IAAK5L,IAL7B,EAAY,+BAAgC,CAAEu7J,cAEvC,EAG4B,IAGtCv9J,KAAKk5B,OAASA,EACdl5B,KAAKs9J,oBAAsBA,CAC5B,CAOO,QAAAE,CAAUnvB,GAChB,QAASruI,KAAKy9J,UAAWpvB,EAC1B,CAWO,mBAAAmuB,CACNhiI,EACA6zG,EACA9pI,GAEA,OAAOvE,KAAKy9J,UAAWpvB,GAAOqvB,eAAgBljI,EAAQj2B,EACvD,CAQQ,SAAAk5J,CAAWpvB,GAClB,IAAMA,EACL,OAAO,IAAIsvB,GAAO39J,KAAKk5B,QAGxBm1G,EAAMA,EAAIxgH,OAEV,UAAY3lB,KAAclI,KAAKs9J,oBAAsB,CACpD,MAAMM,EAAkB11J,EAAW48C,KAC7BjnB,EAAU,GAAS31B,EAAWmmI,KAEpC,UAAYwvB,KAAchgI,EAAU,CACnC,MAAMhhC,EAAQmD,KAAK89J,eAAgBzvB,EAAKwvB,GAExC,GAAKhhK,EACJ,OAAO,IAAI8gK,GAAO39J,KAAKk5B,OAAQm1G,EAAKxxI,EAAO+gK,EAE7C,CACD,CAEA,OAAO,IACR,CAQQ,cAAAE,CAAgBzvB,EAAaxwG,GAEpC,IAAIhhC,EAAQwxI,EAAIxxI,MAAOghC,GAEvB,GAAKhhC,EACJ,OAAOA,EAIR,IAAIkhK,EAAS1vB,EAAIp1H,QAAS,eAAgB,IAG1C,OAFApc,EAAQkhK,EAAOlhK,MAAOghC,GAEjBhhC,IAKLkhK,EAASA,EAAO9kJ,QAAS,SAAU,IACnCpc,EAAQkhK,EAAOlhK,MAAOghC,GAEjBhhC,GAIE,KACR,EAQD,MAAM8gK,GAuBL,WAAA57J,CAAam3B,EAAgBm1G,EAAcxxI,EAA0B+gK,GACpE59J,KAAKquI,IAAMruI,KAAKg+J,aAAc3vB,GAC9BruI,KAAKi+J,QAAU/kI,EACfl5B,KAAKk+J,OAASrhK,EACdmD,KAAKm+J,iBAAmBP,CACzB,CAOO,cAAAF,CACNljI,EACAj2B,GAEA,MAAMgB,EAAsC,CAAC,EAC7C,IAAIgyC,EAEJ,GAAKhzC,EAAQ65J,sBAA0B75J,EAAQ85J,oBAAsBr+J,KAAKquI,KAAOruI,KAAKm+J,iBAAqB,CACrGn+J,KAAKquI,MACT9oI,EAAY,mBAAsBvF,KAAKquI,KAGnC9pI,EAAQ65J,uBACZ74J,EAAWg5B,MAAQ,qBAGpB,MAAM+/H,EAAYt+J,KAAKu+J,gBAAiBh6J,GAExCgzC,EAAc/c,EAAOga,iBAAkB,MAAOjvC,GAAY,CAAEwsC,EAAYF,KACvEA,EAAcgT,aAAc9S,EAAYusH,EAAW,GAErD,MACMt+J,KAAKquI,MACT9oI,EAAW8oI,IAAMruI,KAAKquI,KAGvB92F,EAAc/c,EAAO2Z,mBAAoB5vC,EAAQogD,YAAap/C,GAK/D,OAFAi1B,EAAOoa,kBAAmB,iBAAiB,EAAM2C,GAE1CA,CACR,CAKQ,eAAAgnH,CAAiBh6J,GACxB,OAAKvE,KAAKm+J,iBACFn+J,KAAKm+J,iBAAkBn+J,KAAKk+J,QAI9Bl+J,KAAKquI,KAAO9pI,EAAQ65J,qBACjBp+J,KAAKw+J,sBAGN,EAET,CAKQ,mBAAAA,GACP,MAAM90D,EAAO,IAAIf,GACX3tG,EAAIgF,KAAKi+J,QAAQjjK,EAEvB0uG,EAAKxpG,QCpQP,oyCDqQEwpG,EAAKtB,QApP6B,YAyRlC,OAnCoB,IAAI,GAAU,CACjCjxF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,yCAERjnB,SAAU,CACT,CACCH,IAAK,MACL5R,WAAY,CACXg5B,MAAO,+BAERjnB,SAAU,CAAEoyF,IAEb,CACCvyF,IAAK,IACL5R,WAAY,CACXg5B,MAAO,6BACP36B,OAAQ,SACR6sJ,IAAK,sBACLrwC,KAAMpgH,KAAKquI,IACX,wBAAyBrzI,EAAG,0BAE7Bsc,SAAU,CACT,CACCH,IAAK,OACL5R,WAAY,CACXg5B,MAAO,oCAERjnB,SAAU,CAAEtX,KAAKquI,WAKlB18F,SAEe8sH,SACpB,CAOQ,YAAAT,CAAc3vB,GACrB,OAAMA,EAIDA,EAAIxxI,MAAO,WACRwxI,EAGD,WAAaA,EAPZ,IAQT,E,eEzTG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQv0G,OCYR,MAAM4kI,WAA0B,GAgB9C,WAAA38J,CAAauyB,GACZ9mB,MAAO8mB,GACPA,EAAOtL,OAAOtyB,OAAQ,aAAc,CACnCiuD,YAAa,SACbu4G,UAAW,CACV,CACCl7J,KAAM,cACNqsI,IAAK,kCACLvpF,KAAMjoD,GAIJ,2HAHUA,EAAO,yKAapB,CACCmF,KAAM,UACNqsI,IAAK,CACJ,qCACA,oCACA,qCAEDvpF,KAAMjoD,GAIJ,uIAHUA,EAAO,gKAapB,CACCmF,KAAM,UACNqsI,IAAK,CACJ,wDACA,mDACA,mDACA,sCAEDvpF,KAAMjoD,IACL,MAAMgD,EAAKhD,EAAO,GACZ8vD,EAAO9vD,EAAO,GAEpB,MACC,0IACgDgD,IAAO8sD,EAAO,UAAWA,IAAU,kKAStF,CACC3qD,KAAM,QACNqsI,IAAK,CACJ,qBACA,0CACA,0CACA,sCACA,4CACA,sCACA,qCAEDvpF,KAAMjoD,GAIJ,2IAHUA,EAAO,wKAapB,CACCmF,KAAM,YACNqsI,IAAK,6BAEN,CACCrsI,KAAM,UACNqsI,IAAK,iBAEN,CACCrsI,KAAM,aACNqsI,IAAK,CACJ,qBACA,iBACA,qBACA,wBAGF,CACCrsI,KAAM,SACNqsI,IAAK,gBAEN,CACCrsI,KAAM,WACNqsI,IAAK,qBAKRruI,KAAKo8J,SAAW,IAAIa,GAAe3oI,EAAO4E,OAAQ5E,EAAOtL,OAAO5gB,IAAK,cACtE,CApIA,qBAAkBkuB,GACjB,MAAO,mBACR,CAuIO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdovC,EAASpvC,EAAOl8B,MAAMsrE,OACtB1oE,EAAIs5B,EAAOt5B,EACXqD,EAAai2B,EAAOj2B,WACpBggK,EAAqB/pI,EAAOtL,OAAO5gB,IAAK,6BACxCu8C,EAAcrwB,EAAOtL,OAAO5gB,IAAK,0BAEjCg0J,EAAWp8J,KAAKo8J,SAEtB9nI,EAAO0qE,SAASnxF,IAAK,aAAc,IAAIgvJ,GAAmBvoI,IAG1DovC,EAAOwR,SAAU,QAAS,CACzBsE,eAAgB,eAChB5D,gBAAiB,CAAE,SAIpBv3E,EAAWuhF,IAAK,gBAAiB1W,mBAAoB,CACpD9wE,MAAO,QACP6hC,KAAM,CAAEkiC,GAAgB3hC,aACvB,MAAM6zG,EAAMlyE,EAAa9gC,aAAc,OAEvC,OAAOohI,GAA0BjiI,EAAQ4hI,EAAU/tB,EAAK,CACvD1pF,cACA05G,qBAAsBhwB,GAAOgwB,GAC3B,IAKLhgK,EAAWuhF,IAAK,gBAAiB/xE,IAChCsuJ,GAAkCC,EAAU,CAC3Cz3G,cACA05G,wBAIFhgK,EAAWuhF,IAAK,mBAAoB1W,mBAAoB,CACvD9wE,MAAO,QACP6hC,KAAM,CAAEkiC,GAAgB3hC,aACvB,MAAM6zG,EAAMlyE,EAAa9gC,aAAc,OAMvC,OLtLG,SAAwBkc,EAA0B/c,EAAwB+T,GAGhF,OAFA/T,EAAOoa,kBAAmB,SAAS,EAAM2C,GAElCwlF,GAAUxlF,EAAa/c,EAAQ,CAAE+T,SACzC,CKkLWowH,CALQlC,GAA0BjiI,EAAQ4hI,EAAU/tB,EAAK,CAC/D1pF,cACAy5G,sBAAsB,IAGO5jI,EAAQx/B,EAAG,gBAAkB,IAK7DqD,EAAWuhF,IAAK,mBAAoB/xE,IACnCsuJ,GAAkCC,EAAU,CAC3Cz3G,cACAy5G,sBAAsB,KAIxB//J,EAAWuhF,IAAK,UAEdvX,iBAAkB,CAClBpuC,KAAMxiB,GAAW,CAAE,SAAUktC,GAAcp1B,SAAU9X,EAAQzV,OAAUyV,EAAQ4jB,aAAc,OAC5F,CAAEr5B,MAAM,GACR,KACD5J,MAAO,CAAEwmK,GAAapkI,aACrB,MAAM6zG,EAAMuwB,EAAUvjI,aAAc,OAEpC,OAAK+gI,EAASoB,SAAUnvB,GAChB7zG,EAAOl1B,cAAe,QAAS,CAAE+oI,QAGlC,IAAI,IAIZhmE,iBAAkB,CAClBpuC,KAAM,CACLj4B,KAAM,MACNuD,WAAY,CACX,mBAAmB,IAGrBnN,MAAO,CAAEwmK,GAAapkI,aACrB,MAAM6zG,EAAMuwB,EAAUvjI,aAAc,mBAEpC,OAAK+gI,EAASoB,SAAUnvB,GAChB7zG,EAAOl1B,cAAe,QAAS,CAAE+oI,QAGlC,IAAI,IAIZxgI,KAAK2xD,IAmBLA,EAAWnwD,GAAwB,kBAlBgB,CAAEgG,EAAKvS,EAAMw8D,KAC/D,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAE9tE,MAAM,EAAM08B,QAAS,UAC7E,OAGD,MAAM,WAAE4+B,EAAU,YAAEuS,GAAgBvQ,EAAc6C,gBAAiBr/D,EAAKgtE,SAAUhtE,EAAK+sE,aAEvF/sE,EAAKw6D,WAAaA,EAClBx6D,EAAK+sE,YAAcA,EAEE,GAAOvS,EAAYpxB,aAIvCozB,EAAcwB,WAAWnC,OAAQ77D,EAAKgtE,SAAU,CAAE9tE,MAAM,EAAM08B,QAAS,SACxE,GAG+D,GAEnE,ECrQD,MAAMmgI,GAAa,2DAMJ,MAAMC,WAAuB,GA8B3C,WAAA/8J,CAAauyB,GACZ9mB,MAAO8mB,GAEPt0B,KAAK++J,WAAa,KAClB/+J,KAAKg/J,kBAAoB,IAC1B,CA/BA,mBAAkB9nI,GACjB,MAAO,CAAE,GAAW,GAAQ,GAC7B,CAKA,qBAAkBZ,GACjB,MAAO,gBACR,CA2BO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdk+F,EAAgBl+F,EAAOl8B,MAAMoL,SAK7B4iI,EAAuC9xG,EAAOqC,QAAQvuB,IAAK,qBACjEpI,KAAKwP,SAAU42H,EAAmB,uBAAuB,KACxD,MAAM74F,EAAailF,EAAcv9F,UAAUoY,gBAErC4xH,EAAmB,GAAar1E,aAAcr8C,EAAW7M,OAC/Du+H,EAAiB5nG,WAAa,aAE9B,MAAM6nG,EAAoB,GAAat1E,aAAcr8C,EAAW5M,KAChEu+H,EAAkB7nG,WAAa,SAE/Bm7D,EAAc/iH,KAAM,eAAe,KAClCzP,KAAKm/J,4BAA6BF,EAAkBC,GAEpDD,EAAiBn+I,SACjBo+I,EAAkBp+I,QAAQ,GACxB,CAAElU,SAAU,QAAU,IAGO0nB,EAAO0qE,SAAS52F,IAAK,QAC1CiH,GAAI,WAAW,KACrBrP,KAAK++J,aACT,GAAOx7J,OAAOsvB,aAAc7yB,KAAK++J,YACjC/+J,KAAKg/J,kBAAmBl+I,SAExB9gB,KAAK++J,WAAa,KAClB/+J,KAAKg/J,kBAAoB,KAC1B,GACE,CAAEpyJ,SAAU,QAChB,CASQ,2BAAAuyJ,CAA6BC,EAA4BC,GAChE,MAAM/qI,EAASt0B,KAAKs0B,OACdgrI,EAAgBhrI,EAAOqC,QAAQvuB,IAAKs2J,IAAoBtC,SAExDmD,EAAW,IAAI,GAAWH,EAAcC,GACxC3oH,EAAS6oH,EAAS10H,UAAW,CAAE9B,kBAAkB,IAEvD,IAAIslG,EAAM,GAEV,UAAYlvI,KAAQu3C,EACdv3C,EAAKc,KAAKiO,GAAI,gBAClBmgI,GAAOlvI,EAAKc,KAAK6C,MAOnB,GAHAurI,EAAMA,EAAIxgH,QAGJwgH,EAAIxxI,MAAOgiK,IAGhB,YAFAU,EAASz+I,SAMV,IAAMw+I,EAAc9B,SAAUnvB,GAG7B,YAFAkxB,EAASz+I,SAKmCwT,EAAO0qE,SAAS52F,IAAK,cAG1CssB,WAOxB10B,KAAKg/J,kBAAoB,GAAap1E,aAAcw1E,GAGpDp/J,KAAK++J,WAAa,GAAOx7J,OAAOwvB,YAAY,KAC3CuB,EAAOl8B,MAAMmiC,QAAQC,IACpBx6B,KAAK++J,WAAa,KAElBvkI,EAAOv0B,OAAQs5J,GACfA,EAASz+I,SAET,IAAIw3B,EAAyC,KAIE,eAA1Ct4C,KAAKg/J,kBAAmB1oK,KAAK8+B,WACjCkjB,EAAoBt4C,KAAKg/J,mBAG1BrC,GAAaroI,EAAOl8B,MAAOi2I,EAAK/1F,GAAmB,GAEnDt4C,KAAKg/J,kBAAmBl+I,SACxB9gB,KAAKg/J,kBAAoB,IAAI,IAG9B1qI,EAAOqC,QAAQvuB,IAAK,IAASiuH,wBAAwB,GACnD,MA/BFkpC,EAASz+I,QAgCX,E,eC9KG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQgZ,OCoBR,MAAM0lI,WAAsB,GA6D1C,WAAAz9J,CAAa09J,EAA+DvmI,GAC3E1rB,MAAO0rB,GAEP,MAAMl+B,EAAIk+B,EAAOl+B,EAEjBgF,KAAKwqG,aAAe,IAAI,GACxBxqG,KAAKo/F,WAAa,IAAI,GACtBp/F,KAAKyR,IAAK,qBAAsB,IAChCzR,KAAK0xJ,aAAe1xJ,KAAK2xJ,kBAEzB3xJ,KAAKu+I,eAAiBv+I,KAAKw+I,cAAexjJ,EAAG,QAAU,GAAMqlG,MAAO,kBACpErgG,KAAKu+I,eAAetzI,KAAO,SAC3BjL,KAAKu+I,eAAev3I,KAAM,aAAczH,GAAIS,KAAM,sBAAsBxH,KAAWA,IAEnFwH,KAAKy+I,iBAAmBz+I,KAAKw+I,cAAexjJ,EAAG,UAAY,GAAMy4B,OAAQ,mBAAoB,UAE7FzzB,KAAK0+I,YAAc,IAAI,GAEvB1+I,KAAK6tG,aAAe,IAAI,GAAa,CACpCnB,WAAY1sG,KAAK0+I,YACjBl0C,aAAcxqG,KAAKwqG,aACnBmC,iBAAkB3sG,KAAKo/F,WACvB1hD,QAAS,CAERyvD,cAAe,cAGfD,UAAW,SAIbltG,KAAK0/J,YAAcD,EAEnBz/J,KAAK4jG,YAAa,CACjBzsF,IAAK,OAEL5R,WAAY,CACXg5B,MAAO,CACN,KACA,gBACA,sBAGD8qE,SAAU,MAGX/xF,SAAU,CACTtX,KAAK0xJ,aACL1xJ,KAAKu+I,eACLv+I,KAAKy+I,mBAGR,CAKgB,MAAA9sG,GACfnkC,MAAMmkC,SAEN,GAAe,CACd1X,KAAMj6B,OAGY,CAClBA,KAAK0xJ,aACL1xJ,KAAKu+I,eACLv+I,KAAKy+I,kBAGK7gJ,SAAStE,IAEnB0G,KAAK0+I,YAAY7wI,IAAKvU,GAGtB0G,KAAKwqG,aAAa38F,IAAKvU,EAAEme,QAAU,IAIpCzX,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,SAE/B,MAAM0b,EAAoBrwB,GAAyBA,EAAKqwB,kBAKxDnzB,KAAKo/F,WAAW3tF,IAAK,aAAc0hB,GACnCnzB,KAAKo/F,WAAW3tF,IAAK,YAAa0hB,GAClCnzB,KAAKo/F,WAAW3tF,IAAK,UAAW0hB,GAChCnzB,KAAKo/F,WAAW3tF,IAAK,YAAa0hB,EACnC,CAKgB,OAAAnN,GACfxY,MAAMwY,UAENhmB,KAAKwqG,aAAaxkF,UAClBhmB,KAAKo/F,WAAWp5E,SACjB,CAKO,KAAAinB,GACNjtC,KAAK6tG,aAAaZ,YACnB,CAQA,OAAWohC,GACV,OAAOruI,KAAK0xJ,aAAaxnD,UAAUzyF,QAASjf,MAAMq1B,MACnD,CAEA,OAAWwgH,CAAKA,GACfruI,KAAK0xJ,aAAaxnD,UAAUzyF,QAASjf,MAAQ61I,EAAIxgH,MAClD,CAKO,OAAA0nF,GACNv1G,KAAK2/J,kBAEL,UAAYC,KAAa5/J,KAAK0/J,YAAc,CAC3C,MAAMp1D,EAAYs1D,EAAW5/J,MAG7B,GAAKsqG,EAIJ,OAFAtqG,KAAK0xJ,aAAapnD,UAAYA,GAEvB,CAET,CAEA,OAAO,CACR,CAQO,eAAAq1D,GACN3/J,KAAK0xJ,aAAapnD,UAAY,KAC9BtqG,KAAK0xJ,aAAannD,SAAWvqG,KAAK6/J,wBACnC,CAOQ,eAAAlO,GACP,MAAM32J,EAAIgF,KAAKk5B,OAAQl+B,EAEjBqjJ,EAAe,IAAI,GAAkBr+I,KAAKk5B,OAAQ,IAClD4mI,EAAazhB,EAAan0C,UAchC,OAZAlqG,KAAK6/J,yBAA2B7kK,EAAG,qCACnCgF,KAAK+/J,qBAAuB/kK,EAAG,wDAE/BqjJ,EAAa9vG,MAAQvzC,EAAG,aACxBqjJ,EAAa9zC,SAAWvqG,KAAK6/J,yBAE7BC,EAAWzwJ,GAAI,SAAS,KAEvBgvI,EAAa9zC,SAAWu1D,EAAWroJ,QAASjf,MAAQwH,KAAK+/J,qBAAwB//J,KAAK6/J,yBACtF7/J,KAAKggK,mBAAqBF,EAAWroJ,QAASjf,MAAMq1B,MAAM,IAGpDwwH,CACR,CAWQ,aAAAG,CAAejwG,EAAem7D,EAAcvjE,EAAmB30B,GACtE,MAAMgkG,EAAS,IAAI,GAAYx1G,KAAKk5B,QAkBpC,OAhBAs8E,EAAO/jG,IAAK,CACX88B,QACAm7D,OACAG,SAAS,IAGV2L,EAAO/S,eAAgB,CACtBl9F,WAAY,CACXg5B,MAAO4H,KAIJ30B,GACJgkG,EAAOpkG,SAAU,WAAY7R,GAAIS,KAAMwR,GAGjCgkG,CACR,ECzRc,MAAMyqD,WAAqB,GAIzC,mBAAkB/oI,GACjB,MAAO,CAAEwnI,GACV,CAKA,qBAAkBpoI,GACjB,MAAO,cACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdoB,EAA6BpB,EAAO0qE,SAAS52F,IAAK,cAExDksB,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,cAAcqrB,IAC7C,MAAMi4E,EAAW,GAAgBj4E,GAIjC,OAFAl5B,KAAKkgK,eAAgB/uD,EAAUz7E,GAExBy7E,CAAQ,GAEjB,CAEQ,cAAA+uD,CAAgB/uD,EAAwBz7E,GAC/C,MAAMpB,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACXw6G,EAASrE,EAAShG,WAClBixD,EAAW9nI,EAAOqC,QAAQvuB,IAAKs2J,IAAoBtC,SAEzDjrD,EAAS1hG,KAAM,iBAAiB,KAC/B,MAAMqgH,EAAO,IAAM1tB,GAA4Bo9D,IAAlC,CAiDhB,SAA4BxkK,EAAoBohK,GAC/C,MAAO,CACNtsC,IACC,IAAMA,EAAKue,IAAI/1I,OACd,OAAO0C,EAAG,6BACX,EAED80H,IACC,IAAMssC,EAASoB,SAAU1tC,EAAKue,KAC7B,OAAOrzI,EAAG,mCACX,EAGH,CA9DqEmlK,CAAmB7rI,EAAOt5B,EAAGohK,GAAY9nI,EAAO4E,QAElHi4E,EAAS/F,UAAU9zF,SAASzJ,IAAKiiH,GAKjCta,EAAOnmG,GAAI,QAAQ,KAClBygH,EAAKztB,wBAOLytB,EAAKue,IAAM34G,EAAQl9B,OAAS,GAC5Bs3H,EAAK4hC,aAAaxnD,UAAUW,SAC5BilB,EAAKvtB,sBAAsB,GACzB,CAAE31F,SAAU,QAEfukG,EAAS9hG,GAAI,UAAU,KACjBygH,EAAKva,YACTjhF,EAAOiB,QAAS,aAAcu6F,EAAKue,KACnC/5G,EAAO6qE,QAAQllE,KAAKgT,QACrB,IAGDkkE,EAAS9hG,GAAI,iBAAiB,IAAMygH,EAAK6vC,oBACzCxuD,EAAS9hG,GAAI,UAAU,KACtBilB,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAG5B6iF,EAAK1+G,SAAU,SAAU,UAAW7R,GAAI4xG,GACxC2e,EAAK4hC,aAAaxnD,UAAUljG,KAAM,SAAUzH,GAAIm2B,EAAS,SAGzDo6F,EAAK4hC,aAAa1qJ,KAAM,aAAczH,GAAIm2B,EAAS,YAAa,IAGjEy7E,EAASnqG,KAAM,aAAczH,GAAIm2B,GAEjC8/E,EAAO/jG,IAAK,CACX88B,MAAOvzC,EAAG,gBACV0uG,KCvGH,4eDwGGG,SAAS,GAEX,E,eEvGG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ/vE,OCmBhB,SAASsmI,GACfzkF,EACAp5C,GAEA,IAAMo5C,EAAiBrgD,WACtB,OAGD,MAAMd,EAAS,IAAI,GAAcmhD,EAAiBn4E,UAC5C68J,EA6EP,SACC1kF,EACAnhD,GAEA,MAAM1V,EAAQ0V,EAAOud,cAAe4jC,GAG9B2kF,EAA0B,IAAI,GAAS,CAC5Ct+J,KAAM,WACN68B,OAAQ,CACP,WAAY,QAIRwhI,EAA2C,GAEjD,UAAY7nK,KAASssB,EACpB,GAAoB,iBAAftsB,EAAMyS,MAA2Bq1J,EAAwBzjK,MAAOrE,EAAMyH,MAAwB,CAClG,MAAMsgK,EAAWC,GAAiBhoK,EAAMyH,MAExCogK,EAAiBzgK,KAAM,CACtB6X,QAASjf,EAAMyH,KACfJ,GAAI0gK,EAAS1gK,GACb4gK,MAAOF,EAASE,MAChBjV,OAAQ+U,EAAS/U,QAEnB,CAGD,OAAO6U,CACR,CA3G0BK,CAAyB/kF,EAAkBnhD,GAEpE,IAAM6lI,EAAiB/nK,OACtB,OAGD,IAAIqoK,EAAkC,KAClCC,EAAqB,EAEzBP,EAAiBziK,SAAS,CAAEijK,EAAiBrmK,KAC5C,MAAMsmK,EAkXR,SAA0BC,EAA+BC,GACxD,IAAMD,EACL,OAAO,EAGR,GAAKA,EAAalhK,KAAOmhK,EAAYnhK,GAKpC,OAAKmhK,EAAYxV,OAASuV,EAAavV,QAAW,EAOnD,MAAM5kI,EAAkBo6I,EAAYvpJ,QAAQmP,gBAE5C,IAAMA,EACL,OAAO,EAIR,OAGgBnP,EAHAmP,IAITnP,EAAQvJ,GAAI,UAAW,OAAUuJ,EAAQvJ,GAAI,UAAW,OADhE,IAAiBuJ,CAFjB,CA3Y0BwpJ,CAAiBZ,EAAkB7lK,EAAI,GAAKqmK,GAC9DK,EAA0BJ,EAAkB,KAAOT,EAAkB7lK,EAAI,GACzE2mK,GAmZiEH,EAnZUH,GAmZhDE,EAnZuBG,GAoZnCF,EAAYxV,OAASuV,EAAavV,OAASwV,EAAYxV,OAAS,GADvF,IAAmCuV,EAAsCC,EA5YvE,GALKF,IACJH,EAAc,KACdC,EAAqB,IAGhBD,GAAyC,IAA1BQ,EAA8B,CAClD,MAAMnJ,EAoHT,SAA0BoJ,EAA+B7+H,GACxD,MAAM8+H,EAAkB,IAAI13J,OAAQ,UAAWy3J,EAAavhK,WAAauhK,EAAa5V,qBAAuB,MACvG8V,EAAqB,2CACrBC,EAAsB,wDAEtBC,EAAiBH,EAAgBxoJ,KAAM0pB,GAE7C,IAAIy2H,EAAgB,UAChB/tJ,EAAO,KACPw2J,EAAa,KAEjB,GAAKD,GAAkBA,EAAgB,GAAM,CAC5C,MAAME,EAAqBJ,EAAmBzoJ,KAAM2oJ,EAAgB,IAWpE,GATKE,GAAsBA,EAAoB,KAC9C1I,EAAgB0I,EAAoB,GAAI7zI,OACxC5iB,EAAyB,WAAlB+tJ,GAAgD,UAAlBA,EAA4B,KAAO,MAOlD,WAAlBA,EAA6B,CACjC,MAAM2I,EAwBT,SAAgClqJ,GAC/B,MAAMmqJ,EAwBP,SAA6BnqJ,GAG5B,GAAKA,EAAQokB,SAAU,GAAK3tB,GAAI,SAC/B,OAAO,KAGR,UAAYysF,KAAaljF,EAAQ+jB,cAAgB,CAGhD,IAAMm/D,EAAUzsF,GAAI,UAAW,QAC9B,SAGD,MAAM2zJ,EAAoBlnE,EAAU9+D,SAAU,GAE9C,GAAMgmI,EAKN,OAAKA,EAAkB3zJ,GAAI,SACnB2zJ,EAGCA,EAA2BhmI,SAAU,EAC/C,CAGA,OAAO,IACR,CAtD2BimI,CAAoBrqJ,GAE9C,IAAMmqJ,EACL,OAAO,KAGR,MAAMG,EAAaH,EAAkBxkI,MAErC,GAAoB,MAAf2kI,EACJ,MAAO,SACD,GAAoB,MAAfA,EACX,MAAO,OACR,GAEyB,MAAfA,EACT,MAAO,SAGR,OAAO,IACR,CA5CyBC,CAAuBZ,EAAa3pJ,SAErDkqJ,IACJ3I,EAAgB2I,EAElB,KAAO,CACN,MAAMM,EAAsBV,EAAoB1oJ,KAAM2oJ,EAAgB,IAEjES,GAAuBA,EAAqB,KAChDR,EAAarkK,SAAU6kK,EAAqB,IAE9C,CACD,CAEA,MAAO,CACNh3J,OACAw2J,aACAp8J,MAAO68J,GAAwBlJ,GAEjC,CA/JqBmJ,CAAiBtB,EAAiBt+H,GAEpD,GAAMo+H,GAEC,GAAKE,EAAgBrV,OAASoV,EAAqB,CACzD,MAAMwB,EAAezB,EAAY9kI,SAAU8kI,EAAYrlI,WAAa,GAC9D+mI,EAAoBD,EAAcvmI,SAAUumI,EAAa9mI,WAAa,GAE5EqlI,EAAc2B,GAAoBtK,EAAWqK,EAAmB7nI,GAChEomI,GAAsB,CACvB,MAAO,GAAKC,EAAgBrV,OAASoV,EAAqB,CACzD,MAAM2B,EAAuB3B,EAAqBC,EAAgBrV,OAElEmV,EAyYJ,SAAgC6B,EAA0BrB,GACzD,MAAM/kI,EAAYomI,EAAYrmI,aAAc,CAAEG,aAAa,IAE3D,IAAImmI,EAAa,KACbC,EAAc,EAElB,UAAY9rH,KAAYxa,EAKvB,IAJKwa,EAAS1oC,GAAI,UAAW,OAAU0oC,EAAS1oC,GAAI,UAAW,QAC9Dw0J,IAGIA,IAAgBvB,EAAwB,CAC5CsB,EAAa7rH,EACb,KACD,CAGD,OAAO6rH,CACR,CA3ZkBE,CAAuBhC,EAAa4B,GAClD3B,EAAqBC,EAAgBrV,MACtC,OAZCmV,EAAc2B,GAAoBtK,EAAW6I,EAAgBppJ,QAAS+iB,GAclEqmI,EAAgBrV,QAAUoV,IACxBD,EAAYzyJ,GAAI,UAAW8pJ,EAAU/sJ,QAC1C01J,EAAcnmI,EAAO6c,OAAQ2gH,EAAU/sJ,KAAM01J,IAGhD,CAEA,MAAMrJ,EAuQR,SAAuC7/I,EAAsB+iB,GAI5D,OA0CD,SAA8B/iB,EAAsB+iB,GAEnD,MAAMooI,EAAgB,IAAI,GAAS,CAClC5gK,KAAM,OACN68B,OAAQ,CACP,WAAY,YAIR/Z,EAAQ0V,EAAOud,cAAetgC,GAEpC,UAAYjf,KAASssB,EACA,iBAAftsB,EAAMyS,MAA2B23J,EAAc/lK,MAAOrE,EAAMyH,OAChEu6B,EAAOv0B,OAAQzN,EAAMyH,KAGxB,CA7DC4iK,CAAqBprJ,EAAS+iB,GAC9BA,EAAOma,YAAa,cAAel9B,GAE5B+iB,EAAO6c,OAAQ,KAAM5/B,EAC7B,CA5QmBqrJ,CAA8BjC,EAAgBppJ,QAAS+iB,GAExEA,EAAO70B,YAAa2xJ,EAAUqJ,EAAc,GAE9C,CAoMA,SAASuB,GAAwB1pK,GAChC,GAAKA,EAAMosD,WAAY,uBACtB,MAAO,uBAGR,OAASpsD,GACR,IAAK,cACJ,MAAO,cACR,IAAK,cACJ,MAAO,cACR,IAAK,cACJ,MAAO,cACR,IAAK,cACJ,MAAO,cACR,IAAK,SACL,IAAK,OACL,IAAK,SACJ,OAAOA,EACR,QACC,OAAO,KAEV,CAUA,SAAS8pK,GACRtK,EACAvgJ,EACA+iB,GAEA,MAAM57B,EAAS6Y,EAAQ7Y,OACjBmB,EAAOy6B,EAAOl1B,cAAe0yJ,EAAU/sJ,MACvCwa,EAAW7mB,EAAOo9B,cAAevkB,GAAY,EAcnD,OAZA+iB,EAAOikE,YAAah5E,EAAU1lB,EAAMnB,GAI/Bo5J,EAAU3yJ,OACdm1B,EAAOka,SAAU,kBAAmBsjH,EAAU3yJ,MAAOtF,GAGjDi4J,EAAUyJ,YAAczJ,EAAUyJ,WAAa,GACnDjnI,EAAO/0B,aAAc,QAASuyJ,EAAUyJ,WAAY1hK,GAG9CA,CACR,CAkCA,SAASygK,GAAiB/oJ,GACzB,MAAM3U,EAAqB,CAAC,EACtBk1J,EAAYvgJ,EAAQsnB,SAAU,YAEpC,GAAKi5H,EAAY,CAChB,MAAM+K,EAAU/K,EAAUn7J,MAAO,wBAC3BmmK,EAAahL,EAAUn7J,MAAO,sBAC9BomK,EAAcjL,EAAUn7J,MAAO,wBAEhCkmK,GAAWC,GAAcC,IAC7BngK,EAAKjD,GAAKkjK,EAAS,GACnBjgK,EAAK29J,MAAQuC,EAAY,GACzBlgK,EAAK0oJ,OAASpuJ,SAAU6lK,EAAa,IAEvC,CAEA,OAAOngK,CACR,CC1WO,SAASogK,GAA+BvnF,EAAwCwnF,GACtF,IAAMxnF,EAAiBrgD,WACtB,OAGD,MAAM8nI,EAAe,IAAI,GAAcznF,EAAiBn4E,UAClD6/J,EAiCP,SAA2B1nF,EAAwCnhD,GAClE,MAAM1V,EAAQ0V,EAAOud,cAAe4jC,GAE9B2nF,EAAuB,IAAI,GAAS,CACzCthK,KAAM,WAGDqhK,EAAY,GAElB,UAAY7qK,KAASssB,EAAQ,CAC5B,GAAmB,gBAAdtsB,EAAMyS,KACV,SAGD,MAAMsb,EAAK/tB,EAAMyH,KACX2mB,EAAkBL,EAAGK,gBACrB28I,EAAkB38I,GAAmBA,EAAgB1Y,GAAI,WAAc0Y,EAAgB5kB,KAAO,KAG/FshK,EAAqBzmK,MAAO0pB,IAAQA,EAAG8U,aAAc,cAAqC,gBAApBkoI,GAC1EF,EAAUzjK,KAAQpH,EAAMyH,KAAsBo7B,aAAc,MAE9D,CAEA,OAAOgoI,CACR,CA1DmBG,CAAkB7nF,EAAkBynF,IAkEvD,SACCC,EACA1nF,EACAnhD,GAEA,MAAM1V,EAAQ0V,EAAOud,cAAe4jC,GAE9B8nF,EAAuB,IAAI,GAAS,CACzCzhK,KAAM,QAGD0hK,EAAO,GAEb,UAAYlrK,KAASssB,EACpB,GAAKtsB,EAAMyH,KAAKiO,GAAI,YAAeu1J,EAAqB5mK,MAAOrE,EAAMyH,MAAS,CAC7E,MAAMsmB,EAAK/tB,EAAMyH,KACX0jK,EAASp9I,EAAG8U,aAAc,YAAe9U,EAAG8U,aAAc,YAAct+B,MAAO,KAAQ,GAExF4mK,EAAOrrK,QAAUqrK,EAAOptJ,OAAOqtJ,GAASP,EAAUr6J,QAAS46J,IAAW,IAC1EF,EAAK9jK,KAAM2mB,GAECA,EAAG8U,aAAc,QAC7BqoI,EAAK9jK,KAAM2mB,EAEb,CAGD,UAAY84H,KAAOqkB,EAClBlpI,EAAOv0B,OAAQo5I,EAEjB,CA9FCwkB,CAAwCR,EAAW1nF,EAAkBynF,GA4HtE,SAA4BU,EAAyBnoF,EAAwCnhD,GAC5F,MAAM1V,EAAQ0V,EAAOud,cAAe4jC,GAE9BgoF,EAA6B,GAEnC,UAAYnrK,KAASssB,EACpB,GAAmB,gBAAdtsB,EAAMyS,MAA0BzS,EAAMyH,KAAKiO,GAAI,UAAW,WAAc,CAC5E,MAAMrO,EAAKrH,EAAMyH,KAAKo7B,aAAc,MAEpC,GAAKyoI,EAASv0I,SAAU1vB,GACvB,SAGKkkK,EAAqBvrK,EAAMyH,KAAKrB,OAAQ48B,cAAe37B,IAC5D8jK,EAAO/jK,KAAMpH,EAAMyH,KAErB,CAGD,UAAY2jK,KAASD,EAAS,CAC7B,MAAMv+H,EAAiC,CACtCwvG,IAAKovB,EAASJ,IAGVA,EAAM39H,aAAc,SACxBb,EAAMlZ,IAAM03I,EAAMvoI,aAAc,QAGjC,MAAMgkH,EAAM7kH,EAAOl1B,cAAe,MAAO8/B,GAEzC5K,EAAOikE,YAAamlE,EAAMh/J,MAAS,EAAGy6I,EAAKukB,EAAMhlK,OAClD,CAEA,SAASmlK,EAAqB78H,EAA2BrnC,GACxD,UAAYV,KAAQ+nC,EAEnB,GAAK/nC,EAAK+O,GAAI,WAAc,CAC3B,GAAkB,OAAb/O,EAAK6C,MAAiB7C,EAAKk8B,aAAc,aAAgBx7B,EAC7D,OAAO,EAGR,GAAKkkK,EAAqB5kK,EAAKq8B,cAAe37B,GAC7C,OAAO,CAET,CAGD,OAAO,CACR,CAEA,SAASmkK,EAASJ,GACjB,UAAYjsJ,KAASisJ,EAAMpoI,cAE1B,GAAK7jB,EAAMzJ,GAAI,YAAeyJ,EAAM0jB,aAAc,OACjD,OAAO1jB,EAAM0jB,aAAc,MAG9B,CACD,CArLC4oI,CAAmBZ,EAAW1nF,EAAkBynF,GAoGjD,SAAiCznF,EAAwCnhD,GACxE,MAAM1V,EAAQ0V,EAAOud,cAAe4jC,GAE9B2nF,EAAuB,IAAI,GAAS,CACzCthK,KAAM,WAGD2hK,EAAS,GAEf,UAAYnrK,KAASssB,EACD,gBAAdtsB,EAAMyS,MAA0Bq4J,EAAqBzmK,MAAOrE,EAAMyH,OACtE0jK,EAAO/jK,KAAMpH,EAAMyH,MAIrB,UAAY2jK,KAASD,EACpBnpI,EAAOv0B,OAAQ29J,EAEjB,CArHCM,CAAwBvoF,EAAkBynF,GAE1C,MAAMrpB,EA0LP,SACCp+D,EACAnhD,GAEA,MAAM1V,EAAQ0V,EAAOud,cAAe4jC,GAE9B8nF,EAAuB,IAAI,GAAS,CACzCzhK,KAAM,QAGD0hK,EAAO,GAEb,UAAYlrK,KAASssB,EACftsB,EAAMyH,KAAKiO,GAAI,YAAeu1J,EAAqB5mK,MAAOrE,EAAMyH,OAC/DzH,EAAMyH,KAAKo7B,aAAc,OAASupB,WAAY,YAClD8+G,EAAK9jK,KAAMpH,EAAMyH,MAKpB,OAAOyjK,CACR,CA/MgBS,CAAqCxoF,EAAkBynF,GAEjErpB,EAAOzhJ,QA+Pb,SACC8rK,EACAC,EACA7pI,GAGA,GAAK4pI,EAAc9rK,SAAW+rK,EAAiB/rK,OAC9C,QAAUkC,EAAI,EAAGA,EAAI4pK,EAAc9rK,OAAQkC,IAAM,CAChD,MAAM8pK,EAAS,QAASD,EAAkB7pK,GAAIyQ,eAAiBs5J,GAAqBF,EAAkB7pK,GAAI3C,OAC1G2iC,EAAO/0B,aAAc,MAAO6+J,EAAQF,EAAe5pK,GACpD,CAEF,CA1QEgqK,CAAiDzqB,EAuNnD,SAAkCopB,GACjC,IAAMA,EACL,MAAO,GAGR,MAAMsB,EAAqB,uFACrBC,EAAe,IAAI/6J,OAAQ,OAAS86J,EAAmBvhK,OAAS,yBAA0B,KAC1F62I,EAASopB,EAAQtmK,MAAO6nK,GACxBvmK,EAAS,GAEf,GAAK47I,EACJ,UAAY/mF,KAAS+mF,EAAS,CAC7B,IAAIqD,GAA4B,EAE3BpqF,EAAMzjC,SAAU,aACpB6tH,EAAY,YACDpqF,EAAMzjC,SAAU,gBAC3B6tH,EAAY,cAGRA,GACJj/I,EAAOyB,KAAM,CACZ/H,IAAKm7D,EAAM/5C,QAASwrJ,EAAoB,IAAKxrJ,QAAS,eAAgB,IACtEhO,KAAMmyI,GAGT,CAGD,OAAOj/I,CACR,CArP2DwmK,CAAyBxB,GAAWC,EAE/F,CASO,SAASmB,GAAqBK,GACpC,OAAOpiK,KAAMoiK,EAAU/nK,MAAO,UAAYG,KAAKC,GACvCge,OAAO+R,aAAc5vB,SAAUH,EAAM,OACzCC,KAAM,IACX,CCzCA,MAAM2nK,GAAe,uEACfC,GAAe,sCAKN,MAAMC,GAQpB,WAAAhjK,CAAayB,GACZxD,KAAKwD,SAAWA,CACjB,CAKO,QAAAwhK,CAAUC,GAChB,OAAOJ,GAAa5iK,KAAMgjK,IAAgBH,GAAa7iK,KAAMgjK,EAC9D,CAKO,OAAA1vI,CAASzyB,GACf,MAAQ0iB,KAAMm2D,EAAgB,aAAEp5C,GAAiBz/B,EAAKoiK,YAEtD9E,GAAwCzkF,EAAkBp5C,GAC1D2gI,GAA+BvnF,EAAkB74E,EAAK2wD,aAAad,QAAS,aC/BvE,SAA4BgpB,GAClC,MAAMynF,EAAe,IAAI,GAAcznF,EAAiBn4E,UAExD,UAAYvD,KAAQ07E,EAAiBngD,cAAgB,CACpD,IAAMv7B,EAAKiO,GAAI,WACd,SAOD,GAAKjO,EAAKiO,GAAI,UAAW,SAAY,CACpCk1J,EAAa39J,aAAc,QAAS,OAAQxF,GAC5C,QACD,CAEA,MAAMklK,EAAQllK,EAAKo7B,aAAc,SAC3B1jB,EAAQ1X,EAAK47B,SAAU,GAGV,QAAd57B,EAAK+B,MAAmBmjK,GAAUxtJ,GAMlCA,EAAMzJ,GAAI,UAAW,UACzBk1J,EAAa39J,aAAc,QAAmB,WAAV0/J,EAAqB,OAASA,EAAOxtJ,EAE3E,CACD,CDCEytJ,CAAmBzpF,GACnB74E,EAAK5C,QAAUy7E,CAChB,EEiBD,SAAS0pF,GACR9tH,EACAhzB,EACAiW,GACA,cAAEqpB,EAAa,qBAAEC,IAEjB,IAAIr+B,EAAW+U,EAAOkd,iBAAkBH,EAA0B,WAAbhzB,EAAyB,QAAU,UAaxF,OANAkB,EAAWA,EAAS+kB,yBAAyB,EAAIvqC,UAChDA,EAAKiO,GAAI,aACR21C,EAAct0B,SAAUtvB,EAAK+B,QAC7B8hD,EAAqBv0B,SAAUtvB,EAAK+B,OACnC,CAAEuiB,cAEe,WAAbA,EAAyBkB,EAASykB,UAAYzkB,EAAS0kB,UAC/D,CAKA,SAASm7H,GAAoBnmK,EAAuB0kD,GACnD,QAAS1kD,GAAQA,EAAK+O,GAAI,YAAe21C,EAAct0B,SAAUpwB,EAAK6C,KACvE,CC9EA,MAAMujK,GAAkB,8CAKT,MAAMC,GAQpB,WAAAzjK,CAAayB,GACZxD,KAAKwD,SAAWA,CACjB,CAKO,QAAAwhK,CAAUC,GAChB,OAAOM,GAAgBtjK,KAAMgjK,EAC9B,CAKO,OAAA1vI,CAASzyB,GACf,MAAM03B,EAAS,IAAI,GAAcx6B,KAAKwD,WAC9BgiB,KAAMm2D,GAAqB74E,EAAKoiK,aC7B3B,SAA4BvpF,EAAwCnhD,GAClF,UAAY7iB,KAASgkE,EAAiBngD,cACrC,GAAK7jB,EAAMzJ,GAAI,UAAW,MAA6C,WAApCyJ,EAAMonB,SAAU,eAA+B,CACjF,MAAMsnE,EAAa1qB,EAAiB3/C,cAAerkB,GAEnD6iB,EAAOv0B,OAAQ0R,GACf6iB,EAAOikE,YAAa4H,EAAY1uF,EAAM6jB,cAAemgD,EACtD,CAEF,CDsBE8pF,CAAmB9pF,EAAkBnhD,GL8ChC,SACNmhD,EACAnhD,GAEA,UAAYhiC,KAASgiC,EAAOud,cAAe4jC,GAAqB,CAC/D,MAAMlkE,EAAUjf,EAAMyH,KAEtB,GAAKwX,EAAQvJ,GAAI,UAAW,MAAS,CAEpC,MAAMvH,EAAa8Q,EAAQokB,SAAU,GAEhCl1B,GAAcA,EAAWuH,GAAI,UAAW,MAC5CssB,EAAO2e,cAAexyC,EAExB,CACD,CACD,CK7DE++J,CAA2B/pF,EAAkBnhD,GDzBhC,SACdmhD,EACAnhD,GAEA,MAAMg5B,EAAe,IAAI,GAAch5B,EAAOh3B,SAASuiC,iBACjD8L,EAAe,IAAIuR,GAAcoQ,EAAc,CAAElQ,cAAe,SAEhEO,EAAgBhS,EAAagS,cAC7BC,EAAuBjS,EAAaiS,qBAEpC6hH,EAAoB,GAE1B,UAAYntK,KAASgiC,EAAOud,cAAe4jC,GAAqB,CAC/D,MAAMlkE,EAAUjf,EAAMyH,KAEtB,GAAKwX,EAAQvJ,GAAI,UAAW,MAAS,CACpC,MAAM+tB,EAAcopI,GAAa5tJ,EAAS,UAAW+iB,EAAQ,CAAEqpB,gBAAeC,yBACxEl9B,EAAkBy+I,GAAa5tJ,EAAS,WAAY+iB,EAAQ,CAAEqpB,gBAAeC,yBAE7E8hH,EAAqBN,GAAoBrpI,EAAa4nB,IAC7ByhH,GAAoB1+I,EAAiBi9B,IAMrC+hH,IAC9BD,EAAkB/lK,KAAM6X,EAE1B,CACD,CAEA,UAAYA,KAAWkuJ,EACjBluJ,EAAQqjB,SAAU,6BACtBN,EAAOv0B,OAAQwR,GAEf+iB,EAAOvhB,QAASxB,EAAS+iB,EAAOl1B,cAAe,KAGlD,CCbEugK,CAA+BlqF,EAAkBnhD,GAEjD13B,EAAK5C,QAAUy7E,CAChB,EEnCD,MAAMmqF,GAAoB,8BAKX,MAAMC,GAQpB,WAAAhkK,CAAayB,GACZxD,KAAKwD,SAAWA,CACjB,CAKO,QAAAwhK,CAAUC,GAChB,OAAOa,GAAkB7jK,KAAMgjK,EAChC,CAKO,OAAA1vI,CAASzyB,GACf,MAAM03B,EAAS,IAAI,GAAcx6B,KAAKwD,WAC9BgiB,KAAMm2D,GAAqB74E,EAAKoiK,aC9B3B,SAAgCvpF,EAAwCnhD,GACtF,UAAY7iB,KAASgkE,EAAiBngD,cACrC,GAAK7jB,EAAMzJ,GAAI,UAAW,6BAAgC,CACzD,MAAMm4F,EAAa1qB,EAAiB3/C,cAAerkB,GAEnD6iB,EAAOv0B,OAAQ0R,GACf6iB,EAAOikE,YAAa4H,EAAY1uF,EAAM6jB,cAAemgD,EACtD,CAEF,CDuBEqqF,CAAuBrqF,EAAkBnhD,GEhC5B,SAAsBmhD,EAAwCnhD,GAC5E,UAAY7iB,KAASgkE,EAAiBngD,cAChC7jB,EAAMzJ,GAAI,UAAW,UAAayJ,EAAMsuB,aAAc,UAC1DzL,EAAO9zB,gBAAiB,QAASiR,EAGpC,CF2BEsuJ,CAAatqF,EAAkBnhD,GGjClB,SAAkCmhD,EAAwCnhD,GACxF,UAAY7iB,KAASgkE,EAAiBngD,cAChC7jB,EAAMzJ,GAAI,UAAW,UAA2C,QAA9ByJ,EAAMonB,SAAU,UACtDvE,EAAOma,YAAa,QAASh9B,EAGhC,CH4BEuuJ,CAAyBvqF,EAAkBnhD,GIlC9B,SAA2BmhD,EAAwCnhD,GACjF,UAAY7iB,KAAShX,MAAMrB,KAAMq8E,EAAiBngD,eAC5C7jB,EAAMzJ,GAAI,UAAW,UACzBssB,EAAOv0B,OAAQ0R,EAGlB,CJ6BEwuJ,CAAkBxqF,EAAkBnhD,GAEpC13B,EAAK5C,QAAUy7E,CAChB,EKMD,SAASyqF,GAA2BnB,GACnC,OAAOA,EAAWhsJ,QAAS,2DAA2D,CAAEm5G,EAAWC,IACzE,IAAlBA,EAAO/5H,OAAe,IAAMqI,MAAO0xH,EAAO/5H,OAAS,GAAI4E,KAAM,MAAY+U,OAAQ,EAAGogH,EAAO/5H,SAEpG,CCvCO,SAAS+tK,GAAWpB,EAAoBl/H,GAC9C,MAAMi3C,EAAY,IAAIj4B,UAKhBuhH,EDbA,SAA2BrB,GAEjC,OAAOmB,GAA2BA,GAA2BnB,IAE3DhsJ,QAAS,mFAAoF,QAC7FA,QAAS,mDAAoD,IAC7DA,QAAS,oEAAqE,SAC9EA,QAAS,QAAS,OAClBA,QAAS,iBAAkB,gBAE3BA,QAAS,+BAAgC,IAEzCA,QAAS,2BAA4B,KACxC,CCAwBstJ,CAsGxB,SAAgCtB,GAC/B,MAAMuB,EAAe,UACfC,EAAe,UAEfC,EAAiBzB,EAAWj8J,QAASw9J,GAE3C,GAAKE,EAAiB,EACrB,OAAOzB,EAGR,MAAM0B,EAAiB1B,EAAWj8J,QAASy9J,EAAcC,EAAiBF,EAAaluK,QAEvF,OAAO2sK,EAAWroK,UAAW,EAAG8pK,EAAiBF,EAAaluK,SAC3DquK,GAAkB,EAAI1B,EAAWroK,UAAW+pK,GAAmB,GACnE,CApH0CC,CAFzC3B,EAAaA,EAAWhsJ,QAAS,wBAAyB,MAKpD4tJ,EAAe7pF,EAAUh4B,gBAAiBshH,EAAgB,cDM1D,SAAiCO,GACvCA,EAAan+D,iBAAkB,yBAA0B9qG,SAAS2oB,IACjE,MAAMugJ,EAAcvgJ,EACdwgJ,EAAkBD,EAAY7rH,UAAU3iD,QAAU,EAExDwuK,EAAY7rH,UAAYt6C,MAAOomK,EAAkB,GAAI7pK,KAAM,MAAY+U,OAAQ,EAAG80J,EAAiB,GAErG,CCXCC,CAAwBH,GAGxB,MAAMI,EAAaJ,EAAarhJ,KAAKiB,UAG/BygJ,EA4CP,SAAyBL,EAAwB9gI,GAChD,MAAMytB,EAAe,IAAI,GAAcztB,GACjC8L,EAAe,IAAIuR,GAAcoQ,EAAc,CAAElQ,cAAe,SAChE2B,EAAW4hH,EAAanzH,yBACxBxM,EAAQ2/H,EAAarhJ,KAAKlf,WAEhC,KAAQ4gC,EAAM5uC,OAAS,GACtB2sD,EAASt/C,YAAauhC,EAAO,IAG9B,OAAO2K,EAAa8N,UAAWsF,EAAU,CAAEmE,cAAc,GAC1D,CAvDkB+9G,CAAgBN,EAAc9gI,GAGzCd,EA2DP,SAAwB4hI,GACvB,MAAMhoI,EAAS,GACT0D,EAAe,GACf6kI,EAAYzmK,MAAMrB,KAAMunK,EAAaQ,qBAAsB,UAEjE,UAAYhiK,KAAS+hK,EACf/hK,EAAMiiK,OAASjiK,EAAMiiK,MAAMC,UAAYliK,EAAMiiK,MAAMC,SAASjvK,SAChEumC,EAAOj/B,KAAMyF,EAAMiiK,OACnB/kI,EAAa3iC,KAAMyF,EAAMohB,YAI3B,MAAO,CACNoY,SACA0D,aAAcA,EAAarlC,KAAM,KAEnC,CA3EsBsqK,CAAeX,GAEpC,MAAO,CACNrhJ,KAAM0hJ,EACND,aACApoI,OAAQoG,EAAapG,OACrB0D,aAAc0C,EAAa1C,aAE7B,C,yHCxBO,SAASklI,GAAwB1wK,EAAayB,EAAgByH,EAAYu6B,EAAgBoG,EAAwB,GACnHpoC,eAAyCooC,GAAuDpoC,EAAQooC,EAC5GpG,EAAO/0B,aAAc1O,EAAKyB,EAAOyH,GAEjCu6B,EAAO9zB,gBAAiB3P,EAAKkJ,EAE/B,CAUO,SAASynK,GAAsBltI,EAAgBk/B,EAA0Bn0D,EAAsC,CAAC,GACtH,MAAMoiK,EAAYntI,EAAOl1B,cAAe,YAAaC,GAKrD,OAHAi1B,EAAOquC,cAAe,YAAa8+F,GACnCntI,EAAO90B,OAAQiiK,EAAWjuG,GAEnBiuG,CACR,CAKO,SAASC,GAAqBC,EAAwBF,GAC5D,MAAMG,EAAQH,EAAU/oK,OAAQA,OAC1BmpK,EAAiB3qK,SAAU0qK,EAAMzsI,aAAc,mBAAgC,MAC/E,OAAE2sI,GAAWH,EAAWI,gBAAiBN,GAE/C,QAASI,GAAkBC,EAASD,CACrC,CCUe,SAASG,KACvB,OAAS1oG,IACRA,EAAWnwD,GAAwB,iBAAiB,CAAEgG,EAAKvS,EAAMw8D,KAChE,MAAM6oG,EAAYrlK,EAAKgtE,SAGvB,IAAMxQ,EAAcwB,WAAW7+D,KAAMkmK,EAAW,CAAEnmK,MAAM,IACvD,OAGD,MAAM,KAAEomK,EAAI,YAAEC,EAAW,eAAEN,GAuH9B,SAAoBI,GACnB,IACIJ,EADAM,EAAc,EAelB,MAAMC,EAAW,GACXC,EAAW,GAIjB,IAAIC,EAEJ,UAAYC,KAAc9nK,MAAMrB,KAAM6oK,EAAU3sI,eAAmD,CAGlG,GAAyB,UAApBitI,EAAWzmK,MAAwC,UAApBymK,EAAWzmK,MAAwC,UAApBymK,EAAWzmK,KAC7E,SAIwB,UAApBymK,EAAWzmK,MAAqBwmK,IACpCA,EAAoBC,GAKrB,MAAMC,EAAM/nK,MAAMrB,KAAMmpK,EAAWjtI,eAAgBz1B,QAChDwgB,GAAsDA,EAAGrY,GAAI,UAAW,QAG3E,UAAYy6J,KAAMD,EAEjB,GACGF,GAAqBC,IAAeD,GAEjB,UAApBC,EAAWzmK,MACXrB,MAAMrB,KAAMqpK,EAAGntI,eAAgBljC,QAC/BqI,MAAMrB,KAAMqpK,EAAGntI,eAAgBjlB,OAAOrS,GAAKA,EAAEgK,GAAI,UAAW,QAG7Dm6J,IACAC,EAAS1oK,KAAM+oK,OACT,CACNJ,EAAS3oK,KAAM+oK,GAGf,MAAMC,EAAcC,GAA0BF,KAExCZ,GAAkBa,EAAcb,KACrCA,EAAiBa,EAEnB,CAEF,CAEA,MAAO,CACNP,cACAN,eAAgBA,GAAkB,EAClCK,KAAM,IAAKE,KAAaC,GAE1B,CA9LiDO,CAAWX,GAGnD5iK,EAAgE,CAAC,EAElEwiK,IACJxiK,EAAWwiK,eAAiBA,GAGxBM,IACJ9iK,EAAW8iK,YAAcA,GAG1B,MAAMP,EAAQxoG,EAAc9kC,OAAOl1B,cAAe,QAASC,GAE3D,GAAM+5D,EAAc8Q,WAAY03F,EAAOhlK,EAAK+sE,aAA5C,CAaA,GATAvQ,EAAcwB,WAAWrC,QAAS0pG,EAAW,CAAEnmK,MAAM,IAGrDomK,EAAKxqK,SAASmrK,GAAOzpG,EAAc4C,YAAa6mG,EAAKzpG,EAAc9kC,OAAOkd,iBAAkBowH,EAAO,UAGnGxoG,EAAc6C,gBAAiBgmG,EAAW7oG,EAAc9kC,OAAOkd,iBAAkBowH,EAAO,QAGnFA,EAAM3lI,QAAU,CACpB,MAAM4mI,EAAMzpG,EAAc9kC,OAAOl1B,cAAe,YAChDg6D,EAAc9kC,OAAO90B,OAAQqjK,EAAKzpG,EAAc9kC,OAAOkd,iBAAkBowH,EAAO,QAEhFJ,GAAsBpoG,EAAc9kC,OAAQ8kC,EAAc9kC,OAAOkd,iBAAkBqxH,EAAK,OACzF,CAEAzpG,EAAc+Q,uBAAwBy3F,EAAOhlK,EAlB7C,CAkBmD,GACjD,CAEL,CA6BO,SAASkmK,GAA4BrkH,GAC3C,OAAS6a,IACRA,EAAWnwD,GAAwB,WAAYs1C,KAAgB,CAAEtvC,EAAKvS,GAAQ03B,aAE7E,IAAM13B,EAAKw6D,WACV,OAGD,MAAMqqG,EAAY7kK,EAAKw6D,WAAW58B,MAAMwJ,UAClC2lC,EAAcr1C,EAAOkd,iBAAkBiwH,EAAW,GAGxD,GAAK7kK,EAAKgtE,SAAS3tC,QAGlB,YAFA3H,EAAOquC,cAAe,YAAagH,GAKpC,MAAMvpE,EAAa3F,MAAMrB,KAAMqoK,EAAUnsI,eAGzC,GAAKl1B,EAAWiQ,OAAOpX,GAAQA,EAAK+O,GAAI,UAAW,aAAgB,CAClE,MAAM8/D,EAAYxzC,EAAOl1B,cAAe,aAExCk1B,EAAO90B,OAAQsoE,EAAWxzC,EAAOkd,iBAAkBiwH,EAAW,IAE9D,UAAYxoK,KAAQmH,EACnBk0B,EAAO8a,KAAM9a,EAAOsd,cAAe34C,GAAQq7B,EAAOkd,iBAAkBs2B,EAAW,OAEjF,IACE,CAAEphE,SAAU,OAAS,CAE1B,CAqGA,SAASi8J,GAA0BF,GAClC,IAAIZ,EAAiB,EACjBnjK,EAAQ,EAGZ,MAAM0S,EAAW3W,MAAMrB,KAAMqpK,EAAGntI,eAC9Bz1B,QAAQ4R,GAAwB,OAAfA,EAAM3V,MAAgC,OAAf2V,EAAM3V,OAGhD,KAAQ4C,EAAQ0S,EAAShf,QAAqC,OAA3Bgf,EAAU1S,GAAQ5C,MAAgB,CACpE,MAAMinK,EAAK3xJ,EAAU1S,GAKrBmjK,GAFgB3qK,SAAU6rK,EAAG5tI,aAAc,YAAyB,KAGpEz2B,GACD,CAEA,OAAOmjK,CACR,CChPe,MAAM,GAyJpB,WAAAhmK,CAAa+lK,EAAgBvjK,EAA8B,CAAC,GAC3DvE,KAAKkpK,OAASpB,EACd9nK,KAAKmpK,eAA4B,IAAhB5kK,EAAQwkK,IAAoBxkK,EAAQwkK,IAAMxkK,EAAQ6kK,UAAY,EAC/EppK,KAAKqpK,aAA0B,IAAhB9kK,EAAQwkK,IAAoBxkK,EAAQwkK,IAAMxkK,EAAQ+kK,OACjEtpK,KAAKupK,kBAAkC,IAAnBhlK,EAAQyjK,OAAuBzjK,EAAQyjK,OAASzjK,EAAQilK,aAAe,EAC3FxpK,KAAKypK,gBAAgC,IAAnBllK,EAAQyjK,OAAuBzjK,EAAQyjK,OAASzjK,EAAQmlK,UAC1E1pK,KAAK2pK,mBAAqBplK,EAAQqlK,gBAClC5pK,KAAK6pK,UAAY,IAAIj2J,IACrB5T,KAAK8pK,KAAO,EACZ9pK,KAAK+pK,UAAY,EACjB/pK,KAAKgqK,QAAU,EACfhqK,KAAKiqK,WAAa,EAClBjqK,KAAKkqK,cAAgB,IAAI34J,IACzBvR,KAAKmqK,mBAAqB,CAC3B,CAKA,CAASrpK,OAAOC,YACf,OAAOf,IACR,CAOO,IAAAsB,GACN,MAAMynK,EAAM/oK,KAAKkpK,OAAOrtI,SAAU77B,KAAK+pK,WAGvC,IAAMhB,GAAO/oK,KAAKoqK,gBACjB,MAAO,CAAE7oK,MAAM,EAAM/I,WAAO,GAI7B,IAAMuwK,EAAI76J,GAAI,UAAW,YAGxB,OAFAlO,KAAK+pK,YAEE/pK,KAAKsB,OAGb,GAAKtB,KAAKqqK,mBACT,OAAOrqK,KAAKsqK,oBAGb,IAAIC,EAAkD,KAEtD,MAAMC,EAAWxqK,KAAKyqK,cAEtB,GAAKD,EACCxqK,KAAK2pK,mBAAqB3pK,KAAK0qK,oBACnCH,EAAWvqK,KAAK2qK,gBAAiBH,EAASI,KAAMJ,EAASzB,IAAKyB,EAASxC,aAElE,CACN,MAAM4C,EAAO7B,EAAIltI,SAAU77B,KAAKiqK,YAEhC,IAAMW,EAEL,OAAO5qK,KAAKsqK,oBAGb,MAAMO,EAAUztK,SAAUwtK,EAAKvvI,aAAc,YAAyB,KAChEyvI,EAAU1tK,SAAUwtK,EAAKvvI,aAAc,YAAyB,MAGjEwvI,EAAU,GAAKC,EAAU,IAC7B9qK,KAAK+qK,aAAcH,EAAME,EAASD,GAG7B7qK,KAAK0qK,oBACVH,EAAWvqK,KAAK2qK,gBAAiBC,IAGlC5qK,KAAKmqK,kBAAoBnqK,KAAKgqK,QAAUa,CACzC,CAUA,OAPA7qK,KAAKgqK,UAEAhqK,KAAKgqK,SAAWhqK,KAAKmqK,mBACzBnqK,KAAKiqK,aAICM,GAAYvqK,KAAKsB,MACzB,CAQO,OAAA0pK,CAASjC,GACf/oK,KAAK6pK,UAAUh8J,IAAKk7J,EACrB,CAKQ,iBAAAuB,GAOP,OANAtqK,KAAK8pK,OACL9pK,KAAK+pK,YACL/pK,KAAKgqK,QAAU,EACfhqK,KAAKiqK,WAAa,EAClBjqK,KAAKmqK,mBAAqB,EAEnBnqK,KAAKsB,MACb,CAKQ,aAAA8oK,GAEP,YAAwB,IAAjBpqK,KAAKqpK,SAAyBrpK,KAAK8pK,KAAO9pK,KAAKqpK,OACvD,CAKQ,gBAAAgB,GAEP,YAA2B,IAApBrqK,KAAKypK,YAA4BzpK,KAAKgqK,QAAUhqK,KAAKypK,UAC7D,CASQ,eAAAkB,CAAiBC,EAAeK,EAAYjrK,KAAK8pK,KAAMoB,EAAelrK,KAAKgqK,SAClF,MAAO,CACNzoK,MAAM,EACN/I,MAAO,IAAI2yK,GAAWnrK,KAAM4qK,EAAMK,EAAWC,GAE/C,CAKQ,eAAAR,GACP,MAAMU,EAAuBprK,KAAK6pK,UAAUj8J,IAAK5N,KAAK8pK,MAChDuB,EAAsBrrK,KAAK8pK,KAAO9pK,KAAKmpK,UAEvCmC,EAA4BtrK,KAAKgqK,QAAUhqK,KAAKupK,aAChDgC,OAA6C,IAApBvrK,KAAKypK,YAA4BzpK,KAAKgqK,QAAUhqK,KAAKypK,WAEpF,OAAO2B,GAAwBC,GAAuBC,GAA6BC,CACpF,CAKQ,WAAAd,GACP,MAAMe,EAASxrK,KAAKkqK,cAAc9hK,IAAKpI,KAAK8pK,MAG5C,OAAM0B,GAKCA,EAAOpjK,IAAKpI,KAAKgqK,UAJhB,IAKT,CASQ,YAAAe,CAAcH,EAAeE,EAAiBD,GACrD,MAAM/nK,EAAO,CACZ8nK,OACA7B,IAAK/oK,KAAK8pK,KACV9B,OAAQhoK,KAAKgqK,SAGd,QAAUyB,EAAczrK,KAAK8pK,KAAM2B,EAAczrK,KAAK8pK,KAAOgB,EAASW,IACrE,QAAUC,EAAiB1rK,KAAKgqK,QAAS0B,EAAiB1rK,KAAKgqK,QAAUa,EAASa,IAC5ED,GAAezrK,KAAK8pK,MAAQ4B,GAAkB1rK,KAAKgqK,SACvDhqK,KAAK2rK,iBAAkBF,EAAaC,EAAgB5oK,EAIxD,CASQ,gBAAA6oK,CAAkB5C,EAAaf,EAAgBllK,GAChD9C,KAAKkqK,cAAct8J,IAAKm7J,IAC7B/oK,KAAKkqK,cAAcz4J,IAAKs3J,EAAK,IAAIx3J,KAGjBvR,KAAKkqK,cAAc9hK,IAAK2gK,GAEhCt3J,IAAKu2J,EAAQllK,EACvB,EAMD,MAAMqoK,GAiDL,WAAAppK,CAAa6pK,EAA0BhB,EAAeK,EAAmBC,GACxElrK,KAAK4qK,KAAOA,EACZ5qK,KAAK+oK,IAAM6C,EAAY9B,KACvB9pK,KAAKgoK,OAAS4D,EAAY5B,QAC1BhqK,KAAK6rK,cAAgBZ,EACrBjrK,KAAK8rK,iBAAmBZ,EACxBlrK,KAAKiqK,WAAa2B,EAAY3B,WAC9BjqK,KAAK+pK,UAAY6B,EAAY7B,UAC7B/pK,KAAKkpK,OAAS0C,EAAY1C,MAC3B,CAUA,YAAW6C,GACV,OAAO/rK,KAAK+oK,MAAQ/oK,KAAK6rK,eAAiB7rK,KAAKgoK,SAAWhoK,KAAK8rK,gBAChE,CAKA,aAAWE,GACV,OAAO5uK,SAAU4C,KAAK4qK,KAAKvvI,aAAc,YAAyB,IACnE,CAKA,cAAW4wI,GACV,OAAO7uK,SAAU4C,KAAK4qK,KAAKvvI,aAAc,YAAyB,IACnE,CAKA,YAAW6wI,GACV,OAAOlsK,KAAK+pK,SACb,CAKO,iBAAAoC,GAGN,OAFcnsK,KAAKkpK,OAAO5yK,KAAKkN,SAAUpL,MAE5Bs/C,iBAAkB13C,KAAKkpK,OAAOrtI,SAAU77B,KAAK+oK,KAAkB/oK,KAAKiqK,WAClF,EC/fM,SAASmC,GAAevE,EAAwBtjK,GACtD,MAAO,CAAEujK,GAASttI,aACjB,MAAM6tI,EAAcP,EAAMzsI,aAAc,gBAAmB,EACrDgxI,EAAe7xI,EAAOuZ,uBAAwB,QAAS,KAAM,IAC7Du4H,EAAgB9xI,EAAOuZ,uBAAwB,SAAU,CAAExV,MAAO,SAAW8tI,GAG9EhE,EAAc,GAClB7tI,EAAO90B,OACN80B,EAAOkd,iBAAkB20H,EAAc,OACvC7xI,EAAOuZ,uBACN,QACA,KACAvZ,EAAOyd,YAAYxgC,GAAWA,EAAQvJ,GAAI,UAAW,aAAgBuJ,EAAQ7S,MAASyjK,MAMpFA,EAAcR,EAAW0E,QAASzE,IACtCttI,EAAO90B,OACN80B,EAAOkd,iBAAkB20H,EAAc,OACvC7xI,EAAOuZ,uBACN,QACA,KACAvZ,EAAOyd,YAAYxgC,GAAWA,EAAQvJ,GAAI,UAAW,aAAgBuJ,EAAQ7S,OAAUyjK,MAM1F,UAAY,eAAE7yH,EAAc,OAAEzvC,KAAYxB,EAAQioK,gBACjDhyI,EAAO90B,OACN80B,EAAOkd,iBAAkB20H,EAAc72H,GACvChb,EAAOyd,WAAYlyC,IAgBrB,OAXAy0B,EAAO90B,OACN80B,EAAOkd,iBAAkB20H,EAAc,SACvC7xI,EAAOyd,YAAYxgC,IACbA,EAAQvJ,GAAI,UAAW,cAIpB3J,EAAQioK,gBAAgBn4I,MAAM,EAAItuB,YAAcA,EAAQ0R,QAI3DlT,EAAQkoK,SAkHjB,SAAwBl1H,EAA0B/c,GAGjD,OAFAA,EAAOoa,kBAAmB,SAAS,EAAM2C,GAElCwlF,GAAUxlF,EAAa/c,EAAQ,CAAE2iG,oBAAoB,GAC7D,CAtH4BuvC,CAAeJ,EAAe9xI,GAAW8xI,CAAa,CAElF,CAwBO,SAASK,GAAcpoK,EAAkC,CAAC,GAChE,MAAO,CAAEojK,GAAantI,aACrB,MAAMoyI,EAAWjF,EAAU/oK,OACrBkpK,EAAQ8E,EAAShuK,OACjBstK,EAAWpE,EAAM9rI,cAAe4wI,GAEhChB,EAAc,IAAI,GAAa9D,EAAO,CAAEiB,IAAKmD,IAC7C7D,EAAcP,EAAMzsI,aAAc,gBAA6B,EAC/D0sI,EAAiBD,EAAMzsI,aAAc,mBAAgC,EAE3E,IAAIl9B,EAA6B,KAGjC,UAAY0uK,KAAajB,EACxB,GAAKiB,EAAUjC,MAAQjD,EAAY,CAClC,MACMmF,EADYD,EAAU9D,IAAMV,GAAewE,EAAU7E,OAASD,EAChC,KAAO,KAE3C5pK,EAASoG,EAAQkoK,SAChB,GAAkBjyI,EAAO0Z,sBAAuB44H,GAAmBtyI,GACnEA,EAAOuZ,uBAAwB+4H,GAChC,KACD,CAGD,OAAO3uK,CAAM,CAEf,CAcO,SAAS4uK,GAA6BxoK,EAAkC,CAAC,GAC/E,MAAO,CAAE43D,GAAgB3hC,aACxB,IAAM2hC,EAAav9D,OAAQsP,GAAI,UAAW,aACzC,OAAO,KAGR,IAAM8+J,GAAoC7wG,GACzC,OAAO,KAGR,GAAK53D,EAAQkoK,SACZ,OAAOjyI,EAAOuZ,uBAAwB,OAAQ,CAAExV,MAAO,6BACjD,CAEN,MAAMgZ,EAAc/c,EAAOuZ,uBAAwB,KAInD,OAFAvZ,EAAOoa,kBAAmB,qCAAqC,EAAM2C,GAE9DA,CACR,EAEF,CAUO,SAASy1H,GAAoC7wG,GAKnD,OAFkD,GAFhCA,EAAav9D,OAEK08B,cAEU6gC,EAqBjB79B,mBAAmBh9B,OAE3BC,IAtBtB,CC/Ie,MAAM0rK,WAA2B,GAI/B,OAAAl4I,GACf,MAAM38B,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3ByuC,EAAStrE,EAAMsrE,OAErB1jE,KAAK00B,UAoDP,SAA4BO,EAA0CyuC,GACrE,MAAMjuB,EAAiBxgB,EAAUE,mBAAoBv2B,OAC/CsuK,EAAcz3H,IAAmBA,EAAen/C,KAAOm/C,EAAiBA,EAAe72C,OAE7F,OAAO8kE,EAAOyF,WAAY+jG,EAAwB,QACnD,CAzDmB,CAAmBj4I,EAAWyuC,EAChD,CAegB,OAAAnuC,CACfhxB,EAKI,CAAC,GAEL,MAAM+vB,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfyvK,EAAyBvzI,EAAOqC,QAAQvuB,IAAK,cAE7C+kK,EAAc74I,EAAOtL,OAAO5gB,IAAK,8BACjCglK,EAAiB94I,EAAOtL,OAAO5gB,IAAK,sCAEb,IAAxB7D,EAAQ8jK,aAA6B8E,IACzC5oK,EAAQ8jK,YAAc8E,QAGS,IAA3B5oK,EAAQwjK,gBAAgCqF,IAC5C7oK,EAAQwjK,eAAiBqF,GAG1Bh1K,EAAMmiC,QAAQC,IACb,MAAMstI,EAAQD,EAAWwF,YAAa7yI,EAAQj2B,GAE9CnM,EAAM4iG,aAAc8sE,EAAO,KAAM,KAAM,CAAE3sE,oBAAqB,SAE9D3gE,EAAOgZ,aAAchZ,EAAOkd,iBAAkBowH,EAAM5wG,cAAe,CAAE,EAAG,EAAG,IAAO,GAAK,GAEzF,ECvDc,MAAMo2G,WAAyB,GAa7C,WAAAvrK,CAAauyB,EAAgB/vB,EAAyC,CAAC,GACtEiJ,MAAO8mB,GAEPt0B,KAAKygK,MAAQl8J,EAAQk8J,OAAS,OAC/B,CAKgB,OAAA1rI,GACf,MAAME,EAAYj1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAEvCs4I,IADyBvtK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cACjBolK,+BAAgCv4I,GAAY38B,OAEnF0H,KAAK00B,UAAY64I,CAClB,CASgB,OAAAh4I,GACf,MAAMjB,EAASt0B,KAAKs0B,OACdW,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAClC4yI,EAAyBvzI,EAAOqC,QAAQvuB,IAAK,cAC7CqlK,EAA6B,UAAfztK,KAAKygK,MAEnBiN,EAAqB7F,EAAW2F,+BAAgCv4I,GAChE04I,EAAa9F,EAAW+F,cAAeF,GAEvC3E,EAAM0E,EAAcE,EAAWl9I,MAAQk9I,EAAWj9I,KAClDo3I,EAAQ4F,EAAoB,GAAIpnI,aAAc,SAEpDuhI,EAAWgG,WAAY/F,EAAO,CAAEgG,GAAIL,EAAc1E,EAAMA,EAAM,EAAGgF,wBAAyBN,GAC3F,EClDc,MAAMO,WAA4B,GAahD,WAAAjsK,CAAauyB,EAAgB/vB,EAAwC,CAAC,GACrEiJ,MAAO8mB,GAEPt0B,KAAKygK,MAAQl8J,EAAQk8J,OAAS,OAC/B,CAKgB,OAAA1rI,GACf,MAAME,EAAYj1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAEvCs4I,IADyBvtK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cACjBolK,+BAAgCv4I,GAAY38B,OAEnF0H,KAAK00B,UAAY64I,CAClB,CAUgB,OAAAh4I,GACf,MAAMjB,EAASt0B,KAAKs0B,OACdW,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAClC4yI,EAAyBvzI,EAAOqC,QAAQvuB,IAAK,cAC7C5B,EAA8B,SAAfxG,KAAKygK,MAEpBiN,EAAqB7F,EAAW2F,+BAAgCv4I,GAChEg5I,EAAgBpG,EAAWqG,iBAAkBR,GAE7C1F,EAASxhK,EAAeynK,EAAcx9I,MAAQw9I,EAAcv9I,KAC5Do3I,EAAQ4F,EAAoB,GAAIpnI,aAAc,SAEpDuhI,EAAWsG,cAAerG,EAAO,CAAEsG,QAAS,EAAGN,GAAItnK,EAAewhK,EAASA,EAAS,GACrF,ECzDc,MAAMqG,WAAyB,GAY7C,WAAAtsK,CAAauyB,EAAgB/vB,EAAyD,CAAC,GACtFiJ,MAAO8mB,GAEPt0B,KAAKukB,UAAYhgB,EAAQggB,WAAa,cACvC,CAKgB,OAAAwQ,GACf,MACMu5I,EADyBtuK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cACvBolK,+BAAgCxtK,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WAE5Fj1B,KAAK00B,UAAqC,IAAzB45I,EAAch2K,MAChC,CAKgB,OAAAi9B,GACf,MAAMsyI,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDu/J,EAAYE,EAAW2F,+BAAgCxtK,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WAAa,GAC7D,iBAAnBj1B,KAAKukB,UAGzBsjJ,EAAW0G,sBAAuB5G,EAAW,GAE7CE,EAAW2G,oBAAqB7G,EAAW,EAE7C,EChBM,SAAS8G,GACfC,EACAC,EAMAn0I,GAEA,MAAM,SAAE4uI,EAAQ,YAAEI,EAAW,OAAEF,EAAM,UAAEI,GAAciF,EAG/CC,EAAep0I,EAAOl1B,cAAe,SACrCupK,EAAavF,EAASF,EAAW,EAEvC,QAAU5uK,EAAI,EAAGA,EAAIq0K,EAAYr0K,IAChCggC,EAAOquC,cAAe,WAAY+lG,EAAc,OAGjD,MAAME,EAAW,IAAK,IAAI,GAAaJ,EAAa,CAAEtF,WAAUE,SAAQE,cAAaE,YAAWE,iBAAiB,KAGjH,UAAcb,IAAKgG,EAAW/G,OAAQgH,EAAcpE,KAAMjD,EAAS,SAAEoE,EAAQ,cAAEF,EAAa,iBAAEC,KAAsBgD,EAAW,CAE9H,MAAMG,EAAoBF,EAAY3F,EAChCL,EAAM6F,EAAa/yI,SAAUozI,GAGnC,GAAMlD,EAQD,CACJ,MAAMmD,EAAgB10I,EAAOg4D,aAAcm1E,GAE3CntI,EAAOqrB,OAAQqpH,EAAenG,GAI9BoG,GAAuBD,EAAeH,EAAWC,EAAc1F,EAAQI,EAAWlvI,EACnF,MAbMqxI,EAAgBzC,GAAY0C,EAAmBtC,IACnD9B,GAAsBltI,EAAQA,EAAOkd,iBAAkBqxH,EAAK,OAa/D,CAKA,OA6MD,SAAoC6F,EAAuBF,EAAsBtF,EAAkBI,EAAqBhvI,GACvH,MAAM6tI,EAAcjrK,SAAUsxK,EAAYrzI,aAAc,gBAA6B,KAErF,GAAKgtI,EAAc,EAAI,CAEtBZ,GAAwB,cADEY,EAAce,EACkBwF,EAAcp0I,EAAQ,EACjF,CAEA,MAAMutI,EAAiB3qK,SAAUsxK,EAAYrzI,aAAc,mBAAgC,KAE3F,GAAK0sI,EAAiB,EAAI,CAEzBN,GAAwB,iBADKM,EAAiByB,EACkBoF,EAAcp0I,EAAQ,EACvF,CACD,CA7NC40I,CAA2BR,EAAcF,EAAatF,EAAUI,EAAahvI,GAEtEo0I,CACR,CAyBO,SAASS,GAA+BvH,EAAgBwH,EAAoBlG,EAAmB,GACrG,MAAMmG,EAA0B,GAE1B3D,EAAc,IAAI,GAAa9D,EAAO,CAAEsB,WAAUE,OAAQgG,EAAa,IAE7E,UAAYE,KAAY5D,EAAc,CACrC,MAAM,IAAE7C,EAAG,WAAEkD,GAAeuD,EAGvBzG,EAAMuG,GAAcA,GAFNvG,EAAMkD,EAAa,GAGrCsD,EAAM3vK,KAAM4vK,EAEd,CAEA,OAAOD,CACR,CAOO,SAASE,GAAmB9H,EAAoB+H,EAAkBl1I,GACxE,MAAMoyI,EAAWjF,EAAU/oK,OACrBkpK,EAAQ8E,EAAShuK,OACjBstK,EAAWU,EAAShoK,MAGpB+qK,EAAaD,EAAWxD,EAExB0D,EAAoC,CAAC,EACrCC,EAJUzyK,SAAUuqK,EAAUtsI,aAAc,YAIjBs0I,EAE5BE,EAAiB,IACrBD,EAAkB9E,QAAU+E,GAG7B,MAAMhF,EAAUztK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KAEtEwvI,EAAU,IACd+E,EAAkB/E,QAAUA,GAG7B,MACMvB,EADW4C,EACSyD,EACpBb,EAAW,IAAK,IAAI,GAAahH,EAAO,CAAEsB,SAF/B8C,EAEyC5C,SAAQM,iBAAiB,KAEnF,IACIkG,EADAC,EAAU,KAGd,UAAYlD,KAAaiC,EAAW,CACnC,MAAM,IAAE/F,EAAG,OAAEf,EAAM,KAAE4C,GAASiC,EAEzBjC,IAASjD,QAA6B,IAAhBmI,IAC1BA,EAAc9H,QAGM,IAAhB8H,GAA6BA,IAAgB9H,GAAUe,IAAQO,IACnEyG,EAAUrI,GAAsBltI,EAAQqyI,EAAUV,oBAAqByD,GAEzE,CAKA,OAFAnI,GAAwB,UAAWkI,EAAYhI,EAAWntI,GAEnDu1I,CACR,CA2BO,SAASC,GAAiClI,EAAgBmI,GAChE,MAAMC,EAAe,GAEftE,EAAc,IAAI,GAAa9D,GAErC,UAAY0H,KAAY5D,EAAc,CACrC,MAAM,OAAE5D,EAAM,UAAEgE,GAAcwD,EAGzBxH,EAASiI,GAAiBA,GAFTjI,EAASgE,EAAY,GAG1CkE,EAAatwK,KAAM4vK,EAErB,CAEA,OAAOU,CACR,CASO,SAASC,GAAiBxI,EAAoBmI,EAAqBM,EAAqB51I,GAC9F,MACM61I,EAAaD,EAAcN,EAE3BF,EAAoC,CAAC,EACrCU,EAJUlzK,SAAUuqK,EAAUtsI,aAAc,YAIjBg1I,EAE5BC,EAAiB,IACrBV,EAAkB/E,QAAUyF,GAG7B,MAAMxF,EAAU1tK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KAEtEyvI,EAAU,IACd8E,EAAkB9E,QAAUA,GAG7B,MAAMiF,EAAUrI,GAAsBltI,EAAQA,EAAOmd,oBAAqBgwH,GAAaiI,GAKvF,OAFAnI,GAAwB,UAAW4I,EAAY1I,EAAWntI,GAEnDu1I,CACR,CAQO,SAASZ,GACfxH,EACA4I,EACAC,EACAC,EACAC,EACAl2I,GAEA,MAAMqwI,EAAUztK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KACrEyvI,EAAU1tK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KAI3E,GAFkBm1I,EAAa3F,EAAU,EAExB6F,EAAc,CAG9BjJ,GAAwB,UAFJiJ,EAAcF,EAAa,EAEC7I,EAAWntI,EAAQ,EACpE,CAIA,GAFe+1I,EAAUzF,EAAU,EAErB2F,EAAW,CAGxBhJ,GAAwB,UAFJgJ,EAAWF,EAAU,EAEO5I,EAAWntI,EAAQ,EACpE,CACD,CA2CO,SAASm2I,GAAoB7I,EAAgBD,GACnD,MAAMhlJ,EAAQglJ,EAAW+I,WAAY9I,GAC/B+I,EAAa,IAAIlwK,MAAOkiB,GAAQ/X,KAAM,GAE5C,UAAY,OAAEk9J,KAAY,IAAI,GAAaF,GAC1C+I,EAAY7I,KAGb,MAAM8I,EAAeD,EAAW5iJ,QAAQ,CAAE9vB,EAAQ4yK,EAAY/I,IACtD+I,EAAa5yK,EAAS,IAAKA,EAAQ6pK,IACxC,IAEH,GAAK8I,EAAax4K,OAAS,EAAI,CAE9B,MAAM04K,EAAcF,EAAcA,EAAax4K,OAAS,GAKxD,OAFAuvK,EAAWoJ,cAAenJ,EAAO,CAAEgG,GAAIkD,KAEhC,CACR,CAEA,OAAO,CACR,CA+BO,SAASE,GAAiBpJ,EAAgBD,GAChD,MAAMsJ,EAAY,GACZC,EAAgBvJ,EAAW0E,QAASzE,GAE1C,QAAUoE,EAAW,EAAGA,EAAWkF,EAAelF,IAAa,CAC7CpE,EAAMjsI,SAAUqwI,GAEnB/pI,SACbgvI,EAAUvxK,KAAMssK,EAElB,CAEA,GAAKiF,EAAU74K,OAAS,EAAI,CAE3B,MAAM+4K,EAAWF,EAAWA,EAAU74K,OAAS,GAK/C,OAFAuvK,EAAWyJ,WAAYxJ,EAAO,CAAEgG,GAAIuD,KAE7B,CACR,CAEA,OAAO,CACR,CA0BO,SAASE,GAAwBzJ,EAAgBD,GAChC8I,GAAoB7I,EAAOD,IAIjDqJ,GAAiBpJ,EAAOD,EAE1B,CAoBO,SAAS2J,GACf1J,EACA2J,GAOA,MAAMC,EAAa/wK,MAAMrB,KAAM,IAAI,GAAawoK,EAAO,CACtD0B,YAAaiI,EAAWE,YACxBjI,UAAW+H,EAAWG,WACtB7I,IAAK0I,EAAWI,WAMjB,GAHkCH,EAAWn7J,OAAO,EAAI01J,gBAAiC,IAAfA,IAIzE,OAAOwF,EAAWI,QAInB,MAAMC,EAAoBJ,EAAY,GAAIzF,WAAa,EACvD,OAAOwF,EAAWI,QAAUC,CAC7B,CAuBO,SAASC,GACfjK,EACA2J,GAOA,MAAMO,EAAgBrxK,MAAMrB,KAAM,IAAI,GAAawoK,EAAO,CACzDsB,SAAUqI,EAAWQ,SACrB3I,OAAQmI,EAAWI,QACnB7J,OAAQyJ,EAAWG,cAMpB,GAHkCI,EAAcz7J,OAAO,EAAIy1J,eAA+B,IAAdA,IAI3E,OAAOyF,EAAWG,WAInB,MAAMM,EAAoBF,EAAe,GAAIhG,UAAY,EACzD,OAAOyF,EAAWG,WAAaM,CAChC,CCzfe,MAAMC,WAAyB,GAuB7C,WAAApwK,CAAauyB,EAAgB/vB,GAC5BiJ,MAAO8mB,GAEPt0B,KAAKukB,UAAYhgB,EAAQggB,UACzBvkB,KAAKoyK,aAAiC,SAAlBpyK,KAAKukB,WAA0C,QAAlBvkB,KAAKukB,SACvD,CAKgB,OAAAwQ,GACf,MAAMs9I,EAAcryK,KAAKsyK,oBAEzBtyK,KAAKxH,MAAQ65K,EACbryK,KAAK00B,YAAc29I,CACpB,CASgB,OAAA98I,GACf,MAAMn9B,EAAQ4H,KAAKs0B,OAAOl8B,MACpBif,EAAMjf,EAAMoL,SAEZmkK,EADyB3nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAC3BmqK,iCAAkCl7J,EAAI4d,WAAa,GAE1Eo9I,EAAcryK,KAAKxH,MACnB+rB,EAAYvkB,KAAKukB,UAEvBnsB,EAAMmiC,QAAQC,IACb,MAAMg4I,EAA2B,SAAbjuJ,GAAqC,QAAbA,EAGtCkuJ,EAAiBD,EAAc7K,EAAY0K,EAC3CK,EAAiBF,EAAcH,EAAc1K,EAG7CgL,EAAsBD,EAAa9zK,QAmJ5C,SAA0B8zK,EAAuBD,EAAuBj4I,GACjE2H,GAASuwI,KACTvwI,GAASswI,IACbj4I,EAAOv0B,OAAQu0B,EAAOud,cAAe06H,IAGtCj4I,EAAO8a,KAAM9a,EAAOud,cAAe26H,GAAgBl4I,EAAOkd,iBAAkB+6H,EAAc,SAI3Fj4I,EAAOv0B,OAAQysK,EAChB,CA5JGE,CAAiBF,EAAcD,EAAcj4I,GAE7C,MAAMq4I,EAAgB7yK,KAAKoyK,aAAe,UAAY,UAChDU,EAAW11K,SAAUuqK,EAAUtsI,aAAcw3I,IAA6B,KAC1EE,EAAkB31K,SAAUi1K,EAAYh3I,aAAcw3I,IAA6B,KAGzFr4I,EAAO/0B,aAAcotK,EAAeC,EAAWC,EAAiBN,GAChEj4I,EAAOgZ,aAAchZ,EAAOud,cAAe06H,IAE3C,MAAM5K,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAIxDmpK,GAHcoB,EAAoBrsI,aAAc,SAGjBuhI,EAAY,GAE7C,CAKQ,iBAAAyK,GACP,MACMj7J,EADQrX,KAAKs0B,OAAOl8B,MACRoL,SACZqkK,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDu/J,EAAYE,EAAW0K,iCAAkCl7J,EAAI4d,WAAa,GAEhF,IAAM0yI,EACL,OAID,MAAM0K,EAAcryK,KAAKoyK,aAuB3B,SAA4BzK,EAAoBpjJ,EAAkCsjJ,GACjF,MAAM+E,EAAWjF,EAAU/oK,OACrBkpK,EAAQ8E,EAAShuK,OACjBo0K,EAA8B,SAAbzuJ,EAAuBojJ,EAAU1rI,YAAc0rI,EAAU/gJ,gBAC1EqsJ,GAAsBnL,EAAMzsI,aAAc,mBAAsB,GAAM,EAE5E,IAAM23I,EACL,OAID,MAAME,EAA4B,SAAb3uJ,EAAuBojJ,EAAYqL,EAClDG,EAA6B,SAAb5uJ,EAAuByuJ,EAAiBrL,GAGtDK,OAAQoL,GAAmBvL,EAAWI,gBAAiBiL,IACvDlL,OAAQqL,GAAoBxL,EAAWI,gBAAiBkL,GAE1DG,EAAel2K,SAAU81K,EAAW73I,aAAc,YAAyB,KAE3Ek4I,EAA8B3L,GAAqBC,EAAYqL,GAC/DM,EAA+B5L,GAAqBC,EAAYsL,GAGtE,GAAKF,GAAqBM,GAA+BC,EACxD,OAOD,OAHyBJ,EAAiBE,IAAiBD,EAGjCL,OAAiB,CAC5C,CAvDGS,CAAmB9L,EAAW3nK,KAAKukB,UAAWsjJ,GA4DjD,SAA0BF,EAAoBpjJ,EAAkCsjJ,GAC/E,MAAM+E,EAAWjF,EAAU/oK,OACrBkpK,EAAQ8E,EAAShuK,OAEjBstK,EAAWpE,EAAM9rI,cAAe4wI,GAGtC,GAAoB,QAAbroJ,GAAuB2nJ,IAAarE,EAAW0E,QAASzE,GAAU,GAAsB,MAAbvjJ,GAAkC,IAAb2nJ,EACtG,OAAO,KAGR,MAAMpB,EAAU1tK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KACrEgtI,EAAcP,EAAMzsI,aAAc,gBAAmB,EAErDq4I,EAAmC,QAAbnvJ,GAAyB2nJ,EAAWpB,IAAczC,EACxEsL,EAAmC,MAAbpvJ,GAAqB2nJ,IAAa7D,EAG9D,GAAKA,IAAiBqL,GAAuBC,GAC5C,OAAO,KAGR,MAAMC,EAAqBx2K,SAAUuqK,EAAUtsI,aAAc,YAAyB,KAChFw4I,EAAgC,QAAbtvJ,EAAsB2nJ,EAAW0H,EAAqB1H,EAEzE4C,EAAW,IAAK,IAAI,GAAahH,EAAO,CAAEwB,OAAQuK,KAElDC,EAAkBhF,EAASv9I,MAAM/4B,GAASA,EAAMoyK,OAASjD,IACzDoM,EAAcD,EAAgB9L,OAE9BgM,EAAkBlF,EAASv9I,MAAM,EAAIw3I,MAAKkD,aAAYjE,YACtDA,IAAW+L,IAIE,QAAbxvJ,EAEGwkJ,IAAQ8K,EAGRA,IAAqB9K,EAAMkD,KAIpC,OAAO+H,GAAmBA,EAAgBpJ,KAAOoJ,EAAgBpJ,KAAO,IACzE,CAxGGqJ,CAAiBtM,EAAW3nK,KAAKukB,UAAWsjJ,GAE7C,IAAMwK,EACL,OAID,MAAMQ,EAAgB7yK,KAAKoyK,aAAe,UAAY,UAChDt3H,EAAO19C,SAAUuqK,EAAUtsI,aAAcw3I,IAA6B,KAI5E,OAFwBz1K,SAAUi1K,EAAYh3I,aAAcw3I,IAA6B,OAEhE/3H,EACjBu3H,OADR,CAGD,EAgHD,SAASlwI,GAASwlI,GACjB,MAAMuM,EAAkBvM,EAAU9rI,SAAU,GAE5C,OAA+B,GAAxB8rI,EAAUrsI,YAAmB44I,EAAgBhmK,GAAI,UAAW,cAAiBgmK,EAAgB/xI,OACrG,CCxPe,MAAMgyI,WAAyB,GAI7B,OAAAp/I,GACf,MAAM8yI,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDkmK,EAAgBzG,EAAW2F,+BAAgCxtK,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WACtFm/I,EAAY9F,EAAe,GAEjC,GAAK8F,EAAY,CAChB,MAAMtM,EAAQsM,EAAU9tI,aAAc,SAEhC+tI,EADgBxM,EAAW0E,QAASzE,GACL,EAE/BwM,EAAqBzM,EAAW+F,cAAeU,GAE/CiG,EAAkD,IAA7BD,EAAmB7jJ,OAAe6jJ,EAAmB5jJ,OAAS2jJ,EAGzFr0K,KAAK00B,WAAa6/I,CACnB,MACCv0K,KAAK00B,WAAY,CAEnB,CAKgB,OAAAa,GACf,MAAMn9B,EAAQ4H,KAAKs0B,OAAOl8B,MACpByvK,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAElDosK,EAAiB3M,EAAW2F,+BAAgCp1K,EAAMoL,SAASyxB,WAC3Ew/I,EAAoB5M,EAAW+F,cAAe4G,GAE9CJ,EAAYI,EAAgB,GAC5B1M,EAAQsM,EAAU9tI,aAAc,SAEhCouI,EAAqB7M,EAAWI,gBAAiBmM,GAAYpM,OAEnE5vK,EAAMmiC,QAAQC,IACb,MAAMm6I,EAAeF,EAAkB/jJ,KAAO+jJ,EAAkBhkJ,MAAQ,EAExEo3I,EAAWyJ,WAAYxJ,EAAO,CAC7BgG,GAAI2G,EAAkBhkJ,MACtB23I,KAAMuM,IAGP,MAAMC,EAYT,SAAyB9M,EAAgB+M,EAAyBC,EAAuB1D,GAExF,MAAMrI,EAAMjB,EAAMjsI,SAAUhjC,KAAKD,IAAKi8K,EAAiBzD,EAAgB,IAGvE,IAAIwD,EAAc7L,EAAIltI,SAAU,GAC5BmsI,EAAS,EAEb,UAAYL,KAAaoB,EAAIvtI,cAAgB,CAC5C,GAAKwsI,EAAS8M,EACb,OAAOF,EAGRA,EAAcjN,EACdK,GAAU5qK,SAAUuqK,EAAUtsI,aAAc,YAAyB,IACtE,CAEA,OAAOu5I,CACR,CA9BuBG,CAAgBjN,EAAO2M,EAAkBhkJ,MAAOikJ,EAAoB7M,EAAW0E,QAASzE,IAE5GttI,EAAOgZ,aAAchZ,EAAOkd,iBAAkBk9H,EAAa,GAAK,GAElE,EC9Cc,MAAMI,WAA4B,GAIhC,OAAAjgJ,GACf,MAAM8yI,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDkmK,EAAgBzG,EAAW2F,+BAAgCxtK,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WACtFm/I,EAAY9F,EAAe,GAEjC,GAAK8F,EAAY,CAChB,MAAMtM,EAAQsM,EAAU9tI,aAAc,SAChC2uI,EAAmBpN,EAAW+I,WAAY9I,IAE1C,MAAEr3I,EAAK,KAAEC,GAASm3I,EAAWqG,iBAAkBI,GAErDtuK,KAAK00B,UAAYhE,EAAOD,EAAUwkJ,EAAmB,CACtD,MACCj1K,KAAK00B,WAAY,CAEnB,CAKgB,OAAAa,GACf,MAAMsyI,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,eAChDgsK,EAAWc,GA4ErB,SAA2BjgJ,EAA0C4yI,GACpE,MAAM2M,EAAiB3M,EAAW2F,+BAAgCv4I,GAC5Dm/I,EAAYI,EAAgB,GAC5BU,EAAWV,EAAex1K,MAE1Bm2K,EAAc,CAAEf,EAAWc,GAEjC,OAAOd,EAAU13I,SAAUw4I,GAAaC,EAAcA,EAAY9pK,SACnE,CApFkC+pK,CAAkBp1K,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAAW4yI,GAClFC,EAAQsM,EAAUx1K,OAAQA,OAG1BkwK,EAAW,IAAK,IAAI,GAAahH,IAGjCuN,EAAuB,CAC5B5kJ,MAAOq+I,EAASv9I,MAAM/4B,GAASA,EAAMoyK,OAASwJ,IAAapM,OAC3Dt3I,KAAMo+I,EAASv9I,MAAM/4B,GAASA,EAAMoyK,OAASsK,IAAYlN,QAGpD4M,EAmBR,SACC9F,EACAsF,EACAc,EACAG,GAKA,MAAMxK,EAAUztK,SAAU83K,EAAS75I,aAAc,YAAyB,KAI1E,OAAKwvI,EAAU,EACPqK,EAKEd,EAAUxtJ,iBAAmBsuJ,EAASj5I,YACxCi5I,EAASj5I,aAAem4I,EAAUxtJ,gBAOpCyuJ,EAAqB5kJ,MAClBq+I,EAASzjK,UAAUkmB,MAAM,EAAIy2I,YAC5BA,EAASqN,EAAqB5kJ,QACjCm6I,KAKEkE,EAASzjK,UAAUkmB,MAAM,EAAIy2I,YAC5BA,EAASqN,EAAqB3kJ,OACjCk6I,IAGR,CA3DsB,CAAgBkE,EAAUsF,EAAWc,EAAUG,GAEnEr1K,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzB,MAAM86I,EAAkBD,EAAqB3kJ,KAAO2kJ,EAAqB5kJ,MAAQ,EAEjFo3I,EAAWoJ,cAAenJ,EAAO,CAChCgG,GAAIuH,EAAqB5kJ,MACzB29I,QAASkH,IAGV96I,EAAOgZ,aAAchZ,EAAOkd,iBAAkBk9H,EAAa,GAAK,GAElE,EClDc,MAAMW,WAA4B,GAYhC,OAAAxgJ,GACf,MAAM8yI,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDhQ,EAAQ4H,KAAKs0B,OAAOl8B,MACpBk2K,EAAgBzG,EAAW2F,+BAAgCp1K,EAAMoL,SAASyxB,WAC1EugJ,EAAYlH,EAAch2K,OAAS,EAEzC0H,KAAK00B,UAAY8gJ,EACjBx1K,KAAKxH,MAAQg9K,GAAalH,EAAc/3J,OAAOq0J,GAAQ5qK,KAAKy1K,aAAc7K,EAAMA,EAAKhsK,OAAQA,SAC9F,CAagB,OAAA22B,CAAShxB,EAAoC,CAAC,GAC7D,GAAKA,EAAQksI,aAAezwI,KAAKxH,MAChC,OAGD,MAAMqvK,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDhQ,EAAQ4H,KAAKs0B,OAAOl8B,MAEpBk2K,EAAgBzG,EAAW2F,+BAAgCp1K,EAAMoL,SAASyxB,WAC1E6yI,EAAQwG,EAAe,GAAIhoI,aAAc,UAEzC,MAAE7V,EAAK,KAAEC,GAASm3I,EAAW+F,cAAeU,GAC5CoH,EAAmB11K,KAAKxH,MAAQi4B,EAAQC,EAAO,EAC/CilJ,EAAqB7N,EAAMzsI,aAAc,gBAA6B,EAE5EjjC,EAAMmiC,QAAQC,IACb,GAAKk7I,EAAmB,CAGvB,MACME,EAAmBvG,GAA+BvH,EAAO4N,EAD9CA,EAAmBC,EAAqBA,EAAqB,GAG9E,UAAY,KAAE/K,KAAUgL,EACvBnG,GAAmB7E,EAAM8K,EAAkBl7I,EAE7C,CAEAitI,GAAwB,cAAeiO,EAAkB5N,EAAOttI,EAAQ,EAAG,GAE7E,CAKQ,YAAAi7I,CAAc9N,EAAoBG,GACzC,MAAMO,EAAcjrK,SAAU0qK,EAAMzsI,aAAc,gBAA6B,KAE/E,QAASgtI,GAAiBV,EAAU/oK,OAAoBgG,MAASyjK,CAClE,ECpEc,MAAMwN,WAA+B,GAYnC,OAAA9gJ,GACf,MAAM38B,EAAQ4H,KAAKs0B,OAAOl8B,MACpByvK,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAElDkmK,EAAgBzG,EAAW2F,+BAAgCp1K,EAAMoL,SAASyxB,WAC1EugJ,EAAYlH,EAAch2K,OAAS,EAEzC0H,KAAK00B,UAAY8gJ,EACjBx1K,KAAKxH,MAAQg9K,GAAalH,EAAc/3J,OAAOq0J,GAAQhD,GAAqBC,EAAY+C,IACzF,CAagB,OAAAr1I,CAAShxB,EAAoC,CAAC,GAC7D,GAAKA,EAAQksI,aAAezwI,KAAKxH,MAChC,OAGD,MAAMqvK,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDhQ,EAAQ4H,KAAKs0B,OAAOl8B,MACpBk2K,EAAgBzG,EAAW2F,+BAAgCp1K,EAAMoL,SAASyxB,WAC1E6yI,EAAQwG,EAAe,GAAIhoI,aAAc,UAEzC,MAAE7V,EAAK,KAAEC,GAASm3I,EAAWqG,iBAAkBI,GAC/CwH,EAAsB91K,KAAKxH,MAAQi4B,EAAQC,EAAO,EAExDt4B,EAAMmiC,QAAQC,IACb,GAAKs7I,EAAsB,CAG1B,MAAMF,EAAmB5F,GAAiClI,EAAOgO,GAEjE,UAAY,KAAElL,EAAI,OAAE5C,KAAY4N,EAC/BzF,GAAiBvF,EAAM5C,EAAQ8N,EAAqBt7I,EAEtD,CAEAitI,GAAwB,iBAAkBqO,EAAqBhO,EAAOttI,EAAQ,EAAG,GAEnF,EC3Dc,MAAM,WAAmB,GAIvC,qBAAkBlE,GACjB,MAAO,YACR,CAKO,IAAAI,GACN12B,KAAKiV,SAAU,iBACfjV,KAAKiV,SAAU,aAChB,CA8BO,eAAAgzJ,CAAiBN,GACvB,MAAMiF,EAAWjF,EAAU/oK,OACrBkpK,EAAQ8E,EAAShuK,OAEjBstK,EAAWpE,EAAM9rI,cAAe4wI,GAEhChB,EAAc,IAAI,GAAa9D,EAAO,CAAEiB,IAAKmD,IAEnD,UAAY,KAAEtB,EAAI,IAAE7B,EAAG,OAAEf,KAAY4D,EACpC,GAAKhB,IAASjD,EACb,MAAO,CAAEoB,MAAKf,SAOjB,CAuBO,WAAAqF,CACN7yI,EACAj2B,GAOA,MAAMujK,EAAQttI,EAAOl1B,cAAe,SAE9B8iK,EAAO7jK,EAAQ6jK,MAAQ,EACvBgG,EAAU7pK,EAAQ6pK,SAAW,EAYnC,OAVA2H,GAAiBv7I,EAAQstI,EAAO,EAAGM,EAAMgG,GAEpC7pK,EAAQ8jK,aACZZ,GAAwB,cAAe5uK,KAAKD,IAAK2L,EAAQ8jK,YAAaD,GAAQN,EAAOttI,EAAQ,GAGzFj2B,EAAQwjK,gBACZN,GAAwB,iBAAkB5uK,KAAKD,IAAK2L,EAAQwjK,eAAgBqG,GAAWtG,EAAOttI,EAAQ,GAGhGstI,CACR,CA8BO,UAAA+F,CAAY/F,EAAgBvjK,EAA4E,CAAC,GAC/G,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MAEpByuB,EAAWtiB,EAAQupK,IAAM,EACzBkI,EAAezxK,EAAQ6jK,MAAQ,EAC/B6N,OAAqD,IAAnC1xK,EAAQwpK,uBAC1BmI,EAAoB3xK,EAAQwpK,uBAAyBlnJ,EAAW,EAAIA,EAEpEuhJ,EAAOpoK,KAAKusK,QAASzE,GACrBsG,EAAUpuK,KAAK4wK,WAAY9I,GAEjC,GAAKjhJ,EAAWuhJ,EAMf,MAAM,IAAI,EACT,4CACApoK,KACA,CAAEuE,YAIJnM,EAAMmiC,QAAQC,IACb,MAAM6tI,EAAcP,EAAMzsI,aAAc,gBAA6B,EAQrE,GALKgtI,EAAcxhJ,GAClB4gJ,GAAwB,cAAeY,EAAc2N,EAAclO,EAAOttI,EAAQ,IAI7Ey7I,IAAkC,IAAbpvJ,GAAkBA,IAAauhJ,GAGzD,YAFA2N,GAAiBv7I,EAAQstI,EAAOjhJ,EAAUmvJ,EAAc5H,GAMzD,MAAM+H,EAAeF,EAAkBp9K,KAAKC,IAAK+tB,EAAUqvJ,GAAsBrvJ,EAC3EuvJ,EAAgB,IAAI,GAAatO,EAAO,CAAEwB,OAAQ6M,IAGlDE,EAAiB,IAAI11K,MAAOytK,GAAUtjK,KAAM,GAElD,UAAY,IAAEi+J,EAAG,OAAEf,EAAM,WAAEiE,EAAU,UAAED,EAAS,KAAEpB,KAAUwL,EAAgB,CAC3E,MAAME,EAAcvN,EAAMkD,EAAa,EAGjCsK,EAAiBxN,GAAOmN,GAAqBA,GAAqBI,EADvCvN,EAAMliJ,GAAYA,GAAYyvJ,GAM9D97I,EAAO/0B,aAAc,UAAWwmK,EAAa+J,EAAcpL,GAG3DyL,EAAgBrO,IAAYgE,GAGnBiK,GAAmBM,IAC5BF,EAAgBrO,GAAWgE,EAE7B,CAEA,QAAUE,EAAW,EAAGA,EAAW8J,EAAc9J,IAAa,CAC7D,MAAMU,EAAWpyI,EAAOl1B,cAAe,YAEvCk1B,EAAO90B,OAAQknK,EAAU9E,EAAOjhJ,GAEhC,QAAU2vJ,EAAY,EAAGA,EAAYH,EAAe/9K,OAAQk+K,IAAc,CACzE,MAAM3L,EAAUwL,EAAgBG,GAC1B98G,EAAiBl/B,EAAOkd,iBAAkBk1H,EAAU,OAGrD/B,EAAU,GACdnD,GAAsBltI,EAAQk/B,EAAgBmxG,EAAU,EAAI,CAAEA,gBAAY,GAI3E2L,GAAa39K,KAAK49K,IAAK5L,GAAY,CACpC,CACD,IAEF,CA6BO,aAAAsD,CAAerG,EAAgBvjK,EAA6C,CAAC,GACnF,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MAEpByuB,EAAWtiB,EAAQupK,IAAM,EACzB4I,EAAkBnyK,EAAQ6pK,SAAW,EAE3Ch2K,EAAMmiC,QAAQC,IACb,MAAMutI,EAAiBD,EAAMzsI,aAAc,kBAGtCxU,EAAWkhJ,GACfvtI,EAAO/0B,aAAc,iBAAkBsiK,EAAiB2O,EAAiB5O,GAG1E,MAAM6O,EAAe32K,KAAK4wK,WAAY9I,GAGtC,GAAkB,IAAbjhJ,GAAkB8vJ,IAAiB9vJ,EAAW,CAClD,UAAY+lJ,KAAY9E,EAAMtsI,cAEvBoxI,EAAS1+J,GAAI,UAAW,aAI9B0oK,GAAaF,EAAiBl8I,EAAQA,EAAOkd,iBAAkBk1H,EAAU/lJ,EAAW,MAAQ,IAG7F,MACD,CAEA,MAAM+kJ,EAAc,IAAI,GAAa9D,EAAO,CAAEE,OAAQnhJ,EAAU+iJ,iBAAiB,IAEjF,UAAYiD,KAAajB,EAAc,CACtC,MAAM,IAAE7C,EAAG,KAAE6B,EAAI,iBAAEkB,EAAgB,cAAED,EAAa,UAAEG,EAAS,WAAEC,GAAeY,EAO9E,GAAKf,EAAmBjlJ,EAAW,CAGlC2T,EAAO/0B,aAAc,UAAWumK,EAAY0K,EAAiB9L,GAG7D,MAAM0L,EAAczK,EAAgBI,EAAa,EAEjD,QAAUzxK,EAAIuuK,EAAKvuK,GAAK87K,EAAa97K,IACpCoxK,EAAYZ,QAASxwK,EAEvB,MAGCo8K,GAAaF,EAAiBl8I,EAAQqyI,EAAUV,oBAElD,IAEF,CA8BO,UAAAmF,CAAYxJ,EAAgBvjK,GAClC,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MAEpBu8K,EAAepwK,EAAQ6jK,MAAQ,EAC/ByO,EAAW72K,KAAKusK,QAASzE,GACzBr3I,EAAQlsB,EAAQupK,GAChBp9I,EAAOD,EAAQkkJ,EAAe,EAEpC,GAAKjkJ,EAAOmmJ,EAAW,EAMtB,MAAM,IAAI,EACT,+CACA72K,KACA,CAAE8nK,QAAOvjK,YAIXnM,EAAMmiC,QAAQC,IACb,MAAMs8I,EAAgB,CAAErmJ,QAAOC,SAMzB,YAAEqmJ,EAAW,YAAEC,GAivBxB,SAA2ClP,GAAgB,MAAEr3I,EAAK,KAAEC,IACnE,MAAMqmJ,EAA2B,IAAIxlK,IAC/BylK,EAA2B,GAEjC,UAAY,IAAEjO,EAAG,OAAEf,EAAM,WAAEiE,EAAU,KAAErB,KAAU,IAAI,GAAa9C,EAAO,CAAEwB,OAAQ54I,IAAW,CAC7F,MAAMumJ,EAAgBlO,EAAMkD,EAAa,EAIzC,GAFyClD,GAAOt4I,GAASs4I,GAAOr4I,GAAQumJ,EAAgBvmJ,EAEhD,CACvC,MACMwmJ,EAAejL,GADWv7I,EAAOq4I,EAAM,GAG7CgO,EAAYtlK,IAAKu2J,EAAQ,CACxB4C,OACAE,QAASoM,GAEX,CAIA,GAFqCnO,EAAMt4I,GAASwmJ,GAAiBxmJ,EAEjC,CACnC,IAAIqhJ,EAIHA,EADImF,GAAiBvmJ,EACDA,EAAOD,EAAQ,EAIfwmJ,EAAgBxmJ,EAAQ,EAG7CumJ,EAAYp3K,KAAM,CACjBgrK,OACAE,QAASmB,EAAa6F,GAExB,CACD,CAEA,MAAO,CAAEiF,cAAaC,cACvB,CA1xBwCG,CAAkCrP,EAAOgP,GAM9E,GAAKC,EAAYljK,KAAO,EAsxB3B,SAAyBi0J,EAAgBsP,EAAwBL,EAA0Bv8I,GAC1F,MAAMoxI,EAAc,IAAI,GAAa9D,EAAO,CAC3C8B,iBAAiB,EACjBb,IAAKqO,IAGAC,EAAc,IAAKzL,GACnB7C,EAAMjB,EAAMjsI,SAAUu7I,GAE5B,IAAIE,EAEJ,UAAY,OAAEtP,EAAM,KAAE4C,EAAI,SAAEmB,KAAcsL,EACzC,GAAKN,EAAYnpK,IAAKo6J,GAAW,CAChC,MAAQ4C,KAAM2M,EAAU,QAAEzM,GAAYiM,EAAY3uK,IAAK4/J,GAEjD3yH,EAAiBiiI,EACtB98I,EAAOmd,oBAAqB2/H,GAC5B98I,EAAOkd,iBAAkBqxH,EAAK,GAE/BvuI,EAAO8a,KAAM9a,EAAOsd,cAAey/H,GAAcliI,GACjDoyH,GAAwB,UAAWqD,EAASyM,EAAY/8I,GAExD88I,EAAeC,CAChB,MAAYxL,IAEXuL,EAAe1M,EAGlB,CAhzBI4M,CAAgB1P,EADep3I,EAAO,EACSqmJ,EAAav8I,EAC7D,CAGA,QAAUhgC,EAAIk2B,EAAMl2B,GAAKi2B,EAAOj2B,IAC/BggC,EAAOv0B,OAAQ6hK,EAAMjsI,SAAUrhC,IAIhC,UAAY,QAAEswK,EAAO,KAAEF,KAAUoM,EAChCvP,GAAwB,UAAWqD,EAASF,EAAMpwI,IA8rBtD,SAA4BstI,GAAgB,MAAEr3I,EAAK,KAAEC,GAAuB8J,GAC3E,MAAM6tI,EAAcP,EAAMzsI,aAAc,gBAA6B,EAErE,GAAK5K,EAAQ43I,EAAc,CAG1BZ,GAAwB,cAFR/2I,EAAO23I,EAAcA,GAAgB33I,EAAOD,EAAQ,GAAMA,EAE1Bq3I,EAAOttI,EAAQ,EAChE,CACD,CAlsBGi9I,CAAmB3P,EAAOgP,EAAet8I,GAGnCm2I,GAAoB7I,EAAO9nK,OAGhCkxK,GAAiBpJ,EAAO9nK,KACzB,GAEF,CA+BO,aAAAixK,CAAenJ,EAAgBvjK,GACrC,MAAMnM,EAAQ4H,KAAKs0B,OAAOl8B,MACpBq4B,EAAQlsB,EAAQupK,GAChBwH,EAAkB/wK,EAAQ6pK,SAAW,EACrC19I,EAAOnsB,EAAQupK,GAAKwH,EAAkB,EAE5Cl9K,EAAMmiC,QAAQC,KA8nBhB,SAA+BstI,EAAgBuN,EAAqC76I,GACnF,MAAMutI,EAAiBD,EAAMzsI,aAAc,mBAAgC,EAE3E,GAAK0sI,GAAkBsN,EAAqB5kJ,MAAQs3I,EAAiB,CACpE,MAAM2P,EAAkB7+K,KAAKD,IAAKmvK,EAAiB,EAAmCsN,EAAqB3kJ,MAC1G2kJ,EAAqB5kJ,MAAQ,EAE9B+J,EAAO/0B,aAAc,iBAAkBsiK,EAAiB2P,EAAiB5P,EAC1E,CACD,CAtoBG6P,CAAsB7P,EAAO,CAAEr3I,QAAOC,QAAQ8J,GAE9C,QAAUo9I,EAAqBlnJ,EAAMknJ,GAAsBnnJ,EAAOmnJ,IACjE,UAAY,KAAEhN,EAAI,OAAE5C,EAAM,UAAEgE,IAAe,IAAK,IAAI,GAAalE,IAE3DE,GAAU4P,GAAsB5L,EAAY,GAAKhE,EAASgE,EAAY4L,EAC1EnQ,GAAwB,UAAWuE,EAAY,EAAGpB,EAAMpwI,GAC7CwtI,IAAW4P,GAEtBp9I,EAAOv0B,OAAQ2kK,GAMZsG,GAAiBpJ,EAAO9nK,OAG7B2wK,GAAoB7I,EAAO9nK,KAC5B,GAEF,CA2CO,mBAAAwuK,CAAqB7G,EAAoBkQ,EAAgB,GAC/D,MAAMz/K,EAAQ4H,KAAKs0B,OAAOl8B,MAEpB0vK,EADWH,EAAU/oK,OACJA,OAEjBksK,EAAU1tK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KACrEwvI,EAAUztK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KAE3EjjC,EAAMmiC,QAAQC,IAEb,GAAKqwI,EAAU,EAAI,CAElB,MAAM,aAAEiN,EAAY,YAAEC,GAAgBC,GAAiBnN,EAASgN,GAEhEpQ,GAAwB,UAAWsQ,EAAapQ,EAAWntI,GAG3D,MAAMy9I,EAA6D,CAAC,EAG/DH,EAAe,IACnBG,EAAmBpN,QAAUiN,GAIzBhN,EAAU,IACdmN,EAAmBnN,QAAUA,GAI9B8L,GADsB/L,EAAUgN,EAAgBA,EAAgB,EAAIhN,EAAU,EAClDrwI,EAAQA,EAAOmd,oBAAqBgwH,GAAasQ,EAC9E,CAGA,GAAKpN,EAAUgN,EAAgB,CAC9B,MAAMK,EAAgBL,EAAgBhN,EAGhCiE,EAAW,IAAK,IAAI,GAAahH,KAG/BE,OAAQmQ,GAAoBrJ,EAASv9I,MAAM,EAAIq5I,UAAYA,IAASjD,IAGtEyQ,EAAgBtJ,EAAS/oK,QAAQ,EAAI6kK,OAAMoB,YAAWhE,YACpC4C,IAASjD,GAAaK,IAAWmQ,GAC9BnQ,EAASmQ,GAAmBnQ,EAASgE,EAAYmM,IAM5E,UAAY,KAAEvN,EAAI,UAAEoB,KAAeoM,EAClC59I,EAAO/0B,aAAc,UAAWumK,EAAYkM,EAAetN,GAM5D,MAAMqN,EAA2C,CAAC,EAK7CnN,EAAU,IACdmN,EAAmBnN,QAAUA,GAG9B8L,GAAasB,EAAe19I,EAAQA,EAAOmd,oBAAqBgwH,GAAasQ,GAE7E,MAAMlQ,EAAiBD,EAAMzsI,aAAc,mBAAgC,EAGtE0sI,EAAiBoQ,GACrB1Q,GAAwB,iBAAkBM,EAAiBmQ,EAAepQ,EAAOttI,EAEnF,IAEF,CAuDO,qBAAA+zI,CAAuB5G,EAAoBkQ,EAAgB,GACjE,MAAMz/K,EAAQ4H,KAAKs0B,OAAOl8B,MAEpBw0K,EAAWjF,EAAU/oK,OACrBkpK,EAAQ8E,EAAShuK,OACjBy5K,EAAevQ,EAAM9rI,cAAe4wI,GAEpC9B,EAAU1tK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KACrEwvI,EAAUztK,SAAUuqK,EAAUtsI,aAAc,YAAyB,KAE3EjjC,EAAMmiC,QAAQC,IAEb,GAAKswI,EAAU,EAAI,CAElB,MAAMgE,EAAW,IAAK,IAAI,GAAahH,EAAO,CAC7CsB,SAAUiP,EACV/O,OAAQ+O,EAAevN,EAAU,EACjClB,iBAAiB,MAIZ,aAAEkO,EAAY,YAAEC,GAAgBC,GAAiBlN,EAAS+M,GAEhEpQ,GAAwB,UAAWsQ,EAAapQ,EAAWntI,GAE3D,MAAQwtI,OAAQwI,GAAe1B,EAASv9I,MAAM,EAAIq5I,UAAYA,IAASjD,IAGjEsQ,EAA6D,CAAC,EAG/DH,EAAe,IACnBG,EAAmBnN,QAAUgN,GAIzBjN,EAAU,IACdoN,EAAmBpN,QAAUA,GAG9B,UAAYgC,KAAaiC,EAAW,CACnC,MAAM,OAAE9G,EAAM,IAAEe,GAAQ8D,EAMC9D,GAAOsP,EAAeN,GAExB/P,IAAWwI,IAELzH,EAAMsP,EAAeN,GAAgBD,GAAiB,GAGlFlB,GAAa,EAAGp8I,EAAQqyI,EAAUV,oBAAqB8L,EAEzD,CACD,CAGA,GAAKnN,EAAU+M,EAAgB,CAE9B,MAAMK,EAAgBL,EAAgB/M,EAGhCgE,EAAW,IAAK,IAAI,GAAahH,EAAO,CAAEsB,SAAU,EAAGE,OAAQ+O,KAGrE,UAAY,KAAEzN,EAAI,WAAEqB,EAAU,IAAElD,KAAS+F,EAIxC,GAAKlE,IAASjD,GAAaoB,EAAMkD,EAAaoM,EAAe,CAC5D,MAAMC,EAAerM,EAAaiM,EAElC19I,EAAO/0B,aAAc,UAAW6yK,EAAc1N,EAC/C,CAID,MAAMqN,EAA2C,CAAC,EAG7CpN,EAAU,IACdoN,EAAmBpN,QAAUA,GAG9BkL,GAAiBv7I,EAAQstI,EAAOuQ,EAAe,EAAGH,EAAe,EAAGD,GAGpE,MAAM5P,EAAcP,EAAMzsI,aAAc,gBAA6B,EAEhEgtI,EAAcgQ,GAClB5Q,GAAwB,cAAeY,EAAc6P,EAAepQ,EAAOttI,EAE7E,IAEF,CAWO,UAAAo2I,CAAY9I,GAMlB,MAAO,IAFKA,EAAMjsI,SAAU,GAEZL,eAAgBvN,QAAQ,CAAEmgJ,EAASrF,IAG3CqF,EAFahxK,SAAU2rK,EAAI1tI,aAAc,YAAyB,MAGvE,EACJ,CAWO,OAAAkxI,CAASzE,GAEf,OAAOnnK,MAAMrB,KAAMwoK,EAAMtsI,eACvBvN,QAAQ,CAAE4oJ,EAAUl/J,IAAWA,EAAMzJ,GAAI,UAAW,YAAe2oK,EAAW,EAAIA,GAAU,EAC/F,CAcO,iBAAA0B,CAAmBzQ,EAAgBvjK,EAA8B,CAAC,GACxE,OAAO,IAAI,GAAaujK,EAAOvjK,EAChC,CASO,qBAAAi0K,CAAuBvjJ,GAC7B,MAAMs6I,EAAQ,GAEd,UAAYzqJ,KAAS9kB,KAAKy4K,WAAYxjJ,EAAUmY,aAAgB,CAC/D,MAAM31B,EAAUqN,EAAMmnB,sBAEjBx0B,GAAWA,EAAQvJ,GAAI,UAAW,cACtCqhK,EAAM3vK,KAAM6X,EAEd,CAEA,OAAO83J,CACR,CASO,gCAAAgD,CAAkCt9I,GACxC,MAAMs6I,EAAQ,GAEd,UAAYzqJ,KAASmQ,EAAUmY,YAAc,CAC5C,MAAMsrI,EAAoB5zJ,EAAM4b,MAAM4F,aAAc,aAE/CoyI,GACJnJ,EAAM3vK,KAAM84K,EAEd,CAEA,OAAOnJ,CACR,CAUO,8BAAA/B,CAAgCv4I,GACtC,MAAMq5I,EAAgBtuK,KAAKw4K,sBAAuBvjJ,GAElD,OAAKq5I,EAAch2K,OACXg2K,EAGDtuK,KAAKuyK,iCAAkCt9I,EAC/C,CAeO,aAAA24I,CAAe+K,GACrB,MAAMC,EAAUD,EAAW37K,KAAK4tK,GAAUA,EAAKhsK,OAAoBgG,QAEnE,OAAO5E,KAAK64K,2BAA4BD,EACzC,CAeO,gBAAA1K,CAAkByK,GACxB,MAAM7Q,EAAQ6Q,EAAY,GAAIryI,aAAc,SAGtCsyI,EAFW,IAAK,IAAI,GAAa9Q,IAGrC/hK,QAAQ2S,GAASigK,EAAWppJ,SAAU7W,EAAMkyJ,QAC5C5tK,KAAK0b,GAASA,EAAMsvJ,SAEtB,OAAOhoK,KAAK64K,2BAA4BD,EACzC,CAyBO,sBAAAE,CAAwBC,GAC9B,GAAKA,EAAmBzgL,OAAS,IAAM0H,KAAKg5K,8BAA+BD,GAC1E,OAAO,EAMR,MAAM3Q,EAAoB,IAAIx0J,IACxBw6J,EAAuB,IAAIx6J,IAEjC,IAAIqlK,EAAsB,EAE1B,UAAYtR,KAAaoR,EAAqB,CAC7C,MAAM,IAAEhQ,EAAG,OAAEf,GAAWhoK,KAAKioK,gBAAiBN,GACxCmD,EAAU1tK,SAAUuqK,EAAUtsI,aAAc,aAA2B,EACvEwvI,EAAUztK,SAAUuqK,EAAUtsI,aAAc,aAA2B,EAG7E+sI,EAAKv6J,IAAKk7J,GACVqF,EAAQvgK,IAAKm6J,GAGR8C,EAAU,GACd1C,EAAKv6J,IAAKk7J,EAAM+B,EAAU,GAItBD,EAAU,GACduD,EAAQvgK,IAAKm6J,EAAS6C,EAAU,GAGjCoO,GAAyBnO,EAAUD,CACpC,CAGA,MAAMqO,EAoQR,SAAkC9Q,EAAmBgG,GACpD,MAAM+K,EAAcx4K,MAAMrB,KAAM8oK,EAAKl9J,UAC/B+iK,EAAgBttK,MAAMrB,KAAM8uK,EAAQljK,UAEpC2mK,EAAUh5K,KAAKC,OAAQqgL,GACvBlH,EAAWp5K,KAAKD,OAAQugL,GACxBvH,EAAa/4K,KAAKC,OAAQm1K,GAC1B0D,EAAc94K,KAAKD,OAAQq1K,GAEjC,OAAS4D,EAAUI,EAAW,IAAQL,EAAaD,EAAc,EAClE,CA9Q+ByH,CAAyBhR,EAAMgG,GAE5D,OAAO8K,GAAwBD,CAChC,CAKO,UAAAR,CAAY5sI,GAClB,OAAOlrC,MAAMrB,KAAMusC,GAAS1qB,KAAMk4J,GACnC,CAKQ,0BAAAR,CAA4BD,GACnC,MAAMU,EAAmBV,EAAQz3J,MAAM,CAAEo4J,EAAQ30F,IAAY20F,EAAS30F,IAKtE,MAAO,CAAEn0D,MAHK6oJ,EAAkB,GAGhB5oJ,KAFH4oJ,EAAkBA,EAAiBhhL,OAAS,GAG1D,CAkBQ,6BAAA0gL,CAA+BL,GACtC,MAAM7Q,EAAQ6Q,EAAY,GAAIryI,aAAc,SAEtCqnI,EAAa3tK,KAAK4tK,cAAe+K,GACjCtQ,EAAcjrK,SAAU0qK,EAAMzsI,aAAc,iBAA+B,EAGjF,IAAMr7B,KAAKw5K,yBAA0B7L,EAAYtF,GAChD,OAAO,EAGR,MAAM4F,EAAgBjuK,KAAKkuK,iBAAkByK,GACvC5Q,EAAiB3qK,SAAU0qK,EAAMzsI,aAAc,oBAAkC,EAGvF,OAAOr7B,KAAKw5K,yBAA0BvL,EAAelG,EACtD,CAKQ,wBAAAyR,EAA0B,MAAE/oJ,EAAK,KAAEC,GAAuB+oJ,GAIjE,OAH6BhpJ,EAAQgpJ,IACT/oJ,EAAO+oJ,CAGpC,EAUD,SAAS1D,GAAiBv7I,EAAgBstI,EAAgBjhJ,EAAkBuhJ,EAAcsR,EAA2Bn0K,EAAa,CAAC,GAClI,QAAU/K,EAAI,EAAGA,EAAI4tK,EAAM5tK,IAAM,CAChC,MAAMoyK,EAAWpyI,EAAOl1B,cAAe,YAEvCk1B,EAAO90B,OAAQknK,EAAU9E,EAAOjhJ,GAEhC+vJ,GAAa8C,EAAmBl/I,EAAQA,EAAOkd,iBAAkBk1H,EAAU,OAASrnK,EACrF,CACD,CAOA,SAASqxK,GAAarH,EAAe/0I,EAAgBk/B,EAA0Bn0D,EAAa,CAAC,GAC5F,QAAU/K,EAAI,EAAGA,EAAI+0K,EAAO/0K,IAC3BktK,GAAsBltI,EAAQk/B,EAAgBn0D,EAEhD,CAiBA,SAASyyK,GAAiBl9H,EAAc+8H,GACvC,GAAK/8H,EAAO+8H,EACX,MAAO,CAAEC,aAAc,EAAGC,YAAa,GAGxC,MAAMD,EAAej/K,KAAK+B,MAAOkgD,EAAO+8H,GAGxC,MAAO,CAAEC,eAAcC,YAFDj9H,EAAOg9H,EAAeD,EAAkBC,EAG/D,CA6HA,SAASuB,GAAmBvrI,EAAeC,GAE1C,MAAMgvD,EAAOjvD,EAAOpN,MACds8D,EAAOjvD,EAAOrN,MAKpB,OAAOq8D,EAAKrgE,SAAUsgE,IAAU,EAAI,CACrC,CCxrCe,MAAM28E,WAA0B,GAI9B,OAAA5kJ,GACf,MAAM8yI,EAAa7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IAEtC2wK,EAAqBlR,EAAW2Q,sBAAuBx4K,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WACxFj1B,KAAK00B,UAAYmzI,EAAWiR,uBAAwBC,EACrD,CAOgB,OAAAxjJ,GACf,MAAMn9B,EAAQ4H,KAAKs0B,OAAOl8B,MACpByvK,EAAa7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IAE5ChQ,EAAMmiC,QAAQC,IACb,MAAMu+I,EAAqBlR,EAAW2Q,sBAAuBpgL,EAAMoL,SAASyxB,WAGtE2kJ,EAAiBb,EAAmB5sJ,SAGpC,WAAE0tJ,EAAU,YAAEC,GA6CvB,SAA6BF,EAAyBb,EAAoClR,GACzF,IAAIkS,EAAiB,EACjBC,EAAkB,EAEtB,UAAYrS,KAAaoR,EAAqB,CAC7C,MAAM,IAAEhQ,EAAG,OAAEf,GAAWH,EAAWI,gBAAiBN,GAEpDoS,EAAiBE,GAActS,EAAWK,EAAQ+R,EAAgB,WAClEC,EAAkBC,GAActS,EAAWoB,EAAKiR,EAAiB,UAClE,CAGA,MAAQjR,IAAKmR,EAAclS,OAAQmS,GAAoBtS,EAAWI,gBAAiB2R,GAE7EC,EAAaE,EAAiBI,EAC9BL,EAAcE,EAAkBE,EAEtC,MAAO,CAAEL,aAAYC,cACtB,CA/DuCM,CAAoBR,EAAgBb,EAAoBlR,GAC5FJ,GAAwB,UAAWoS,EAAYD,EAAgBp/I,GAC/DitI,GAAwB,UAAWqS,EAAaF,EAAgBp/I,GAEhE,UAAYmtI,KAAaoR,EACxB,GAAiBpR,EAAWiS,EAAgBp/I,GAM7C+2I,GAHcqI,EAAetzI,aAAc,SAGZuhI,GAE/BrtI,EAAOgZ,aAAcomI,EAAgB,KAAM,GAE7C,EAQD,SAAS,GAAiBS,EAA0BC,EAAqB9/I,GAClE,GAAS6/I,KACT,GAASC,IACb9/I,EAAOv0B,OAAQu0B,EAAOud,cAAeuiI,IAGtC9/I,EAAO8a,KAAM9a,EAAOud,cAAesiI,GAAmB7/I,EAAOkd,iBAAkB4iI,EAAY,SAI5F9/I,EAAOv0B,OAAQo0K,EAChB,CAKA,SAAS,GAAS1S,GACjB,MAAMuM,EAAkBvM,EAAU9rI,SAAU,GAE5C,OAA+B,GAAxB8rI,EAAUrsI,YAAmB44I,EAAiBhmK,GAAI,UAAW,cAAiBgmK,EAAgB/xI,OACtG,CAsBA,SAAS83I,GAActS,EAAoBjnI,EAAe65I,EAA0BC,GACnF,MAAMC,EAAiBr9K,SAAUuqK,EAAUtsI,aAAcm/I,IAAqB,KAE9E,OAAO3hL,KAAKC,IAAKyhL,EAAkB75I,EAAQ+5I,EAC5C,CCtGe,MAAMC,WAAyB,GAI7C,WAAA34K,CAAauyB,GACZ9mB,MAAO8mB,GAGPt0B,KAAKg1B,aAAc,CACpB,CAKgB,OAAAD,GACf,MACMu5I,EADyBtuK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cACvBolK,+BAAgCxtK,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WAE5Fj1B,KAAK00B,UAAY45I,EAAch2K,OAAS,CACzC,CAKgB,OAAAi9B,GACf,MAAMn9B,EAAQ4H,KAAKs0B,OAAOl8B,MACpByvK,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDosK,EAAiB3M,EAAW2F,+BAAgCp1K,EAAMoL,SAASyxB,WAC3E04I,EAAa9F,EAAW+F,cAAe4G,GAEvC1M,EAAQ0M,EAAgB,GAAIluI,aAAc,SAC1Cq0I,EAA+B,GAErC,QAAUzO,EAAWyB,EAAWl9I,MAAOy7I,GAAYyB,EAAWj9I,KAAMw7I,IACnE,UAAYtB,KAAU9C,EAAMjsI,SAAUqwI,GAAwB1wI,cAC7Dm/I,EAAe/6K,KAAMxH,EAAM0/C,cAAe8yH,IAI5CxyK,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAcmnI,EAAgB,GAEvC,ECxCc,MAAMC,WAA4B,GAIhD,WAAA74K,CAAauyB,GACZ9mB,MAAO8mB,GAGPt0B,KAAKg1B,aAAc,CACpB,CAKgB,OAAAD,GACf,MACMu5I,EADyBtuK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cACvBolK,+BAAgCxtK,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WAE5Fj1B,KAAK00B,UAAY45I,EAAch2K,OAAS,CACzC,CAKgB,OAAAi9B,GACf,MAAMsyI,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDhQ,EAAQ4H,KAAKs0B,OAAOl8B,MACpBo8K,EAAiB3M,EAAW2F,+BAAgCp1K,EAAMoL,SAASyxB,WAC3Em/I,EAAYI,EAAgB,GAC5BU,EAAWV,EAAex1K,MAC1B8oK,EAAQsM,EAAU9tI,aAAc,SAEhCu0I,EAAgBhT,EAAWI,gBAAiBmM,GAC5C0G,EAAcjT,EAAWI,gBAAiBiN,GAE1C1L,EAAc3wK,KAAKD,IAAKiiL,EAAc7S,OAAQ8S,EAAY9S,QAC1D0B,EAAY7wK,KAAKC,IAAK+hL,EAAc7S,OAAQ8S,EAAY9S,QAExD2S,EAA+B,GAErC,UAAYI,KAAY,IAAI,GAAajT,EAAO,CAAE0B,cAAaE,cAC9DiR,EAAe/6K,KAAMxH,EAAM0/C,cAAeijI,EAASnQ,OAGpDxyK,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAcmnI,EAAgB,GAEvC,ECoKc,SAASK,GAA4B5iL,GACnDA,EAAMoL,SAASk3B,mBAAmBF,GAMnC,SAA+BA,EAAgBpiC,GAC9C,MAAM6nE,EAAU7nE,EAAMoL,SAASm8D,OAAOQ,aAEtC,IAAIlvB,GAAW,EAGf,MAAMgqI,EAAiB,IAAIrnK,IAE3B,UAAY8E,KAASunD,EAAU,CAC9B,IAAI6nG,EAAwB,KAET,UAAdpvJ,EAAMzN,MAAkC,SAAdyN,EAAM1W,OACpC8lK,EAAQpvJ,EAAM+M,SAASykB,WAIH,UAAdxxB,EAAMzN,MAAkC,UAAdyN,EAAMzN,MAAsC,YAAdyN,EAAM1W,MAAoC,aAAd0W,EAAM1W,OAChG8lK,EAAQpvJ,EAAM+M,SAAS6gB,aAAc,UAIjC40I,GAAuBxiK,KAC3BovJ,EAAQpvJ,EAAMoM,MAAM4b,MAAM4F,aAAc,UAGpCwhI,IAAUmT,EAAertK,IAAKk6J,KAGlC72H,EAAWkqI,GAAsBrT,EAAOttI,IAAYyW,EAEpDA,EAAWmqI,GAAmBtT,EAAOttI,IAAYyW,EAEjDgqI,EAAeptK,IAAKi6J,GAEtB,CAEA,OAAO72H,CACR,CA3C6CoqI,CAAsB7gJ,EAAQpiC,IAC3E,CAiDA,SAAS+iL,GAAsBrT,EAAgBttI,GAC9C,IAAIyW,GAAW,EAEf,MAAM+lI,EA+EP,SAA0BlP,GACzB,MAAMO,EAAcjrK,SAAU0qK,EAAMzsI,aAAc,gBAA6B,KACzEigJ,EAAU36K,MAAMrB,KAAMwoK,EAAMtsI,eAChCvN,QAAQ,CAAEtpB,EAAOokK,IAASA,EAAI76J,GAAI,UAAW,YAAevJ,EAAQ,EAAIA,GAAO,GAE3EqyK,EAAc,GAEpB,UAAY,IAAEjO,EAAG,KAAE6B,EAAI,WAAEqB,KAAgB,IAAI,GAAanE,GAAU,CAEnE,GAAKmE,EAAa,EACjB,SAGD,MAGMsP,EAHaxS,EAAMV,EAGKA,EAAciT,EAG5C,GAAKvS,EAAMkD,EAAasP,EAAW,CAClC,MAAM5L,EAAa4L,EAAWxS,EAE9BiO,EAAYp3K,KAAM,CAAEgrK,OAAME,QAAS6E,GACpC,CACD,CAEA,OAAOqH,CACR,CA1GqBwE,CAAiB1T,GAErC,GAAKkP,EAAY1+K,OAAS,CAGzB24C,GAAW,EAEX,UAAYnuC,KAAQk0K,EACnBvP,GAAwB,UAAW3kK,EAAKgoK,QAAShoK,EAAK8nK,KAAMpwI,EAAQ,EAEtE,CAEA,OAAOyW,CACR,CAOA,SAASmqI,GAAmBtT,EAAgBttI,GAC3C,IAAIyW,GAAW,EAEf,MAAMwqI,EAwFP,SAA6B3T,GAE5B,MAAM4T,EAAU,IAAI/6K,MAAOmnK,EAAMxsI,YAAaxwB,KAAM,GAEpD,UAAY,SAAEohK,KAAc,IAAI,GAAapE,EAAO,CAAE8B,iBAAiB,IACtE8R,EAASxP,KAGV,OAAOwP,CACR,CAjGyBC,CAAoB7T,GACtC6M,EAAe,GAGrB,UAAczI,EAAUr4J,KAAU4nK,EAAgBhjK,WAE3C5E,GAAQi0J,EAAMjsI,SAAUqwI,GAAYh+J,GAAI,UAAW,aACxDymK,EAAa/0K,KAAMssK,GAKrB,GAAKyI,EAAar8K,OAAS,CAG1B24C,GAAW,EAEX,UAAYi7H,KAAYyI,EAAatpK,UACpCmvB,EAAOv0B,OAAQ6hK,EAAMjsI,SAAUqwI,IAC/BuP,EAAgBj0K,OAAQ0kK,EAAU,EAEpC,CAGA,MAAM0P,EAAcH,EAAgB11K,QAAQ,CAAEgjK,EAAKmD,IAAcpE,EAAMjsI,SAAUqwI,GAAYh+J,GAAI,UAAW,cAGtG2tK,EAAYD,EAAa,GAG/B,IAFgBA,EAAYrlK,OAAOje,GAAUA,IAAWujL,IAExC,CAIf,MAAMC,EAAaF,EAAY3tJ,QAAQ,CAAE4nB,EAAM92C,IAAaA,EAAU82C,EAAO92C,EAAU82C,GAAM,GAE7F,UAAcq2H,EAAUr4J,KAAU+nK,EAAYnjK,UAAY,CACzD,MAAMi+J,EAAkBoF,EAAajoK,EAErC,GAAK6iK,EAAkB,CACtB,QAAUl8K,EAAI,EAAGA,EAAIk8K,EAAiBl8K,IACrCktK,GAAsBltI,EAAQA,EAAOkd,iBAAkBowH,EAAMjsI,SAAUqwI,GAAa,QAGrFj7H,GAAW,CACZ,CACD,CACD,CAEA,OAAOA,CACR,CAoDA,SAASiqI,GAAuBxiK,GAC/B,GAAoB,cAAfA,EAAMzN,KACV,OAAO,EAGR,MAAMlU,EAAM2hB,EAAM+rC,aAElB,MAAe,gBAAR1tD,GAAiC,YAARA,GAA6B,YAARA,CACtD,CCtYe,SAASglL,GAAmC3jL,GAC1DA,EAAMoL,SAASk3B,mBAAmBF,GAMnC,SAAqCA,EAAgBpiC,GACpD,MAAM6nE,EAAU7nE,EAAMoL,SAASm8D,OAAOQ,aAEtC,IAAIlvB,GAAW,EAEf,UAAYv4B,KAASunD,EACD,UAAdvnD,EAAMzN,MAAkC,SAAdyN,EAAM1W,OACpCivC,EAAW+qI,GAAUtjK,EAAM+M,SAASykB,UAAsB1P,IAAYyW,GAGpD,UAAdv4B,EAAMzN,MAAkC,YAAdyN,EAAM1W,OACpCivC,EAAWgrI,GAAavjK,EAAM+M,SAASykB,UAAsB1P,IAAYyW,GAGvD,UAAdv4B,EAAMzN,MAAkC,aAAdyN,EAAM1W,OACpCivC,EAAWirI,GAAqBxjK,EAAM+M,SAASykB,UAAsB1P,IAAYyW,GAG7D,UAAdv4B,EAAMzN,MAAkC,UAAdyN,EAAMzN,OAAsBkxK,GAAsBzjK,KAClFu4B,EAAWirI,GAAqBxjK,EAAM+M,SAAS7mB,OAAmB47B,IAAYyW,GAIhF,OAAOA,CACR,CA9B6CmrI,CAA4B5hJ,EAAQpiC,IACjF,CAkCA,SAAS4jL,GAAUlU,EAAgBttI,GAClC,IAAIyW,GAAW,EAEf,UAAY83H,KAAOjB,EAAMtsI,cACnButI,EAAI76J,GAAI,UAAW,cACvB+iC,EAAWgrI,GAAalT,EAAKvuI,IAAYyW,GAI3C,OAAOA,CACR,CAKA,SAASgrI,GAAarP,EAAmBpyI,GACxC,IAAIyW,GAAW,EAEf,UAAY02H,KAAaiF,EAASpxI,cACjCyV,EAAWirI,GAAqBvU,EAAWntI,IAAYyW,EAGxD,OAAOA,CACR,CAOA,SAASirI,GAAqBvU,EAAoBntI,GAEjD,GAA6B,GAAxBmtI,EAAUrsI,WAKd,OAFAd,EAAOquC,cAAe,YAAa8+F,IAE5B,EAKR,MAAM0U,EAAY17K,MAAMrB,KAAMqoK,EAAUnsI,eAAgBz1B,QAAQ4R,GAASA,EAAMzJ,GAAI,WAInF,UAAYyJ,KAAS0kK,EACpB7hJ,EAAOsc,KAAMtc,EAAOsd,cAAengC,GAAS,aAI7C,QAAS0kK,EAAU/jL,MACpB,CAOA,SAAS6jL,GAAsBzjK,GAC9B,QAAMA,EAAM+M,SAAS7mB,OAAOsP,GAAI,UAAW,eAItB,UAAdwK,EAAMzN,MAAkC,SAAdyN,EAAM1W,MAAiC,UAAd0W,EAAMzN,KACjE,CCjFA,SAASqxK,GAAe3kK,EAAgB6lD,GACvC,IAAM7lD,EAAMzJ,GAAI,UAAW,aAC1B,OAAO,EAGR,MAAMqpC,EAAcimB,EAAOf,cAAe9kD,GAE1C,QAAM4/B,GAICy1H,GAAoCr1J,KAAY4/B,EAAYrpC,GAAI,UAAW,OACnF,C,eC/DI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ4rB,OC4BR,MAAM,WAAqB,GAuBzC,WAAA/3B,CAAauyB,GACZ9mB,MAAO8mB,GAEPt0B,KAAKu8K,iBAAmB,EACzB,CAlBA,qBAAkBjmJ,GACjB,MAAO,cACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GACV,CAcO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACfsrE,EAAStrE,EAAMsrE,OACfrlE,EAAai2B,EAAOj2B,WACpBwpK,EAAavzI,EAAOqC,QAAQvuB,IAAK,IAEvCs7D,EAAOwR,SAAU,QAAS,CACzBsE,eAAgB,eAChB5D,gBAAiB,CAAE,cAAe,oBAGnClS,EAAOwR,SAAU,WAAY,CAC5BqD,QAAS,QACTzU,SAAS,IAGVJ,EAAOwR,SAAU,YAAa,CAC7B4D,eAAgB,aAChBP,QAAS,WACT3C,gBAAiB,CAAE,UAAW,WAC9B9R,SAAS,EACT2O,cAAc,IAIfp0E,EAAWuhF,IAAK,UAAW/xE,KrBtEnB2xD,IACRA,EAAWnwD,GAAwB,kBAAkB,CAAEgG,EAAKvS,EAAMw8D,KAEjE,IAAMA,EAAcwB,WAAW7+D,KAAMa,EAAKgtE,SAAU,CAAE9tE,MAAM,EAAM08B,QAAS,UAC1E,OAID,MAAMypI,EAuJT,SAAiCnqB,GAChC,UAAYw+B,KAAex+B,EAAWxiH,cACrC,GAAKghJ,EAAYtuK,GAAI,UAAW,SAC/B,OAAOsuK,CAGV,CA7JqBC,CAAwB35K,EAAKgtE,UAG/C,IAAMq4F,IAAc7oG,EAAcwB,WAAW7+D,KAAMkmK,EAAW,CAAEnmK,MAAM,IACrE,OAIDs9D,EAAcwB,WAAWrC,QAAS37D,EAAKgtE,SAAU,CAAE9tE,MAAM,EAAM08B,QAAS,UAGxE,MAGMg+I,EAAa,GAHMp9G,EAAc4C,YAAaimG,EAAWrlK,EAAK+sE,aAGzBvS,WAAYpxB,YAGjDwwI,GAONp9G,EAAc6C,gBAAiBr/D,EAAKgtE,SAAUxQ,EAAc9kC,OAAOkd,iBAAkBglI,EAAY,QACjGp9G,EAAc+Q,uBAAwBqsG,EAAY55K,IANjDw8D,EAAcwB,WAAWnC,OAAQ77D,EAAKgtE,SAAU,CAAE9tE,MAAM,EAAM08B,QAAS,SAMhB,GACtD,IqBuCHrgC,EAAWuhF,IAAK,UAAW/xE,IAAKq6J,MAEhC7pK,EAAWuhF,IAAK,mBAAoB1W,mBAAoB,CACvD9wE,MAAO,CACN4J,KAAM,QACNuD,WAAY,CAAE,gBAEf00B,KAAMmyI,GAAevE,EAAY,CAChC4E,UAAU,EACVD,gBAAiBxsK,KAAKu8K,qBAGxBl+K,EAAWuhF,IAAK,gBAAiB1W,mBAAoB,CACpD9wE,MAAO,CACN4J,KAAM,QACNuD,WAAY,CAAE,gBAEf00B,KAAMmyI,GAAevE,EAAY,CAChC2E,gBAAiBxsK,KAAKu8K,qBAKxBl+K,EAAWuhF,IAAK,UAAWvX,iBAAkB,CAAEjwE,MAAO,WAAY6hC,KAAM,OACxE57B,EAAWuhF,IAAK,UAAW/xE,KrBWnB2xD,IACRA,EAAWnwD,GAAwB,cAAc,CAAEgG,EAAKvS,KAClDA,EAAKgtE,SAAS3tC,SAAqC,GAA1Br/B,EAAK+sE,YAAYjrE,OAC9CyQ,EAAIlJ,MACL,GACE,CAAES,SAAU,QAAU,IqBdzBvO,EAAWuhF,IAAK,YAAavX,iBAAkB,CAC9CjwE,MAAO,WACP6hC,KnBlDK,CAAE2yI,GAAYpyI,YACboyI,EAASzqI,QACf3H,EAAO2Z,mBAAoB,MAC3B3Z,EAAOuZ,uBAAwB,QmBmDhC11C,EAAWuhF,IAAK,UAAWvX,iBAAkB,CAAEjwE,MAAO,YAAa6hC,KAAM,OACzE57B,EAAWuhF,IAAK,UAAWvX,iBAAkB,CAAEjwE,MAAO,YAAa6hC,KAAM,OACzE57B,EAAWuhF,IAAK,UAAW/xE,IAAKm7J,GAA4B,OAC5D3qK,EAAWuhF,IAAK,UAAW/xE,IAAKm7J,GAA4B,OAE5D3qK,EAAWuhF,IAAK,mBAAoBvX,iBAAkB,CACrDjwE,MAAO,YACP6hC,KAAM0yI,GAAc,CAAEF,UAAU,MAEjCpuK,EAAWuhF,IAAK,gBAAiBvX,iBAAkB,CAClDjwE,MAAO,YACP6hC,KAAM0yI,OAIPtuK,EAAWuhF,IAAK,mBAAoBvX,iBAAkB,CACrDjwE,MAAO,YACP6hC,KAAM8yI,GAA6B,CAAEN,UAAU,IAC/C1jG,kBAAmB,SAEpB1qE,EAAWuhF,IAAK,gBAAiBvX,iBAAkB,CAClDjwE,MAAO,YACP6hC,KAAM8yI,KACNhkG,kBAAmB,SAIpB1qE,EAAWuhF,IAAK,YAAarV,qBAAsB,CAAEnyE,MAAO,UAAW6hC,KAAM,YAC7E57B,EAAWuhF,IAAK,UAAWrV,qBAAsB,CAChDnyE,MAAO,CAAErB,IAAK,UAAWyB,MAAOmkL,GAAgB,YAChD1iJ,KAAM,YAGP57B,EAAWuhF,IAAK,YAAarV,qBAAsB,CAAEnyE,MAAO,UAAW6hC,KAAM,YAC7E57B,EAAWuhF,IAAK,UAAWrV,qBAAsB,CAChDnyE,MAAO,CAAErB,IAAK,UAAWyB,MAAOmkL,GAAgB,YAChD1iJ,KAAM,YAIP3F,EAAOtL,OAAOtyB,OAAQ,6BAA8B,GACpD49B,EAAOtL,OAAOtyB,OAAQ,gCAAiC,GAGvD49B,EAAO0qE,SAASnxF,IAAK,cAAe,IAAIo/J,GAAoB34I,IAC5DA,EAAO0qE,SAASnxF,IAAK,sBAAuB,IAAIy/J,GAAkBh5I,EAAQ,CAAEmsI,MAAO,WACnFnsI,EAAO0qE,SAASnxF,IAAK,sBAAuB,IAAIy/J,GAAkBh5I,EAAQ,CAAEmsI,MAAO,WACnFnsI,EAAO0qE,SAASnxF,IAAK,wBAAyB,IAAImgK,GAAqB15I,EAAQ,CAAEmsI,MAAO,UACxFnsI,EAAO0qE,SAASnxF,IAAK,yBAA0B,IAAImgK,GAAqB15I,EAAQ,CAAEmsI,MAAO,WAEzFnsI,EAAO0qE,SAASnxF,IAAK,iBAAkB,IAAIsmK,GAAkB7/I,IAC7DA,EAAO0qE,SAASnxF,IAAK,oBAAqB,IAAImnK,GAAqB1gJ,IAEnEA,EAAO0qE,SAASnxF,IAAK,2BAA4B,IAAIwgK,GAAkB/5I,EAAQ,CAAE/P,UAAW,gBAC5F+P,EAAO0qE,SAASnxF,IAAK,6BAA8B,IAAIwgK,GAAkB/5I,EAAQ,CAAE/P,UAAW,kBAE9F+P,EAAO0qE,SAASnxF,IAAK,kBAAmB,IAAI8rK,GAAmBrlJ,IAE/DA,EAAO0qE,SAASnxF,IAAK,sBAAuB,IAAIskK,GAAkB79I,EAAQ,CAAE/P,UAAW,WACvF+P,EAAO0qE,SAASnxF,IAAK,qBAAsB,IAAIskK,GAAkB79I,EAAQ,CAAE/P,UAAW,UACtF+P,EAAO0qE,SAASnxF,IAAK,qBAAsB,IAAIskK,GAAkB79I,EAAQ,CAAE/P,UAAW,UACtF+P,EAAO0qE,SAASnxF,IAAK,mBAAoB,IAAIskK,GAAkB79I,EAAQ,CAAE/P,UAAW,QAEpF+P,EAAO0qE,SAASnxF,IAAK,uBAAwB,IAAIgoK,GAAwBvhJ,IACzEA,EAAO0qE,SAASnxF,IAAK,oBAAqB,IAAI0nK,GAAqBjhJ,IAEnEA,EAAO0qE,SAASnxF,IAAK,iBAAkB,IAAI6sK,GAAkBpmJ,IAC7DA,EAAO0qE,SAASnxF,IAAK,oBAAqB,IAAI+sK,GAAqBtmJ,IAEnE0mJ,GAA4B5iL,GAC5B2jL,GAAmC3jL,GAEnC4H,KAAKwP,SAAUpX,EAAMoL,SAAU,eAAe,MCpLjC,SAAsCpL,EAAc+mG,GAClE,MAAMx/B,EAASvnE,EAAMoL,SAASm8D,OAE9B,UAAYplC,KAAUolC,EAAOQ,aAAe,CAC3C,IAAI2nG,EACA8U,GAAc,EAElB,GAAoB,aAAfriJ,EAAOtvB,KAAsB,CACjC,MAAMwM,EAAU8iB,EAAOzV,MAAM4b,MAAMwJ,UAEnC,IAAMzyB,IAAYA,EAAQvJ,GAAI,UAAW,SACxC,SAGD,GAA4B,eAAvBqsB,EAAOkqB,cAAwD,kBAAvBlqB,EAAOkqB,aACnD,SAGDqjH,EAAQrwJ,EACRmlK,EAAqC,eAAvBriJ,EAAOkqB,YACtB,KAA2B,YAAflqB,EAAOv4B,MAAqC,aAAfu4B,EAAOv4B,OAC/C8lK,EAAQvtI,EAAO9U,SAAS6gB,aAAc,SACtCs2I,EAA6B,YAAfriJ,EAAOv4B,MAGtB,IAAM8lK,EACL,SAGD,MAAMO,EAAcP,EAAMzsI,aAAc,gBAAmB,EACrD0sI,EAAiBD,EAAMzsI,aAAc,mBAAsB,EAE3DuwI,EAAc,IAAI,GAAa9D,GAErC,UAAY+E,KAAajB,EAAc,CACtC,MACMiR,EADYhQ,EAAU9D,IAAMV,GAAewE,EAAU7E,OAASD,EAC5B,KAAO,KAEzCxwH,EAAc4nD,EAAQ3hC,OAAOf,cAAeowG,EAAUjC,MAEvDrzH,GAAeA,EAAYrpC,GAAI,YAAeqpC,EAAYv1C,MAAQ66K,GACtE19E,EAAQrrB,cAAiB8oG,EAAc/P,EAAUjC,KAAKhsK,OAASiuK,EAAUjC,KAE3E,CACD,CACD,CDwIGkS,CAA6B1kL,EAAOk8B,EAAO6qE,SFnL/B,SAAkC/mG,EAAc+mG,GAC9D,MAAMx/B,EAASvnE,EAAMoL,SAASm8D,OAGxBo9G,EAAe,IAAInpK,IAEzB,UAAY2mB,KAAUolC,EAAOQ,aAAe,CAC3C,MAAMvhE,EAAwB,aAAf27B,EAAOtvB,KAAsBsvB,EAAOzV,MAAM4b,MAAM9hC,OAAS27B,EAAO9U,SAAS7mB,OAEnFA,EAAOsP,GAAI,UAAW,cAC1B6uK,EAAalvK,IAAKjP,EAEpB,CAEA,UAAY+oK,KAAaoV,EAAa7xK,SAAgC,CACrE,MAAM8xK,EAAsBr8K,MAAMrB,KAAMqoK,EAAUnsI,eAChDz1B,QAAQ4R,GAAS2kK,GAAe3kK,EAAkBwnF,EAAQ3hC,UAE5D,UAAYwQ,KAAagvG,EACxB79E,EAAQrrB,cAAe9F,EAEzB,CACD,CE8JGivG,CAAyB7kL,EAAOk8B,EAAO6qE,QAAS,GAElD,CAKO,sBAAA+9E,CAAwBC,GAC9Bn9K,KAAKu8K,iBAAiB38K,KAAMu9K,EAC7B,EASD,SAASR,GAAgB1xK,GACxB,OAAS2/J,IACR,MAAM9vH,EAAO19C,SAAUwtK,EAAKvvI,aAAcpwB,IAE1C,OAAKma,OAAO05F,MAAOhkE,IAAUA,GAAQ,EAC7B,KAGDA,CAAI,CAEb,C,eExOI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQhhB,OCQR,MAAMsjJ,WAAwB,GAwC5C,WAAAr7K,CAAam3B,GACZ1rB,MAAO0rB,GAEP,MAAMlyB,EAAOhH,KAAK0iG,aAClB1iG,KAAK4wB,MAAQ5wB,KAAKq9K,wBAClBr9K,KAAKo/F,WAAa,IAAI,GACtBp/F,KAAKwqG,aAAe,IAAI,GACxBxqG,KAAKyR,IAAK,OAAQ,GAClBzR,KAAKyR,IAAK,UAAW,GACrBzR,KAAKgH,KAAM,SAAUzH,GAAIS,KAAM,UAAWA,KAAM,QAAQ,CAAEouK,EAAShG,IAAU,GAAIA,OAAYgG,MAE7FpuK,KAAK4jG,YAAa,CACjBzsF,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CAAE,OAGVjnB,SAAU,CACT,CACCH,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CAAE,mCAEVlvB,GAAI,CACH,+CAAgDrI,EAAKzH,GAAI,YAE1D+X,SAAUtX,KAAK4wB,OAEhB,CACCzZ,IAAK,MACL5R,WAAY,CACXg5B,MAAO,CACN,KACA,mCAED,eAAe,GAEhBjnB,SAAU,CACT,CACC4iB,KAAMlzB,EAAKzH,GAAI,aAMnB8P,GAAI,CACHm6F,UAAWxiG,EAAKzH,IAAI8V,IACnBA,EAAI6d,gBAAgB,IAGrBq2E,MAAOviG,EAAKzH,IAAI,KACfS,KAAKwQ,KAAM,UAAW,OAMzBxQ,KAAKqP,GAAI,WAAW,CAAEgG,EAAK2L,KAC1B,MAAM,IAAE+nJ,EAAG,OAAEf,GAAWhnJ,EAAOpd,OAAOm3C,QACtC/6C,KAAK4wB,MAAMxoB,IAAmC,IAA5BhL,SAAU2rK,EAAK,IAAO,IAAa3rK,SAAU4qK,EAAQ,IAAO,IAAO/6H,OAAO,IAM7FjtC,KAAKwqG,aAAan7F,GAAI,yBAAyB,CAAEgG,EAAKrT,EAAM4wB,KAC3D,IAAMA,EACL,OAGD,MAAM,IAAEm2I,EAAG,OAAEf,GAAWp1I,EAAemoB,QAGvC/6C,KAAKyR,IAAK,CACT22J,KAAMhrK,SAAU2rK,GAChBqF,QAAShxK,SAAU4qK,IACjB,IAGJhoK,KAAKqP,GAAI,kBAAkB,IAAMrP,KAAKs9K,wBACtCt9K,KAAKqP,GAAI,eAAe,IAAMrP,KAAKs9K,uBACpC,CAEgB,MAAA3rI,GACfnkC,MAAMmkC,SCxHO,UACd,iBAAEg7D,EAAgB,aAAEnC,EAAY,UAAE+yE,EAAS,gBAAEC,EAAe,oBAAE7tJ,IAQ9D,MAAM8tJ,EAAgD,iBAApBD,EAA+B,IAAMA,EAAkBA,EAsCzF,SAASE,EAAoBC,GAC5B,OAAStoK,IACR,MAAMud,EAAiB2qJ,EAAUhsJ,MAAMtxB,GAAQA,EAAKwX,UAAY+yF,EAAa53E,iBACvEgrJ,EAAsBL,EAAUpsJ,SAAUyB,GAC1CirJ,EAAmBF,EAAiBC,EAAqBL,GAE7DA,EAAUn1K,IAAKy1K,GAAsC5wI,QAEvD53B,EAAI8d,kBACJ9d,EAAI6d,gBAAgB,CAEtB,CAoBA,SAAS4qJ,EAAsBC,EAAsB3wE,GACpD,OAAK2wE,IAAiB3wE,EAAmB,EACjC,EAEA2wE,EAAe,CAExB,CAoBA,SAASC,EAAqBD,EAAsB3wE,GACnD,OAAsB,IAAjB2wE,EACG3wE,EAAmB,EAEnB2wE,EAAe,CAExB,CAnGApxE,EAAiBl7F,IAAK,aAAcisK,GAAoB,CAAEE,EAAqBL,IAC/C,QAAxB5tJ,EACNquJ,EAAqBJ,EAAqBL,EAAUjlL,QACpDwlL,EAAsBF,EAAqBL,EAAUjlL,WAGvDq0G,EAAiBl7F,IAAK,YAAaisK,GAAoB,CAAEE,EAAqBL,IAC9C,QAAxB5tJ,EACNmuJ,EAAsBF,EAAqBL,EAAUjlL,QACrD0lL,EAAqBJ,EAAqBL,EAAUjlL,WAGtDq0G,EAAiBl7F,IAAK,UAAWisK,GAAoB,CAAEE,EAAqBL,KAC3E,IAAIr3D,EAAY03D,EAAsBH,IAUtC,OARKv3D,EAAY,IAChBA,EAAY03D,EAAsBH,IAAuB5kL,KAAK+B,MAAO2iL,EAAUjlL,OAASmlL,KAEnFv3D,EAAYq3D,EAAUjlL,OAAS,IACnC4tH,GAAau3D,MAIRv3D,CAAS,KAGjBvZ,EAAiBl7F,IAAK,YAAaisK,GAAoB,CAAEE,EAAqBL,KAC7E,IAAIr3D,EAAY03D,EAAsBH,IAMtC,OAJKv3D,EAAYq3D,EAAUjlL,OAAS,IACnC4tH,EAAY03D,EAAsBH,KAG5Bv3D,CAAS,IAmElB,CDWE,CAA4B,CAC3BvZ,iBAAkB3sG,KAAKo/F,WACvBoL,aAAcxqG,KAAKwqG,aACnB+yE,UAAWv9K,KAAK4wB,MAChB4sJ,gBAAiB,GACjB7tJ,oBAAqB3vB,KAAKk5B,QAAUl5B,KAAKk5B,OAAOvJ,sBAGjD,UAAY1vB,KAAQD,KAAK4wB,MACxB5wB,KAAKwqG,aAAa38F,IAAK5N,EAAKwX,SAG7BzX,KAAKo/F,WAAW5vF,SAAUxP,KAAKyX,QAChC,CAKO,KAAAw1B,GACNjtC,KAAK4wB,MAAMxoB,IAAK,GAAK6kC,OACtB,CAKO,SAAAi+D,GACNlrG,KAAK4wB,MAAMxoB,IAAK,GAAK6kC,OACtB,CAKQ,mBAAAqwI,GACP,MAAMlV,EAAOpoK,KAAKooK,KACZgG,EAAUpuK,KAAKouK,QAErBpuK,KAAK4wB,MAAM5zB,KAAK,CAAEihL,EAASr5K,KAE1B,MAIMytG,EAJUx5G,KAAK+B,MAAOgK,EAAQ,IAIbwjK,GAHJxjK,EAAQ,GAGiBwpK,EAE5C6P,EAAQxsK,IAAK,OAAQ4gG,EAAM,GAE7B,CAUQ,iBAAA6rE,CAAmBhlJ,EAAgB6vI,EAAaf,EAAgBz5H,GACvE,MAAMinE,EAAS,IAAI,GAAYt8E,GAc/B,OAZAs8E,EAAO/jG,IAAK,CACX88B,QACAhQ,MAAO,sCAGRi3E,EAAO/S,eAAgB,CACtBl9F,WAAY,CACX,WAAYwjK,EACZ,cAAef,KAIVxyD,CACR,CAKQ,qBAAA6nE,GACP,MAAMc,EAAQ,GAGd,QAAUv5K,EAAQ,EAAGA,EAAQ,IAAKA,IAAU,CAC3C,MAAMmkK,EAAMlwK,KAAK+B,MAAOgK,EAAQ,IAC1BojK,EAASpjK,EAAQ,GACjB2pC,EAAQ,GAAIw6H,EAAM,OAASf,EAAS,IAE1CmW,EAAMv+K,KAAMI,KAAKk+K,kBAAmBl+K,KAAKk5B,OAAS6vI,EAAM,EAAGf,EAAS,EAAGz5H,GACxE,CAEA,OAAOvuC,KAAKsjG,iBAAkB66E,EAC/B,EEpMc,MAAMC,WAAgB,GAIpC,qBAAkB9nJ,GACjB,MAAO,SACR,CAKO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdt5B,EAAIgF,KAAKs0B,OAAOt5B,EAEhBqjL,EAA4C,QADjB/pJ,EAAO4E,OAAO3K,yBAG/C+F,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,eAAeqrB,IAC9C,MAAMxD,EAA8BpB,EAAO0qE,SAAS52F,IAAK,eACnD8mG,EAAe,GAAgBh2E,GAWrC,IAAIolJ,EAmBJ,OA5BApvE,EAAaloG,KAAM,aAAczH,GAAIm2B,GAGrCw5E,EAAa/D,WAAW15F,IAAK,CAC5Bi4F,KCjEJ,8TDkEIn7D,MAAOvzC,EAAG,gBACV6uG,SAAS,IAKVqF,EAAa7/F,GAAI,iBAAiB,KAC5BivK,IAKLA,EAAkB,IAAIlB,GAAiBlkJ,GACvCg2E,EAAa9D,UAAU9zF,SAASzJ,IAAKywK,GAErCA,EAAgBltK,SAAU,WAAY7R,GAAI2vG,GAE1CA,EAAa7/F,GAAI,WAAW,KAC3BilB,EAAOiB,QAAS,cAAe,CAAE6yI,KAAMkW,EAAgBlW,KAAMgG,QAASkQ,EAAgBlQ,UACtF95I,EAAO6qE,QAAQllE,KAAKgT,OAAO,IACzB,IAGGiiE,CAAY,IAGpB56E,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,eAAeqrB,IAC9C,MAAM30B,EAAU,CACf,CACC0G,KAAM,eACN7S,MAAO,CACNwhC,YAAa,uBACb2U,MAAOvzC,EAAG,iBACVujL,UAAU,IAGZ,CAAEtzK,KAAM,aACR,CACCA,KAAM,SACN7S,MAAO,CACNwhC,YAAaykJ,EAAe,wBAA0B,yBACtD9vI,MAAOvzC,EAAG,wBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAaykJ,EAAe,yBAA2B,wBACvD9vI,MAAOvzC,EAAG,yBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAa,oBACb2U,MAAOvzC,EAAG,mBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAa,oBACb2U,MAAOvzC,EAAG,oBAKb,OAAOgF,KAAKw+K,iBAAkBxjL,EAAG,UErIpC,yYFqIiEuJ,EAAS20B,EAAQ,IAGhF5E,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,YAAYqrB,IAC3C,MAAM30B,EAAU,CACf,CACC0G,KAAM,eACN7S,MAAO,CACNwhC,YAAa,oBACb2U,MAAOvzC,EAAG,cACVujL,UAAU,IAGZ,CAAEtzK,KAAM,aACR,CACCA,KAAM,SACN7S,MAAO,CACNwhC,YAAa,sBACb2U,MAAOvzC,EAAG,sBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAa,sBACb2U,MAAOvzC,EAAG,sBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAa,iBACb2U,MAAOvzC,EAAG,gBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAa,iBACb2U,MAAOvzC,EAAG,iBAKb,OAAOgF,KAAKw+K,iBAAkBxjL,EAAG,OGjLpC,sYHiL2DuJ,EAAS20B,EAAQ,IAG1E5E,EAAO6E,GAAG61E,iBAAiBnhG,IAAK,mBAAmBqrB,IAClD,MAAM30B,EAAU,CACf,CACC0G,KAAM,SACN7S,MAAO,CACNwhC,YAAa,mBACb2U,MAAOvzC,EAAG,mBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAaykJ,EAAe,sBAAwB,qBACpD9vI,MAAOvzC,EAAG,sBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAa,qBACb2U,MAAOvzC,EAAG,qBAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAaykJ,EAAe,qBAAuB,sBACnD9vI,MAAOvzC,EAAG,qBAGZ,CAAEiQ,KAAM,aACR,CACCA,KAAM,SACN7S,MAAO,CACNwhC,YAAa,2BACb2U,MAAOvzC,EAAG,2BAGZ,CACCiQ,KAAM,SACN7S,MAAO,CACNwhC,YAAa,6BACb2U,MAAOvzC,EAAG,8BAKb,OAAOgF,KAAKy+K,iCAAkCzjL,EAAG,eInOpD,yYJmOyFuJ,EAAS20B,EAAQ,GAEzG,CASQ,gBAAAslJ,CAAkBjwI,EAAem7D,EAAcnlG,EAA4C20B,GAClG,MAAM5E,EAASt0B,KAAKs0B,OACd46E,EAAe,GAAgBh2E,GAC/B8lE,EAAWh/F,KAAK0+K,6BAA8BxvE,EAAc3qG,GAuBlE,OApBA2qG,EAAa/D,WAAW15F,IAAK,CAC5B88B,QACAm7D,OACAG,SAAS,IAIVqF,EAAaloG,KAAM,aAAcmN,OAAQ6qF,EAAU,aAAa,IAAKq9C,IAC7DA,EAAWhoH,MAAMK,GAAaA,MAGtC10B,KAAKwP,SAAU0/F,EAAc,WAAW75F,IACvCif,EAAOiB,QAAWlgB,EAAInS,OAAgB02B,aAG9BvkB,EAAInS,kBAAkB,IAC7BoxB,EAAO6qE,QAAQllE,KAAKgT,OACrB,IAGMiiE,CACR,CAUQ,gCAAAuvE,CAAkClwI,EAAem7D,EAAcnlG,EAA4C20B,GAClH,MAAM5E,EAASt0B,KAAKs0B,OACd46E,EAAe,GAAgBh2E,EAAQ,IACvCylJ,EAAmB,kBAGnBC,EAAkCtqJ,EAAO0qE,SAAS52F,IAAKu2K,GAGvD3/E,EAAWh/F,KAAK0+K,6BAA8BxvE,EAAc3qG,GA0BlE,OAxBA2qG,EAAa/D,WAAW15F,IAAK,CAC5B88B,QACAm7D,OACAG,SAAS,EACTn1E,WAAW,IAIZw6E,EAAaloG,KAAM,aAAcmN,OAAQ,CAAEyqK,KAAiB5/E,GAAY,aAAa,IAAKq9C,IAClFA,EAAWhoH,MAAMK,GAAaA,MAItC10B,KAAKwP,SAAU0/F,EAAa/D,WAAY,WAAW,KAClD72E,EAAOiB,QAASopJ,GAChBrqJ,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAI5BjtC,KAAKwP,SAAU0/F,EAAc,WAAW75F,IACvCif,EAAOiB,QAAWlgB,EAAInS,OAAgB02B,aACtCtF,EAAO6qE,QAAQllE,KAAKgT,OAAO,IAGrBiiE,CACR,CASQ,4BAAAwvE,CAA8BxvE,EAA4B3qG,GACjE,MAAM+vB,EAASt0B,KAAKs0B,OACd0qE,EAA2B,GAC3Bi9C,EAAkB,IAAI,GAE5B,UAAY/6H,KAAU3c,EACrBs6K,GAAe39J,EAAQoT,EAAQ0qE,EAAUi9C,GAK1C,OAFA,GAAmB/sC,EAAc+sC,GAE1Bj9C,CACR,EAUD,SAAS6/E,GACR39J,EACAoT,EACA0qE,EACAi9C,GAEA,GAAqB,WAAhB/6H,EAAOjW,MAAqC,iBAAhBiW,EAAOjW,KAA0B,CACjE,MAAM7S,EAAQ8oB,EAAO9oB,MAAQ,IAAI,GAAO8oB,EAAO9oB,QACzC,YAAEwhC,EAAW,SAAE2kJ,GAAar9J,EAAO9oB,MACnCs9B,EAAUpB,EAAO0qE,SAAS52F,IAAKwxB,GAErColE,EAASp/F,KAAM81B,GAEft9B,EAAMqZ,IAAK,CAAEmoB,gBAEbxhC,EAAM4O,KAAM,aAAczH,GAAIm2B,GAEzB6oJ,GACJnmL,EAAM4O,KAAM,QAASzH,GAAIm2B,EAAS,SAGnCt9B,EAAMqZ,IAAK,CACVw9F,UAAU,GAEZ,CAEAgtC,EAAgBpuI,IAAKqT,EACtB,C,eK9WI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ4Y,OCyBR,MAAMglJ,WAAuB,GAI3C,qBAAkBxoJ,GACjB,MAAO,gBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GAAY,GACtB,CAKO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf6hC,EAAO3F,EAAO6qE,QAAQllE,KAE5Bj6B,KAAKwP,SACJpX,EACA,iBACA,CAAEid,EAAKnZ,IAAU8D,KAAK++K,qBAAsB1pK,EAAKnZ,IACjD,CAAE0Q,SAAU,SAGb5M,KAAKwP,SACJyqB,EAAKz2B,SACL,cACA,CAAE6R,EAAKvS,IAAU9C,KAAKg/K,uBAAwB3pK,EAAKvS,IACnD,CAAE8J,SAAU,SAGb5M,KAAKi/K,4BACLj/K,KAAKk/K,wBACN,CAKO,qBAAA1G,GACN,MAAM3Q,EAAa7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IACtC6sB,EAAYj1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAEvCq5I,EAAgBzG,EAAW2Q,sBAAuBvjJ,GAExD,OAA6B,GAAxBq5I,EAAch2K,OACX,KAQDg2K,CACR,CAKO,sBAAA6Q,GACN,MAAMtX,EAAa7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IACtCkmK,EAAgBtuK,KAAKw4K,wBAE3B,OAAMlK,EAICtuK,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IAChC,MAAMmhD,EAAmBnhD,EAAOkZ,0BAExBjjB,MAAOkhJ,EAAajhJ,KAAMkhJ,GAAe/J,EAAWqG,iBAAkBI,IACtE79I,MAAOwhJ,EAAUvhJ,KAAMmhJ,GAAYhK,EAAW+F,cAAeU,GAE/DI,EAAcJ,EAAe,GAAIhoI,aAAc,SAErD,IAAI84I,EAAkBvN,EAClBwN,EAAqBzN,EAIzB,GAAK/J,EAAWiR,uBAAwBxK,GAAkB,CACzD,MAAMmD,EAAa,CAClBE,cACAC,aACAK,WACAJ,WAGDuN,EAAkB5N,GAAoB9C,EAAa+C,GACnD4N,EAAqBtN,GAAuBrD,EAAa+C,EAC1D,CAEA,MAOM3J,EAAQ2G,GAAuBC,EAPd,CACtBtF,SAAU6I,EACVzI,YAAamI,EACbrI,OAAQ8V,EACR1V,UAAW2V,GAGsD7kJ,GAIlE,OAFAA,EAAO90B,OAAQoiK,EAAOnsF,EAAkB,GAEjCA,CAAgB,IAvChB,IAyCT,CAeO,gBAAA2jG,CAAkBC,EAAqBjF,GAC7C,MAAMkF,EAAgBx/K,KAAKy/K,kBAAmBF,EAAYjF,GAE1Dt6K,KAAKs0B,OAAOl8B,MAAMmiC,QAAQC,IACzBA,EAAOgZ,aACNgsI,EAAcjQ,MAAMvyK,KAAK4tK,GAAQpwI,EAAOsd,cAAe8yH,KACvD,CAAEp8H,SAAUgxI,EAAchxI,UAC1B,GAEH,CAKO,YAAAkxI,GACN,MAEMjoK,EADiB,IADLzX,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UACPmY,aAAcpuC,MACrBitC,sBAE/B,OAAKx0B,GAAWA,EAAQvJ,GAAI,UAAW,aAC/BuJ,EAGD,IACR,CAKO,aAAAkoK,GACN,MAEMloK,EADkB,GADNzX,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UACJmY,aACTnB,sBAEhC,OAAKx0B,GAAWA,EAAQvJ,GAAI,UAAW,aAC/BuJ,EAGD,IACR,CAWQ,yBAAAwnK,GACP,MAAM3qJ,EAASt0B,KAAKs0B,OACdsrJ,EAAgC,IAAIhsK,IAE1C0gB,EAAOj2B,WAAWuhF,IAAK,mBAAoB/xE,KAAK2xD,GAAcA,EAAWnwD,GAAI,aAAa,CAAEgG,EAAKvS,EAAMw8D,KACtG,MAAM+K,EAAa/K,EAAc9kC,QAqBlC,SAAqC6vC,GACpC,UAAYw1G,KAAyBD,EACpCv1G,EAAWtvC,YAAa,+BAAgC8kJ,GAGzDD,EAAYhuK,OACb,CAzBCkuK,CAA4Bz1G,GAE5B,MAAMikG,EAAgBtuK,KAAKw4K,wBAE3B,IAAMlK,EACL,OAGD,UAAY3G,KAAa2G,EAAgB,CACxC,MAAM/2H,EAAc+nB,EAAc9B,OAAOf,cAAekrG,GAExDt9F,EAAWzuC,SAAU,+BAAgC2b,GACrDqoI,EAAY/xK,IAAK0pC,EAClB,CAEA,MAAMwoI,EAAezgH,EAAc9B,OAAOf,cAAe6xG,EAAeA,EAAch2K,OAAS,IAC/F+xE,EAAW72B,aAAcusI,EAAc,EAAG,GACxC,CAAEnzK,SAAU,YAShB,CASQ,sBAAAsyK,GACP,MAAM5qJ,EAASt0B,KAAKs0B,OAEpBt0B,KAAKqP,GAAI,oBAAoB,KAC5B,IAAMrP,KAAK00B,UAAY,CACtB,MAAM45I,EAAgBtuK,KAAKw4K,wBAE3B,IAAMlK,EACL,OAGDh6I,EAAOl8B,MAAMmiC,QAAQC,IACpB,MAAM/U,EAAW+U,EAAOkd,iBAAkB42H,EAAe,GAAK,GACxDxpJ,EAAQwP,EAAOl8B,MAAMsrE,OAAOiE,yBAA0BliD,GAE5D+U,EAAOgZ,aAAc1uB,EAAO,GAE9B,IAEF,CAOQ,oBAAAi6J,CAAsBzvK,EAAkBpT,GAC/C,MAAM2rK,EAAa7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IACtC6sB,EAAY/4B,EAAM,GAClBqI,EAAUrI,EAAM,GAChB9D,EAAQ4H,KAAKs0B,OAAOl8B,MACpB+0C,GAAc5oC,GAAgC,YAArBA,EAAQggB,UACjCw0J,EAAqBlR,EAAW2Q,sBAAuBvjJ,GAEvD8jJ,EAAmBzgL,SAIzBgX,EAAMnD,OAEN/T,EAAMmiC,QAAQC,IACb,MAAMwlJ,EAAoBjH,EAAoB5rI,EAAa4rI,EAAmBzgL,OAAS,EAAI,GAE3FF,EAAMmiC,QAAQC,IACb,UAAYmtI,KAAaoR,EACxB3gL,EAAM29F,cAAev7D,EAAOwd,gBAAiB2vH,EAAW,MACzD,IAGD,MAAMsY,EAAgB7nL,EAAMsrE,OAAOiE,yBAA0BntC,EAAOkd,iBAAkBsoI,EAAmB,IAKpG/qJ,EAAU/mB,GAAI,qBAClBssB,EAAOgZ,aAAcysI,GAErBhrJ,EAAUmN,MAAO69I,EAClB,IAEF,CAkBQ,sBAAAjB,CAAwB3pK,EAAgBvS,GAC/C,MAAMwxB,EAASt0B,KAAKs0B,OACdg6I,EAAgBtuK,KAAKw4K,wBAE3B,IAAMlK,EACL,OAGD,MAAMr0I,EAAO3F,EAAO6qE,QAAQllE,KACtBujC,EAASlpC,EAAO6qE,QAAQ3hC,OACxB/V,EAAa6mH,EAActxK,KAAK2qK,GAAa1tI,EAAK6d,cAAe0lB,EAAOf,cAAekrG,MAE7F7kK,EAAKmyB,UAAYgF,EAAK+d,gBAAiByP,EACxC,CAQQ,iBAAAg4H,CAAmBF,EAAqBjF,GAC/C,MAAMzS,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,cAClDyyK,EAAgBhT,EAAWI,gBAAiBsX,GAC5CzE,EAAcjT,EAAWI,gBAAiBqS,GAE1ClR,EAAWvwK,KAAKD,IAAKiiL,EAAc9R,IAAK+R,EAAY/R,KACpDO,EAASzwK,KAAKC,IAAK+hL,EAAc9R,IAAK+R,EAAY/R,KAElDS,EAAc3wK,KAAKD,IAAKiiL,EAAc7S,OAAQ8S,EAAY9S,QAC1D0B,EAAY7wK,KAAKC,IAAK+hL,EAAc7S,OAAQ8S,EAAY9S,QAGxDkY,EAAsC,IAAIv/K,MAAO2oK,EAASF,EAAW,GAAIt+J,KAAM,MAAO9N,KAAK,IAAM,KAEjGu6J,EAAgB,CACrB6R,WACAE,SACAE,cACAE,aAGD,UAAY,IAAEX,EAAG,KAAE6B,KAAU,IAAI,GAAa2U,EAAWj5I,aAAc,SAAYixH,GAClF2oB,EAAcnX,EAAMK,GAAWxpK,KAAMgrK,GAGtC,MAAMuV,EAAiBrF,EAAY/R,IAAM8R,EAAc9R,IACjDqX,EAAmBtF,EAAY9S,OAAS6S,EAAc7S,OAU5D,OARKmY,GACJD,EAAa70K,UAGT+0K,GACJF,EAAatiL,SAASmrK,GAAOA,EAAI19J,YAG3B,CACNkkK,MAAO2Q,EAAa12G,OACpBh7B,SAAU2xI,GAAkBC,EAE9B,ECvVc,MAAMC,WAAuB,GAI3C,qBAAkB/pJ,GACjB,MAAO,gBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE4nJ,GAAgB,GAC1B,CAKO,IAAApoJ,GACN,MAAMpC,EAASt0B,KAAKs0B,OACdk/B,EAAel/B,EAAO6qE,QAAQllE,KAAKz2B,SAEzCxD,KAAKwP,SAAiCgkD,EAAc,QAAQ,CAAEn+C,EAAKvS,IAAU9C,KAAKsgL,WAAYjrK,EAAKvS,KACnG9C,KAAKwP,SAAgCgkD,EAAc,OAAO,CAAEn+C,EAAKvS,IAAU9C,KAAKsgL,WAAYjrK,EAAKvS,KACjG9C,KAAKwP,SACJ8kB,EAAOl8B,MACP,iBACA,CAAEid,GAAOnV,EAAS+tC,KAAkBjuC,KAAKugL,iBAAkBlrK,EAAKnV,EAAS+tC,IACzE,CAAErhC,SAAU,SAGb5M,KAAKiV,SAAU,wBAChB,CAQQ,UAAAqrK,CAAYjrK,EAAgCvS,GACnD,MAAM09K,EAAiBxgL,KAAKs0B,OAAOqC,QAAQvuB,IAAK02K,IAEhD,IAAM0B,EAAehI,wBACpB,OAGD,GAAiB,OAAZnjK,EAAIrT,OAAkBhC,KAAKs0B,OAAOl8B,MAAMi9B,UAAWr1B,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,WAClF,OAGDnyB,EAAKowB,iBACL7d,EAAIlJ,OAEJ,MAAMs0K,EAAiBzgL,KAAKs0B,OAAOxxB,KAC7B0wD,EAAexzD,KAAKs0B,OAAO6qE,QAAQllE,KAAKz2B,SAExCtD,EAAUugL,EAAetjG,OAAQqjG,EAAerB,0BAEtD3rH,EAAahjD,KAAwC,kBAAmB,CACvEijD,aAAc3wD,EAAK2wD,aACnBvzD,UACAq4B,OAAQljB,EAAIrT,MAEd,CAcQ,gBAAAu+K,CAAkBlrK,EAAgBnV,EAAkC+tC,GAC3E,GAAKA,IAAeA,EAAW//B,GAAI,qBAClC,OAGD,MAAM9V,EAAQ4H,KAAKs0B,OAAOl8B,MACpByvK,EAAa7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IAG5C,IAAIs4K,EAAc1gL,KAAK2gL,6BAA8BzgL,EAAS9H,GAE9D,IAAMsoL,EACL,OAGD,MAAM3H,EAAqBlR,EAAW2F,+BAAgCp1K,EAAMoL,SAASyxB,WAE/E8jJ,EAAmBzgL,QAOzB+c,EAAIlJ,OAEJ/T,EAAMmiC,QAAQC,IACb,MAAMomJ,EAAmB,CACxB/9J,MAAOglJ,EAAW+I,WAAY8P,GAC9B59J,OAAQ+kJ,EAAW0E,QAASmU,IAIvBzrJ,EA2NT,SACC8jJ,EACA6H,EAIApmJ,EACAqtI,GAEA,MAAMgZ,EAAgB9H,EAAoB,GAAIzyI,aAAc,SAEtD2nI,EAAgBpG,EAAWqG,iBAAkB6K,GAC7CpL,EAAa9F,EAAW+F,cAAemL,GAEvC9jJ,EAAY,CACjB08I,YAAa1D,EAAcx9I,MAC3BmhJ,WAAY3D,EAAcv9I,KAC1BuhJ,SAAUtE,EAAWl9I,MACrBohJ,QAASlE,EAAWj9I,MAIfowJ,EAAsD,IAA9B/H,EAAmBzgL,OAE5CwoL,IACJ7rJ,EAAU48I,SAAW+O,EAAiB99J,OAAS,EAC/CmS,EAAU28I,YAAcgP,EAAiB/9J,MAAQ,EAsCnD,SAA0BilJ,EAAgBiZ,EAAwBC,EAAuBnZ,GACxF,MAAMoZ,EAAapZ,EAAW+I,WAAY9I,GACpCoZ,EAAcrZ,EAAW0E,QAASzE,GAEnCkZ,EAAgBC,GACpBpZ,EAAWsG,cAAerG,EAAO,CAChCgG,GAAImT,EACJ7S,QAAS4S,EAAgBC,IAItBF,EAAiBG,GACrBrZ,EAAWgG,WAAY/F,EAAO,CAC7BgG,GAAIoT,EACJ9Y,KAAM2Y,EAAiBG,GAG1B,CArDEC,CAAiBN,EAAe5rJ,EAAU48I,QAAU,EAAG58I,EAAU28I,WAAa,EAAG/J,IAK7EiZ,IAA0BjZ,EAAWiR,uBAAwBC,GAuInE,SAA2CjR,EAAgB2J,EAAoCj3I,GAC9F,MAAM,SAAEy3I,EAAQ,QAAEJ,EAAO,YAAEF,EAAW,WAAEC,GAAeH,EAEjD9D,EAAa,CAAEl9I,MAAOwhJ,EAAUvhJ,KAAMmhJ,GACtC5D,EAAgB,CAAEx9I,MAAOkhJ,EAAajhJ,KAAMkhJ,GAGlDwP,GAAiBtZ,EAAO6J,EAAahE,EAAYnzI,GACjD4mJ,GAAiBtZ,EAAO8J,EAAa,EAAGjE,EAAYnzI,GAGpD6mJ,GAAmBvZ,EAAOmK,EAAUhE,EAAezzI,GACnD6mJ,GAAmBvZ,EAAO+J,EAAU,EAAG5D,EAAezzI,EAAQy3I,EAC/D,CAhJEqP,CAAkCT,EAAe5rJ,EAAWuF,IAiB5DvF,EAAU48I,QAAUL,GAAoBqP,EAAe5rJ,GACvDA,EAAU28I,WAAaG,GAAuB8O,EAAe5rJ,IAG9D,OAAOA,CACR,CAtRqBssJ,CAAwBxI,EAAoB6H,EAAkBpmJ,EAAQqtI,GAIlF2Z,EAAkBvsJ,EAAU48I,QAAU58I,EAAUg9I,SAAW,EAC3DwP,EAAiBxsJ,EAAU28I,WAAa38I,EAAU08I,YAAc,EAShEhD,EAAiB,CACtBvF,SAAU,EACVI,YAAa,EACbF,OAAQzwK,KAAKD,IAAK4oL,EAAiBZ,EAAiB99J,QAAW,EAC/D4mJ,UAAW7wK,KAAKD,IAAK6oL,EAAgBb,EAAiB/9J,OAAU,GAGjE69J,EAAcjS,GAAuBiS,EAAa/R,EAAgBn0I,GAGlE,MAAMqmJ,EAAgB9H,EAAoB,GAAIzyI,aAAc,SAEtDk5I,EAAgBx/K,KAAK0hL,gCAAiChB,EAAaE,EAAkBC,EAAe5rJ,EAAWuF,GAErH,GAAKx6B,KAAKs0B,OAAOqC,QAAQvuB,IAAK,kBAAmBssB,UAAY,CAG5D,MAAM6zG,EAAkBs/B,EAAW4Q,WAAY+G,EAAcxiL,KAAK4tK,GAAQpwI,EAAOsd,cAAe8yH,MAEhGpwI,EAAOgZ,aAAc+0F,EACtB,MAEC/tG,EAAOgZ,aAAcgsI,EAAe,GAAK,EAC1C,KApDAjO,GAAwBmP,EAAa7Y,EAsDvC,CAKQ,+BAAA6Z,CACPhB,EACAE,EACAC,EACA5rJ,EACAuF,GAEA,MAAQ3X,MAAO8+J,EAAa7+J,OAAQ8+J,GAAiBhB,EAG/CiB,EA0RR,SAA4B/Z,EAAgBjlJ,EAAeC,GAE1D,MAAM9lB,EAAM,IAAI2D,MAAOmiB,GAAShY,KAAM,MACpC9N,KAAK,IAAM,IAAI2D,MAAOkiB,GAAQ/X,KAAM,QAEtC,UAAY,OAAEk9J,EAAM,IAAEe,EAAG,KAAE6B,KAAU,IAAI,GAAa9C,GACrD9qK,EAAK+rK,GAAOf,GAAW4C,EAGxB,OAAO5tK,CACR,CApSiC8kL,CAAmBpB,EAAaiB,EAAaC,GAEtEG,EAAmB,IAAK,IAAI,GAAalB,EAAe,CAC7DzX,SAAUn0I,EAAUg9I,SACpB3I,OAAQr0I,EAAU48I,QAClBrI,YAAav0I,EAAU08I,YACvBjI,UAAWz0I,EAAU28I,WACrBhI,iBAAiB,KAIZ4V,EAAgC,GAGtC,IAAI9lH,EAQJ,UAAYmzG,KAAakV,EAAmB,CAC3C,MAAM,IAAEhZ,EAAG,OAAEf,GAAW6E,EAGnB7E,IAAW/yI,EAAU08I,cACzBj4G,EAAiBmzG,EAAUV,qBAI5B,MAAM6V,EAAYjZ,EAAM9zI,EAAUg9I,SAC5BgQ,EAAeja,EAAS/yI,EAAU08I,YAClCuQ,EAAaL,EAAwBG,EAAYJ,GAAgBK,EAAeN,GAIhFQ,EAAeD,EAAa1nJ,EAAOg4D,aAAc0vF,GAAe,KAGhEE,EAAepiL,KAAKqiL,sBAAuBxV,EAAWsV,EAAczoH,EAAiBl/B,GAGrF4nJ,IAKNjT,GAAuBiT,EAAcrZ,EAAKf,EAAQ/yI,EAAU48I,QAAS58I,EAAU28I,WAAYp3I,GAE3FglJ,EAAc5/K,KAAMwiL,GAEpB1oH,EAAiBl/B,EAAOmd,oBAAqByqI,GAC9C,CAGA,MAAM/Z,EAAcjrK,SAAUyjL,EAAcxlJ,aAAc,gBAA6B,KACjF0sI,EAAiB3qK,SAAUyjL,EAAcxlJ,aAAc,mBAAgC,KAEvFinJ,EAAsCrtJ,EAAUg9I,SAAW5J,GAAeA,GAAepzI,EAAU48I,QACnG0Q,EAAyCttJ,EAAU08I,YAAc5J,GAAkBA,GAAkB9yI,EAAU28I,WAErH,GAAK0Q,EAAsC,CAC1C,MACME,EAAWnB,GAAmBR,EAAexY,EAD9B,CAAE53I,MAAOwE,EAAU08I,YAAajhJ,KAAMuE,EAAU28I,YACSp3I,EAAQvF,EAAUg9I,UAEhGuN,EAAc5/K,QAAS4iL,EACxB,CAEA,GAAKD,EAAyC,CAC7C,MACMC,EAAWpB,GAAiBP,EAAe9Y,EAD/B,CAAEt3I,MAAOwE,EAAUg9I,SAAUvhJ,KAAMuE,EAAU48I,SACar3I,GAE5EglJ,EAAc5/K,QAAS4iL,EACxB,CAEA,OAAOhD,CACR,CAQO,qBAAA6C,CACNxV,EACAsV,EACAzoH,EACAl/B,GAEA,MAAM,KAAEowI,EAAI,SAAEmB,GAAac,EAW3B,OALKd,GACJvxI,EAAOv0B,OAAQ2kK,GAIVuX,GAIN3nJ,EAAO90B,OAAQy8K,EAAczoH,GAEtByoH,GALC,IAMT,CAQO,4BAAAxB,CAA8BzgL,EAAkC9H,GACtE,IAAM8H,EAAQgO,GAAI,sBAAyBhO,EAAQgO,GAAI,WACtD,OAAO,KAIR,GAAKhO,EAAQgO,GAAI,UAAW,SAC3B,OAAOhO,EAKR,GAA2B,GAAtBA,EAAQo7B,YAAmBp7B,EAAQ27B,SAAU,GAAK3tB,GAAI,UAAW,SACrE,OAAOhO,EAAQ27B,SAAU,GAK1B,MAAM4mJ,EAAerqL,EAAM2/C,cAAe73C,GAE1C,UAAYuX,KAAWgrK,EAAav2I,WACnC,GAAKz0B,EAAQvJ,GAAI,UAAW,SAAY,CAEvC,MAAMw0K,EAActqL,EAAMy/C,YAAa4qI,EAAa/hJ,MAAOtoC,EAAMw/C,qBAAsBngC,IAEvF,GAAKrf,EAAMylF,WAAY6kG,EAAa,CAAE5kG,mBAAmB,IACxD,OAAO,KAIR,MAAM6kG,EAAavqL,EAAMy/C,YAAaz/C,EAAMu/C,oBAAqBlgC,GAAWgrK,EAAa9hJ,KAEzF,OAAKvoC,EAAMylF,WAAY8kG,EAAY,CAAE7kG,mBAAmB,IAChD,KAIDrmE,CACR,CAGD,OAAO,IACR,EA6LD,SAAS4pK,GAAmBvZ,EAAgB4H,EAAkBkT,EAAsCpoJ,EAAgB4uI,EAAmB,GAEtI,GAAKsG,EAAW,EACf,OAQD,OALyBL,GAA+BvH,EAAO4H,EAAUtG,GAGnCrjK,QAAQ,EAAIiiK,SAAQgE,eAAiB6W,GAAuB7a,EAAQgE,EAAW4W,KAEjG5lL,KAAK,EAAI4tK,UAAY6E,GAAmB7E,EAAM8E,EAAUl1I,IAC7E,CAEA,SAAS4mJ,GAAiBtZ,EAAgBsI,EAAqB0S,EAAmCtoJ,GAEjG,GAAK41I,EAAc,EAClB,OAQD,OALyBJ,GAAiClI,EAAOsI,GAG3BrqK,QAAQ,EAAIgjK,MAAKkD,gBAAkB4W,GAAuB9Z,EAAKkD,EAAY6W,KAE7F9lL,KAAK,EAAI4tK,OAAM5C,YAAcmI,GAAiBvF,EAAM5C,EAAQoI,EAAa51I,IAC9F,CAOA,SAASqoJ,GAAuBj+K,EAAek2C,EAAc63E,GAC5D,MAAMowD,EAAWn+K,EAAQk2C,EAAO,GAC1B,MAAErqB,EAAK,KAAEC,GAASiiG,EAKxB,OAH0B/tH,GAAS6rB,GAAS7rB,GAAS8rB,GAChB9rB,EAAQ6rB,GAASsyJ,GAAYtyJ,CAGnE,CC1jBe,MAAMuyJ,WAAsB,GAI1C,qBAAkB1sJ,GACjB,MAAO,eACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE4nJ,GAAgB,GAC1B,CAKO,IAAApoJ,GACN,MACM88B,EADOxzD,KAAKs0B,OAAO6qE,QAAQllE,KACPz2B,SAE1BxD,KAAKwP,SACJgkD,EACA,YACA,IAAKt3D,IAAU8D,KAAKijL,eAAgB/mL,IACpC,CAAEqR,QAAS,UAGZvN,KAAKwP,SACJgkD,EACA,OACA,IAAKt3D,IAAU8D,KAAKkjL,6BAA8BhnL,IAClD,CAAEqR,QAAS,WAGZvN,KAAKwP,SACJgkD,EACA,OACA,IAAKt3D,IAAU8D,KAAKmjL,cAAejnL,IACnC,CAAEqR,QAAS,CAAE,KAAM,OAErB,CAMQ,yBAAA21K,CAA2BE,EAAsCvjD,GACxE,MAAMvrG,EAASt0B,KAAKs0B,OAEd4b,EADY5b,EAAOl8B,MAAMoL,SAASyxB,UACN+Y,qBAE5BkC,GAAoBA,EAAgBhiC,GAAI,UAAW,WAIzD2xH,EAAa3sG,iBACb2sG,EAAa1sG,kBACbiwJ,EAAkBj3K,OAElBmoB,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAOgZ,aAAchZ,EAAOud,cAAiB7H,EAAgBrU,SAAU,GAAiBA,SAAU,IAAkB,IAEtH,CAMQ,UAAAsnJ,CAAYC,EAAsCvjD,GACzD,MAAMvrG,EAASt0B,KAAKs0B,OACduzI,EAAyB7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IAClDo4K,EAAiCxgL,KAAKs0B,OAAOqC,QAAQvuB,IAAK,kBAE1D6sB,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAClC+mE,GAAa6jC,EAAapyG,SAEhC,IAAIk6I,EAA4BE,EAAW0K,iCAAkCt9I,GAAa,GAM1F,GAJM0yI,IACLA,EAAY6Y,EAAed,iBAGtB/X,EACL,OAGD9nC,EAAa3sG,iBACb2sG,EAAa1sG,kBACbiwJ,EAAkBj3K,OAElB,MAAMygK,EAAWjF,EAAU/oK,OACrBkpK,EAAQ8E,EAAShuK,OAEjBykL,EAAkBvb,EAAM9rI,cAAe4wI,GACvC0W,EAAmB1W,EAAS5wI,cAAe2rI,GAE3C4b,EAAwC,IAArBD,EAEzB,IAAMtnF,GAAaunF,GAAwC,IAApBF,EAMtC,YAJA/uJ,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAOgZ,aAAchZ,EAAOsd,cAAegwH,GAAS,IAMtD,MAAM0b,EAAkBF,IAAqB1W,EAAStxI,WAAa,EAC7DmoJ,EAAYJ,IAAoBxb,EAAW0E,QAASzE,GAAU,EAEpE,GAAK9rE,GAAaynF,GAAaD,IAC9BlvJ,EAAOiB,QAAS,uBAIX8tJ,IAAoBxb,EAAW0E,QAASzE,GAAU,GAKtD,YAJAxzI,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAOgZ,aAAchZ,EAAOsd,cAAegwH,GAAS,IAOvD,IAAI8M,EAGJ,GAAK54E,GAAawnF,EAAkB,CACnC,MAAME,EAAU5b,EAAMjsI,SAAUwnJ,EAAkB,GAElDzO,EAAc8O,EAAQ7nJ,SAAU,EACjC,UAEWmgE,GAAaunF,EAAmB,CAC1C,MAAMI,EAAc7b,EAAMjsI,SAAUwnJ,EAAkB,GAEtDzO,EAAc+O,EAAY9nJ,SAAU8nJ,EAAYroJ,WAAa,EAC9D,MAGCs5I,EAAchI,EAAS/wI,SAAUynJ,GAAqBtnF,EAAY,GAAK,IAGxE1nE,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAOgZ,aAAchZ,EAAOud,cAAe68H,GAAe,GAE5D,CAKQ,WAAAqO,CAAavyK,EAAsBmvH,GAC1C,MAAMvrG,EAASt0B,KAAKs0B,OAGd/P,EAAY,GAFFs7G,EAAavyG,QAEiCgH,EAAO4E,OAAO3K,0BACzDvuB,KAAK4jL,iBAAkBr/J,EAAWs7G,EAAapyG,YAGjEoyG,EAAa3sG,iBACb2sG,EAAa1sG,kBACbziB,EAAUvE,OAEZ,CASQ,gBAAAy3K,CAAkBr/J,EAAkCo9G,GAC3D,MAAMkmC,EAAa7nK,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IACtCo4K,EAAiCxgL,KAAKs0B,OAAOqC,QAAQvuB,IAAK,kBAC1DhQ,EAAQ4H,KAAKs0B,OAAOl8B,MACpB68B,EAAY78B,EAAMoL,SAASyxB,UAC3B+mE,EAAY,CAAE,QAAS,QAASzsE,SAAUhL,GAI1C+pJ,EAAgBzG,EAAW2Q,sBAAuBvjJ,GAExD,GAAKq5I,EAAch2K,OAAS,CAC3B,IAAIurL,EAUJ,OAPCA,EADIliD,EACQ6+C,EAAed,eAEf1jF,EAAYsyE,EAAeA,EAAch2K,OAAS,GAAMg2K,EAAe,GAGpFtuK,KAAK8jL,6BAA8BD,EAAYt/J,EAAWo9G,IAEnD,CACR,CAGA,MAAMgmC,EAAY1yI,EAAUgY,MAAO3G,aAAc,aAGjD,IAAMqhI,EACL,OAAO,EAIR,IAAM1yI,EAAU+V,YACf,GAAK22F,GAMJ,GAAK1sG,EAAUkY,YAAc6uD,IAAc/mE,EAAUmuC,sBAAuBukG,GAC3E,OAAO,MAEF,CACN,MAAMz3H,EAAkBjb,EAAU+Y,qBAGlC,IAAMkC,IAAoB93C,EAAMsrE,OAAOgE,SAAUx3B,GAChD,OAAO,CAET,CAID,QAAKlwC,KAAK+jL,uBAAwB9uJ,EAAW0yI,EAAW3rE,KACvDh8F,KAAK8jL,6BAA8Bnc,EAAWpjJ,EAAWo9G,IAElD,EAIT,CASQ,sBAAAoiD,CAAwB9uJ,EAA0C0yI,EAAoB3rE,GAC7F,MAAM5jG,EAAQ4H,KAAKs0B,OAAOl8B,MACpBsrE,EAAS1jE,KAAKs0B,OAAOl8B,MAAMsrE,OAE3Bz2B,EAAQ+uD,EAAY/mE,EAAUuY,kBAAqBvY,EAAUE,mBAInE,IAAMuuC,EAAOyO,gBAAiBllC,GAAQ/+B,GAAI,UAAW,aAAgB,CAGpE,OAFyB9V,EAAMs/C,iBAAkBiwH,EAAW3rE,EAAY,MAAQ,GAExD9jC,WAAYjrB,EACrC,CAEA,MAAMk0F,EAAQ/oI,EAAM4/C,gBAAiB/K,GAKrC,OAHA70C,EAAMo+F,gBAAiB2qC,EAAO,CAAE58G,UAAWy3E,EAAY,UAAY,aAG5D/uD,EAAMhpB,QAASk9G,EAAMl0F,MAC7B,CASU,4BAAA62I,CAA8BD,EAAoBt/J,EAAkCo9G,GAAkB,GAC/G,MAAMvpI,EAAQ4H,KAAKs0B,OAAOl8B,MAEpB0vK,EAAQ+b,EAAUv9I,aAAc,SAChCwoI,EAAW,IAAK,IAAI,GAAahH,EAAO,CAAE8B,iBAAiB,MACzDb,IAAK8I,EAAS7J,OAAQ4J,GAAe9C,EAAUA,EAASx2K,OAAS,GAEnE0rL,EAAkBlV,EAASv9I,MAAM,EAAIq5I,UAAYA,GAAQiZ,IAC/D,IAAI,IAAE9a,EAAG,OAAEf,GAAWgc,EAEtB,OAASz/J,GACR,IAAK,OACJyjJ,IACA,MAED,IAAK,KACJe,IACA,MAED,IAAK,QACJf,GAAUgc,EAAgBhY,UAC1B,MAED,IAAK,OACJjD,GAAOib,EAAgB/X,WAWzB,GAP4BlD,EAAM,GAAKA,EAAM8I,GACnB7J,EAAS,GAAKe,GAAO,GACvBf,EAAS4J,GAAc7I,GAAO8I,EAUrD,YAJAz5K,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAchZ,EAAOsd,cAAegwH,GAAS,IAMjDE,EAAS,GACbA,EAASrmC,EAAkB,EAAIiwC,EAC/B7I,KACWf,EAAS4J,IACpB5J,EAASrmC,EAAkBiwC,EAAa,EACxC7I,KAGD,MAAMkb,EAAenV,EAASv9I,MAAMwpJ,GAAYA,EAAShS,KAAOA,GAAOgS,EAAS/S,QAAUA,IAAU4C,KAC9F5uE,EAAY,CAAE,QAAS,QAASzsE,SAAUhL,GAC1Ci8J,EAAiCxgL,KAAKs0B,OAAOqC,QAAQvuB,IAAK,kBAEhE,GAAKu5H,GAAmB6+C,EAAe9rJ,UAAY,CAClD,MAAM6qJ,EAAaiB,EAAeb,iBAAmBkE,EAErDrD,EAAelB,iBAAkBC,EAAY0E,EAC9C,KAAO,CACN,MAAMC,EAAmB9rL,EAAMs/C,iBAAkBusI,EAAcjoF,EAAY,EAAI,OAE/E5jG,EAAMmiC,QAAQC,IACbA,EAAOgZ,aAAc0wI,EAAkB,GAEzC,CACD,EC/Vc,MAAM,WAA4B/4H,GAAjD,kCACC,KAAgBC,aAAe,CAC9B,YAAa,aACd,CAKO,UAAAC,CAAYJ,GAClBjrD,KAAKwQ,KAAMy6C,EAAShgD,KAAMggD,EAC3B,ECdc,MAAMk5H,WAAmB,GAIvC,qBAAkB7tJ,GACjB,MAAO,YACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE4nJ,GAAgB,GAC1B,CAKO,IAAApoJ,GACS12B,KAAKs0B,OAIb6qE,QAAQllE,KAAKu6B,YAAa,IAEjCx0D,KAAKokL,6BACLpkL,KAAKqkL,2BACN,CAMQ,0BAAAD,GACP,MAAM9vJ,EAASt0B,KAAKs0B,OACduzI,EAAavzI,EAAOqC,QAAQvuB,IAAK,IACvC,IAAIk8K,GAAuB,EAE3B,MAAM9D,EAAiBlsJ,EAAOqC,QAAQvuB,IAAK02K,IAE3C9+K,KAAKwP,SAAU8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,aAAa,CAAE6R,EAAKwqH,KAChE,MAAM5qG,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAExC,IAAMj1B,KAAK00B,YAAc8rJ,EAAe9rJ,UACvC,OAGD,IAAMmrG,EAAa50E,SAASx9B,SAC3B,OAGD,MAAM8xJ,EAAaiB,EAAeb,iBAAmB9X,EAAW0K,iCAAkCt9I,GAAa,GAE/G,IAAMsqJ,EACL,OAGD,MAAMjF,EAAat6K,KAAKukL,+BAAgC1kD,GAEnDy6C,GAAckK,GAAqBjF,EAAYjF,KACnDgK,GAAuB,EACvB9D,EAAelB,iBAAkBC,EAAYjF,GAE7Cz6C,EAAa3sG,iBACd,IAGDlzB,KAAKwP,SAAU8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,WAAW,KACvD8gL,GAAuB,CAAK,IAmB7BtkL,KAAKwP,SAAU8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,mBAAmB6R,IAC1DivK,GAGJjvK,EAAIlJ,MACL,GACE,CAAES,SAAU,WAChB,CAUQ,yBAAAy3K,GACP,MAAM/vJ,EAASt0B,KAAKs0B,OACpB,IAAIirJ,EAA4BjF,EAC5BmK,GAAqB,EACrBH,GAAuB,EAE3B,MAAM9D,EAAiBlsJ,EAAOqC,QAAQvuB,IAAK02K,IAE3C9+K,KAAKwP,SAAU8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,aAAa,CAAE6R,EAAKwqH,KAC1D7/H,KAAK00B,WAAc8rJ,EAAe9rJ,YAKnCmrG,EAAa50E,SAASx9B,UAAYoyG,EAAa50E,SAASz9B,SAAWqyG,EAAa50E,SAAS19B,SAI9FgyJ,EAAav/K,KAAKukL,+BAAgC1kD,IAAc,IAGjE7/H,KAAKwP,SAAU8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,aAAa,CAAE6R,EAAKwqH,KAChE,IAAMA,EAAa50E,SAASqnD,QAC3B,OAGD,IAAMitE,EACL,OAGD,MAAMmF,EAAgB1kL,KAAKukL,+BAAgC1kD,GAEtD6kD,GAAiBF,GAAqBjF,EAAYmF,KACtDpK,EAAaoK,EAIPD,GAAsBnK,GAAciF,IACzCkF,GAAqB,IAKjBA,IAINH,GAAuB,EACvB9D,EAAelB,iBAAkBC,EAAYjF,GAE7Cz6C,EAAa3sG,iBAAgB,IAG9BlzB,KAAKwP,SAAU8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,WAAW,KACvDihL,GAAqB,EACrBH,GAAuB,EACvB/E,EAAa,KACbjF,EAAa,IAAI,IAIlBt6K,KAAKwP,SAAU8kB,EAAO6qE,QAAQllE,KAAKz2B,SAAU,mBAAmB6R,IAC1DivK,GAGJjvK,EAAIlJ,MACL,GACE,CAAES,SAAU,WAChB,CAOQ,8BAAA23K,CAAgC1kD,GAEvC,MAAM8kD,EAAoB9kD,EAAaj8H,OACjC+uC,EAAe3yC,KAAKs0B,OAAO6qE,QAAQllE,KAAKyd,iBAAkBitI,EAAmB,GAInF,OAHsB3kL,KAAKs0B,OAAO6qE,QAAQ3hC,OAAOJ,gBAAiBzqB,GAC/B/zC,OAEf0nC,aAAc,YAAa,CAAEjK,aAAa,GAC/D,EAGD,SAASmoJ,GAAqBI,EAAgBC,GAC7C,OAAOD,EAAMhmL,OAAQA,QAAUimL,EAAMjmL,OAAQA,MAC9C,C,eCjNI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQk7B,OCIhB,SAASgrJ,GAAwB7vJ,GACvC,MAAMsiB,EAActiB,EAAU+Y,qBAE9B,OAAKuJ,GAAewtI,GAAextI,GAC3BA,EAGD,IACR,CAKO,SAAS,GAAwBtiB,GACvC,MAAMspB,EAAoBtpB,EAAUE,mBAEpC,IAAMopB,EACL,OAAO,KAGR,IAAI3/C,EAAiD2/C,EAAkB3/C,OAEvE,KAAQA,GAAS,CAChB,GAAKA,EAAOsP,GAAI,YAAe62K,GAAenmL,GAC7C,OAAOA,EAGRA,EAASA,EAAOA,MACjB,CAEA,OAAO,IACR,CAKA,SAASmmL,GAAextI,GACvB,QAASA,EAAY/Q,kBAAmB,UAAa,GAAU+Q,EAChE,C,eCnDI,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQzd,O,iMCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,OCIvB,MAAMkrJ,GAA4BzrE,GAAiBtB,iBAGlD+sE,GAA0BzsE,gBAC1BysE,GAA0BtsE,oBAC1BssE,GAA0BrsE,oBAC1BqsE,GAA0B9sE,gBAC1B8sE,GAA0B3sE,oBAC1B2sE,GAA0B1sE,oBAC1B0sE,GAA0BpsE,oB,iQCtBvB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQ9+E,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,O,eCTnB,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQA,O,0VCEvB,MAAM,GAAkB,CAAEmrJ,aAAa,GACjCC,GAAqC,KAM5B,MAAMC,WAAchyK,KA4BlC,WAAApR,CAAaqjL,EAAkC7gL,EAAwB,CAAC,GAGvE,GAFAiJ,SAEM43K,EAML,MAAM,IAAI,EACT,0BACAplL,MAIGuE,EAAQ8gL,WACZrlL,KAAKslL,oBAAqB/gL,EAAQ8gL,WAGnCrlL,KAAKyR,IAAK,QAASlN,EAAQ8gL,WAG1BrlL,KAAK6zE,SADiC,mBAA3BuxG,EACKA,EAEA,KAAMG,OAyJKlyC,EAzJgB+xC,EA0JtC,IAAI3sJ,SAAiB,CAAElc,EAASyuH,KACtC,MAAMyD,EAAM,IAAIC,eAEhBD,EAAInB,KAAM,MAAO+F,GAEjB5E,EAAIn+H,iBAAkB,QAAQ,KAC7B,MAAM2nI,EAAaxJ,EAAI3B,OACjB04C,EAAc/2C,EAAII,SAExB,OAAKoJ,EAAa,KAAOA,EAAa,IAM9BjN,EACN,IAAI,EAAe,kCAAmC,OAIjDzuH,EAASipK,EAAa,IAG9B/2C,EAAIn+H,iBAAkB,SAAS,IAAM06H,EAAQ,IAAI3yI,MAAO,oBACxDo2I,EAAIn+H,iBAAkB,SAAS,IAAM06H,EAAQ,IAAI3yI,MAAO,YAExDo2I,EAAIQ,MAAM,IA3BZ,IAA8BoE,CAzJwC,EAGpErzI,KAAKygB,SAAW,SAAK,IAAoBlc,EAC1C,CAKO,IAAAmyB,GACN,OAAO,IAAI+B,SAAS,CAAElc,EAASyuH,KACxBhrI,KAAKxH,OAQNwH,KAAKygB,SAASwkK,aAClBjlL,KAAKylL,+BAGNlpK,EAASvc,OAXRA,KAAK0lL,eACH5tJ,KAAMvb,GACNgvG,MAAOyf,EASyB,GAErC,CAKO,YAAA06C,GACN,OAAO1lL,KAAK6zE,WACV/7C,MAAMt/B,IACNwH,KAAKslL,oBAAqB9sL,GAC1BwH,KAAKyR,IAAK,QAASjZ,GAEdwH,KAAKygB,SAASwkK,aAClBjlL,KAAKylL,+BAGCzlL,OAEV,CAKO,OAAAgmB,GACN6M,aAAc7yB,KAAK2lL,qBACpB,CAOQ,mBAAAL,CAAqBM,GAE5B,MAAMC,EAAiC,iBAAfD,EAGlBE,GAAiB,SAAS7jL,KAAM2jL,GAIhCG,EAAcF,GAA+C,IAAnCD,EAAW7oL,MAAO,KAAMzE,OAExD,IAAQwtL,IAAiBC,EAMxB,MAAM,IAAI,EAAe,0BAA2B/lL,KAEtD,CAKQ,4BAAAylL,GACP,MAAMO,EAA0BhmL,KAAKimL,8BAErCpzJ,aAAc7yB,KAAK2lL,sBAEnB3lL,KAAK2lL,qBAAuB5yJ,YAAY,KACvC/yB,KAAK0lL,cAAc,GACjBM,EACJ,CAOQ,2BAAAC,GACP,IACC,MAAQ,CAAE9vC,GAAuBn2I,KAAKxH,MAAOuE,MAAO,MAC5CmpL,IAAKC,GAAoBvjL,KAAK+7E,MAAOj7E,KAAMyyI,IAEnD,IAAMgwC,EACL,OAAOjB,GAKR,OAFgCrsL,KAAK+B,OAA6B,IAAlBurL,EAA2Bt3K,KAAK2yB,OAAU,EAG3F,CAAE,MAAQhgC,GACT,OAAO0jL,EACR,CACD,CAQA,aAAc5nK,CAAQ8nK,EAAkC7gL,EAAwB,CAAC,GAGhF,OAFc,IAAI4gL,GAAOC,EAAwB7gL,GAEpCmyB,MACd,ECpLD,MAAM0vJ,GAAwB,uBAKf,MAAMC,WAAqBj3K,KAyBzC,WAAArN,CACCukL,EACAnoE,EACAooE,GAIA,GAFA/4K,SAEM84K,EAML,MAAM,IAAI,EAAe,4BAA6B,MAGvD,IAAMnoE,EAML,MAAM,IAAI,EAAe,6BAA8B,MAGxD,IAAMooE,EAML,MAAM,IAAI,EAAe,mCAAoC,MAG9DvmL,KAAK8qI,KA8LP,SAAoBruI,GACnB,GAAuB,iBAAXA,EACX,OAAO,EAGR,MAAMI,EAAQJ,EAAOI,MAAOupL,IAC5B,SAAWvpL,IAASA,EAAMvE,OAC3B,CArMckuL,CAAWF,GA8JzB,SAAwB7jL,EAAgBgkL,EAAY,KACnD,IACC,MAAMC,EAAcjkL,EAAO5F,MAAOupL,IAA0B,GACtDO,EAAajjL,KAAMjB,EAAOwW,QAASmtK,GAAuB,KAE1DQ,EAAgC,GAEtC,QAAUhrK,EAAS,EAAGA,EAAS+qK,EAAWruL,OAAQsjB,GAAU6qK,EAAY,CACvE,MAAM3kL,EAAQ6kL,EAAW7kL,MAAO8Z,EAAQA,EAAS6qK,GAC3CI,EAAc,IAAIlmL,MAAemB,EAAMxJ,QAE7C,QAAUkC,EAAI,EAAGA,EAAIsH,EAAMxJ,OAAQkC,IAClCqsL,EAAarsL,GAAMsH,EAAMmrB,WAAYzyB,GAGtCosL,EAAWhnL,KAAM,IAAI8c,WAAYmqK,GAClC,CAEA,OAAO,IAAIC,KAAMF,EAAY,CAAE37K,KAAMy7K,GACtC,CAAE,MAAQ98K,GAMT,MAAM,IAAI,EAAe,yCAA0C,KACpE,CACD,CAzLwCm9K,CAAeT,GAAeA,EAEpEtmL,KAAKs4I,OAASn6B,EACdn+G,KAAKgnL,YAAcT,CACpB,CAKO,UAAAU,CAAY13K,GAGlB,OAFAvP,KAAKqP,GAAoC,YAAY,CAAEC,EAAOxM,IAAUyM,EAAUzM,KAE3E9C,IACR,CAKO,OAAAknL,CAAS33K,GAGf,OAFAvP,KAAKyP,KAA8B,SAAS,CAAEH,EAAOxM,IAAUyM,EAAUzM,KAElE9C,IACR,CAKO,KAAAqrI,GACNrrI,KAAKyuI,IAAKpD,OACX,CAKO,IAAA4D,GAIN,OAHAjvI,KAAKmnL,kBACLnnL,KAAKonL,sBAEEpnL,KAAKwuI,cACb,CAKQ,eAAA24C,GACP,MAAM14C,EAAM,IAAIC,eAEhBD,EAAInB,KAAM,OAAQttI,KAAKgnL,aACvBv4C,EAAIsJ,iBAAkB,gBAAiB/3I,KAAKs4I,OAAO9/I,OACnDi2I,EAAIE,aAAe,OAEnB3uI,KAAKyuI,IAAMA,CACZ,CAKQ,mBAAA24C,GACP,MAAM34C,EAAMzuI,KAAKyuI,IAEXy4C,EAAY94K,GACV,IAAMpO,KAAKwQ,KAA8B,QAASpC,GAG1DqgI,EAAIn+H,iBAAkB,QAAS42K,EAAS,kBACxCz4C,EAAIn+H,iBAAkB,QAAS42K,EAAS,UAGnCz4C,EAAI1B,QACR0B,EAAI1B,OAAOz8H,iBAAkB,YAAYhB,IACnCA,EAAMy/H,kBACV/uI,KAAKwQ,KAAsC,WAAY,CACtDu6H,MAAOz7H,EAAMy7H,MACbW,SAAUp8H,EAAMs7H,QAElB,IAIF6D,EAAIn+H,iBAAkB,QAAQ,KAC7B,MAAM2nI,EAAaxJ,EAAI3B,OACjB04C,EAAc/2C,EAAII,SAExB,GAAKoJ,EAAa,KAAOA,EAAa,IACrC,OAAOj4I,KAAKwQ,KAA8B,QAASg1K,EAAYp3K,SAAWo3K,EAAY57K,MACvF,GAEF,CAKQ,YAAA4kI,GACP,MAAMoJ,EAAW,IAAI5I,SACfP,EAAMzuI,KAAKyuI,IAIjB,OAFAmJ,EAAS/xF,OAAQ,OAAQ7lD,KAAK8qI,MAEvB,IAAIryG,SAAyB,CAAElc,EAASyuH,KAC9CyD,EAAIn+H,iBAAkB,QAAQ,KAC7B,MAAM2nI,EAAaxJ,EAAI3B,OACjB04C,EAAc/2C,EAAII,SAExB,OAAKoJ,EAAa,KAAOA,EAAa,IAChCutC,EAAYp3K,QAMT48H,EAAQ,IAAI,EAClB,qCACAhrI,KACA,CAAEoO,QAASo3K,EAAYp3K,WAIlB48H,EAAQw6C,EAAY57K,OAGrB2S,EAASipK,EAAa,IAG9B/2C,EAAIn+H,iBAAkB,SAAS,IAAM06H,EAAQ,IAAI3yI,MAAO,oBACxDo2I,EAAIn+H,iBAAkB,SAAS,IAAM06H,EAAQ,IAAI3yI,MAAO,YAExDo2I,EAAIQ,KAAM2I,EAAU,GAEtB,EC/Lc,MAAMyvC,GAiBpB,WAAAtlL,CAAao8G,EAAyBooE,GACrC,IAAMpoE,EAML,MAAM,IAAI,EAAe,8BAA+B,MAGzD,IAAMooE,EAML,MAAM,IAAI,EAAe,oCAAqC,MAG/DvmL,KAAKs4I,OAASn6B,EACdn+G,KAAKgnL,YAAcT,CACpB,CAmBO,MAAAx5C,CAAQu5C,GACd,OAAO,IAAID,GAAcC,EAAYtmL,KAAKs4I,OAAQt4I,KAAKgnL,YACxD,EC1Dc,MAAMM,WAA0B5tJ,GAI9C,qBAAkBpD,GACjB,MAAO,mBACR,CAUO,WAAAiiH,CAAa6sC,EAAkC7gL,GACrD,OAAO,IAAI4gL,GAAOC,EAAwB7gL,EAC3C,CAQO,mBAAAq2I,CAAqBz8B,EAAyBooE,GACpD,OAAO,IAAIc,GAAelpE,EAAOooE,EAClC,E,+NC1CG,GAAU,CAAC,WAAa,oBAAoB,WAAa,CAAC,YAAW,GAEzE,OAAiB,OACjB,WAAoB,GAEP,KAAI,KAAS,IAIX,KAAQzsJ,OCsBR,MAAMytJ,WAAsB,IAAtBA,GACGvuJ,eAAiB,CCF1B,cAAyB,GAIvC,mBAAkB9B,GACjB,MAAO,CAAE,GAAW,GAAO6wG,GAAW9L,GAAY3F,GAAQ,GAC3D,CAKA,qBAAkBhgG,GACjB,MAAO,YACR,G9Mdc,cAAoC,GAIlD,mBAAkBY,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,uBACR,CAKO,IAAAI,GACN,MAAM23G,EAAMruI,KAAKs0B,OAAOtL,OAAO5gB,IAAK,sBAE9BimI,IAKNruI,KAAKs0B,OAAOqC,QAAQvuB,IAAK,IAAiB0jI,oBAAsBC,GAAU,IAAIqC,GAAerC,EAAQsC,EAAKruI,KAAKs0B,OAAOt5B,GACvH,GGpCc,cAAyB,GAIvC,mBAAkBk8B,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,YACR,CAKO,SAAAmxF,GACNznH,KAAKwnL,sBACLxnL,KAAKynL,6BACLznL,KAAK0nL,yBACL1nL,KAAK2nL,4BACL3nL,KAAK4nL,2BACL5nL,KAAK6nL,+BACN,CAWQ,mBAAAL,GACP,MAAMxoF,EAAWh/F,KAAKs0B,OAAO0qE,SAExBA,EAAS52F,IAAK,iBAClB8mI,GAAwBlvI,KAAKs0B,OAAQt0B,KAAM,WAAY,gBAGnDg/F,EAAS52F,IAAK,iBAClB8mI,GAAwBlvI,KAAKs0B,OAAQt0B,KAAM,aAAc,gBAGrDg/F,EAAS52F,IAAK,aAClB8mI,GAAwBlvI,KAAKs0B,OAAQt0B,KAAM,cAAe,YAGtDg/F,EAAS52F,IAAK,kBAClB8mI,GAAwBlvI,KAAKs0B,OAAQt0B,KAAM,mBAAmB,KAC7DA,KAAKs0B,OAAOiB,QAAS,YACrBv1B,KAAKs0B,OAAOiB,QAAS,gBAAiB,GAGzC,CAeQ,0BAAAkyJ,GACP,MAAMzoF,EAAWh/F,KAAKs0B,OAAO0qE,SAE7B,GAAKA,EAAS52F,IAAK,QAAW,CAC7B,MAAM0/K,EAAex3C,GAAwCtwI,KAAKs0B,OAAQ,QAE1Ek7G,GAAyBxvI,KAAKs0B,OAAQt0B,KAAM,gCAAiC8nL,GAC7Et4C,GAAyBxvI,KAAKs0B,OAAQt0B,KAAM,4BAA6B8nL,EAC1E,CAEA,GAAK9oF,EAAS52F,IAAK,UAAa,CAC/B,MAAM2/K,EAAiBz3C,GAAwCtwI,KAAKs0B,OAAQ,UAI5Ek7G,GAAyBxvI,KAAKs0B,OAAQt0B,KAAM,6BAA8B+nL,GAC1Ev4C,GAAyBxvI,KAAKs0B,OAAQt0B,KAAM,0BAA2B+nL,EACxE,CAEA,GAAK/oF,EAAS52F,IAAK,QAAW,CAC7B,MAAM4/K,EAAe13C,GAAwCtwI,KAAKs0B,OAAQ,QAE1Ek7G,GAAyBxvI,KAAKs0B,OAAQt0B,KAAM,kBAAmBgoL,EAChE,CAEA,GAAKhpF,EAAS52F,IAAK,iBAAoB,CACtC,MAAM6/K,EAAwB33C,GAAwCtwI,KAAKs0B,OAAQ,iBAEnFk7G,GAAyBxvI,KAAKs0B,OAAQt0B,KAAM,oBAAqBioL,EAClE,CACD,CAWQ,sBAAAP,GACP,MAAMhyJ,EAAsC11B,KAAKs0B,OAAO0qE,SAAS52F,IAAK,WAEjEstB,GACJA,EAAQ2lH,cACNt1I,QAAQ/D,GAAQA,EAAKnF,MAAO,oBAC5Be,SAASsqL,IACT,MAAM9vH,EAAQ8vH,EAAW,GACnBrqJ,EAAU,IAAIl0B,OAAQ,OAAQyuD,WAEpC82E,GAAwBlvI,KAAKs0B,OAAQt0B,KAAM69B,GAAS,KAEnD,IAAMnI,EAAQhB,WAAagB,EAAQl9B,QAAU0vL,EAC5C,OAAO,EAGRloL,KAAKs0B,OAAOiB,QAAS,UAAW,CAAE/8B,MAAO0vL,GAAa,GACpD,GAGP,CAQQ,yBAAAP,GACF3nL,KAAKs0B,OAAO0qE,SAAS52F,IAAK,eAC9B8mI,GAAwBlvI,KAAKs0B,OAAQt0B,KAAM,QAAS,aAEtD,CAQQ,wBAAA4nL,GACP,MAAMtzJ,EAASt0B,KAAKs0B,OACdW,EAAYX,EAAOl8B,MAAMoL,SAASyxB,UAEnCX,EAAO0qE,SAAS52F,IAAK,cACzB8mI,GAAwB56G,EAAQt0B,KAAM,SAAS,KAC9C,GAAKi1B,EAAUE,mBAAoBv2B,OAAOsP,GAAI,UAAW,YACxD,OAAO,EAERlO,KAAKs0B,OAAOiB,QAAS,YAAa,CACjC4yJ,2BAA2B,GACzB,GAGN,CAQQ,6BAAAN,GACF7nL,KAAKs0B,OAAO0qE,SAAS52F,IAAK,mBAC9B8mI,GAAwBlvI,KAAKs0B,OAAQt0B,KAAM,QAAS,iBAEtD,G5EpHc,cAA2B,GA0CzC,WAAA+B,CAAauyB,GACZ9mB,MAAO8mB,GAlBR,KAAQonF,gBAAyC,KAoBhD17G,KAAKooL,oBAAsB76E,GAAwBvtG,KAAKs0B,OAAOtL,OAAO5gB,IAAK,iBAC3EpI,KAAKovG,YAAcpvG,KAAKmnH,qBACxBnnH,KAAKorG,UAAYprG,KAAK+kH,mBACtB/kH,KAAKmrG,WAAanrG,KAAK4mH,oBAGvB,GAAqB,CACpBh3G,QAAS5P,KAAKorG,UACdnJ,gBAAiB,CAAEjiG,KAAKorG,UAAU3zF,QAAUzX,KAAKmrG,WAAW1zF,SAC5DuqF,UAAW,IAAMhiG,KAAKorG,UAAUlkF,UAChC3X,SAAU,IAAMvP,KAAKqoL,cAEvB,CAtBA,qBAAkB/xJ,GACjB,MAAO,cACR,CAyBO,IAAAI,GACN,MAAMpC,EAASt0B,KAAKs0B,OAGpBt0B,KAAKwP,SAA6C8kB,EAAOl8B,MAAMoL,SAASyxB,UAAW,gBAAgB,CAAE5f,EAAKvS,KACpGA,EAAK2/D,cACTziE,KAAKqoL,YACN,IAGDroL,KAAKwP,SAA+B8kB,EAAO6E,GAAI,UAAU,IAAMn5B,KAAKsoL,kBAEpEtoL,KAAKwP,SAAiC8kB,EAAQ,qBAAqB,IAAMt0B,KAAKsoL,iBAAiB,CAAE17K,SAAU,QAC3G5M,KAAKwP,SAAiC8kB,EAAO6E,GAAGqxE,aAAc,oBAAoB,IAAMxqG,KAAKsoL,kBAG7FtoL,KAAKwP,SAA0CxP,KAAKmrG,WAAY,oBAAoB,CAAE91F,EAAKrT,EAAMklB,KAC3FA,EAEJlnB,KAAKmrG,WAAW37F,SAAUjM,OAAQ,UAAU,IAAMvD,KAAKsoL,mBAGvDtoL,KAAKmrG,WAAWx7F,cAAepM,OAAQ,UAGvCvD,KAAKqoL,aACN,IAID/zJ,EAAO6E,GAAG8oF,WAAYjiH,KAAKovG,YAAa,CACvC4T,YAAa,IAAMhjH,KAAKuoL,aACxB3lE,UAAW,IAAM5iH,KAAKqoL,cAExB,CAOO,SAAA5gE,GACNznH,KAAKovG,YAAYhB,eAAgBpuG,KAAKooL,oBAAqBpoL,KAAKs0B,OAAO6E,GAAG61E,kBAG1E,UAAY/uG,KAAQD,KAAKovG,YAAYx+E,MACpC3wB,EAAKoP,GAAwB,WAAW,IAAMrP,KAAKqoL,YAAY,IAAQ,CAAEz7K,SAAU,QAErF,CAKgB,OAAAoZ,GACfxY,MAAMwY,UAGNhmB,KAAKorG,UAAUplF,UACfhmB,KAAKmrG,WAAWnlF,UAChBhmB,KAAKovG,YAAYppF,UAEZhmB,KAAK07G,iBACT17G,KAAK07G,gBAAgB11F,SAEvB,CAKQ,kBAAAmhG,GACP,MAAMnsH,EAAIgF,KAAKs0B,OAAO4E,OAAOl+B,EACvB8yG,GAAuB9tG,KAAKooL,oBAAoB7gE,uBAChDnY,EAAc,IAAI,GAAapvG,KAAKs0B,OAAO4E,OAAQ,CACxD40E,sBACAC,YAAY,IAKb,OAFAqB,EAAYmD,UAAYv3G,EAAG,gCAEpBo0G,CACR,CAKQ,gBAAA2V,GACP,MAAMzwF,EAASt0B,KAAKs0B,OACd82E,EAAY,IAAImO,GAAkBjlF,EAAO4E,QAa/C,OAXAkyE,EAAUlrG,QAAQ2N,IAAK7N,KAAKovG,aAC5BhE,EAAU7sE,MAAQ,uBAClBjK,EAAO6E,GAAGc,KAAKzU,KAAK3X,IAAKu9F,GACzB92E,EAAO6E,GAAGqxE,aAAa38F,IAAKu9F,EAAU3zF,SAGtCzX,KAAKovG,YAAYhQ,WAAW3tF,IAAK,OAAO,CAAE4D,EAAKoe,KAC9CzzB,KAAKqoL,YAAY,GACjB50J,GAAQ,IAGF23E,CACR,CAKQ,iBAAAwb,GACP,MAAMtyF,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACXmwG,EAAa,IAAI6c,GAAiB1zF,EAAO4E,QAwB/C,OAtBAiyE,EAAW15F,IAAK,CACf88B,MAAOvzC,EAAG,cACV0uG,KAAM,GACNuF,UAAU,IAIX9D,EAAWnkG,KAAM,QAASzH,GAAIS,KAAKorG,UAAW,aAC9CD,EAAWnkG,KAAM,WAAYzH,GAAIS,KAAKorG,UAAW,aAAalkF,IAAcA,IAG5ElnB,KAAKwP,SAA8B27F,EAAY,WAAW,KACnDnrG,KAAKorG,UAAUlkF,UAGpBlnB,KAAKqoL,YAAY,GAFjBroL,KAAKuoL,YAGN,IAGDj0J,EAAO6E,GAAGc,KAAKzU,KAAK3X,IAAKs9F,GACzB72E,EAAO6E,GAAGqxE,aAAa38F,IAAKs9F,EAAW1zF,SAEhC0zF,CACR,CAMQ,aAAAm9E,GACP,MAAMh0J,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf6hC,EAAO3F,EAAO6qE,QAAQllE,KAG5B,IAAM3F,EAAO6E,GAAGqxE,aAAa13E,UAG5B,YAFA9yB,KAAKwoL,cAMN,IAAMl0J,EAAOl8B,MAAMi9B,UAAWf,EAAOl8B,MAAMoL,SAASyxB,WAGnD,YAFAj1B,KAAKwoL,cAMN,MAAMC,EAAc9nL,MAAMrB,KAAMlH,EAAMoL,SAASyxB,UAAU0tC,qBAAuB,GAGhF,IAAM8lH,GAAe9nL,MAAMrB,KAAMU,KAAKovG,YAAYx+E,OAAQra,OAAStW,IAAgBA,EAAKy0B,YAGvF,YAFA10B,KAAKwoL,cAMN,MAAMt2I,EAAYjY,EAAK4X,aAAayK,aAAchoB,EAAO6qE,QAAQ3hC,OAAOf,cAAegsH,IAGvFzoL,KAAKmrG,WAAWjkF,WAAY,EAG5BlnB,KAAK0oL,sBAGL1oL,KAAK2oL,uBAAwBz2I,GAGxBlyC,KAAKorG,UAAUlkF,WACnBlnB,KAAKuoL,YAEP,CAKQ,WAAAC,GACPxoL,KAAKmrG,WAAWjkF,WAAY,CAC7B,CAMQ,UAAAqhK,GAKP,IAAMvoL,KAAKmrG,WAAWjkF,UACrB,OAGD,MAAM0hK,EAAa5oL,KAAKorG,UAAUlkF,UAyBlClnB,KAAKorG,UAAU0M,OAEf,MAAM3vE,EAAkBnoC,KAAK6oL,8BAE7B7oL,KAAKovG,YAAYpB,SAAWhuG,KAAK8oL,oBAAqB3gJ,GAEtDnoC,KAAKorG,UAAU2N,IAAK,CACnBn1G,OAAQ5D,KAAKmrG,WAAW1zF,QACxB4P,QAAS8gB,IAGJygJ,GACH5oL,KAAKovG,YAAYx+E,MAAMxoB,IAAK,GAAa6kC,OAE7C,CAKQ,2BAAA47I,GACP,MAAME,EAAwB/oL,KAAKs0B,OAAOl8B,MAAMoL,SAASyxB,UAAUoY,gBAAiB/2C,KAAK8+B,SAEzF,OAAOp1B,KAAKs0B,OAAO6E,GAAG6mF,mBAAoB+oE,EAC3C,CAOQ,UAAAV,CAAYnpC,GACnBl/I,KAAKorG,UAAUlkF,WAAY,EAEtBg4H,GACJl/I,KAAKs0B,OAAO6qE,QAAQllE,KAAKgT,OAE3B,CAOQ,sBAAA07I,CAAwBjnG,GAC/B,MAAMsnG,EAAgBzlL,OAAOie,iBAAkBkgE,GAEzCunG,EAAe,IAAI,GAAMjpL,KAAK6oL,+BAC9BK,EAAoB9rL,SAAU4rL,EAAcG,WAAY,IAGxDC,EAAoBhsL,SAAU4rL,EAAcK,WAAY,KAAiD,IAAzCjsL,SAAU4rL,EAAcM,SAAU,IAElG7jK,EAAW0B,GAAoB,CACpC1P,QAASzX,KAAKmrG,WAAW1zF,QACzB7T,OAAQ89E,EACRt6D,UAAW,CACV,CAAE8pF,EAAa5E,KACd,IAAIvqF,EAQJ,OALCA,EAD+C,QAA3C/hB,KAAKs0B,OAAO4E,OAAOvJ,oBAChBs5J,EAAalnK,KAAOuqF,EAAWzpF,MAE/BomK,EAAatnK,MAGd,CACNF,IAAKyvF,EAAYzvF,IAAMynK,GAAsBE,EAAoB98E,EAAWxpF,QAAW,EACvFf,OACA,KAKJ/hB,KAAKmrG,WAAW1pF,IAAMgE,EAAShE,IAC/BzhB,KAAKmrG,WAAWppF,KAAO0D,EAAS1D,IACjC,CAKQ,mBAAA2mK,GACP,MAAMvgJ,EAAkBnoC,KAAK6oL,8BAGvB7oL,KAAKooL,oBAAoB7gE,yBAEzBvnH,KAAK07G,iBAAmB17G,KAAK07G,gBAAgBjkG,UAAY0wB,IAC7DnoC,KAAK07G,gBAAgB11F,UACrBhmB,KAAK07G,gBAAkB,MAGlB17G,KAAK07G,kBACV17G,KAAK07G,gBAAkB,IAAI,GAAgBvzE,GAAiB,KAC3DnoC,KAAKovG,YAAYpB,SAAWhuG,KAAK8oL,oBAAqB3gJ,EAAiB,KAI3E,CAQQ,mBAAA2gJ,CAAqB3gJ,GAC5B,MAAM8gJ,EAAe,IAAI,GAAM9gJ,GACzBmkE,EAAa,IAAI,GAAMtsG,KAAKmrG,WAAW1zF,SAEvCmE,EADmD,QAA3C5b,KAAKs0B,OAAO4E,OAAOvJ,oBACR28E,EAAWvqF,KAAOknK,EAAatnK,MAAU2qF,EAAWzpF,MAAQomK,EAAalnK,KAAOuqF,EAAWvqF,KAEpH,OAAO,GAAMknK,EAAapmK,MAAQjH,EACnC,GwRtdc,cAAmB,GAIjC,mBAAkBsb,GACjB,MAAO,CAAEy5G,GAAaE,GACvB,CAKA,qBAAkBv6G,GACjB,MAAO,MACR,GCbc,cAAqB,GAInC,mBAAkBY,GACjB,MAAO,CAAE65G,GAAeC,GACzB,CAKA,qBAAkB16G,GACjB,MAAO,QACR,GCXc,cAAyB,GAIvC,mBAAkBY,GACjB,MAAO,CAAEg7G,GAAmBE,GAC7B,CAKA,qBAAkB97G,GACjB,MAAO,YACR,GCVc,cAAoB,GAIlC,qBAAkBA,GACjB,MAAO,OACR,CAKA,mBAAkBY,GACjB,MAAO,CAAEk+G,GAAc/C,GACxB,GCVc,cAAuB,GAIrC,qBAAkB/7G,GACjB,MAAO,UACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,OAAQ,wBAAyBujH,GAAiBnB,GAC5D,GCtBc,cAA4B5/G,GAA3C,kCAoCC,KAAOykF,MAAiC,KAKxC,KAAiBorE,QAAU,IAAIh4K,GAAgC,CAK/D,qBAAkB+kB,GACjB,MAAO,eACR,CAKA,mBAAkBY,GACjB,MAAO,CAAEowJ,GACV,CAKa,IAAA5wJ,GAAsB,iCAClC,MAEMnyB,EAFSvE,KAAKuN,QAAQyb,OAEL5gB,IAAK,kBAAqB,CAAC,EAElD,UAAcrR,EAAKyB,KAAWxB,OAAOyhB,QAASlU,GAC3CvE,KAA+BjJ,GAAQyB,EAG1C,IAAMwH,KAAKqzI,SAGV,YAFArzI,KAAKm+G,MAAQ,MAKd,MAAMg6B,EAAuCn4I,KAAKuN,QAAQopB,QAAQvuB,IAAK,qBAEvEpI,KAAKm+G,YAAcg6B,EAAkBI,YAAav4I,KAAKqzI,UAAW38G,OAElE12B,KAAKupL,QAAQ93K,IAAKzR,KAAKqzI,SAAUrzI,KAAKm+G,MACvC,IAQa,gBAAAqrE,CAAkBn2C,GAAgD,iCAE9E,GAAKrzI,KAAKupL,QAAQ37K,IAAKylI,GACtB,OAAOrzI,KAAKypL,YAAap2C,GAG1B,MAAM8E,EAAuCn4I,KAAKuN,QAAQopB,QAAQvuB,IAAK,qBACjE+1G,QAAcg6B,EAAkBI,YAAalF,GAAW38G,OAI9D,OAFA12B,KAAKupL,QAAQ93K,IAAK4hI,EAAUl1B,GAErBA,CACR,IAOO,WAAAsrE,CAAap2C,GACnB,MAAMl1B,EAAQn+G,KAAKupL,QAAQnhL,IAAKirI,GAEhC,IAAMl1B,EAML,MAAM,IAAI,EAAe,qCAAsCn+G,MAGhE,OAAOm+G,CACR,CAKgB,OAAAn4F,GACfxY,MAAMwY,UAEN,UAAYm4F,KAASn+G,KAAKupL,QAAQr+K,SACjCizG,EAAMn4F,SAER,GClHc,cAAwB,GAItC,qBAAkBsQ,GACjB,MAAO,WACR,CAKA,mBAAkBY,GACjB,MAAO,CAAEwjH,GAA4B,cACtC,CAKO,IAAAhkH,GACN,MAAMpC,EAASt0B,KAAKs0B,OAEdA,EAAOqC,QAAQ/oB,IAAK,sBAA0B0mB,EAAOqC,QAAQ/oB,IAAK,uBAavE,EAAY,mCAAoC0mB,EAElD,GC/Cc,cAAsB,GAIpC,mBAAkB4C,GACjB,MAAO,CAAEukH,GAAgBG,GAC1B,CAKA,qBAAkBtlH,GACjB,MAAO,SACR,GCZc,cAAoB,GAIlC,mBAAkBY,GACjB,MAAO,CAAE2pH,GAAYK,GACtB,CAKA,qBAAkB5qH,GACjB,MAAO,OACR,GCrBc,cAA2B,GAIzC,mBAAkBY,GACjB,MAAO,CAAEmrH,GAAqBY,GAC/B,CAKA,qBAAkB3sH,GACjB,MAAO,cACR,GCRc,cAAyB,GAIvC,mBAAkBY,GACjB,MAAO,CAAEkyH,GAAmBa,GAC7B,CAKA,qBAAkB3zH,GACjB,MAAO,YACR,GCVc,cAA2B,GAIzC,mBAAkBY,GACjB,MAAO,CAAE,GAAyB,GACnC,CAKA,qBAAkBZ,GACjB,MAAO,cACR,CAKO,SAAAmxF,GACN,MAAMnzF,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACX0uL,EAA0Bp1J,EAAOqC,QAAQvuB,IAAK,IAC9Cu0I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAcrD,IAAqC4gB,EAZnC0gK,EAAwBx0G,SAAU,QAAS,CAC1Cq9B,UAAWv3G,EAAG,iBACd41B,OAUkC5H,EAVCsL,EAAOtL,OAAO5gB,IAAK,kBAAqB,GAWtE4gB,EAAOhsB,KAAKiD,GAAQ,EAAUA,GAASA,EAAK+B,KAAO/B,KAVxD8jI,kBAAmB9uG,GAAa0nH,EAAWW,8BAA+BroH,IAE5E,GChCc,cAA0B,GAIxC,qBAAkBqB,GACjB,MAAO,aACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE6uH,GAAoB7B,GAAeO,GAC7C,GCJc,cAAqB,GAInC,qBAAkBnuH,GACjB,MAAO,QACR,CAKA,mBAAkBY,GACjB,MAAO,CAAEm0H,GAAeC,GACzB,GC3Bc,cAAmB,GAIjC,mBAAkBp0H,GACjB,MAAO,CAAE,GAAa,GAAQ69H,GAC/B,CAKA,qBAAkBz+H,GACjB,MAAO,MACR,GCbc,cAAmB,GAIjC,mBAAkBY,GACjB,MAAO,CAAE,GAAa,GACvB,CAKA,qBAAkBZ,GACjB,MAAO,MACR,GCJc,cAAyB,GAIvC,mBAAkBY,GACjB,MAAO,CAAEwnI,GAAmBuB,GAAcnB,GAAgB,GAC3D,CAKA,qBAAkBxoI,GACjB,MAAO,YACR,GlBeC,GmBxBa,cAA8B,GAI5C,qBAAkBA,GACjB,MAAO,iBACR,CAKA,mBAAkBY,GACjB,MAAO,CAAE,GACV,CAKO,IAAAR,GACN,MAAMpC,EAASt0B,KAAKs0B,OACd8xG,EAAuC9xG,EAAOqC,QAAQvuB,IAAK,qBAC3DorD,EAAel/B,EAAO6qE,QAAQllE,KAAKz2B,SACnCmmL,EAAiC,GAEvCA,EAAY/pL,KAAM,IAAImlK,GAAkBvxG,IACxCm2H,EAAY/pL,KAAM,IAAI4lK,GAAsBhyG,IAC5Cm2H,EAAY/pL,KAAM,IAAImmK,GAAwBvyG,IAE9C4yE,EAAkB/2H,GACjB,uBACA,CAAEgG,EAAKvS,KACN,GAAKA,EAAK8mL,kCACT,OAKD,GAFkBt1J,EAAOl8B,MAAMoL,SAASyxB,UAAUE,mBAAoBv2B,OAEvDsP,GAAI,UAAW,aAC7B,OAGD,MAAM+2J,EAAaniK,EAAK2wD,aAAad,QAAS,aACxCk3H,EAAmBF,EAAYp4J,MAAM0S,GAAcA,EAAW+gI,SAAUC,KAEzE4kB,IACE/mL,EAAKoiK,cACVpiK,EAAKoiK,YAAcmB,GAAWpB,EAAYzxG,EAAaztB,kBAGxD8jJ,EAAiBt0J,QAASzyB,GAE1BA,EAAK8mL,mCAAoC,EAC1C,GAED,CAAEh9K,SAAU,QAEd,GCpBc,cAA6B,GAI3C,mBAAkBsqB,GACjB,MAAO,CAAE0oH,GAAc,GACxB,CAKA,qBAAkBtpH,GACjB,MAAO,gBACR,CAKO,SAAAmxF,GACN,MAAMnzF,EAASt0B,KAAKs0B,OAEfA,EAAOqC,QAAQ/oB,IAAK,sBACxB0mB,EAAOl8B,MAAMsrE,OAAOxwB,OAAQ,aAAc,CACzC0iC,gBAAiB,CAAE,aAIhBthD,EAAOqC,QAAQ/oB,IAAK,uBACxB0mB,EAAOl8B,MAAMsrE,OAAOxwB,OAAQ,cAAe,CAC1C0iC,gBAAiB,CAAE,aAIrB51E,KAAKsgJ,mBACLtgJ,KAAK8pL,qCACN,CAMQ,gBAAAxpC,GACP,MAAMhsH,EAASt0B,KAAKs0B,OACdj2B,EAAai2B,EAAOj2B,WACpBs+I,EAAyBroH,EAAOqC,QAAQvuB,IAAK,cAEnD/J,EAAWuhF,IAAK,UAAW/xE,IpLhBtB,SAAwB8uI,GAC9B,MAAMotC,EAAuB,CAAE,SAAU,QAAS,OAAQ,SAEpD17G,EAA6C,CAAEh5D,EAAKvS,EAAMw8D,KAC/D,MAAM0qH,EAAqBlnL,EAAKgtE,SAGhC,IAAMxQ,EAAcwB,WAAW7+D,KAAM+nL,EAAoB,CAAEhoL,MAAM,IAChE,OAGD,MAAMiB,EAAU,IAAIsO,IAGpB,UAAY04K,KAAsBD,EAAmBxuJ,cACpD,GAAKyuJ,EAAmB/7K,GAAI,UAAW,UAAa,CACnD,MAAM3I,EAAiD,CAAC,EAExD,UAAYvD,KAAQ+nL,EACdE,EAAmBhkJ,aAAcjkC,IAEhCs9D,EAAcwB,WAAW7+D,KAAMgoL,EAAoB,CAAE1kL,WAAYvD,MACrEuD,EAAYvD,GAASioL,EAAmB5uJ,aAAcr5B,IAKpDhL,OAAOC,KAAMsO,GAAajN,QAC9B2K,EAAQwO,IAAKw4K,EAAoB1kL,EAEnC,CAGD,MAAM2kL,EAAiBvtC,EAAWoB,mBAAoBisC,GAGtD,IAAME,EACL,OAGD,IAAI1pC,EAAa19I,EAAK+sE,YAAYjxE,OAOlC,IAAM4hJ,EAAWtyI,GAAI,UAAW,cAAiB,CAChD,MAAMi8K,EAAmB7qH,EAAc4C,YAAagoH,EAAgBpnL,EAAK+sE,aAGzE/sE,EAAKw6D,WAAa6sH,EAAiB7sH,WAGnCx6D,EAAK+sE,YAAcs6G,EAAiBt6G,YAEpC2wE,EAAa,GAAO2pC,EAAiB7sH,WAAYpxB,WAClD,CAEAozB,EAAcwB,WAAWrC,QAASurH,EAAoB,CAAEhoL,MAAM,IAI9D,UAAcy/E,EAAel8E,KAAgBtC,EAC5Cq8D,EAAcwB,WAAWrC,QAASgjB,EAAe,CAAEl8E,WAAYvO,OAAOC,KAAMsO,KAGxEtC,EAAQ4Q,MACZyrD,EAAc9kC,OAAO/0B,aAAc,UAAW9E,MAAMrB,KAAM2D,EAAQiI,UAAYs1I,GAI/ElhF,EAAc6C,gBAAiB6nH,EAAoBxpC,EAAY,EAGhE,OAAOhhF,IACNA,EAAWnwD,GAAwB,kBAAmBg/D,EAAW,CAEnE,CoL9DkC+7G,CAAeztC,IAC/Ct+I,EAAWuhF,IAAK,YAAa/xE,IpLuHxB,SAAmC8uI,GACzC,MAAMtuE,EAA0D,CAAEh5D,EAAKvS,EAAMw8D,KAC5E,IAAMA,EAAcwB,WAAWrC,QAAS37D,EAAK7C,KAAMoV,EAAIrT,MACtD,OAGD,MAAMqoE,EAAa/K,EAAc9kC,OAC3B/iB,EAAU6nD,EAAc9B,OAAOf,cAAe35D,EAAK7C,MACnDoqL,EAAa1tC,EAAWoB,mBAAoBtmI,GAC5CgpD,EAAoB39D,EAAK29D,kBAE/B,GAAKA,GAAqBA,EAAkBnoE,OAAS,CAEpD,MAAMgyL,EAAiBjgH,EAAWt2B,uBAAwB,UAAW,KACpE0sB,EAAkBzjE,KAAKutL,GACflgH,EAAWl2B,mBAAoB,SAAUo2I,MAK5CC,EAAoB,GAC1B,IAAIjzI,EAAc8yI,EAAWzrL,OAE7B,KAAQ24C,GAAeA,EAAYrpC,GAAI,qBAAuB,CAC7D,MAAM4Y,EAAgBywB,EAAY34C,OAElCyrE,EAAWnzB,OAAQmzB,EAAWvyB,cAAeuyI,GAAc9yI,GAE3DizI,EAAkBprL,QAASm4C,GAC3BA,EAAczwB,CACf,CAGAujD,EAAW3kE,OAAQ2kE,EAAWzyB,qBAAsByyI,GAAcC,GAClEjgH,EAAW/0B,KAAM+0B,EAAWvyB,cAAeuyI,GAAchgH,EAAW3yB,iBAAkB4yI,EAAgB,QAGtG,UAAYz2I,KAAoB22I,EAC/BngH,EAAWvzB,KAAMuzB,EAAWvyB,cAAewyI,GAAkBz2I,EAE/D,SAGUw2I,EAAWzrL,OAAQsP,GAAI,UAAW,WAAc,CACzD,MAAMo8K,EAAiBD,EAAWzrL,OAElCyrE,EAAW/0B,KAAM+0B,EAAWvyB,cAAeuyI,GAAchgH,EAAWzyB,qBAAsB0yI,IAC1FjgH,EAAWpkE,OAAQqkL,EACpB,GAGD,OAAO9qH,IACNA,EAAWnwD,GAAqC,+BAAgCg/D,GAChF7O,EAAWnwD,GAAqC,gCAAiCg/D,EAAW,CAE9F,CoL9KoCo8G,CAA0B9tC,GAC7D,CAOQ,mCAAAmtC,GACP,MAAMx1J,EAASt0B,KAAKs0B,OAEpB,IAAMA,EAAOqC,QAAQ/oB,IAAK,sBACzB,OAGD,MAAM8nI,EAAyCphH,EAAOqC,QAAQvuB,IAAK,sBAEnEpI,KAAKwP,SACJkmI,EACA,kBACA,CAAErgI,GAAOsgI,eAAc7yI,WACtB,MAAMG,EAAUH,EAAKG,QAEfA,GAINqxB,EAAOl8B,MAAMmiC,QAAQC,IACpBA,EAAO+4D,cAAe,CACrBtwF,WACE0yI,EAAc,GACf,GAEN,GClHc,cAAoB,GAIlC,mBAAkBz+G,GACjB,MAAO,CAAE,GAAcknJ,GAASU,GAAgBqF,GAAYnB,GAAe3C,GAAgB,GAC5F,CAKA,qBAAkB/pJ,GACjB,MAAO,OACR,GC1Bc,cAA2B,GAIzC,mBAAkBY,GACjB,MAAO,CAAE,GACV,CAKA,qBAAkBZ,GACjB,MAAO,cACR,CAKO,SAAAmxF,GACN,MAAMnzF,EAASt0B,KAAKs0B,OACdt5B,EAAIs5B,EAAOt5B,EACX0uL,EAA0Bp1J,EAAOqC,QAAQvuB,IAAK,IAE9CsiL,EAA2Bp2J,EAAOtL,OAAO5gB,IAAK,wBAE9CuiL,EAAoBr2J,EAAOtL,OAAO5gB,IAAK,sBAExCsiL,GACJhB,EAAwBx0G,SAAU,eAAgB,CACjDq9B,UAAWv3G,EAAG,iBACd41B,MAAO85J,EACP3mD,kBAAmB,KAIhB4mD,GACJjB,EAAwBx0G,SAAU,QAAS,CAC1Cq9B,UAAWv3G,EAAG,iBACd41B,MAAO+5J,EACP5mD,kBAAmB+gD,IAGtB,G/QkBc,cAAiC,GAI/C,mBAAkB5tJ,GACjB,MAAO,CAAE,SAAU,QACpB,CAKA,qBAAkBZ,GACjB,MAAO,oBACR,CAKA,WAAAv0B,CAAauyB,GACZ9mB,MAAO8mB,GAEPA,EAAOtL,OAAOtyB,OAAQ,SAAU,CAC/B0sF,gBAAiB,CAChBwnG,QAASzwD,KAGZ,CAKO,IAAAzjG,GACN,MACM48C,EADQtzE,KAAKs0B,OAAOl8B,MACGoL,SAASyxB,UAEtCq+C,EAAejkE,GAAI,gBAAgB,KAElCrP,KAAK00B,WAAa4+C,EAAe53C,OAAQ98B,OAAOsP,GAAI,UAAW,YAAa,IAG7ElO,KAAK6qL,+BACN,CAKQ,6BAAAA,GACP,MAAMv2J,EAASt0B,KAAKs0B,OACdl8B,EAAQk8B,EAAOl8B,MACf29J,EAAuBzhI,EAAOqC,QAAQvuB,IAAK,UAC3C0iL,EAiHR,SAAmC9hK,GAClC,MAAM+hK,EAAQ/hK,EAAO+hK,OAAS,GACxB9kL,EAAS+iB,EAAO/iB,QAAU,GAC1B+kL,EAAiBC,IAA6DhlL,EAAOspB,SAAU07J,GAIrG,OAiBD,SACCnoE,GAGA,MAAMooE,EAAyB,IAAIt3K,IAEnC,UAAYu3K,KAAyBroE,EACpC,GAAqC,iBAAzBqoE,GAAqCrxD,GAAuBqxD,GACvE,UAAYF,KAAkBnxD,GAAuBqxD,GACpDD,EAAuBr9K,IAAKo9K,QAG7BC,EAAuBr9K,IAAKs9K,GAI9B,OAAOxqL,MAAMrB,KAAM4rL,EACpB,CAlCQE,CAFYpiK,EAAO4hK,QAAQzqL,OAAQ4qL,GAAQhlL,OAAQilL,IAGxDjlL,OAAQilL,GACRhuL,KAAKiuL,GACoB,iBAAlBA,GAA8B3yD,GAAiB2yD,GAAmB3yD,GAAiB2yD,GAAmBA,IAG7GllL,QAAUklL,GAA+F,iBAAnBA,IACtFjuL,KAAKiuL,IAAA,CACL3rL,KAAM86H,GAAe6wD,EAAe3rL,MACpCC,GAAI86H,GAAa4wD,EAAe1rL,OAEnC,CAnIoC8rL,CAA0B/2J,EAAOtL,OAAO5gB,IAAK,2BAazE+sJ,EAAU,IAAI3+B,GAAaliG,EAAOl8B,OAXjB8hC,IACtB,UAAYoxJ,KAA4BR,EAA4B,CAInE,GAHaQ,EAAyBhsL,KACnB2C,KAAMi4B,GAGxB,MAAO,CAAEoxJ,2BAEX,KAKDn2B,EAAQ9lJ,GAEH,gBAAgB,CAAEgG,EAAKvS,KAC3B,IAAMA,EAAKojE,MAAM+jB,SAChB,OAGD,MAAM,KAAE3qF,EAAI,GAAEC,GAAOuD,EAAKwoL,yBAEpBn3J,EAAU70B,EAAKuZ,KAAM/V,EAAKo3B,MAC1BqxJ,EAAWhsL,EAAI40B,EAAQryB,MAAO,IAE9B0pL,EAAe1oL,EAAKgiB,MAE1B,IAAIqoD,EAAch5C,EAAQvvB,MAE1BxM,EAAMguE,eAAe5rC,IACpB,QAAUhgC,EAAI,EAAGA,EAAI25B,EAAQ77B,OAAQkC,IAAM,CAC1C,MAAMqC,EAAQs3B,EAAS35B,GACjBmrD,EAAc4lI,EAAU/wL,EAAI,GAElC,GAAoB,MAAfmrD,EAAsB,CAC1BwnB,GAAetwE,EAAMvE,OAErB,QACD,CAEA,MAAMmzL,EAAkBD,EAAa9qJ,MAAM4J,aAAc6iC,GACnDu+G,EAAetzL,EAAMy/C,YAAa4zI,EAAiBA,EAAgBnhJ,aAAcztC,EAAMvE,SACvFiN,EAAa+0H,GAAgCmxD,GAEnDrzL,EAAMqjG,cAAejhE,EAAOmZ,WAAYgS,EAAapgD,GAAcmmL,GAEnEv+G,GAAexnB,EAAYrtD,MAC5B,CAEAF,EAAMguE,eAAe,KACpB2vF,EAAa1/B,wBAAwB,GACnC,GACD,IAGJ8+B,EAAQnuJ,KAAM,aAAczH,GAAIS,KACjC,IyP5JoBunL,GA+BGxuJ,cAAgB,CACtC4yJ,aAAc,CACb,OAAQ,OACR,IAAK,UACL,IAAK,cAAe,cAAe,aAAc,aACjD,IAAK,eAAgB,eAAgB,UAAW,UAEjD9mD,QAAS,CACRj0G,MAAO,CACN,OAAQ,SAAU,SAGpBoiC,MAAO,CACN6xE,QAAS,CACR,oBACA,mBACA,kBACA,IACA,qBACA,yBAGFijC,MAAO,CACN8jB,eAAgB,CACf,cACA,WACA,oBAIFl9J,SAAU,K,kB1zBrFZ","sources":["webpack://BalloonEditor/webpack/universalModuleDefinition","webpack://BalloonEditor/../node_modules/color-convert/conversions.js","webpack://BalloonEditor/../node_modules/color-convert/index.js","webpack://BalloonEditor/../node_modules/color-convert/route.js","webpack://BalloonEditor/../node_modules/color-name/index.js","webpack://BalloonEditor/../ckeditor5-basic-styles/theme/code.css","webpack://BalloonEditor/../ckeditor5-block-quote/theme/blockquote.css","webpack://BalloonEditor/./theme/theme.css","webpack://BalloonEditor/../ckeditor5-clipboard/theme/clipboard.css","webpack://BalloonEditor/../ckeditor5-engine/theme/placeholder.css","webpack://BalloonEditor/../ckeditor5-engine/theme/renderer.css","webpack://BalloonEditor/../ckeditor5-heading/theme/heading.css","webpack://BalloonEditor/../ckeditor5-image/theme/image.css","webpack://BalloonEditor/../ckeditor5-image/theme/imagecaption.css","webpack://BalloonEditor/../ckeditor5-image/theme/imageinsert.css","webpack://BalloonEditor/../ckeditor5-image/theme/imageinsertformrowview.css","webpack://BalloonEditor/../ckeditor5-image/theme/imageresize.css","webpack://BalloonEditor/../ckeditor5-image/theme/imagestyle.css","webpack://BalloonEditor/../ckeditor5-image/theme/imageuploadicon.css","webpack://BalloonEditor/../ckeditor5-image/theme/imageuploadloader.css","webpack://BalloonEditor/../ckeditor5-image/theme/imageuploadprogress.css","webpack://BalloonEditor/../ckeditor5-image/theme/textalternativeform.css","webpack://BalloonEditor/../ckeditor5-link/theme/link.css","webpack://BalloonEditor/../ckeditor5-link/theme/linkactions.css","webpack://BalloonEditor/../ckeditor5-link/theme/linkform.css","webpack://BalloonEditor/../ckeditor5-link/theme/linkimage.css","webpack://BalloonEditor/../ckeditor5-list/theme/collapsible.css","webpack://BalloonEditor/../ckeditor5-list/theme/documentlist.css","webpack://BalloonEditor/../ckeditor5-list/theme/list.css","webpack://BalloonEditor/../ckeditor5-list/theme/listproperties.css","webpack://BalloonEditor/../ckeditor5-list/theme/liststyles.css","webpack://BalloonEditor/../ckeditor5-list/theme/todolist.css","webpack://BalloonEditor/../ckeditor5-media-embed/theme/mediaembed.css","webpack://BalloonEditor/../ckeditor5-media-embed/theme/mediaembedediting.css","webpack://BalloonEditor/../ckeditor5-media-embed/theme/mediaform.css","webpack://BalloonEditor/../ckeditor5-table/theme/colorinput.css","webpack://BalloonEditor/../ckeditor5-table/theme/form.css","webpack://BalloonEditor/../ckeditor5-table/theme/formrow.css","webpack://BalloonEditor/../ckeditor5-table/theme/inserttable.css","webpack://BalloonEditor/../ckeditor5-table/theme/table.css","webpack://BalloonEditor/../ckeditor5-table/theme/tablecaption.css","webpack://BalloonEditor/../ckeditor5-table/theme/tablecellproperties.css","webpack://BalloonEditor/../ckeditor5-table/theme/tablecolumnresize.css","webpack://BalloonEditor/../ckeditor5-table/theme/tableediting.css","webpack://BalloonEditor/../ckeditor5-table/theme/tableform.css","webpack://BalloonEditor/../ckeditor5-table/theme/tableproperties.css","webpack://BalloonEditor/../ckeditor5-table/theme/tableselection.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/button/button.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/button/switchbutton.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/colorgrid/colorgrid.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/colorpicker/colorpicker.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/colorselector/colorselector.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/dropdown/dropdown.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/dropdown/listdropdown.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/dropdown/splitbutton.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/dropdown/toolbardropdown.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/editorui/editorui.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/formheader/formheader.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/icon/icon.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/input/input.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/label/label.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/labeledfield/labeledfieldview.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/list/list.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/panel/balloonpanel.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/panel/balloonrotator.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/panel/fakepanel.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/panel/stickypanel.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/responsive-form/responsiveform.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/toolbar/blocktoolbar.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/toolbar/toolbar.css","webpack://BalloonEditor/../ckeditor5-ui/theme/components/tooltip/tooltip.css","webpack://BalloonEditor/../ckeditor5-ui/theme/globals/globals.css","webpack://BalloonEditor/../ckeditor5-widget/theme/widget.css","webpack://BalloonEditor/../ckeditor5-widget/theme/widgetresize.css","webpack://BalloonEditor/../ckeditor5-widget/theme/widgettypearound.css","webpack://BalloonEditor/../node_modules/css-loader/dist/runtime/api.js","webpack://BalloonEditor/../node_modules/css-loader/dist/runtime/cssWithMappingToString.js","webpack://BalloonEditor/../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js","webpack://BalloonEditor/webpack/bootstrap","webpack://BalloonEditor/webpack/runtime/compat get default export","webpack://BalloonEditor/webpack/runtime/define property getters","webpack://BalloonEditor/webpack/runtime/global","webpack://BalloonEditor/webpack/runtime/hasOwnProperty shorthand","webpack://BalloonEditor/webpack/runtime/nonce","webpack://BalloonEditor/../ckeditor5-utils/src/env.ts","webpack://BalloonEditor/../ckeditor5-utils/src/fastdiff.ts","webpack://BalloonEditor/../ckeditor5-utils/src/diff.ts","webpack://BalloonEditor/../ckeditor5-utils/src/spy.ts","webpack://BalloonEditor/../ckeditor5-utils/src/eventinfo.ts","webpack://BalloonEditor/../ckeditor5-utils/src/uid.ts","webpack://BalloonEditor/../ckeditor5-utils/src/priorities.ts","webpack://BalloonEditor/../ckeditor5-utils/src/inserttopriorityarray.ts","webpack://BalloonEditor/../ckeditor5-utils/src/ckeditorerror.ts","webpack://BalloonEditor/../ckeditor5-utils/src/version.ts","webpack://BalloonEditor/../ckeditor5-utils/src/emittermixin.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/isObject.js","webpack://BalloonEditor/../ckeditor5-utils/src/observablemixin.ts","webpack://BalloonEditor/../ckeditor5-utils/src/count.ts","webpack://BalloonEditor/../ckeditor5-utils/src/comparearrays.ts","webpack://BalloonEditor/../ckeditor5-utils/src/isiterable.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/_freeGlobal.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_root.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_Symbol.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getRawTag.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_objectToString.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseGetTag.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isObjectLike.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isString.js","webpack://BalloonEditor/../ckeditor5-utils/src/dom/createelement.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/_overArg.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getPrototype.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isPlainObject.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_listCacheClear.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/eq.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_assocIndexOf.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_listCacheDelete.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_listCacheGet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_listCacheHas.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_listCacheSet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_ListCache.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_stackClear.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_stackDelete.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_stackGet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_stackHas.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isFunction.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_coreJsData.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_isMasked.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_toSource.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseIsNative.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getValue.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getNative.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_Map.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_nativeCreate.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_hashClear.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_hashDelete.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_hashGet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_hashHas.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_hashSet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_Hash.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_mapCacheClear.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_isKeyable.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getMapData.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_mapCacheDelete.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_mapCacheGet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_mapCacheHas.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_mapCacheSet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_MapCache.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_stackSet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_Stack.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_arrayEach.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_defineProperty.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseAssignValue.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_assignValue.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_copyObject.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseTimes.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseIsArguments.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isArguments.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/stubFalse.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isBuffer.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_isIndex.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isLength.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseIsTypedArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseUnary.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_nodeUtil.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isTypedArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_arrayLikeKeys.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_isPrototype.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_nativeKeys.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseKeys.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isArrayLike.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/keys.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseAssign.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_nativeKeysIn.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseKeysIn.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/keysIn.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseAssignIn.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_cloneBuffer.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_copyArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_arrayFilter.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/stubArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getSymbols.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_copySymbols.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_arrayPush.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getSymbolsIn.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_copySymbolsIn.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseGetAllKeys.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getAllKeys.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getAllKeysIn.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_DataView.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_Promise.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_Set.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_WeakMap.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_getTag.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_initCloneArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_Uint8Array.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_cloneArrayBuffer.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_cloneDataView.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_cloneRegExp.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_cloneSymbol.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_cloneTypedArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_initCloneByTag.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseCreate.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_initCloneObject.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseIsMap.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isMap.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseIsSet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isSet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseClone.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/cloneDeepWith.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isElement.js","webpack://BalloonEditor/../ckeditor5-utils/src/config.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/isnode.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/iswindow.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/emittermixin.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/global.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/istext.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/isrange.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/getborderwidths.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/rect.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/resizeobserver.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/setdatainelement.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/tounit.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/indexof.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/insertat.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/iscomment.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/isvisible.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/position.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/getpositionedancestor.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/remove.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/scroll.ts","webpack://BalloonEditor/../ckeditor5-utils/src/keyboard.ts","webpack://BalloonEditor/../ckeditor5-utils/src/toarray.ts","webpack://BalloonEditor/../ckeditor5-utils/src/translation-service.ts","webpack://BalloonEditor/../ckeditor5-utils/src/language.ts","webpack://BalloonEditor/../ckeditor5-utils/src/locale.ts","webpack://BalloonEditor/../ckeditor5-utils/src/collection.ts","webpack://BalloonEditor/../ckeditor5-utils/src/first.ts","webpack://BalloonEditor/../ckeditor5-utils/src/focustracker.ts","webpack://BalloonEditor/../ckeditor5-utils/src/keystrokehandler.ts","webpack://BalloonEditor/../ckeditor5-utils/src/tomap.ts","webpack://BalloonEditor/../ckeditor5-utils/src/objecttomap.ts","webpack://BalloonEditor/../ckeditor5-utils/src/delay.ts","webpack://BalloonEditor/../ckeditor5-utils/src/unicode.ts","webpack://BalloonEditor/../ckeditor5-core/src/plugin.ts","webpack://BalloonEditor/../ckeditor5-core/src/command.ts","webpack://BalloonEditor/../ckeditor5-core/src/multicommand.ts","webpack://BalloonEditor/../ckeditor5-core/src/plugincollection.ts","webpack://BalloonEditor/../ckeditor5-core/src/context.ts","webpack://BalloonEditor/../ckeditor5-core/src/contextplugin.ts","webpack://BalloonEditor/../ckeditor5-core/src/editingkeystrokehandler.ts","webpack://BalloonEditor/../ckeditor5-engine/theme/placeholder.css?e2af","webpack://BalloonEditor/../ckeditor5-engine/src/view/placeholder.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/typecheckable.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/clone.js","webpack://BalloonEditor/../ckeditor5-engine/src/view/node.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/text.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/textproxy.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/matcher.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/isSymbol.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_isKey.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/memoize.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_memoizeCapped.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_stringToPath.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_arrayMap.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseToString.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/toString.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_castPath.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/last.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_toKey.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseGet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseSlice.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_parent.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseUnset.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/unset.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/get.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_assignMergeValue.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseFor.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_createBaseFor.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isArrayLikeObject.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_safeGet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/toPlainObject.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseMergeDeep.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseMerge.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/identity.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_apply.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_overRest.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/constant.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseSetToString.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_shortOut.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_setToString.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseRest.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_isIterateeCall.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_createAssigner.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/merge.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseSet.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/set.js","webpack://BalloonEditor/../ckeditor5-engine/src/view/stylesmap.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/element.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/containerelement.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/editableelement.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/rooteditableelement.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/treewalker.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/position.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/range.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/selection.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/documentselection.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/bubblingeventinfo.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/bubblingemittermixin.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/document.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/attributeelement.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/emptyelement.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/uielement.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/rawelement.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/documentfragment.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/downcastwriter.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/filler.ts","webpack://BalloonEditor/../ckeditor5-engine/theme/renderer.css?dd62","webpack://BalloonEditor/../ckeditor5-engine/src/view/renderer.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/domconverter.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/isvalidattributename.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/getancestors.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/observer.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/assignIn.js","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/domeventdata.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/domeventobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/keyobserver.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/now.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_trimmedEndIndex.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseTrim.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/toNumber.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/debounce.js","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/fakeselectionobserver.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/_setCacheAdd.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_setCacheHas.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_SetCache.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_arraySome.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_cacheHas.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_equalArrays.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_mapToArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_setToArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_equalByTag.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_equalObjects.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseIsEqualDeep.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_baseIsEqual.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/isEqualWith.js","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/mutationobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/focusobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/selectionobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/compositionobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/datatransfer.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/inputobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/arrowkeysobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/tabobserver.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/cloneDeep.js","webpack://BalloonEditor/../ckeditor5-engine/src/view/view.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/typecheckable.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/node.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/nodelist.ts","webpack://BalloonEditor/../ckeditor5-utils/src/splicearray.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/text.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/textproxy.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/element.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/treewalker.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/position.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/range.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/mapper.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/modelconsumable.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/downcastdispatcher.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/selection.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/liverange.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/documentselection.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/conversionhelpers.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/downcasthelpers.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/utils/autoparagraphing.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/upcasthelpers.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/utils/selection-post-fixer.ts","webpack://BalloonEditor/../ckeditor5-engine/src/controller/editingcontroller.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/viewconsumable.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/schema.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/upcastdispatcher.ts","webpack://BalloonEditor/../ckeditor5-engine/src/dataprocessor/basichtmlwriter.ts","webpack://BalloonEditor/../ckeditor5-engine/src/dataprocessor/htmldataprocessor.ts","webpack://BalloonEditor/../ckeditor5-engine/src/controller/datacontroller.ts","webpack://BalloonEditor/../ckeditor5-engine/src/conversion/conversion.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/operation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/utils.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/moveoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/insertoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/splitoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/mergeoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/markeroperation.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/isEqual.js","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/attributeoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/nooperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/renameoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/rootattributeoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/rootoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/operationfactory.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/transform.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/liveposition.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/batch.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/differ.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/history.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/rootelement.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/document.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/markercollection.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/operation/detachoperation.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/documentfragment.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/writer.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/utils/deletecontent.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/utils/getselectedcontent.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/utils/insertcontent.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/utils/findoptimalinsertionrange.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/utils/insertobject.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/utils/modifyselection.ts","webpack://BalloonEditor/../ckeditor5-engine/src/model/model.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/clickobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/observer/mouseobserver.ts","webpack://BalloonEditor/../ckeditor5-engine/src/view/upcastwriter.ts","webpack://BalloonEditor/../ckeditor5-core/src/commandcollection.ts","webpack://BalloonEditor/../ckeditor5-core/src/editor/editor.ts","webpack://BalloonEditor/../ckeditor5-core/src/editor/utils/dataapimixin.ts","webpack://BalloonEditor/../ckeditor5-core/src/editor/utils/elementapimixin.ts","webpack://BalloonEditor/../ckeditor5-core/src/pendingactions.ts","webpack://BalloonEditor/../ckeditor5-core/theme/icons/cancel.svg","webpack://BalloonEditor/../ckeditor5-core/src/index.ts","webpack://BalloonEditor/../ckeditor5-core/theme/icons/bold.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/caption.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/check.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/cog.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/eraser.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/image.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/low-vision.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/importexport.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/paragraph.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/plus.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/text.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/align-bottom.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/align-middle.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/align-top.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/align-left.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/align-center.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/align-right.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/align-justify.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-inline-left.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-center.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-inline-right.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-full-width.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-inline.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-left.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-right.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-size-full.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-size-large.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-size-small.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/object-size-medium.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/pencil.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/pilcrow.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/quote.svg","webpack://BalloonEditor/../ckeditor5-core/theme/icons/three-vertical-dots.svg","webpack://BalloonEditor/../ckeditor5-ui/src/bindings/clickoutsidehandler.ts","webpack://BalloonEditor/../ckeditor5-ui/src/bindings/csstransitiondisablermixin.ts","webpack://BalloonEditor/../ckeditor5-ui/src/bindings/submithandler.ts","webpack://BalloonEditor/../ckeditor5-ui/src/viewcollection.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/globals/globals.css?236e","webpack://BalloonEditor/../ckeditor5-ui/src/view.ts","webpack://BalloonEditor/../ckeditor5-ui/src/template.ts","webpack://BalloonEditor/../ckeditor5-ui/src/editorui/bodycollection.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/icon/icon.css?00c8","webpack://BalloonEditor/../ckeditor5-ui/src/icon/iconview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/button/button.css?de94","webpack://BalloonEditor/../ckeditor5-ui/src/button/buttonview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/button/switchbutton.css?7bcd","webpack://BalloonEditor/../ckeditor5-ui/src/button/switchbuttonview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/colorgrid/colorgrid.css?94cc","webpack://BalloonEditor/../ckeditor5-ui/theme/components/label/label.css?fd5a","webpack://BalloonEditor/../ckeditor5-ui/src/label/labelview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/labeledfield/labeledfieldview.css?dcc2","webpack://BalloonEditor/../ckeditor5-ui/src/labeledfield/labeledfieldview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/input/input.css?d15d","webpack://BalloonEditor/../ckeditor5-ui/src/input/inputview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/inputtext/inputtextview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/dropdown/dropdownpanelview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/dropdown/dropdown.css?a927","webpack://BalloonEditor/../ckeditor5-ui/src/dropdown/dropdownview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/icons/dropdown-arrow.svg","webpack://BalloonEditor/../ckeditor5-ui/src/dropdown/button/dropdownbuttonview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/focuscycler.ts","webpack://BalloonEditor/../ckeditor5-ui/src/toolbar/toolbarseparatorview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/toolbar/toolbarlinebreakview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/toolbar/normalizetoolbarconfig.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/toolbar/toolbar.css?10d7","webpack://BalloonEditor/../ckeditor5-ui/src/toolbar/toolbarview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/bindings/preventdefault.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/list/list.css?fdbc","webpack://BalloonEditor/../ckeditor5-ui/src/list/listview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/list/listitemview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/list/listseparatorview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/dropdown/splitbutton.css?a37f","webpack://BalloonEditor/../ckeditor5-ui/src/dropdown/button/splitbuttonview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/dropdown/toolbardropdown.css?31da","webpack://BalloonEditor/../ckeditor5-ui/theme/components/dropdown/listdropdown.css?ae60","webpack://BalloonEditor/../ckeditor5-ui/src/dropdown/utils.ts","webpack://BalloonEditor/../ckeditor5-ui/src/labeledfield/utils.ts","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/utils/math.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/utils/convert.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/utils/compare.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/utils/dom.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/components/slider.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/components/hue.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/components/saturation.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/styles/color-picker.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/components/color-picker.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/styles/hue.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/styles/saturation.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/lib/entrypoints/hex.js","webpack://BalloonEditor/javascript/node_modules/vanilla-colorful/hex-color-picker.js","webpack://BalloonEditor/../ckeditor5-ui/theme/components/colorpicker/colorpicker.css?2b84","webpack://BalloonEditor/../ckeditor5-ui/src/colorselector/documentcolorcollection.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/colorselector/colorselector.css?c433","webpack://BalloonEditor/../ckeditor5-ui/src/componentfactory.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/panel/balloonpanel.css?69dc","webpack://BalloonEditor/../ckeditor5-ui/src/panel/balloon/balloonpanelview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/tooltip/tooltip.css?c7bf","webpack://BalloonEditor/../ckeditor5-ui/src/tooltipmanager.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/throttle.js","webpack://BalloonEditor/../ckeditor5-ui/src/editorui/poweredby.ts","webpack://BalloonEditor/../ckeditor5-utils/src/verifylicense.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/icons/project-logo.svg","webpack://BalloonEditor/../ckeditor5-utils/src/dom/findclosestscrollableancestor.ts","webpack://BalloonEditor/../ckeditor5-ui/src/editorui/editorui.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/editorui/editorui.css?e51b","webpack://BalloonEditor/../ckeditor5-ui/src/editorui/editoruiview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/editableui/editableuiview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/editableui/inline/inlineeditableuiview.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/formheader/formheader.css?322e","webpack://BalloonEditor/../ckeditor5-ui/src/notification/notification.ts","webpack://BalloonEditor/../ckeditor5-ui/src/model.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/panel/balloonrotator.css?580a","webpack://BalloonEditor/../ckeditor5-ui/theme/components/panel/fakepanel.css?9740","webpack://BalloonEditor/../ckeditor5-ui/src/panel/balloon/contextualballoon.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/icons/previous-arrow.svg","webpack://BalloonEditor/../ckeditor5-ui/theme/icons/next-arrow.svg","webpack://BalloonEditor/../ckeditor5-ui/theme/components/panel/stickypanel.css?4553","webpack://BalloonEditor/../ckeditor5-ui/src/panel/sticky/stickypanelview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/toolbar/balloon/balloontoolbar.ts","webpack://BalloonEditor/../ckeditor5-ui/theme/components/toolbar/blocktoolbar.css?edd0","webpack://BalloonEditor/../ckeditor5-ui/src/toolbar/block/blockbuttonview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/toolbar/block/blocktoolbar.ts","webpack://BalloonEditor/../ckeditor5-watchdog/src/watchdog.ts","webpack://BalloonEditor/../ckeditor5-watchdog/src/utils/getsubnodes.ts","webpack://BalloonEditor/../ckeditor5-watchdog/src/utils/areconnectedthroughproperties.ts","webpack://BalloonEditor/../ckeditor5-watchdog/src/editorwatchdog.ts","webpack://BalloonEditor/../ckeditor5-watchdog/src/contextwatchdog.ts","webpack://BalloonEditor/../ckeditor5-editor-balloon/src/ballooneditorui.ts","webpack://BalloonEditor/../ckeditor5-editor-balloon/src/ballooneditoruiview.ts","webpack://BalloonEditor/../ckeditor5-editor-balloon/src/ballooneditor.ts","webpack://BalloonEditor/../ckeditor5-utils/src/dom/getdatafromelement.ts","webpack://BalloonEditor/../ckeditor5-core/src/editor/utils/securesourceelement.ts","webpack://BalloonEditor/../ckeditor5-core/src/editor/utils/attachtoform.ts","webpack://BalloonEditor/../ckeditor5-clipboard/src/clipboardobserver.ts","webpack://BalloonEditor/../ckeditor5-clipboard/src/utils/viewtoplaintext.ts","webpack://BalloonEditor/../ckeditor5-clipboard/src/clipboardpipeline.ts","webpack://BalloonEditor/../ckeditor5-clipboard/src/utils/normalizeclipboarddata.ts","webpack://BalloonEditor/../ckeditor5-clipboard/src/utils/plaintexttohtml.ts","webpack://BalloonEditor/../ckeditor5-typing/src/utils/changebuffer.ts","webpack://BalloonEditor/../ckeditor5-typing/src/inserttextcommand.ts","webpack://BalloonEditor/../ckeditor5-typing/src/inserttextobserver.ts","webpack://BalloonEditor/../ckeditor5-typing/src/input.ts","webpack://BalloonEditor/../ckeditor5-typing/src/deletecommand.ts","webpack://BalloonEditor/../ckeditor5-typing/src/deleteobserver.ts","webpack://BalloonEditor/../ckeditor5-typing/src/delete.ts","webpack://BalloonEditor/../ckeditor5-typing/src/typing.ts","webpack://BalloonEditor/../ckeditor5-typing/src/utils/getlasttextline.ts","webpack://BalloonEditor/../ckeditor5-typing/src/textwatcher.ts","webpack://BalloonEditor/../ckeditor5-typing/src/twostepcaretmovement.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/escapeRegExp.js","webpack://BalloonEditor/../ckeditor5-typing/src/texttransformation.ts","webpack://BalloonEditor/../ckeditor5-typing/src/utils/findattributerange.ts","webpack://BalloonEditor/../ckeditor5-enter/src/utils.ts","webpack://BalloonEditor/../ckeditor5-enter/src/entercommand.ts","webpack://BalloonEditor/../ckeditor5-enter/src/enterobserver.ts","webpack://BalloonEditor/../ckeditor5-enter/src/enter.ts","webpack://BalloonEditor/../ckeditor5-enter/src/shiftentercommand.ts","webpack://BalloonEditor/../ckeditor5-enter/src/shiftenter.ts","webpack://BalloonEditor/../ckeditor5-widget/src/highlightstack.ts","webpack://BalloonEditor/../ckeditor5-widget/theme/icons/drag-handle.svg","webpack://BalloonEditor/../ckeditor5-widget/src/utils.ts","webpack://BalloonEditor/../ckeditor5-widget/src/widgettypearound/utils.ts","webpack://BalloonEditor/../ckeditor5-widget/theme/widgettypearound.css?6f95","webpack://BalloonEditor/../ckeditor5-widget/src/widgettypearound/widgettypearound.ts","webpack://BalloonEditor/../ckeditor5-widget/theme/icons/return-arrow.svg","webpack://BalloonEditor/../ckeditor5-widget/src/verticalnavigation.ts","webpack://BalloonEditor/../ckeditor5-widget/theme/widget.css?85b1","webpack://BalloonEditor/../ckeditor5-widget/src/widget.ts","webpack://BalloonEditor/../ckeditor5-widget/src/widgettoolbarrepository.ts","webpack://BalloonEditor/../ckeditor5-widget/src/widgetresize/resizerstate.ts","webpack://BalloonEditor/../ckeditor5-widget/src/widgetresize/resizer.ts","webpack://BalloonEditor/../ckeditor5-widget/theme/widgetresize.css?8599","webpack://BalloonEditor/../ckeditor5-clipboard/theme/clipboard.css?ab7a","webpack://BalloonEditor/../ckeditor5-clipboard/src/dragdrop.ts","webpack://BalloonEditor/../ckeditor5-clipboard/src/pasteplaintext.ts","webpack://BalloonEditor/../ckeditor5-clipboard/src/clipboard.ts","webpack://BalloonEditor/../ckeditor5-clipboard/src/lineview.ts","webpack://BalloonEditor/../ckeditor5-select-all/src/selectallcommand.ts","webpack://BalloonEditor/../ckeditor5-select-all/src/selectallediting.ts","webpack://BalloonEditor/../ckeditor5-select-all/src/selectallui.ts","webpack://BalloonEditor/../ckeditor5-select-all/theme/icons/select-all.svg","webpack://BalloonEditor/../ckeditor5-select-all/src/selectall.ts","webpack://BalloonEditor/../ckeditor5-undo/src/basecommand.ts","webpack://BalloonEditor/../ckeditor5-undo/src/undocommand.ts","webpack://BalloonEditor/../ckeditor5-undo/src/redocommand.ts","webpack://BalloonEditor/../ckeditor5-undo/src/undoediting.ts","webpack://BalloonEditor/../ckeditor5-undo/theme/icons/undo.svg","webpack://BalloonEditor/../ckeditor5-undo/theme/icons/redo.svg","webpack://BalloonEditor/../ckeditor5-undo/src/undoui.ts","webpack://BalloonEditor/../ckeditor5-undo/src/undo.ts","webpack://BalloonEditor/../ckeditor5-upload/src/filereader.ts","webpack://BalloonEditor/../ckeditor5-upload/src/filerepository.ts","webpack://BalloonEditor/../ckeditor5-upload/src/ui/filedialogbuttonview.ts","webpack://BalloonEditor/../ckeditor5-adapter-ckfinder/src/utils.ts","webpack://BalloonEditor/../ckeditor5-adapter-ckfinder/src/uploadadapter.ts","webpack://BalloonEditor/../ckeditor5-autoformat/src/blockautoformatediting.ts","webpack://BalloonEditor/../ckeditor5-autoformat/src/inlineautoformatediting.ts","webpack://BalloonEditor/../ckeditor5-autoformat/src/autoformat.ts","webpack://BalloonEditor/../ckeditor5-basic-styles/src/attributecommand.ts","webpack://BalloonEditor/../ckeditor5-basic-styles/src/bold/boldediting.ts","webpack://BalloonEditor/../ckeditor5-basic-styles/src/bold/boldui.ts","webpack://BalloonEditor/../ckeditor5-basic-styles/theme/code.css?985e","webpack://BalloonEditor/../ckeditor5-basic-styles/src/italic/italicediting.ts","webpack://BalloonEditor/../ckeditor5-basic-styles/theme/icons/italic.svg","webpack://BalloonEditor/../ckeditor5-basic-styles/src/italic/italicui.ts","webpack://BalloonEditor/../ckeditor5-block-quote/src/blockquotecommand.ts","webpack://BalloonEditor/../ckeditor5-block-quote/src/blockquoteediting.ts","webpack://BalloonEditor/../ckeditor5-block-quote/theme/blockquote.css?5a6d","webpack://BalloonEditor/../ckeditor5-block-quote/src/blockquoteui.ts","webpack://BalloonEditor/../ckeditor5-ckbox/src/ckboxui.ts","webpack://BalloonEditor/../ckeditor5-ckbox/theme/icons/browse-files.svg","webpack://BalloonEditor/../ckeditor5-ckbox/src/utils.ts","webpack://BalloonEditor/../ckeditor5-ckbox/src/ckboxcommand.ts","webpack://BalloonEditor/../ckeditor5-ckbox/src/ckboxuploadadapter.ts","webpack://BalloonEditor/../ckeditor5-ckbox/src/ckboxediting.ts","webpack://BalloonEditor/../ckeditor5-ckfinder/src/ckfinderui.ts","webpack://BalloonEditor/../ckeditor5-ckfinder/theme/icons/browse-files.svg","webpack://BalloonEditor/../ckeditor5-ckfinder/src/ckfindercommand.ts","webpack://BalloonEditor/../ckeditor5-ckfinder/src/ckfinderediting.ts","webpack://BalloonEditor/../ckeditor5-easy-image/src/cloudservicesuploadadapter.ts","webpack://BalloonEditor/../ckeditor5-paragraph/src/paragraphcommand.ts","webpack://BalloonEditor/../ckeditor5-paragraph/src/insertparagraphcommand.ts","webpack://BalloonEditor/../ckeditor5-paragraph/src/paragraph.ts","webpack://BalloonEditor/../ckeditor5-heading/src/headingcommand.ts","webpack://BalloonEditor/../ckeditor5-heading/src/headingediting.ts","webpack://BalloonEditor/../ckeditor5-heading/theme/heading.css?c486","webpack://BalloonEditor/../ckeditor5-heading/src/headingui.ts","webpack://BalloonEditor/../ckeditor5-heading/src/utils.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/utils.ts","webpack://BalloonEditor/../ckeditor5-image/src/imageutils.ts","webpack://BalloonEditor/../ckeditor5-image/src/autoimage.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagetextalternative/imagetextalternativecommand.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagetextalternative/imagetextalternativeediting.ts","webpack://BalloonEditor/../ckeditor5-image/theme/textalternativeform.css?1540","webpack://BalloonEditor/../ckeditor5-ui/theme/components/responsive-form/responsiveform.css?643e","webpack://BalloonEditor/../ckeditor5-image/src/imagetextalternative/ui/textalternativeformview.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/ui/utils.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagetextalternative/imagetextalternativeui.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagetextalternative.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/converters.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/imageloadobserver.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/insertimagecommand.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/replaceimagesourcecommand.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/imageediting.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/imagetypecommand.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/imageblockediting.ts","webpack://BalloonEditor/../ckeditor5-image/theme/image.css?c23e","webpack://BalloonEditor/../ckeditor5-image/src/imageblock.ts","webpack://BalloonEditor/../ckeditor5-image/src/image/imageinlineediting.ts","webpack://BalloonEditor/../ckeditor5-image/src/imageinline.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagecaption/imagecaptionutils.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagecaption/toggleimagecaptioncommand.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagecaption/imagecaptionediting.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagecaption/imagecaptionui.ts","webpack://BalloonEditor/../ckeditor5-image/theme/imagecaption.css?fbd1","webpack://BalloonEditor/../ckeditor5-image/src/imageupload/utils.ts","webpack://BalloonEditor/../ckeditor5-image/src/imageupload/imageuploadui.ts","webpack://BalloonEditor/../ckeditor5-image/theme/imageuploadprogress.css?3fdd","webpack://BalloonEditor/../ckeditor5-image/theme/imageuploadicon.css?e799","webpack://BalloonEditor/../ckeditor5-image/theme/imageuploadloader.css?4493","webpack://BalloonEditor/../ckeditor5-image/src/imageupload/imageuploadprogress.ts","webpack://BalloonEditor/../ckeditor5-image/src/imageupload/uploadimagecommand.ts","webpack://BalloonEditor/../ckeditor5-image/src/imageupload/imageuploadediting.ts","webpack://BalloonEditor/../ckeditor5-image/theme/imageinsertformrowview.css?4e22","webpack://BalloonEditor/../ckeditor5-image/theme/imageinsert.css?2858","webpack://BalloonEditor/../ckeditor5-image/theme/imageresize.css?f537","webpack://BalloonEditor/../ckeditor5-image/src/imagestyle/imagestylecommand.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagestyle/utils.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagestyle/converters.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagestyle/imagestyleediting.ts","webpack://BalloonEditor/../ckeditor5-image/theme/imagestyle.css?bc8e","webpack://BalloonEditor/../ckeditor5-image/src/imagestyle/imagestyleui.ts","webpack://BalloonEditor/../ckeditor5-indent/src/indentediting.ts","webpack://BalloonEditor/../ckeditor5-indent/theme/icons/indent.svg","webpack://BalloonEditor/../ckeditor5-indent/theme/icons/outdent.svg","webpack://BalloonEditor/../ckeditor5-indent/src/indentui.ts","webpack://BalloonEditor/../ckeditor5-link/src/utils/automaticdecorators.ts","webpack://BalloonEditor/javascript/node_modules/lodash-es/_castSlice.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_hasUnicode.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_asciiToArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_unicodeToArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_stringToArray.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/upperFirst.js","webpack://BalloonEditor/javascript/node_modules/lodash-es/_createCaseFirst.js","webpack://BalloonEditor/../ckeditor5-link/src/utils.ts","webpack://BalloonEditor/../ckeditor5-link/src/linkcommand.ts","webpack://BalloonEditor/../ckeditor5-link/src/unlinkcommand.ts","webpack://BalloonEditor/../ckeditor5-link/src/utils/manualdecorator.ts","webpack://BalloonEditor/../ckeditor5-link/theme/link.css?2bbb","webpack://BalloonEditor/../ckeditor5-link/src/linkediting.ts","webpack://BalloonEditor/../ckeditor5-typing/src/utils/inlinehighlight.ts","webpack://BalloonEditor/../ckeditor5-link/theme/linkform.css?91e4","webpack://BalloonEditor/../ckeditor5-link/src/ui/linkformview.ts","webpack://BalloonEditor/../ckeditor5-link/theme/linkactions.css?bbc6","webpack://BalloonEditor/../ckeditor5-link/src/ui/linkactionsview.ts","webpack://BalloonEditor/../ckeditor5-link/theme/icons/unlink.svg","webpack://BalloonEditor/../ckeditor5-link/theme/icons/link.svg","webpack://BalloonEditor/../ckeditor5-link/src/linkui.ts","webpack://BalloonEditor/../ckeditor5-link/src/autolink.ts","webpack://BalloonEditor/../ckeditor5-link/theme/linkimage.css?4624","webpack://BalloonEditor/../ckeditor5-list/src/documentlist/utils/listwalker.ts","webpack://BalloonEditor/../ckeditor5-list/theme/documentlist.css?2655","webpack://BalloonEditor/../ckeditor5-list/theme/list.css?89de","webpack://BalloonEditor/../ckeditor5-list/src/list/utils.ts","webpack://BalloonEditor/../ckeditor5-list/src/list/listui.ts","webpack://BalloonEditor/../ckeditor5-list/theme/icons/numberedlist.svg","webpack://BalloonEditor/../ckeditor5-list/theme/icons/bulletedlist.svg","webpack://BalloonEditor/../ckeditor5-list/src/documentlistproperties/utils/style.ts","webpack://BalloonEditor/../ckeditor5-list/theme/collapsible.css?bc89","webpack://BalloonEditor/../ckeditor5-list/theme/listproperties.css?617f","webpack://BalloonEditor/../ckeditor5-list/theme/liststyles.css?4e4d","webpack://BalloonEditor/../ckeditor5-list/src/list/indentcommand.ts","webpack://BalloonEditor/../ckeditor5-list/src/list/listcommand.ts","webpack://BalloonEditor/../ckeditor5-list/src/list/listutils.ts","webpack://BalloonEditor/../ckeditor5-list/src/list/converters.ts","webpack://BalloonEditor/../ckeditor5-list/src/list/listediting.ts","webpack://BalloonEditor/../ckeditor5-list/src/todolist/todolistediting.ts","webpack://BalloonEditor/../ckeditor5-list/theme/todolist.css?b9a9","webpack://BalloonEditor/../ckeditor5-media-embed/src/converters.ts","webpack://BalloonEditor/../ckeditor5-media-embed/src/utils.ts","webpack://BalloonEditor/../ckeditor5-media-embed/src/mediaembedcommand.ts","webpack://BalloonEditor/../ckeditor5-media-embed/src/mediaregistry.ts","webpack://BalloonEditor/../ckeditor5-media-embed/theme/icons/media-placeholder.svg","webpack://BalloonEditor/../ckeditor5-media-embed/theme/mediaembedediting.css?5708","webpack://BalloonEditor/../ckeditor5-media-embed/src/mediaembedediting.ts","webpack://BalloonEditor/../ckeditor5-media-embed/src/automediaembed.ts","webpack://BalloonEditor/../ckeditor5-media-embed/theme/mediaform.css?2610","webpack://BalloonEditor/../ckeditor5-media-embed/src/ui/mediaformview.ts","webpack://BalloonEditor/../ckeditor5-media-embed/src/mediaembedui.ts","webpack://BalloonEditor/../ckeditor5-media-embed/theme/icons/media.svg","webpack://BalloonEditor/../ckeditor5-media-embed/theme/mediaembed.css?9783","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/list.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/image.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/normalizers/mswordnormalizer.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/table.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/br.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/normalizers/googledocsnormalizer.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/removeboldwrapper.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/normalizers/googlesheetsnormalizer.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/removegooglesheetstag.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/removexmlns.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/removeinvalidtablewidth.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/removestyleblock.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/space.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/filters/parse.ts","webpack://BalloonEditor/../ckeditor5-table/src/utils/common.ts","webpack://BalloonEditor/../ckeditor5-table/src/converters/upcasttable.ts","webpack://BalloonEditor/../ckeditor5-table/src/tablewalker.ts","webpack://BalloonEditor/../ckeditor5-table/src/converters/downcast.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/inserttablecommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/insertrowcommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/insertcolumncommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/splitcellcommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/utils/structure.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/mergecellcommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/removerowcommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/removecolumncommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/setheaderrowcommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/setheadercolumncommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/tableutils.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/mergecellscommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/selectrowcommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/commands/selectcolumncommand.ts","webpack://BalloonEditor/../ckeditor5-table/src/converters/table-layout-post-fixer.ts","webpack://BalloonEditor/../ckeditor5-table/src/converters/table-cell-paragraph-post-fixer.ts","webpack://BalloonEditor/../ckeditor5-table/src/converters/table-cell-refresh-handler.ts","webpack://BalloonEditor/../ckeditor5-table/theme/tableediting.css?f65e","webpack://BalloonEditor/../ckeditor5-table/src/tableediting.ts","webpack://BalloonEditor/../ckeditor5-table/src/converters/table-headings-refresh-handler.ts","webpack://BalloonEditor/../ckeditor5-table/theme/inserttable.css?466f","webpack://BalloonEditor/../ckeditor5-table/src/ui/inserttableview.ts","webpack://BalloonEditor/../ckeditor5-ui/src/bindings/addkeyboardhandlingforgrid.ts","webpack://BalloonEditor/../ckeditor5-table/src/tableui.ts","webpack://BalloonEditor/../ckeditor5-table/theme/icons/table.svg","webpack://BalloonEditor/../ckeditor5-table/theme/icons/table-column.svg","webpack://BalloonEditor/../ckeditor5-table/theme/icons/table-row.svg","webpack://BalloonEditor/../ckeditor5-table/theme/icons/table-merge-cell.svg","webpack://BalloonEditor/../ckeditor5-table/theme/tableselection.css?ee02","webpack://BalloonEditor/../ckeditor5-table/src/tableselection.ts","webpack://BalloonEditor/../ckeditor5-table/src/tableclipboard.ts","webpack://BalloonEditor/../ckeditor5-table/src/tablekeyboard.ts","webpack://BalloonEditor/../ckeditor5-table/src/tablemouse/mouseeventsobserver.ts","webpack://BalloonEditor/../ckeditor5-table/src/tablemouse.ts","webpack://BalloonEditor/../ckeditor5-table/theme/table.css?fded","webpack://BalloonEditor/../ckeditor5-table/src/utils/ui/widget.ts","webpack://BalloonEditor/../ckeditor5-table/theme/colorinput.css?2edd","webpack://BalloonEditor/../ckeditor5-table/theme/formrow.css?813a","webpack://BalloonEditor/../ckeditor5-table/theme/form.css?6b23","webpack://BalloonEditor/../ckeditor5-table/theme/tableform.css?6102","webpack://BalloonEditor/../ckeditor5-table/theme/tablecellproperties.css?e7e9","webpack://BalloonEditor/../ckeditor5-table/src/utils/ui/contextualballoon.ts","webpack://BalloonEditor/../ckeditor5-table/theme/tableproperties.css?fbdb","webpack://BalloonEditor/../ckeditor5-table/theme/tablecaption.css?327b","webpack://BalloonEditor/../ckeditor5-table/theme/tablecolumnresize.css?00bf","webpack://BalloonEditor/../ckeditor5-cloud-services/src/token/token.ts","webpack://BalloonEditor/../ckeditor5-cloud-services/src/uploadgateway/fileuploader.ts","webpack://BalloonEditor/../ckeditor5-cloud-services/src/uploadgateway/uploadgateway.ts","webpack://BalloonEditor/../ckeditor5-cloud-services/src/cloudservicescore.ts","webpack://BalloonEditor/./theme/theme.css?d9e3","webpack://BalloonEditor/./src/ckeditor.ts","webpack://BalloonEditor/../ckeditor5-essentials/src/essentials.ts","webpack://BalloonEditor/../ckeditor5-basic-styles/src/bold.ts","webpack://BalloonEditor/../ckeditor5-basic-styles/src/italic.ts","webpack://BalloonEditor/../ckeditor5-block-quote/src/blockquote.ts","webpack://BalloonEditor/../ckeditor5-ckbox/src/ckbox.ts","webpack://BalloonEditor/../ckeditor5-ckfinder/src/ckfinder.ts","webpack://BalloonEditor/../ckeditor5-cloud-services/src/cloudservices.ts","webpack://BalloonEditor/../ckeditor5-easy-image/src/easyimage.ts","webpack://BalloonEditor/../ckeditor5-heading/src/heading.ts","webpack://BalloonEditor/../ckeditor5-image/src/image.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagecaption.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagestyle.ts","webpack://BalloonEditor/../ckeditor5-image/src/imagetoolbar.ts","webpack://BalloonEditor/../ckeditor5-image/src/imageupload.ts","webpack://BalloonEditor/../ckeditor5-indent/src/indent.ts","webpack://BalloonEditor/../ckeditor5-link/src/link.ts","webpack://BalloonEditor/../ckeditor5-list/src/list.ts","webpack://BalloonEditor/../ckeditor5-media-embed/src/mediaembed.ts","webpack://BalloonEditor/../ckeditor5-paste-from-office/src/pastefromoffice.ts","webpack://BalloonEditor/../ckeditor5-image/src/pictureediting.ts","webpack://BalloonEditor/../ckeditor5-table/src/table.ts","webpack://BalloonEditor/../ckeditor5-table/src/tabletoolbar.ts"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory();\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"BalloonEditor\"] = factory();\n\telse\n\t\troot[\"BalloonEditor\"] = factory();\n})(self, () => {\nreturn ","/* MIT license */\n/* eslint-disable no-mixed-operators */\nconst cssKeywords = require('color-name');\n\n// NOTE: conversions should only return primitive values (i.e. arrays, or\n// values that give correct `typeof` results).\n// do not use box values types (i.e. Number(), String(), etc.)\n\nconst reverseKeywords = {};\nfor (const key of Object.keys(cssKeywords)) {\n\treverseKeywords[cssKeywords[key]] = key;\n}\n\nconst convert = {\n\trgb: {channels: 3, labels: 'rgb'},\n\thsl: {channels: 3, labels: 'hsl'},\n\thsv: {channels: 3, labels: 'hsv'},\n\thwb: {channels: 3, labels: 'hwb'},\n\tcmyk: {channels: 4, labels: 'cmyk'},\n\txyz: {channels: 3, labels: 'xyz'},\n\tlab: {channels: 3, labels: 'lab'},\n\tlch: {channels: 3, labels: 'lch'},\n\thex: {channels: 1, labels: ['hex']},\n\tkeyword: {channels: 1, labels: ['keyword']},\n\tansi16: {channels: 1, labels: ['ansi16']},\n\tansi256: {channels: 1, labels: ['ansi256']},\n\thcg: {channels: 3, labels: ['h', 'c', 'g']},\n\tapple: {channels: 3, labels: ['r16', 'g16', 'b16']},\n\tgray: {channels: 1, labels: ['gray']}\n};\n\nmodule.exports = convert;\n\n// Hide .channels and .labels properties\nfor (const model of Object.keys(convert)) {\n\tif (!('channels' in convert[model])) {\n\t\tthrow new Error('missing channels property: ' + model);\n\t}\n\n\tif (!('labels' in convert[model])) {\n\t\tthrow new Error('missing channel labels property: ' + model);\n\t}\n\n\tif (convert[model].labels.length !== convert[model].channels) {\n\t\tthrow new Error('channel and label counts mismatch: ' + model);\n\t}\n\n\tconst {channels, labels} = convert[model];\n\tdelete convert[model].channels;\n\tdelete convert[model].labels;\n\tObject.defineProperty(convert[model], 'channels', {value: channels});\n\tObject.defineProperty(convert[model], 'labels', {value: labels});\n}\n\nconvert.rgb.hsl = function (rgb) {\n\tconst r = rgb[0] / 255;\n\tconst g = rgb[1] / 255;\n\tconst b = rgb[2] / 255;\n\tconst min = Math.min(r, g, b);\n\tconst max = Math.max(r, g, b);\n\tconst delta = max - min;\n\tlet h;\n\tlet s;\n\n\tif (max === min) {\n\t\th = 0;\n\t} else if (r === max) {\n\t\th = (g - b) / delta;\n\t} else if (g === max) {\n\t\th = 2 + (b - r) / delta;\n\t} else if (b === max) {\n\t\th = 4 + (r - g) / delta;\n\t}\n\n\th = Math.min(h * 60, 360);\n\n\tif (h < 0) {\n\t\th += 360;\n\t}\n\n\tconst l = (min + max) / 2;\n\n\tif (max === min) {\n\t\ts = 0;\n\t} else if (l <= 0.5) {\n\t\ts = delta / (max + min);\n\t} else {\n\t\ts = delta / (2 - max - min);\n\t}\n\n\treturn [h, s * 100, l * 100];\n};\n\nconvert.rgb.hsv = function (rgb) {\n\tlet rdif;\n\tlet gdif;\n\tlet bdif;\n\tlet h;\n\tlet s;\n\n\tconst r = rgb[0] / 255;\n\tconst g = rgb[1] / 255;\n\tconst b = rgb[2] / 255;\n\tconst v = Math.max(r, g, b);\n\tconst diff = v - Math.min(r, g, b);\n\tconst diffc = function (c) {\n\t\treturn (v - c) / 6 / diff + 1 / 2;\n\t};\n\n\tif (diff === 0) {\n\t\th = 0;\n\t\ts = 0;\n\t} else {\n\t\ts = diff / v;\n\t\trdif = diffc(r);\n\t\tgdif = diffc(g);\n\t\tbdif = diffc(b);\n\n\t\tif (r === v) {\n\t\t\th = bdif - gdif;\n\t\t} else if (g === v) {\n\t\t\th = (1 / 3) + rdif - bdif;\n\t\t} else if (b === v) {\n\t\t\th = (2 / 3) + gdif - rdif;\n\t\t}\n\n\t\tif (h < 0) {\n\t\t\th += 1;\n\t\t} else if (h > 1) {\n\t\t\th -= 1;\n\t\t}\n\t}\n\n\treturn [\n\t\th * 360,\n\t\ts * 100,\n\t\tv * 100\n\t];\n};\n\nconvert.rgb.hwb = function (rgb) {\n\tconst r = rgb[0];\n\tconst g = rgb[1];\n\tlet b = rgb[2];\n\tconst h = convert.rgb.hsl(rgb)[0];\n\tconst w = 1 / 255 * Math.min(r, Math.min(g, b));\n\n\tb = 1 - 1 / 255 * Math.max(r, Math.max(g, b));\n\n\treturn [h, w * 100, b * 100];\n};\n\nconvert.rgb.cmyk = function (rgb) {\n\tconst r = rgb[0] / 255;\n\tconst g = rgb[1] / 255;\n\tconst b = rgb[2] / 255;\n\n\tconst k = Math.min(1 - r, 1 - g, 1 - b);\n\tconst c = (1 - r - k) / (1 - k) || 0;\n\tconst m = (1 - g - k) / (1 - k) || 0;\n\tconst y = (1 - b - k) / (1 - k) || 0;\n\n\treturn [c * 100, m * 100, y * 100, k * 100];\n};\n\nfunction comparativeDistance(x, y) {\n\t/*\n\t\tSee https://en.m.wikipedia.org/wiki/Euclidean_distance#Squared_Euclidean_distance\n\t*/\n\treturn (\n\t\t((x[0] - y[0]) ** 2) +\n\t\t((x[1] - y[1]) ** 2) +\n\t\t((x[2] - y[2]) ** 2)\n\t);\n}\n\nconvert.rgb.keyword = function (rgb) {\n\tconst reversed = reverseKeywords[rgb];\n\tif (reversed) {\n\t\treturn reversed;\n\t}\n\n\tlet currentClosestDistance = Infinity;\n\tlet currentClosestKeyword;\n\n\tfor (const keyword of Object.keys(cssKeywords)) {\n\t\tconst value = cssKeywords[keyword];\n\n\t\t// Compute comparative distance\n\t\tconst distance = comparativeDistance(rgb, value);\n\n\t\t// Check if its less, if so set as closest\n\t\tif (distance < currentClosestDistance) {\n\t\t\tcurrentClosestDistance = distance;\n\t\t\tcurrentClosestKeyword = keyword;\n\t\t}\n\t}\n\n\treturn currentClosestKeyword;\n};\n\nconvert.keyword.rgb = function (keyword) {\n\treturn cssKeywords[keyword];\n};\n\nconvert.rgb.xyz = function (rgb) {\n\tlet r = rgb[0] / 255;\n\tlet g = rgb[1] / 255;\n\tlet b = rgb[2] / 255;\n\n\t// Assume sRGB\n\tr = r > 0.04045 ? (((r + 0.055) / 1.055) ** 2.4) : (r / 12.92);\n\tg = g > 0.04045 ? (((g + 0.055) / 1.055) ** 2.4) : (g / 12.92);\n\tb = b > 0.04045 ? (((b + 0.055) / 1.055) ** 2.4) : (b / 12.92);\n\n\tconst x = (r * 0.4124) + (g * 0.3576) + (b * 0.1805);\n\tconst y = (r * 0.2126) + (g * 0.7152) + (b * 0.0722);\n\tconst z = (r * 0.0193) + (g * 0.1192) + (b * 0.9505);\n\n\treturn [x * 100, y * 100, z * 100];\n};\n\nconvert.rgb.lab = function (rgb) {\n\tconst xyz = convert.rgb.xyz(rgb);\n\tlet x = xyz[0];\n\tlet y = xyz[1];\n\tlet z = xyz[2];\n\n\tx /= 95.047;\n\ty /= 100;\n\tz /= 108.883;\n\n\tx = x > 0.008856 ? (x ** (1 / 3)) : (7.787 * x) + (16 / 116);\n\ty = y > 0.008856 ? (y ** (1 / 3)) : (7.787 * y) + (16 / 116);\n\tz = z > 0.008856 ? (z ** (1 / 3)) : (7.787 * z) + (16 / 116);\n\n\tconst l = (116 * y) - 16;\n\tconst a = 500 * (x - y);\n\tconst b = 200 * (y - z);\n\n\treturn [l, a, b];\n};\n\nconvert.hsl.rgb = function (hsl) {\n\tconst h = hsl[0] / 360;\n\tconst s = hsl[1] / 100;\n\tconst l = hsl[2] / 100;\n\tlet t2;\n\tlet t3;\n\tlet val;\n\n\tif (s === 0) {\n\t\tval = l * 255;\n\t\treturn [val, val, val];\n\t}\n\n\tif (l < 0.5) {\n\t\tt2 = l * (1 + s);\n\t} else {\n\t\tt2 = l + s - l * s;\n\t}\n\n\tconst t1 = 2 * l - t2;\n\n\tconst rgb = [0, 0, 0];\n\tfor (let i = 0; i < 3; i++) {\n\t\tt3 = h + 1 / 3 * -(i - 1);\n\t\tif (t3 < 0) {\n\t\t\tt3++;\n\t\t}\n\n\t\tif (t3 > 1) {\n\t\t\tt3--;\n\t\t}\n\n\t\tif (6 * t3 < 1) {\n\t\t\tval = t1 + (t2 - t1) * 6 * t3;\n\t\t} else if (2 * t3 < 1) {\n\t\t\tval = t2;\n\t\t} else if (3 * t3 < 2) {\n\t\t\tval = t1 + (t2 - t1) * (2 / 3 - t3) * 6;\n\t\t} else {\n\t\t\tval = t1;\n\t\t}\n\n\t\trgb[i] = val * 255;\n\t}\n\n\treturn rgb;\n};\n\nconvert.hsl.hsv = function (hsl) {\n\tconst h = hsl[0];\n\tlet s = hsl[1] / 100;\n\tlet l = hsl[2] / 100;\n\tlet smin = s;\n\tconst lmin = Math.max(l, 0.01);\n\n\tl *= 2;\n\ts *= (l <= 1) ? l : 2 - l;\n\tsmin *= lmin <= 1 ? lmin : 2 - lmin;\n\tconst v = (l + s) / 2;\n\tconst sv = l === 0 ? (2 * smin) / (lmin + smin) : (2 * s) / (l + s);\n\n\treturn [h, sv * 100, v * 100];\n};\n\nconvert.hsv.rgb = function (hsv) {\n\tconst h = hsv[0] / 60;\n\tconst s = hsv[1] / 100;\n\tlet v = hsv[2] / 100;\n\tconst hi = Math.floor(h) % 6;\n\n\tconst f = h - Math.floor(h);\n\tconst p = 255 * v * (1 - s);\n\tconst q = 255 * v * (1 - (s * f));\n\tconst t = 255 * v * (1 - (s * (1 - f)));\n\tv *= 255;\n\n\tswitch (hi) {\n\t\tcase 0:\n\t\t\treturn [v, t, p];\n\t\tcase 1:\n\t\t\treturn [q, v, p];\n\t\tcase 2:\n\t\t\treturn [p, v, t];\n\t\tcase 3:\n\t\t\treturn [p, q, v];\n\t\tcase 4:\n\t\t\treturn [t, p, v];\n\t\tcase 5:\n\t\t\treturn [v, p, q];\n\t}\n};\n\nconvert.hsv.hsl = function (hsv) {\n\tconst h = hsv[0];\n\tconst s = hsv[1] / 100;\n\tconst v = hsv[2] / 100;\n\tconst vmin = Math.max(v, 0.01);\n\tlet sl;\n\tlet l;\n\n\tl = (2 - s) * v;\n\tconst lmin = (2 - s) * vmin;\n\tsl = s * vmin;\n\tsl /= (lmin <= 1) ? lmin : 2 - lmin;\n\tsl = sl || 0;\n\tl /= 2;\n\n\treturn [h, sl * 100, l * 100];\n};\n\n// http://dev.w3.org/csswg/css-color/#hwb-to-rgb\nconvert.hwb.rgb = function (hwb) {\n\tconst h = hwb[0] / 360;\n\tlet wh = hwb[1] / 100;\n\tlet bl = hwb[2] / 100;\n\tconst ratio = wh + bl;\n\tlet f;\n\n\t// Wh + bl cant be > 1\n\tif (ratio > 1) {\n\t\twh /= ratio;\n\t\tbl /= ratio;\n\t}\n\n\tconst i = Math.floor(6 * h);\n\tconst v = 1 - bl;\n\tf = 6 * h - i;\n\n\tif ((i & 0x01) !== 0) {\n\t\tf = 1 - f;\n\t}\n\n\tconst n = wh + f * (v - wh); // Linear interpolation\n\n\tlet r;\n\tlet g;\n\tlet b;\n\t/* eslint-disable max-statements-per-line,no-multi-spaces */\n\tswitch (i) {\n\t\tdefault:\n\t\tcase 6:\n\t\tcase 0: r = v; g = n; b = wh; break;\n\t\tcase 1: r = n; g = v; b = wh; break;\n\t\tcase 2: r = wh; g = v; b = n; break;\n\t\tcase 3: r = wh; g = n; b = v; break;\n\t\tcase 4: r = n; g = wh; b = v; break;\n\t\tcase 5: r = v; g = wh; b = n; break;\n\t}\n\t/* eslint-enable max-statements-per-line,no-multi-spaces */\n\n\treturn [r * 255, g * 255, b * 255];\n};\n\nconvert.cmyk.rgb = function (cmyk) {\n\tconst c = cmyk[0] / 100;\n\tconst m = cmyk[1] / 100;\n\tconst y = cmyk[2] / 100;\n\tconst k = cmyk[3] / 100;\n\n\tconst r = 1 - Math.min(1, c * (1 - k) + k);\n\tconst g = 1 - Math.min(1, m * (1 - k) + k);\n\tconst b = 1 - Math.min(1, y * (1 - k) + k);\n\n\treturn [r * 255, g * 255, b * 255];\n};\n\nconvert.xyz.rgb = function (xyz) {\n\tconst x = xyz[0] / 100;\n\tconst y = xyz[1] / 100;\n\tconst z = xyz[2] / 100;\n\tlet r;\n\tlet g;\n\tlet b;\n\n\tr = (x * 3.2406) + (y * -1.5372) + (z * -0.4986);\n\tg = (x * -0.9689) + (y * 1.8758) + (z * 0.0415);\n\tb = (x * 0.0557) + (y * -0.2040) + (z * 1.0570);\n\n\t// Assume sRGB\n\tr = r > 0.0031308\n\t\t? ((1.055 * (r ** (1.0 / 2.4))) - 0.055)\n\t\t: r * 12.92;\n\n\tg = g > 0.0031308\n\t\t? ((1.055 * (g ** (1.0 / 2.4))) - 0.055)\n\t\t: g * 12.92;\n\n\tb = b > 0.0031308\n\t\t? ((1.055 * (b ** (1.0 / 2.4))) - 0.055)\n\t\t: b * 12.92;\n\n\tr = Math.min(Math.max(0, r), 1);\n\tg = Math.min(Math.max(0, g), 1);\n\tb = Math.min(Math.max(0, b), 1);\n\n\treturn [r * 255, g * 255, b * 255];\n};\n\nconvert.xyz.lab = function (xyz) {\n\tlet x = xyz[0];\n\tlet y = xyz[1];\n\tlet z = xyz[2];\n\n\tx /= 95.047;\n\ty /= 100;\n\tz /= 108.883;\n\n\tx = x > 0.008856 ? (x ** (1 / 3)) : (7.787 * x) + (16 / 116);\n\ty = y > 0.008856 ? (y ** (1 / 3)) : (7.787 * y) + (16 / 116);\n\tz = z > 0.008856 ? (z ** (1 / 3)) : (7.787 * z) + (16 / 116);\n\n\tconst l = (116 * y) - 16;\n\tconst a = 500 * (x - y);\n\tconst b = 200 * (y - z);\n\n\treturn [l, a, b];\n};\n\nconvert.lab.xyz = function (lab) {\n\tconst l = lab[0];\n\tconst a = lab[1];\n\tconst b = lab[2];\n\tlet x;\n\tlet y;\n\tlet z;\n\n\ty = (l + 16) / 116;\n\tx = a / 500 + y;\n\tz = y - b / 200;\n\n\tconst y2 = y ** 3;\n\tconst x2 = x ** 3;\n\tconst z2 = z ** 3;\n\ty = y2 > 0.008856 ? y2 : (y - 16 / 116) / 7.787;\n\tx = x2 > 0.008856 ? x2 : (x - 16 / 116) / 7.787;\n\tz = z2 > 0.008856 ? z2 : (z - 16 / 116) / 7.787;\n\n\tx *= 95.047;\n\ty *= 100;\n\tz *= 108.883;\n\n\treturn [x, y, z];\n};\n\nconvert.lab.lch = function (lab) {\n\tconst l = lab[0];\n\tconst a = lab[1];\n\tconst b = lab[2];\n\tlet h;\n\n\tconst hr = Math.atan2(b, a);\n\th = hr * 360 / 2 / Math.PI;\n\n\tif (h < 0) {\n\t\th += 360;\n\t}\n\n\tconst c = Math.sqrt(a * a + b * b);\n\n\treturn [l, c, h];\n};\n\nconvert.lch.lab = function (lch) {\n\tconst l = lch[0];\n\tconst c = lch[1];\n\tconst h = lch[2];\n\n\tconst hr = h / 360 * 2 * Math.PI;\n\tconst a = c * Math.cos(hr);\n\tconst b = c * Math.sin(hr);\n\n\treturn [l, a, b];\n};\n\nconvert.rgb.ansi16 = function (args, saturation = null) {\n\tconst [r, g, b] = args;\n\tlet value = saturation === null ? convert.rgb.hsv(args)[2] : saturation; // Hsv -> ansi16 optimization\n\n\tvalue = Math.round(value / 50);\n\n\tif (value === 0) {\n\t\treturn 30;\n\t}\n\n\tlet ansi = 30\n\t\t+ ((Math.round(b / 255) << 2)\n\t\t| (Math.round(g / 255) << 1)\n\t\t| Math.round(r / 255));\n\n\tif (value === 2) {\n\t\tansi += 60;\n\t}\n\n\treturn ansi;\n};\n\nconvert.hsv.ansi16 = function (args) {\n\t// Optimization here; we already know the value and don't need to get\n\t// it converted for us.\n\treturn convert.rgb.ansi16(convert.hsv.rgb(args), args[2]);\n};\n\nconvert.rgb.ansi256 = function (args) {\n\tconst r = args[0];\n\tconst g = args[1];\n\tconst b = args[2];\n\n\t// We use the extended greyscale palette here, with the exception of\n\t// black and white. normal palette only has 4 greyscale shades.\n\tif (r === g && g === b) {\n\t\tif (r < 8) {\n\t\t\treturn 16;\n\t\t}\n\n\t\tif (r > 248) {\n\t\t\treturn 231;\n\t\t}\n\n\t\treturn Math.round(((r - 8) / 247) * 24) + 232;\n\t}\n\n\tconst ansi = 16\n\t\t+ (36 * Math.round(r / 255 * 5))\n\t\t+ (6 * Math.round(g / 255 * 5))\n\t\t+ Math.round(b / 255 * 5);\n\n\treturn ansi;\n};\n\nconvert.ansi16.rgb = function (args) {\n\tlet color = args % 10;\n\n\t// Handle greyscale\n\tif (color === 0 || color === 7) {\n\t\tif (args > 50) {\n\t\t\tcolor += 3.5;\n\t\t}\n\n\t\tcolor = color / 10.5 * 255;\n\n\t\treturn [color, color, color];\n\t}\n\n\tconst mult = (~~(args > 50) + 1) * 0.5;\n\tconst r = ((color & 1) * mult) * 255;\n\tconst g = (((color >> 1) & 1) * mult) * 255;\n\tconst b = (((color >> 2) & 1) * mult) * 255;\n\n\treturn [r, g, b];\n};\n\nconvert.ansi256.rgb = function (args) {\n\t// Handle greyscale\n\tif (args >= 232) {\n\t\tconst c = (args - 232) * 10 + 8;\n\t\treturn [c, c, c];\n\t}\n\n\targs -= 16;\n\n\tlet rem;\n\tconst r = Math.floor(args / 36) / 5 * 255;\n\tconst g = Math.floor((rem = args % 36) / 6) / 5 * 255;\n\tconst b = (rem % 6) / 5 * 255;\n\n\treturn [r, g, b];\n};\n\nconvert.rgb.hex = function (args) {\n\tconst integer = ((Math.round(args[0]) & 0xFF) << 16)\n\t\t+ ((Math.round(args[1]) & 0xFF) << 8)\n\t\t+ (Math.round(args[2]) & 0xFF);\n\n\tconst string = integer.toString(16).toUpperCase();\n\treturn '000000'.substring(string.length) + string;\n};\n\nconvert.hex.rgb = function (args) {\n\tconst match = args.toString(16).match(/[a-f0-9]{6}|[a-f0-9]{3}/i);\n\tif (!match) {\n\t\treturn [0, 0, 0];\n\t}\n\n\tlet colorString = match[0];\n\n\tif (match[0].length === 3) {\n\t\tcolorString = colorString.split('').map(char => {\n\t\t\treturn char + char;\n\t\t}).join('');\n\t}\n\n\tconst integer = parseInt(colorString, 16);\n\tconst r = (integer >> 16) & 0xFF;\n\tconst g = (integer >> 8) & 0xFF;\n\tconst b = integer & 0xFF;\n\n\treturn [r, g, b];\n};\n\nconvert.rgb.hcg = function (rgb) {\n\tconst r = rgb[0] / 255;\n\tconst g = rgb[1] / 255;\n\tconst b = rgb[2] / 255;\n\tconst max = Math.max(Math.max(r, g), b);\n\tconst min = Math.min(Math.min(r, g), b);\n\tconst chroma = (max - min);\n\tlet grayscale;\n\tlet hue;\n\n\tif (chroma < 1) {\n\t\tgrayscale = min / (1 - chroma);\n\t} else {\n\t\tgrayscale = 0;\n\t}\n\n\tif (chroma <= 0) {\n\t\thue = 0;\n\t} else\n\tif (max === r) {\n\t\thue = ((g - b) / chroma) % 6;\n\t} else\n\tif (max === g) {\n\t\thue = 2 + (b - r) / chroma;\n\t} else {\n\t\thue = 4 + (r - g) / chroma;\n\t}\n\n\thue /= 6;\n\thue %= 1;\n\n\treturn [hue * 360, chroma * 100, grayscale * 100];\n};\n\nconvert.hsl.hcg = function (hsl) {\n\tconst s = hsl[1] / 100;\n\tconst l = hsl[2] / 100;\n\n\tconst c = l < 0.5 ? (2.0 * s * l) : (2.0 * s * (1.0 - l));\n\n\tlet f = 0;\n\tif (c < 1.0) {\n\t\tf = (l - 0.5 * c) / (1.0 - c);\n\t}\n\n\treturn [hsl[0], c * 100, f * 100];\n};\n\nconvert.hsv.hcg = function (hsv) {\n\tconst s = hsv[1] / 100;\n\tconst v = hsv[2] / 100;\n\n\tconst c = s * v;\n\tlet f = 0;\n\n\tif (c < 1.0) {\n\t\tf = (v - c) / (1 - c);\n\t}\n\n\treturn [hsv[0], c * 100, f * 100];\n};\n\nconvert.hcg.rgb = function (hcg) {\n\tconst h = hcg[0] / 360;\n\tconst c = hcg[1] / 100;\n\tconst g = hcg[2] / 100;\n\n\tif (c === 0.0) {\n\t\treturn [g * 255, g * 255, g * 255];\n\t}\n\n\tconst pure = [0, 0, 0];\n\tconst hi = (h % 1) * 6;\n\tconst v = hi % 1;\n\tconst w = 1 - v;\n\tlet mg = 0;\n\n\t/* eslint-disable max-statements-per-line */\n\tswitch (Math.floor(hi)) {\n\t\tcase 0:\n\t\t\tpure[0] = 1; pure[1] = v; pure[2] = 0; break;\n\t\tcase 1:\n\t\t\tpure[0] = w; pure[1] = 1; pure[2] = 0; break;\n\t\tcase 2:\n\t\t\tpure[0] = 0; pure[1] = 1; pure[2] = v; break;\n\t\tcase 3:\n\t\t\tpure[0] = 0; pure[1] = w; pure[2] = 1; break;\n\t\tcase 4:\n\t\t\tpure[0] = v; pure[1] = 0; pure[2] = 1; break;\n\t\tdefault:\n\t\t\tpure[0] = 1; pure[1] = 0; pure[2] = w;\n\t}\n\t/* eslint-enable max-statements-per-line */\n\n\tmg = (1.0 - c) * g;\n\n\treturn [\n\t\t(c * pure[0] + mg) * 255,\n\t\t(c * pure[1] + mg) * 255,\n\t\t(c * pure[2] + mg) * 255\n\t];\n};\n\nconvert.hcg.hsv = function (hcg) {\n\tconst c = hcg[1] / 100;\n\tconst g = hcg[2] / 100;\n\n\tconst v = c + g * (1.0 - c);\n\tlet f = 0;\n\n\tif (v > 0.0) {\n\t\tf = c / v;\n\t}\n\n\treturn [hcg[0], f * 100, v * 100];\n};\n\nconvert.hcg.hsl = function (hcg) {\n\tconst c = hcg[1] / 100;\n\tconst g = hcg[2] / 100;\n\n\tconst l = g * (1.0 - c) + 0.5 * c;\n\tlet s = 0;\n\n\tif (l > 0.0 && l < 0.5) {\n\t\ts = c / (2 * l);\n\t} else\n\tif (l >= 0.5 && l < 1.0) {\n\t\ts = c / (2 * (1 - l));\n\t}\n\n\treturn [hcg[0], s * 100, l * 100];\n};\n\nconvert.hcg.hwb = function (hcg) {\n\tconst c = hcg[1] / 100;\n\tconst g = hcg[2] / 100;\n\tconst v = c + g * (1.0 - c);\n\treturn [hcg[0], (v - c) * 100, (1 - v) * 100];\n};\n\nconvert.hwb.hcg = function (hwb) {\n\tconst w = hwb[1] / 100;\n\tconst b = hwb[2] / 100;\n\tconst v = 1 - b;\n\tconst c = v - w;\n\tlet g = 0;\n\n\tif (c < 1) {\n\t\tg = (v - c) / (1 - c);\n\t}\n\n\treturn [hwb[0], c * 100, g * 100];\n};\n\nconvert.apple.rgb = function (apple) {\n\treturn [(apple[0] / 65535) * 255, (apple[1] / 65535) * 255, (apple[2] / 65535) * 255];\n};\n\nconvert.rgb.apple = function (rgb) {\n\treturn [(rgb[0] / 255) * 65535, (rgb[1] / 255) * 65535, (rgb[2] / 255) * 65535];\n};\n\nconvert.gray.rgb = function (args) {\n\treturn [args[0] / 100 * 255, args[0] / 100 * 255, args[0] / 100 * 255];\n};\n\nconvert.gray.hsl = function (args) {\n\treturn [0, 0, args[0]];\n};\n\nconvert.gray.hsv = convert.gray.hsl;\n\nconvert.gray.hwb = function (gray) {\n\treturn [0, 100, gray[0]];\n};\n\nconvert.gray.cmyk = function (gray) {\n\treturn [0, 0, 0, gray[0]];\n};\n\nconvert.gray.lab = function (gray) {\n\treturn [gray[0], 0, 0];\n};\n\nconvert.gray.hex = function (gray) {\n\tconst val = Math.round(gray[0] / 100 * 255) & 0xFF;\n\tconst integer = (val << 16) + (val << 8) + val;\n\n\tconst string = integer.toString(16).toUpperCase();\n\treturn '000000'.substring(string.length) + string;\n};\n\nconvert.rgb.gray = function (rgb) {\n\tconst val = (rgb[0] + rgb[1] + rgb[2]) / 3;\n\treturn [val / 255 * 100];\n};\n","const conversions = require('./conversions');\nconst route = require('./route');\n\nconst convert = {};\n\nconst models = Object.keys(conversions);\n\nfunction wrapRaw(fn) {\n\tconst wrappedFn = function (...args) {\n\t\tconst arg0 = args[0];\n\t\tif (arg0 === undefined || arg0 === null) {\n\t\t\treturn arg0;\n\t\t}\n\n\t\tif (arg0.length > 1) {\n\t\t\targs = arg0;\n\t\t}\n\n\t\treturn fn(args);\n\t};\n\n\t// Preserve .conversion property if there is one\n\tif ('conversion' in fn) {\n\t\twrappedFn.conversion = fn.conversion;\n\t}\n\n\treturn wrappedFn;\n}\n\nfunction wrapRounded(fn) {\n\tconst wrappedFn = function (...args) {\n\t\tconst arg0 = args[0];\n\n\t\tif (arg0 === undefined || arg0 === null) {\n\t\t\treturn arg0;\n\t\t}\n\n\t\tif (arg0.length > 1) {\n\t\t\targs = arg0;\n\t\t}\n\n\t\tconst result = fn(args);\n\n\t\t// We're assuming the result is an array here.\n\t\t// see notice in conversions.js; don't use box types\n\t\t// in conversion functions.\n\t\tif (typeof result === 'object') {\n\t\t\tfor (let len = result.length, i = 0; i < len; i++) {\n\t\t\t\tresult[i] = Math.round(result[i]);\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t};\n\n\t// Preserve .conversion property if there is one\n\tif ('conversion' in fn) {\n\t\twrappedFn.conversion = fn.conversion;\n\t}\n\n\treturn wrappedFn;\n}\n\nmodels.forEach(fromModel => {\n\tconvert[fromModel] = {};\n\n\tObject.defineProperty(convert[fromModel], 'channels', {value: conversions[fromModel].channels});\n\tObject.defineProperty(convert[fromModel], 'labels', {value: conversions[fromModel].labels});\n\n\tconst routes = route(fromModel);\n\tconst routeModels = Object.keys(routes);\n\n\trouteModels.forEach(toModel => {\n\t\tconst fn = routes[toModel];\n\n\t\tconvert[fromModel][toModel] = wrapRounded(fn);\n\t\tconvert[fromModel][toModel].raw = wrapRaw(fn);\n\t});\n});\n\nmodule.exports = convert;\n","const conversions = require('./conversions');\n\n/*\n\tThis function routes a model to all other models.\n\n\tall functions that are routed have a property `.conversion` attached\n\tto the returned synthetic function. This property is an array\n\tof strings, each with the steps in between the 'from' and 'to'\n\tcolor models (inclusive).\n\n\tconversions that are not possible simply are not included.\n*/\n\nfunction buildGraph() {\n\tconst graph = {};\n\t// https://jsperf.com/object-keys-vs-for-in-with-closure/3\n\tconst models = Object.keys(conversions);\n\n\tfor (let len = models.length, i = 0; i < len; i++) {\n\t\tgraph[models[i]] = {\n\t\t\t// http://jsperf.com/1-vs-infinity\n\t\t\t// micro-opt, but this is simple.\n\t\t\tdistance: -1,\n\t\t\tparent: null\n\t\t};\n\t}\n\n\treturn graph;\n}\n\n// https://en.wikipedia.org/wiki/Breadth-first_search\nfunction deriveBFS(fromModel) {\n\tconst graph = buildGraph();\n\tconst queue = [fromModel]; // Unshift -> queue -> pop\n\n\tgraph[fromModel].distance = 0;\n\n\twhile (queue.length) {\n\t\tconst current = queue.pop();\n\t\tconst adjacents = Object.keys(conversions[current]);\n\n\t\tfor (let len = adjacents.length, i = 0; i < len; i++) {\n\t\t\tconst adjacent = adjacents[i];\n\t\t\tconst node = graph[adjacent];\n\n\t\t\tif (node.distance === -1) {\n\t\t\t\tnode.distance = graph[current].distance + 1;\n\t\t\t\tnode.parent = current;\n\t\t\t\tqueue.unshift(adjacent);\n\t\t\t}\n\t\t}\n\t}\n\n\treturn graph;\n}\n\nfunction link(from, to) {\n\treturn function (args) {\n\t\treturn to(from(args));\n\t};\n}\n\nfunction wrapConversion(toModel, graph) {\n\tconst path = [graph[toModel].parent, toModel];\n\tlet fn = conversions[graph[toModel].parent][toModel];\n\n\tlet cur = graph[toModel].parent;\n\twhile (graph[cur].parent) {\n\t\tpath.unshift(graph[cur].parent);\n\t\tfn = link(conversions[graph[cur].parent][cur], fn);\n\t\tcur = graph[cur].parent;\n\t}\n\n\tfn.conversion = path;\n\treturn fn;\n}\n\nmodule.exports = function (fromModel) {\n\tconst graph = deriveBFS(fromModel);\n\tconst conversion = {};\n\n\tconst models = Object.keys(graph);\n\tfor (let len = models.length, i = 0; i < len; i++) {\n\t\tconst toModel = models[i];\n\t\tconst node = graph[toModel];\n\n\t\tif (node.parent === null) {\n\t\t\t// No possible conversion, or this node is the source model.\n\t\t\tcontinue;\n\t\t}\n\n\t\tconversion[toModel] = wrapConversion(toModel, graph);\n\t}\n\n\treturn conversion;\n};\n\n","'use strict'\r\n\r\nmodule.exports = {\r\n\t\"aliceblue\": [240, 248, 255],\r\n\t\"antiquewhite\": [250, 235, 215],\r\n\t\"aqua\": [0, 255, 255],\r\n\t\"aquamarine\": [127, 255, 212],\r\n\t\"azure\": [240, 255, 255],\r\n\t\"beige\": [245, 245, 220],\r\n\t\"bisque\": [255, 228, 196],\r\n\t\"black\": [0, 0, 0],\r\n\t\"blanchedalmond\": [255, 235, 205],\r\n\t\"blue\": [0, 0, 255],\r\n\t\"blueviolet\": [138, 43, 226],\r\n\t\"brown\": [165, 42, 42],\r\n\t\"burlywood\": [222, 184, 135],\r\n\t\"cadetblue\": [95, 158, 160],\r\n\t\"chartreuse\": [127, 255, 0],\r\n\t\"chocolate\": [210, 105, 30],\r\n\t\"coral\": [255, 127, 80],\r\n\t\"cornflowerblue\": [100, 149, 237],\r\n\t\"cornsilk\": [255, 248, 220],\r\n\t\"crimson\": [220, 20, 60],\r\n\t\"cyan\": [0, 255, 255],\r\n\t\"darkblue\": [0, 0, 139],\r\n\t\"darkcyan\": [0, 139, 139],\r\n\t\"darkgoldenrod\": [184, 134, 11],\r\n\t\"darkgray\": [169, 169, 169],\r\n\t\"darkgreen\": [0, 100, 0],\r\n\t\"darkgrey\": [169, 169, 169],\r\n\t\"darkkhaki\": [189, 183, 107],\r\n\t\"darkmagenta\": [139, 0, 139],\r\n\t\"darkolivegreen\": [85, 107, 47],\r\n\t\"darkorange\": [255, 140, 0],\r\n\t\"darkorchid\": [153, 50, 204],\r\n\t\"darkred\": [139, 0, 0],\r\n\t\"darksalmon\": [233, 150, 122],\r\n\t\"darkseagreen\": [143, 188, 143],\r\n\t\"darkslateblue\": [72, 61, 139],\r\n\t\"darkslategray\": [47, 79, 79],\r\n\t\"darkslategrey\": [47, 79, 79],\r\n\t\"darkturquoise\": [0, 206, 209],\r\n\t\"darkviolet\": [148, 0, 211],\r\n\t\"deeppink\": [255, 20, 147],\r\n\t\"deepskyblue\": [0, 191, 255],\r\n\t\"dimgray\": [105, 105, 105],\r\n\t\"dimgrey\": [105, 105, 105],\r\n\t\"dodgerblue\": [30, 144, 255],\r\n\t\"firebrick\": [178, 34, 34],\r\n\t\"floralwhite\": [255, 250, 240],\r\n\t\"forestgreen\": [34, 139, 34],\r\n\t\"fuchsia\": [255, 0, 255],\r\n\t\"gainsboro\": [220, 220, 220],\r\n\t\"ghostwhite\": [248, 248, 255],\r\n\t\"gold\": [255, 215, 0],\r\n\t\"goldenrod\": [218, 165, 32],\r\n\t\"gray\": [128, 128, 128],\r\n\t\"green\": [0, 128, 0],\r\n\t\"greenyellow\": [173, 255, 47],\r\n\t\"grey\": [128, 128, 128],\r\n\t\"honeydew\": [240, 255, 240],\r\n\t\"hotpink\": [255, 105, 180],\r\n\t\"indianred\": [205, 92, 92],\r\n\t\"indigo\": [75, 0, 130],\r\n\t\"ivory\": [255, 255, 240],\r\n\t\"khaki\": [240, 230, 140],\r\n\t\"lavender\": [230, 230, 250],\r\n\t\"lavenderblush\": [255, 240, 245],\r\n\t\"lawngreen\": [124, 252, 0],\r\n\t\"lemonchiffon\": [255, 250, 205],\r\n\t\"lightblue\": [173, 216, 230],\r\n\t\"lightcoral\": [240, 128, 128],\r\n\t\"lightcyan\": [224, 255, 255],\r\n\t\"lightgoldenrodyellow\": [250, 250, 210],\r\n\t\"lightgray\": [211, 211, 211],\r\n\t\"lightgreen\": [144, 238, 144],\r\n\t\"lightgrey\": [211, 211, 211],\r\n\t\"lightpink\": [255, 182, 193],\r\n\t\"lightsalmon\": [255, 160, 122],\r\n\t\"lightseagreen\": [32, 178, 170],\r\n\t\"lightskyblue\": [135, 206, 250],\r\n\t\"lightslategray\": [119, 136, 153],\r\n\t\"lightslategrey\": [119, 136, 153],\r\n\t\"lightsteelblue\": [176, 196, 222],\r\n\t\"lightyellow\": [255, 255, 224],\r\n\t\"lime\": [0, 255, 0],\r\n\t\"limegreen\": [50, 205, 50],\r\n\t\"linen\": [250, 240, 230],\r\n\t\"magenta\": [255, 0, 255],\r\n\t\"maroon\": [128, 0, 0],\r\n\t\"mediumaquamarine\": [102, 205, 170],\r\n\t\"mediumblue\": [0, 0, 205],\r\n\t\"mediumorchid\": [186, 85, 211],\r\n\t\"mediumpurple\": [147, 112, 219],\r\n\t\"mediumseagreen\": [60, 179, 113],\r\n\t\"mediumslateblue\": [123, 104, 238],\r\n\t\"mediumspringgreen\": [0, 250, 154],\r\n\t\"mediumturquoise\": [72, 209, 204],\r\n\t\"mediumvioletred\": [199, 21, 133],\r\n\t\"midnightblue\": [25, 25, 112],\r\n\t\"mintcream\": [245, 255, 250],\r\n\t\"mistyrose\": [255, 228, 225],\r\n\t\"moccasin\": [255, 228, 181],\r\n\t\"navajowhite\": [255, 222, 173],\r\n\t\"navy\": [0, 0, 128],\r\n\t\"oldlace\": [253, 245, 230],\r\n\t\"olive\": [128, 128, 0],\r\n\t\"olivedrab\": [107, 142, 35],\r\n\t\"orange\": [255, 165, 0],\r\n\t\"orangered\": [255, 69, 0],\r\n\t\"orchid\": [218, 112, 214],\r\n\t\"palegoldenrod\": [238, 232, 170],\r\n\t\"palegreen\": [152, 251, 152],\r\n\t\"paleturquoise\": [175, 238, 238],\r\n\t\"palevioletred\": [219, 112, 147],\r\n\t\"papayawhip\": [255, 239, 213],\r\n\t\"peachpuff\": [255, 218, 185],\r\n\t\"peru\": [205, 133, 63],\r\n\t\"pink\": [255, 192, 203],\r\n\t\"plum\": [221, 160, 221],\r\n\t\"powderblue\": [176, 224, 230],\r\n\t\"purple\": [128, 0, 128],\r\n\t\"rebeccapurple\": [102, 51, 153],\r\n\t\"red\": [255, 0, 0],\r\n\t\"rosybrown\": [188, 143, 143],\r\n\t\"royalblue\": [65, 105, 225],\r\n\t\"saddlebrown\": [139, 69, 19],\r\n\t\"salmon\": [250, 128, 114],\r\n\t\"sandybrown\": [244, 164, 96],\r\n\t\"seagreen\": [46, 139, 87],\r\n\t\"seashell\": [255, 245, 238],\r\n\t\"sienna\": [160, 82, 45],\r\n\t\"silver\": [192, 192, 192],\r\n\t\"skyblue\": [135, 206, 235],\r\n\t\"slateblue\": [106, 90, 205],\r\n\t\"slategray\": [112, 128, 144],\r\n\t\"slategrey\": [112, 128, 144],\r\n\t\"snow\": [255, 250, 250],\r\n\t\"springgreen\": [0, 255, 127],\r\n\t\"steelblue\": [70, 130, 180],\r\n\t\"tan\": [210, 180, 140],\r\n\t\"teal\": [0, 128, 128],\r\n\t\"thistle\": [216, 191, 216],\r\n\t\"tomato\": [255, 99, 71],\r\n\t\"turquoise\": [64, 224, 208],\r\n\t\"violet\": [238, 130, 238],\r\n\t\"wheat\": [245, 222, 179],\r\n\t\"white\": [255, 255, 255],\r\n\t\"whitesmoke\": [245, 245, 245],\r\n\t\"yellow\": [255, 255, 0],\r\n\t\"yellowgreen\": [154, 205, 50]\r\n};\r\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-content code{background-color:hsla(0,0%,78%,.3);border-radius:2px;padding:.15em}.ck.ck-editor__editable .ck-code_selected{background-color:hsla(0,0%,78%,.5)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-basic-styles/theme/code.css\"],\"names\":[],\"mappings\":\"AAKA,iBACC,kCAAuC,CAEvC,iBAAkB,CADlB,aAED,CAEA,0CACC,kCACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-content code {\\n\\tbackground-color: hsla(0, 0%, 78%, 0.3);\\n\\tpadding: .15em;\\n\\tborder-radius: 2px;\\n}\\n\\n.ck.ck-editor__editable .ck-code_selected {\\n\\tbackground-color: hsla(0, 0%, 78%, 0.5);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-content blockquote{border-left:5px solid #ccc;font-style:italic;margin-left:0;margin-right:0;overflow:hidden;padding-left:1.5em;padding-right:1.5em}.ck-content[dir=rtl] blockquote{border-left:0;border-right:5px solid #ccc}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-block-quote/theme/blockquote.css\"],\"names\":[],\"mappings\":\"AAKA,uBAWC,0BAAsC,CADtC,iBAAkB,CAFlB,aAAc,CACd,cAAe,CAPf,eAAgB,CAIhB,kBAAmB,CADnB,mBAOD,CAEA,gCACC,aAAc,CACd,2BACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-content blockquote {\\n\\t/* See #12 */\\n\\toverflow: hidden;\\n\\n\\t/* https://github.com/ckeditor/ckeditor5-block-quote/issues/15 */\\n\\tpadding-right: 1.5em;\\n\\tpadding-left: 1.5em;\\n\\n\\tmargin-left: 0;\\n\\tmargin-right: 0;\\n\\tfont-style: italic;\\n\\tborder-left: solid 5px hsl(0, 0%, 80%);\\n}\\n\\n.ck-content[dir=\\\"rtl\\\"] blockquote {\\n\\tborder-left: 0;\\n\\tborder-right: solid 5px hsl(0, 0%, 80%);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-block-toolbar-button{transform:translateX(calc(var(--ck-spacing-large)*-1))}\", \"\",{\"version\":3,\"sources\":[\"webpack://./theme/theme.css\"],\"names\":[],\"mappings\":\"AAMA,4BACC,sDACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/* Give the block toolbar button some space, moving it a few pixels away from the editable area. */\\n.ck.ck-block-toolbar-button {\\n\\ttransform: translateX( calc(-1 * var(--ck-spacing-large)) );\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-editor__editable .ck.ck-clipboard-drop-target-position{display:inline;pointer-events:none;position:relative}.ck.ck-editor__editable .ck.ck-clipboard-drop-target-position span{position:absolute;width:0}.ck.ck-editor__editable .ck-widget:-webkit-drag>.ck-widget__selection-handle,.ck.ck-editor__editable .ck-widget:-webkit-drag>.ck-widget__type-around{display:none}.ck.ck-clipboard-drop-target-line{pointer-events:none;position:absolute}:root{--ck-clipboard-drop-target-dot-width:12px;--ck-clipboard-drop-target-dot-height:8px;--ck-clipboard-drop-target-color:var(--ck-color-focus-border)}.ck.ck-editor__editable .ck.ck-clipboard-drop-target-position span{background:var(--ck-clipboard-drop-target-color);border:1px solid var(--ck-clipboard-drop-target-color);bottom:calc(var(--ck-clipboard-drop-target-dot-height)*-.5);margin-left:-1px;top:calc(var(--ck-clipboard-drop-target-dot-height)*-.5)}.ck.ck-editor__editable .ck.ck-clipboard-drop-target-position span:after{border-color:var(--ck-clipboard-drop-target-color) transparent transparent transparent;border-style:solid;border-width:calc(var(--ck-clipboard-drop-target-dot-height)) calc(var(--ck-clipboard-drop-target-dot-width)*.5) 0 calc(var(--ck-clipboard-drop-target-dot-width)*.5);content:\\\"\\\";display:block;height:0;left:50%;position:absolute;top:calc(var(--ck-clipboard-drop-target-dot-height)*-.5);transform:translateX(-50%);width:0}.ck.ck-editor__editable .ck-widget.ck-clipboard-drop-target-range{outline:var(--ck-widget-outline-thickness) solid var(--ck-clipboard-drop-target-color)!important}.ck.ck-editor__editable .ck-widget:-webkit-drag{zoom:.6;outline:none!important}.ck.ck-clipboard-drop-target-line{background:var(--ck-clipboard-drop-target-color);border:1px solid var(--ck-clipboard-drop-target-color);height:0;margin-top:-1px}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-clipboard/theme/clipboard.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-clipboard/clipboard.css\"],\"names\":[],\"mappings\":\"AASC,8DACC,cAAe,CAEf,mBAAoB,CADpB,iBAOD,CAJC,mEACC,iBAAkB,CAClB,OACD,CAWA,qJACC,YACD,CAIF,kCAEC,mBAAoB,CADpB,iBAED,CChCA,MACC,yCAA0C,CAC1C,yCAA0C,CAC1C,6DACD,CAOE,mEAIC,gDAAiD,CADjD,sDAAuD,CAFvD,2DAA8D,CAI9D,gBAAiB,CAHjB,wDAqBD,CAfC,yEAWC,sFAAuF,CAEvF,kBAAmB,CADnB,qKAA0K,CAX1K,UAAW,CAIX,aAAc,CAFd,QAAS,CAIT,QAAS,CADT,iBAAkB,CAElB,wDAA2D,CAE3D,0BAA2B,CAR3B,OAYD,CAOF,kEACC,gGACD,CAKA,gDACC,OAAS,CACT,sBACD,CAGD,kCAGC,gDAAiD,CADjD,sDAAuD,CADvD,QAAS,CAGT,eACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-editor__editable {\\n\\t/*\\n\\t * Vertical drop target (in text).\\n\\t */\\n\\t& .ck.ck-clipboard-drop-target-position {\\n\\t\\tdisplay: inline;\\n\\t\\tposition: relative;\\n\\t\\tpointer-events: none;\\n\\n\\t\\t& span {\\n\\t\\t\\tposition: absolute;\\n\\t\\t\\twidth: 0;\\n\\t\\t}\\n\\t}\\n\\n\\t/*\\n\\t * Styles of the widget being dragged (its preview).\\n\\t */\\n\\t& .ck-widget:-webkit-drag {\\n\\t\\t& > .ck-widget__selection-handle {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\n\\t\\t& > .ck-widget__type-around {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\t}\\n}\\n\\n.ck.ck-clipboard-drop-target-line {\\n\\tposition: absolute;\\n\\tpointer-events: none;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-clipboard-drop-target-dot-width: 12px;\\n\\t--ck-clipboard-drop-target-dot-height: 8px;\\n\\t--ck-clipboard-drop-target-color: var(--ck-color-focus-border)\\n}\\n\\n.ck.ck-editor__editable {\\n\\t/*\\n\\t * Vertical drop target (in text).\\n\\t */\\n\\t& .ck.ck-clipboard-drop-target-position {\\n\\t\\t& span {\\n\\t\\t\\tbottom: calc(-.5 * var(--ck-clipboard-drop-target-dot-height));\\n\\t\\t\\ttop: calc(-.5 * var(--ck-clipboard-drop-target-dot-height));\\n\\t\\t\\tborder: 1px solid var(--ck-clipboard-drop-target-color);\\n\\t\\t\\tbackground: var(--ck-clipboard-drop-target-color);\\n\\t\\t\\tmargin-left: -1px;\\n\\n\\t\\t\\t/* The triangle above the marker */\\n\\t\\t\\t&::after {\\n\\t\\t\\t\\tcontent: \\\"\\\";\\n\\t\\t\\t\\twidth: 0;\\n\\t\\t\\t\\theight: 0;\\n\\n\\t\\t\\t\\tdisplay: block;\\n\\t\\t\\t\\tposition: absolute;\\n\\t\\t\\t\\tleft: 50%;\\n\\t\\t\\t\\ttop: calc(var(--ck-clipboard-drop-target-dot-height) * -.5);\\n\\n\\t\\t\\t\\ttransform: translateX(-50%);\\n\\t\\t\\t\\tborder-color: var(--ck-clipboard-drop-target-color) transparent transparent transparent;\\n\\t\\t\\t\\tborder-width: calc(var(--ck-clipboard-drop-target-dot-height)) calc(.5 * var(--ck-clipboard-drop-target-dot-width)) 0 calc(.5 * var(--ck-clipboard-drop-target-dot-width));\\n\\t\\t\\t\\tborder-style: solid;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t/*\\n\\t * Styles of the widget that it a drop target.\\n\\t */\\n\\t& .ck-widget.ck-clipboard-drop-target-range {\\n\\t\\toutline: var(--ck-widget-outline-thickness) solid var(--ck-clipboard-drop-target-color) !important;\\n\\t}\\n\\n\\t/*\\n\\t * Styles of the widget being dragged (its preview).\\n\\t */\\n\\t& .ck-widget:-webkit-drag {\\n\\t\\tzoom: 0.6;\\n\\t\\toutline: none !important;\\n\\t}\\n}\\n\\n.ck.ck-clipboard-drop-target-line {\\n\\theight: 0;\\n\\tborder: 1px solid var(--ck-clipboard-drop-target-color);\\n\\tbackground: var(--ck-clipboard-drop-target-color);\\n\\tmargin-top: -1px;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck .ck-placeholder,.ck.ck-placeholder{position:relative}.ck .ck-placeholder:before,.ck.ck-placeholder:before{content:attr(data-placeholder);left:0;pointer-events:none;position:absolute;right:0}.ck.ck-read-only .ck-placeholder:before{display:none}.ck.ck-reset_all .ck-placeholder{position:relative}.ck .ck-placeholder:before,.ck.ck-placeholder:before{color:var(--ck-color-engine-placeholder-text);cursor:text}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-engine/theme/placeholder.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-engine/placeholder.css\"],\"names\":[],\"mappings\":\"AAMA,uCAEC,iBAWD,CATC,qDAIC,8BAA+B,CAF/B,MAAO,CAKP,mBAAoB,CANpB,iBAAkB,CAElB,OAKD,CAKA,wCACC,YACD,CAQD,iCACC,iBACD,CC5BC,qDAEC,6CAA8C,CAD9C,WAED\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/* See ckeditor/ckeditor5#936. */\\n.ck.ck-placeholder,\\n.ck .ck-placeholder {\\n\\tposition: relative;\\n\\n\\t&::before {\\n\\t\\tposition: absolute;\\n\\t\\tleft: 0;\\n\\t\\tright: 0;\\n\\t\\tcontent: attr(data-placeholder);\\n\\n\\t\\t/* See ckeditor/ckeditor5#469. */\\n\\t\\tpointer-events: none;\\n\\t}\\n}\\n\\n/* See ckeditor/ckeditor5#1987. */\\n.ck.ck-read-only .ck-placeholder {\\n\\t&::before {\\n\\t\\tdisplay: none;\\n\\t}\\n}\\n\\n/*\\n * Rules for the `ck-placeholder` are loaded before the rules for `ck-reset_all` in the base CKEditor 5 DLL build.\\n * This fix overwrites the incorrectly set `position: static` from `ck-reset_all`.\\n * See https://github.com/ckeditor/ckeditor5/issues/11418.\\n */\\n.ck.ck-reset_all .ck-placeholder {\\n\\tposition: relative;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/* See ckeditor/ckeditor5#936. */\\n.ck.ck-placeholder, .ck .ck-placeholder {\\n\\t&::before {\\n\\t\\tcursor: text;\\n\\t\\tcolor: var(--ck-color-engine-placeholder-text);\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-editor__editable span[data-ck-unsafe-element]{display:none}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-engine/theme/renderer.css\"],\"names\":[],\"mappings\":\"AAMA,qDACC,YACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/* Elements marked by the Renderer as hidden should be invisible in the editor. */\\n.ck.ck-editor__editable span[data-ck-unsafe-element] {\\n\\tdisplay: none;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-heading_heading1{font-size:20px}.ck.ck-heading_heading2{font-size:17px}.ck.ck-heading_heading3{font-size:14px}.ck[class*=ck-heading_heading]{font-weight:700}.ck.ck-dropdown.ck-heading-dropdown .ck-dropdown__button .ck-button__label{width:8em}.ck.ck-dropdown.ck-heading-dropdown .ck-dropdown__panel .ck-list__item{min-width:18em}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-heading/theme/heading.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-heading/heading.css\"],\"names\":[],\"mappings\":\"AAKA,wBACC,cACD,CAEA,wBACC,cACD,CAEA,wBACC,cACD,CAEA,+BACC,eACD,CCZC,2EACC,SACD,CAEA,uEACC,cACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-heading_heading1 {\\n\\tfont-size: 20px;\\n}\\n\\n.ck.ck-heading_heading2 {\\n\\tfont-size: 17px;\\n}\\n\\n.ck.ck-heading_heading3 {\\n\\tfont-size: 14px;\\n}\\n\\n.ck[class*=\\\"ck-heading_heading\\\"] {\\n\\tfont-weight: bold;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/* Resize dropdown's button label. */\\n.ck.ck-dropdown.ck-heading-dropdown {\\n\\t& .ck-dropdown__button .ck-button__label {\\n\\t\\twidth: 8em;\\n\\t}\\n\\n\\t& .ck-dropdown__panel .ck-list__item {\\n\\t\\tmin-width: 18em;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-content .image{clear:both;display:table;margin:.9em auto;min-width:50px;text-align:center}.ck-content .image img{display:block;margin:0 auto;max-width:100%;min-width:100%}.ck-content .image-inline{align-items:flex-start;display:inline-flex;max-width:100%}.ck-content .image-inline picture{display:flex}.ck-content .image-inline img,.ck-content .image-inline picture{flex-grow:1;flex-shrink:1;max-width:100%}.ck.ck-editor__editable .image>figcaption.ck-placeholder:before{overflow:hidden;padding-left:inherit;padding-right:inherit;text-overflow:ellipsis;white-space:nowrap}.ck.ck-editor__editable .image-inline.ck-widget_selected,.ck.ck-editor__editable .image.ck-widget_selected{z-index:1}.ck.ck-editor__editable .image-inline.ck-widget_selected ::selection{display:none}.ck.ck-editor__editable td .image-inline img,.ck.ck-editor__editable th .image-inline img{max-width:none}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/image.css\"],\"names\":[],\"mappings\":\"AAMC,mBAEC,UAAW,CADX,aAAc,CAOd,gBAAkB,CAGlB,cAAe,CARf,iBAuBD,CAbC,uBAEC,aAAc,CAGd,aAAc,CAGd,cAAe,CAGf,cACD,CAGD,0BAYC,sBAAuB,CANvB,mBAAoB,CAGpB,cAoBD,CAdC,kCACC,YACD,CAGA,gEAGC,WAAY,CACZ,aAAc,CAGd,cACD,CAUD,gEASC,eAAgB,CARhB,oBAAqB,CACrB,qBAAsB,CAQtB,sBAAuB,CAFvB,kBAGD,CAWA,2GACC,SAUD,CAHC,qEACC,YACD,CAOA,0FACC,cACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-content {\\n\\t& .image {\\n\\t\\tdisplay: table;\\n\\t\\tclear: both;\\n\\t\\ttext-align: center;\\n\\n\\t\\t/* Make sure there is some space between the content and the image. Center image by default. */\\n\\t\\t/* The first value should be equal to --ck-spacing-large variable if used in the editor context\\n\\t \\tto avoid the content jumping (See https://github.com/ckeditor/ckeditor5/issues/9825). */\\n\\t\\tmargin: 0.9em auto;\\n\\n\\t\\t/* Make sure the caption will be displayed properly (See: https://github.com/ckeditor/ckeditor5/issues/1870). */\\n\\t\\tmin-width: 50px;\\n\\n\\t\\t& img {\\n\\t\\t\\t/* Prevent unnecessary margins caused by line-height (see #44). */\\n\\t\\t\\tdisplay: block;\\n\\n\\t\\t\\t/* Center the image if its width is smaller than the content's width. */\\n\\t\\t\\tmargin: 0 auto;\\n\\n\\t\\t\\t/* Make sure the image never exceeds the size of the parent container (ckeditor/ckeditor5-ui#67). */\\n\\t\\t\\tmax-width: 100%;\\n\\n\\t\\t\\t/* Make sure the image is never smaller than the parent container (See: https://github.com/ckeditor/ckeditor5/issues/9300). */\\n\\t\\t\\tmin-width: 100%\\n\\t\\t}\\n\\t}\\n\\n\\t& .image-inline {\\n\\t\\t/*\\n\\t\\t * Normally, the .image-inline would have \\\"display: inline-block\\\" and \\\"img { width: 100% }\\\" (to follow the wrapper while resizing).\\n\\t\\t * Unfortunately, together with \\\"srcset\\\", it gets automatically stretched up to the width of the editing root.\\n\\t\\t * This strange behavior does not happen with inline-flex.\\n\\t\\t */\\n\\t\\tdisplay: inline-flex;\\n\\n\\t\\t/* While being resized, don't allow the image to exceed the width of the editing root. */\\n\\t\\tmax-width: 100%;\\n\\n\\t\\t/* This is required by Safari to resize images in a sensible way. Without this, the browser breaks the ratio. */\\n\\t\\talign-items: flex-start;\\n\\n\\t\\t/* When the picture is present it must act as a flex container to let the img resize properly */\\n\\t\\t& picture {\\n\\t\\t\\tdisplay: flex;\\n\\t\\t}\\n\\n\\t\\t/* When the picture is present, it must act like a resizable img. */\\n\\t\\t& picture,\\n\\t\\t& img {\\n\\t\\t\\t/* This is necessary for the img to span the entire .image-inline wrapper and to resize properly. */\\n\\t\\t\\tflex-grow: 1;\\n\\t\\t\\tflex-shrink: 1;\\n\\n\\t\\t\\t/* Prevents overflowing the editing root boundaries when an inline image is very wide. */\\n\\t\\t\\tmax-width: 100%;\\n\\t\\t}\\n\\t}\\n}\\n\\n.ck.ck-editor__editable {\\n\\t/*\\n\\t * Inhertit the content styles padding of the <figcaption> in case the integration overrides `text-align: center`\\n\\t * of `.image` (e.g. to the left/right). This ensures the placeholder stays at the padding just like the native\\n\\t * caret does, and not at the edge of <figcaption>.\\n\\t */\\n\\t& .image > figcaption.ck-placeholder::before {\\n\\t\\tpadding-left: inherit;\\n\\t\\tpadding-right: inherit;\\n\\n\\t\\t/*\\n\\t\\t * Make sure the image caption placeholder doesn't overflow the placeholder area.\\n\\t\\t * See https://github.com/ckeditor/ckeditor5/issues/9162.\\n\\t\\t */\\n\\t\\twhite-space: nowrap;\\n\\t\\toverflow: hidden;\\n\\t\\ttext-overflow: ellipsis;\\n\\t}\\n\\n\\n\\t/*\\n\\t * Make sure the selected inline image always stays on top of its siblings.\\n\\t * See https://github.com/ckeditor/ckeditor5/issues/9108.\\n\\t */\\n\\t& .image.ck-widget_selected {\\n\\t\\tz-index: 1;\\n\\t}\\n\\n\\t& .image-inline.ck-widget_selected {\\n\\t\\tz-index: 1;\\n\\n\\t\\t/*\\n\\t\\t * Make sure the native browser selection style is not displayed.\\n\\t\\t * Inline image widgets have their own styles for the selected state and\\n\\t\\t * leaving this up to the browser is asking for a visual collision.\\n\\t\\t */\\n\\t\\t& ::selection {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\t}\\n\\n\\t/* The inline image nested in the table should have its original size if not resized.\\n\\tSee https://github.com/ckeditor/ckeditor5/issues/9117. */\\n\\t& td,\\n\\t& th {\\n\\t\\t& .image-inline img {\\n\\t\\t\\tmax-width: none;\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-color-image-caption-background:#f7f7f7;--ck-color-image-caption-text:#333;--ck-color-image-caption-highligted-background:#fd0}.ck-content .image>figcaption{background-color:var(--ck-color-image-caption-background);caption-side:bottom;color:var(--ck-color-image-caption-text);display:table-caption;font-size:.75em;outline-offset:-1px;padding:.6em;word-break:break-word}.ck.ck-editor__editable .image>figcaption.image__caption_highlighted{animation:ck-image-caption-highlight .6s ease-out}@keyframes ck-image-caption-highlight{0%{background-color:var(--ck-color-image-caption-highligted-background)}to{background-color:var(--ck-color-image-caption-background)}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/imagecaption.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,2CAAoD,CACpD,kCAA8C,CAC9C,mDACD,CAGA,8BAKC,yDAA0D,CAH1D,mBAAoB,CAEpB,wCAAyC,CAHzC,qBAAsB,CAMtB,eAAgB,CAChB,mBAAoB,CAFpB,YAAa,CAHb,qBAMD,CAGA,qEACC,iDACD,CAEA,sCACC,GACC,oEACD,CAEA,GACC,yDACD,CACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-image-caption-background: hsl(0, 0%, 97%);\\n\\t--ck-color-image-caption-text: hsl(0, 0%, 20%);\\n\\t--ck-color-image-caption-highligted-background: hsl(52deg 100% 50%);\\n}\\n\\n/* Content styles */\\n.ck-content .image > figcaption {\\n\\tdisplay: table-caption;\\n\\tcaption-side: bottom;\\n\\tword-break: break-word;\\n\\tcolor: var(--ck-color-image-caption-text);\\n\\tbackground-color: var(--ck-color-image-caption-background);\\n\\tpadding: .6em;\\n\\tfont-size: .75em;\\n\\toutline-offset: -1px;\\n}\\n\\n/* Editing styles */\\n.ck.ck-editor__editable .image > figcaption.image__caption_highlighted {\\n\\tanimation: ck-image-caption-highlight .6s ease-out;\\n}\\n\\n@keyframes ck-image-caption-highlight {\\n\\t0% {\\n\\t\\tbackground-color: var(--ck-color-image-caption-highligted-background);\\n\\t}\\n\\n\\t100% {\\n\\t\\tbackground-color: var(--ck-color-image-caption-background);\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-image-insert__panel{padding:var(--ck-spacing-large)}.ck.ck-image-insert__ck-finder-button{border:1px solid #ccc;border-radius:var(--ck-border-radius);display:block;margin:var(--ck-spacing-standard) auto;width:100%}.ck.ck-splitbutton>.ck-file-dialog-button.ck-button{border:none;margin:0;padding:0}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/imageinsert.css\"],\"names\":[],\"mappings\":\"AAKA,2BACC,+BACD,CAEA,sCAIC,qBAAiC,CACjC,qCAAsC,CAJtC,aAAc,CAEd,sCAAuC,CADvC,UAID,CAGA,oDAGC,WAAY,CADZ,QAAS,CADT,SAGD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-image-insert__panel {\\n\\tpadding: var(--ck-spacing-large);\\n}\\n\\n.ck.ck-image-insert__ck-finder-button {\\n\\tdisplay: block;\\n\\twidth: 100%;\\n\\tmargin: var(--ck-spacing-standard) auto;\\n\\tborder: 1px solid hsl(0, 0%, 80%);\\n\\tborder-radius: var(--ck-border-radius);\\n}\\n\\n/* https://github.com/ckeditor/ckeditor5/issues/7986 */\\n.ck.ck-splitbutton > .ck-file-dialog-button.ck-button {\\n\\tpadding: 0;\\n\\tmargin: 0;\\n\\tborder: none;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-image-insert-form:focus{outline:none}.ck.ck-form__row{display:flex;flex-direction:row;flex-wrap:nowrap;justify-content:space-between}.ck.ck-form__row>:not(.ck-label){flex-grow:1}.ck.ck-form__row.ck-image-insert-form__action-row{margin-top:var(--ck-spacing-standard)}.ck.ck-form__row.ck-image-insert-form__action-row .ck-button-cancel,.ck.ck-form__row.ck-image-insert-form__action-row .ck-button-save{justify-content:center}.ck.ck-form__row.ck-image-insert-form__action-row .ck-button .ck-button__label{color:var(--ck-color-text)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/imageinsertformrowview.css\"],\"names\":[],\"mappings\":\"AAMC,+BAEC,YACD,CAGD,iBACC,YAAa,CACb,kBAAmB,CACnB,gBAAiB,CACjB,6BAmBD,CAhBC,iCACC,WACD,CAEA,kDACC,qCAUD,CARC,sIAEC,sBACD,CAEA,+EACC,0BACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-image-insert-form {\\n\\t&:focus {\\n\\t\\t/* See: https://github.com/ckeditor/ckeditor5/issues/4773 */\\n\\t\\toutline: none;\\n\\t}\\n}\\n\\n.ck.ck-form__row {\\n\\tdisplay: flex;\\n\\tflex-direction: row;\\n\\tflex-wrap: nowrap;\\n\\tjustify-content: space-between;\\n\\n\\t/* Ignore labels that work as fieldset legends */\\n\\t& > *:not(.ck-label) {\\n\\t\\tflex-grow: 1;\\n\\t}\\n\\n\\t&.ck-image-insert-form__action-row {\\n\\t\\tmargin-top: var(--ck-spacing-standard);\\n\\n\\t\\t& .ck-button-save,\\n\\t\\t& .ck-button-cancel {\\n\\t\\t\\tjustify-content: center;\\n\\t\\t}\\n\\n\\t\\t& .ck-button .ck-button__label {\\n\\t\\t\\tcolor: var(--ck-color-text);\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-content .image.image_resized{box-sizing:border-box;display:block;max-width:100%}.ck-content .image.image_resized img{width:100%}.ck-content .image.image_resized>figcaption{display:block}.ck.ck-editor__editable td .image-inline.image_resized img,.ck.ck-editor__editable th .image-inline.image_resized img{max-width:100%}[dir=ltr] .ck.ck-button.ck-button_with-text.ck-resize-image-button .ck-button__icon{margin-right:var(--ck-spacing-standard)}[dir=rtl] .ck.ck-button.ck-button_with-text.ck-resize-image-button .ck-button__icon{margin-left:var(--ck-spacing-standard)}.ck.ck-dropdown .ck-button.ck-resize-image-button .ck-button__label{width:4em}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/imageresize.css\"],\"names\":[],\"mappings\":\"AAKA,iCAQC,qBAAsB,CADtB,aAAc,CANd,cAkBD,CATC,qCAEC,UACD,CAEA,4CAEC,aACD,CAQC,sHACC,cACD,CAIF,oFACC,uCACD,CAEA,oFACC,sCACD,CAEA,oEACC,SACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-content .image.image_resized {\\n\\tmax-width: 100%;\\n\\t/*\\n\\tThe `<figure>` element for resized images must not use `display:table` as browsers do not support `max-width` for it well.\\n\\tSee https://stackoverflow.com/questions/4019604/chrome-safari-ignoring-max-width-in-table/14420691#14420691 for more.\\n\\tFortunately, since we control the width, there is no risk that the image will look bad.\\n\\t*/\\n\\tdisplay: block;\\n\\tbox-sizing: border-box;\\n\\n\\t& img {\\n\\t\\t/* For resized images it is the `<figure>` element that determines the image width. */\\n\\t\\twidth: 100%;\\n\\t}\\n\\n\\t& > figcaption {\\n\\t\\t/* The `<figure>` element uses `display:block`, so `<figcaption>` also has to. */\\n\\t\\tdisplay: block;\\n\\t}\\n}\\n\\n.ck.ck-editor__editable {\\n\\t/* The resized inline image nested in the table should respect its parent size.\\n\\tSee https://github.com/ckeditor/ckeditor5/issues/9117. */\\n\\t& td,\\n\\t& th {\\n\\t\\t& .image-inline.image_resized img {\\n\\t\\t\\tmax-width: 100%;\\n\\t\\t}\\n\\t}\\n}\\n\\n[dir=\\\"ltr\\\"] .ck.ck-button.ck-button_with-text.ck-resize-image-button .ck-button__icon {\\n\\tmargin-right: var(--ck-spacing-standard);\\n}\\n\\n[dir=\\\"rtl\\\"] .ck.ck-button.ck-button_with-text.ck-resize-image-button .ck-button__icon {\\n\\tmargin-left: var(--ck-spacing-standard);\\n}\\n\\n.ck.ck-dropdown .ck-button.ck-resize-image-button .ck-button__label {\\n\\twidth: 4em;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-image-style-spacing:1.5em;--ck-inline-image-style-spacing:calc(var(--ck-image-style-spacing)/2)}.ck-content .image-style-block-align-left,.ck-content .image-style-block-align-right{max-width:calc(100% - var(--ck-image-style-spacing))}.ck-content .image-style-align-left,.ck-content .image-style-align-right{clear:none}.ck-content .image-style-side{float:right;margin-left:var(--ck-image-style-spacing);max-width:50%}.ck-content .image-style-align-left{float:left;margin-right:var(--ck-image-style-spacing)}.ck-content .image-style-align-center{margin-left:auto;margin-right:auto}.ck-content .image-style-align-right{float:right;margin-left:var(--ck-image-style-spacing)}.ck-content .image-style-block-align-right{margin-left:auto;margin-right:0}.ck-content .image-style-block-align-left{margin-left:0;margin-right:auto}.ck-content p+.image-style-align-left,.ck-content p+.image-style-align-right,.ck-content p+.image-style-side{margin-top:0}.ck-content .image-inline.image-style-align-left,.ck-content .image-inline.image-style-align-right{margin-bottom:var(--ck-inline-image-style-spacing);margin-top:var(--ck-inline-image-style-spacing)}.ck-content .image-inline.image-style-align-left{margin-right:var(--ck-inline-image-style-spacing)}.ck-content .image-inline.image-style-align-right{margin-left:var(--ck-inline-image-style-spacing)}.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open>.ck-splitbutton__action:not(.ck-disabled),.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open>.ck-splitbutton__arrow:not(.ck-disabled),.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open>.ck-splitbutton__arrow:not(.ck-disabled):not(:hover),.ck.ck-splitbutton.ck-splitbutton_flatten:hover>.ck-splitbutton__action:not(.ck-disabled),.ck.ck-splitbutton.ck-splitbutton_flatten:hover>.ck-splitbutton__arrow:not(.ck-disabled),.ck.ck-splitbutton.ck-splitbutton_flatten:hover>.ck-splitbutton__arrow:not(.ck-disabled):not(:hover){background-color:var(--ck-color-button-on-background)}.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open>.ck-splitbutton__action:not(.ck-disabled):after,.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open>.ck-splitbutton__arrow:not(.ck-disabled):after,.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open>.ck-splitbutton__arrow:not(.ck-disabled):not(:hover):after,.ck.ck-splitbutton.ck-splitbutton_flatten:hover>.ck-splitbutton__action:not(.ck-disabled):after,.ck.ck-splitbutton.ck-splitbutton_flatten:hover>.ck-splitbutton__arrow:not(.ck-disabled):after,.ck.ck-splitbutton.ck-splitbutton_flatten:hover>.ck-splitbutton__arrow:not(.ck-disabled):not(:hover):after{display:none}.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open:hover>.ck-splitbutton__action:not(.ck-disabled),.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open:hover>.ck-splitbutton__arrow:not(.ck-disabled),.ck.ck-splitbutton.ck-splitbutton_flatten.ck-splitbutton_open:hover>.ck-splitbutton__arrow:not(.ck-disabled):not(:hover){background-color:var(--ck-color-button-on-hover-background)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/imagestyle.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,8BAA+B,CAC/B,qEACD,CAMC,qFAEC,oDACD,CAIA,yEAEC,UACD,CAEA,8BACC,WAAY,CACZ,yCAA0C,CAC1C,aACD,CAEA,oCACC,UAAW,CACX,0CACD,CAEA,sCACC,gBAAiB,CACjB,iBACD,CAEA,qCACC,WAAY,CACZ,yCACD,CAEA,2CAEC,gBAAiB,CADjB,cAED,CAEA,0CACC,aAAc,CACd,iBACD,CAGA,6GAGC,YACD,CAGC,mGAGC,kDAAmD,CADnD,+CAED,CAEA,iDACC,iDACD,CAEA,kDACC,gDACD,CAUC,0lBAGC,qDAKD,CAHC,8nBACC,YACD,CAKD,oVAGC,2DACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-image-style-spacing: 1.5em;\\n\\t--ck-inline-image-style-spacing: calc(var(--ck-image-style-spacing) / 2);\\n}\\n\\n.ck-content {\\n\\t/* Provides a minimal side margin for the left and right aligned images, so that the user has a visual feedback\\n\\tconfirming successful application of the style if image width exceeds the editor's size.\\n\\tSee https://github.com/ckeditor/ckeditor5/issues/9342 */\\n\\t& .image-style-block-align-left,\\n\\t& .image-style-block-align-right {\\n\\t\\tmax-width: calc(100% - var(--ck-image-style-spacing));\\n\\t}\\n\\n\\t/* Allows displaying multiple floating images in the same line.\\n\\tSee https://github.com/ckeditor/ckeditor5/issues/9183#issuecomment-804988132 */\\n\\t& .image-style-align-left,\\n\\t& .image-style-align-right {\\n\\t\\tclear: none;\\n\\t}\\n\\n\\t& .image-style-side {\\n\\t\\tfloat: right;\\n\\t\\tmargin-left: var(--ck-image-style-spacing);\\n\\t\\tmax-width: 50%;\\n\\t}\\n\\n\\t& .image-style-align-left {\\n\\t\\tfloat: left;\\n\\t\\tmargin-right: var(--ck-image-style-spacing);\\n\\t}\\n\\n\\t& .image-style-align-center {\\n\\t\\tmargin-left: auto;\\n\\t\\tmargin-right: auto;\\n\\t}\\n\\n\\t& .image-style-align-right {\\n\\t\\tfloat: right;\\n\\t\\tmargin-left: var(--ck-image-style-spacing);\\n\\t}\\n\\n\\t& .image-style-block-align-right {\\n\\t\\tmargin-right: 0;\\n\\t\\tmargin-left: auto;\\n\\t}\\n\\n\\t& .image-style-block-align-left {\\n\\t\\tmargin-left: 0;\\n\\t\\tmargin-right: auto;\\n\\t}\\n\\n\\t/* Simulates margin collapsing with the preceding paragraph, which does not work for the floating elements. */\\n\\t& p + .image-style-align-left,\\n\\t& p + .image-style-align-right,\\n\\t& p + .image-style-side {\\n\\t\\tmargin-top: 0;\\n\\t}\\n\\n\\t& .image-inline {\\n\\t\\t&.image-style-align-left,\\n\\t\\t&.image-style-align-right {\\n\\t\\t\\tmargin-top: var(--ck-inline-image-style-spacing);\\n\\t\\t\\tmargin-bottom: var(--ck-inline-image-style-spacing);\\n\\t\\t}\\n\\n\\t\\t&.image-style-align-left {\\n\\t\\t\\tmargin-right: var(--ck-inline-image-style-spacing);\\n\\t\\t}\\n\\n\\t\\t&.image-style-align-right {\\n\\t\\t\\tmargin-left: var(--ck-inline-image-style-spacing);\\n\\t\\t}\\n\\t}\\n}\\n\\n.ck.ck-splitbutton {\\n\\t/* The button should display as a regular drop-down if the action button\\n\\tis forced to fire the same action as the arrow button. */\\n\\t&.ck-splitbutton_flatten {\\n\\t\\t&:hover,\\n\\t\\t&.ck-splitbutton_open {\\n\\t\\t\\t& > .ck-splitbutton__action:not(.ck-disabled),\\n\\t\\t\\t& > .ck-splitbutton__arrow:not(.ck-disabled),\\n\\t\\t\\t& > .ck-splitbutton__arrow:not(.ck-disabled):not(:hover) {\\n\\t\\t\\t\\tbackground-color: var(--ck-color-button-on-background);\\n\\n\\t\\t\\t\\t&::after {\\n\\t\\t\\t\\t\\tdisplay: none;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t&.ck-splitbutton_open:hover {\\n\\t\\t\\t& > .ck-splitbutton__action:not(.ck-disabled),\\n\\t\\t\\t& > .ck-splitbutton__arrow:not(.ck-disabled),\\n\\t\\t\\t& > .ck-splitbutton__arrow:not(.ck-disabled):not(:hover) {\\n\\t\\t\\t\\tbackground-color: var(--ck-color-button-on-hover-background);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-image-upload-complete-icon{border-radius:50%;display:block;position:absolute;right:min(var(--ck-spacing-medium),6%);top:min(var(--ck-spacing-medium),6%);z-index:1}.ck-image-upload-complete-icon:after{content:\\\"\\\";position:absolute}:root{--ck-color-image-upload-icon:#fff;--ck-color-image-upload-icon-background:#008a00;--ck-image-upload-icon-size:20;--ck-image-upload-icon-width:2px;--ck-image-upload-icon-is-visible:clamp(0px,100% - 50px,1px)}.ck-image-upload-complete-icon{animation-delay:0ms,3s;animation-duration:.5s,.5s;animation-fill-mode:forwards,forwards;animation-name:ck-upload-complete-icon-show,ck-upload-complete-icon-hide;background:var(--ck-color-image-upload-icon-background);font-size:calc(1px*var(--ck-image-upload-icon-size));height:calc(var(--ck-image-upload-icon-is-visible)*var(--ck-image-upload-icon-size));opacity:0;overflow:hidden;width:calc(var(--ck-image-upload-icon-is-visible)*var(--ck-image-upload-icon-size))}.ck-image-upload-complete-icon:after{animation-delay:.5s;animation-duration:.5s;animation-fill-mode:forwards;animation-name:ck-upload-complete-icon-check;border-right:var(--ck-image-upload-icon-width) solid var(--ck-color-image-upload-icon);border-top:var(--ck-image-upload-icon-width) solid var(--ck-color-image-upload-icon);box-sizing:border-box;height:0;left:25%;opacity:0;top:50%;transform:scaleX(-1) rotate(135deg);transform-origin:left top;width:0}@keyframes ck-upload-complete-icon-show{0%{opacity:0}to{opacity:1}}@keyframes ck-upload-complete-icon-hide{0%{opacity:1}to{opacity:0}}@keyframes ck-upload-complete-icon-check{0%{height:0;opacity:1;width:0}33%{height:0;width:.3em}to{height:.45em;opacity:1;width:.3em}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/imageuploadicon.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-image/imageuploadicon.css\"],\"names\":[],\"mappings\":\"AAKA,+BAUC,iBAAkB,CATlB,aAAc,CACd,iBAAkB,CAOlB,sCAAwC,CADxC,oCAAsC,CAGtC,SAMD,CAJC,qCACC,UAAW,CACX,iBACD,CChBD,MACC,iCAA8C,CAC9C,+CAA4D,CAG5D,8BAA+B,CAC/B,gCAAiC,CACjC,4DACD,CAEA,+BAWC,sBAA4B,CAN5B,0BAAgC,CADhC,qCAAuC,CADvC,wEAA0E,CAD1E,uDAAwD,CAMxD,oDAAuD,CAWvD,oFAAuF,CAlBvF,SAAU,CAgBV,eAAgB,CAChB,mFA0BD,CAtBC,qCAgBC,mBAAsB,CADtB,sBAAyB,CAEzB,4BAA6B,CAH7B,4CAA6C,CAF7C,sFAAuF,CADvF,oFAAqF,CASrF,qBAAsB,CAdtB,QAAS,CAJT,QAAS,CAGT,SAAU,CADV,OAAQ,CAKR,mCAAoC,CACpC,yBAA0B,CAH1B,OAcD,CAGD,wCACC,GACC,SACD,CAEA,GACC,SACD,CACD,CAEA,wCACC,GACC,SACD,CAEA,GACC,SACD,CACD,CAEA,yCACC,GAGC,QAAS,CAFT,SAAU,CACV,OAED,CACA,IAEC,QAAS,CADT,UAED,CACA,GAGC,YAAc,CAFd,SAAU,CACV,UAED,CACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-image-upload-complete-icon {\\n\\tdisplay: block;\\n\\tposition: absolute;\\n\\n\\t/*\\n\\t * Smaller images should have the icon closer to the border.\\n\\t * Match the icon position with the linked image indicator brought by the link image feature.\\n\\t */\\n\\ttop: min(var(--ck-spacing-medium), 6%);\\n\\tright: min(var(--ck-spacing-medium), 6%);\\n\\tborder-radius: 50%;\\n\\tz-index: 1;\\n\\n\\t&::after {\\n\\t\\tcontent: \\\"\\\";\\n\\t\\tposition: absolute;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-image-upload-icon: hsl(0, 0%, 100%);\\n\\t--ck-color-image-upload-icon-background: hsl(120, 100%, 27%);\\n\\n\\t/* Match the icon size with the linked image indicator brought by the link image feature. */\\n\\t--ck-image-upload-icon-size: 20;\\n\\t--ck-image-upload-icon-width: 2px;\\n\\t--ck-image-upload-icon-is-visible: clamp(0px, 100% - 50px, 1px);\\n}\\n\\n.ck-image-upload-complete-icon {\\n\\topacity: 0;\\n\\tbackground: var(--ck-color-image-upload-icon-background);\\n\\tanimation-name: ck-upload-complete-icon-show, ck-upload-complete-icon-hide;\\n\\tanimation-fill-mode: forwards, forwards;\\n\\tanimation-duration: 500ms, 500ms;\\n\\n\\t/* To make animation scalable. */\\n\\tfont-size: calc(1px * var(--ck-image-upload-icon-size));\\n\\n\\t/* Hide completed upload icon after 3 seconds. */\\n\\tanimation-delay: 0ms, 3000ms;\\n\\n\\t/*\\n\\t * Use CSS math to simulate container queries.\\n\\t * https://css-tricks.com/the-raven-technique-one-step-closer-to-container-queries/#what-about-showing-and-hiding-things\\n\\t */\\n\\toverflow: hidden;\\n\\twidth: calc(var(--ck-image-upload-icon-is-visible) * var(--ck-image-upload-icon-size));\\n\\theight: calc(var(--ck-image-upload-icon-is-visible) * var(--ck-image-upload-icon-size));\\n\\n\\t/* This is check icon element made from border-width mixed with animations. */\\n\\t&::after {\\n\\t\\t/* Because of border transformation we need to \\\"hard code\\\" left position. */\\n\\t\\tleft: 25%;\\n\\n\\t\\ttop: 50%;\\n\\t\\topacity: 0;\\n\\t\\theight: 0;\\n\\t\\twidth: 0;\\n\\n\\t\\ttransform: scaleX(-1) rotate(135deg);\\n\\t\\ttransform-origin: left top;\\n\\t\\tborder-top: var(--ck-image-upload-icon-width) solid var(--ck-color-image-upload-icon);\\n\\t\\tborder-right: var(--ck-image-upload-icon-width) solid var(--ck-color-image-upload-icon);\\n\\n\\t\\tanimation-name: ck-upload-complete-icon-check;\\n\\t\\tanimation-duration: 500ms;\\n\\t\\tanimation-delay: 500ms;\\n\\t\\tanimation-fill-mode: forwards;\\n\\n\\t\\t/* #1095. While reset is not providing proper box-sizing for pseudoelements, we need to handle it. */\\n\\t\\tbox-sizing: border-box;\\n\\t}\\n}\\n\\n@keyframes ck-upload-complete-icon-show {\\n\\tfrom {\\n\\t\\topacity: 0;\\n\\t}\\n\\n\\tto {\\n\\t\\topacity: 1;\\n\\t}\\n}\\n\\n@keyframes ck-upload-complete-icon-hide {\\n\\tfrom {\\n\\t\\topacity: 1;\\n\\t}\\n\\n\\tto {\\n\\t\\topacity: 0;\\n\\t}\\n}\\n\\n@keyframes ck-upload-complete-icon-check {\\n\\t0% {\\n\\t\\topacity: 1;\\n\\t\\twidth: 0;\\n\\t\\theight: 0;\\n\\t}\\n\\t33% {\\n\\t\\twidth: 0.3em;\\n\\t\\theight: 0;\\n\\t}\\n\\t100% {\\n\\t\\topacity: 1;\\n\\t\\twidth: 0.3em;\\n\\t\\theight: 0.45em;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck .ck-upload-placeholder-loader{align-items:center;display:flex;justify-content:center;left:0;position:absolute;top:0}.ck .ck-upload-placeholder-loader:before{content:\\\"\\\";position:relative}:root{--ck-color-upload-placeholder-loader:#b3b3b3;--ck-upload-placeholder-loader-size:32px;--ck-upload-placeholder-image-aspect-ratio:2.8}.ck .ck-image-upload-placeholder{margin:0;width:100%}.ck .ck-image-upload-placeholder.image-inline{width:calc(var(--ck-upload-placeholder-loader-size)*2*var(--ck-upload-placeholder-image-aspect-ratio))}.ck .ck-image-upload-placeholder img{aspect-ratio:var(--ck-upload-placeholder-image-aspect-ratio)}.ck .ck-upload-placeholder-loader{height:100%;width:100%}.ck .ck-upload-placeholder-loader:before{animation:ck-upload-placeholder-loader 1s linear infinite;border-radius:50%;border-right:2px solid transparent;border-top:3px solid var(--ck-color-upload-placeholder-loader);height:var(--ck-upload-placeholder-loader-size);width:var(--ck-upload-placeholder-loader-size)}@keyframes ck-upload-placeholder-loader{to{transform:rotate(1turn)}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/imageuploadloader.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-image/imageuploadloader.css\"],\"names\":[],\"mappings\":\"AAKA,kCAGC,kBAAmB,CADnB,YAAa,CAEb,sBAAuB,CAEvB,MAAO,CALP,iBAAkB,CAIlB,KAOD,CAJC,yCACC,UAAW,CACX,iBACD,CCXD,MACC,4CAAqD,CACrD,wCAAyC,CACzC,8CACD,CAEA,iCAGC,QAAS,CADT,UAgBD,CAbC,8CACC,sGACD,CAEA,qCAOC,4DACD,CAGD,kCAEC,WAAY,CADZ,UAWD,CARC,yCAMC,yDAA0D,CAH1D,iBAAkB,CAElB,kCAAmC,CADnC,8DAA+D,CAF/D,+CAAgD,CADhD,8CAMD,CAGD,wCACC,GACC,uBACD,CACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck .ck-upload-placeholder-loader {\\n\\tposition: absolute;\\n\\tdisplay: flex;\\n\\talign-items: center;\\n\\tjustify-content: center;\\n\\ttop: 0;\\n\\tleft: 0;\\n\\n\\t&::before {\\n\\t\\tcontent: '';\\n\\t\\tposition: relative;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-upload-placeholder-loader: hsl(0, 0%, 70%);\\n\\t--ck-upload-placeholder-loader-size: 32px;\\n\\t--ck-upload-placeholder-image-aspect-ratio: 2.8;\\n}\\n\\n.ck .ck-image-upload-placeholder {\\n\\t/* We need to control the full width of the SVG gray background. */\\n\\twidth: 100%;\\n\\tmargin: 0;\\n\\n\\t&.image-inline {\\n\\t\\twidth: calc( 2 * var(--ck-upload-placeholder-loader-size) * var(--ck-upload-placeholder-image-aspect-ratio) );\\n\\t}\\n\\n\\t& img {\\n\\t\\t/*\\n\\t\\t * This is an arbitrary aspect for a 1x1 px GIF to display to the user. Not too tall, not too short.\\n\\t\\t * There's nothing special about this number except that it should make the image placeholder look like\\n\\t\\t * a real image during this short period after the upload started and before the image was read from the\\n\\t\\t * file system (and a rich preview was loaded).\\n\\t\\t */\\n\\t\\taspect-ratio: var(--ck-upload-placeholder-image-aspect-ratio);\\n\\t}\\n}\\n\\n.ck .ck-upload-placeholder-loader {\\n\\twidth: 100%;\\n\\theight: 100%;\\n\\n\\t&::before {\\n\\t\\twidth: var(--ck-upload-placeholder-loader-size);\\n\\t\\theight: var(--ck-upload-placeholder-loader-size);\\n\\t\\tborder-radius: 50%;\\n\\t\\tborder-top: 3px solid var(--ck-color-upload-placeholder-loader);\\n\\t\\tborder-right: 2px solid transparent;\\n\\t\\tanimation: ck-upload-placeholder-loader 1s linear infinite;\\n\\t}\\n}\\n\\n@keyframes ck-upload-placeholder-loader {\\n\\tto {\\n\\t\\ttransform: rotate( 360deg );\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-editor__editable .image,.ck.ck-editor__editable .image-inline{position:relative}.ck.ck-editor__editable .image .ck-progress-bar,.ck.ck-editor__editable .image-inline .ck-progress-bar{left:0;position:absolute;top:0}.ck.ck-editor__editable .image-inline.ck-appear,.ck.ck-editor__editable .image.ck-appear{animation:fadeIn .7s}.ck.ck-editor__editable .image .ck-progress-bar,.ck.ck-editor__editable .image-inline .ck-progress-bar{background:var(--ck-color-upload-bar-background);height:2px;transition:width .1s;width:0}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/imageuploadprogress.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-image/imageuploadprogress.css\"],\"names\":[],\"mappings\":\"AAMC,qEAEC,iBACD,CAGA,uGAIC,MAAO,CAFP,iBAAkB,CAClB,KAED,CCRC,yFACC,oBACD,CAID,uGAIC,gDAAiD,CAFjD,UAAW,CAGX,oBAAuB,CAFvB,OAGD,CAGD,kBACC,GAAO,SAAY,CACnB,GAAO,SAAY,CACpB\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-editor__editable {\\n\\t& .image,\\n\\t& .image-inline {\\n\\t\\tposition: relative;\\n\\t}\\n\\n\\t/* Upload progress bar. */\\n\\t& .image .ck-progress-bar,\\n\\t& .image-inline .ck-progress-bar {\\n\\t\\tposition: absolute;\\n\\t\\ttop: 0;\\n\\t\\tleft: 0;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-editor__editable {\\n\\t& .image,\\n\\t& .image-inline {\\n\\t\\t/* Showing animation. */\\n\\t\\t&.ck-appear {\\n\\t\\t\\tanimation: fadeIn 700ms;\\n\\t\\t}\\n\\t}\\n\\n\\t/* Upload progress bar. */\\n\\t& .image .ck-progress-bar,\\n\\t& .image-inline .ck-progress-bar {\\n\\t\\theight: 2px;\\n\\t\\twidth: 0;\\n\\t\\tbackground: var(--ck-color-upload-bar-background);\\n\\t\\ttransition: width 100ms;\\n\\t}\\n}\\n\\n@keyframes fadeIn {\\n\\tfrom { opacity: 0; }\\n\\tto { opacity: 1; }\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-text-alternative-form{display:flex;flex-direction:row;flex-wrap:nowrap}.ck.ck-text-alternative-form .ck-labeled-field-view{display:inline-block}.ck.ck-text-alternative-form .ck-label{display:none}@media screen and (max-width:600px){.ck.ck-text-alternative-form{flex-wrap:wrap}.ck.ck-text-alternative-form .ck-labeled-field-view{flex-basis:100%}.ck.ck-text-alternative-form .ck-button{flex-basis:50%}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-image/theme/textalternativeform.css\",\"webpack://./../ckeditor5-ui/theme/mixins/_rwd.css\"],\"names\":[],\"mappings\":\"AAOA,6BACC,YAAa,CACb,kBAAmB,CACnB,gBAqBD,CAnBC,oDACC,oBACD,CAEA,uCACC,YACD,CCZA,oCDCD,6BAcE,cAUF,CARE,oDACC,eACD,CAEA,wCACC,cACD,CCrBD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_rwd.css\\\";\\n\\n.ck.ck-text-alternative-form {\\n\\tdisplay: flex;\\n\\tflex-direction: row;\\n\\tflex-wrap: nowrap;\\n\\n\\t& .ck-labeled-field-view {\\n\\t\\tdisplay: inline-block;\\n\\t}\\n\\n\\t& .ck-label {\\n\\t\\tdisplay: none;\\n\\t}\\n\\n\\t@mixin ck-media-phone {\\n\\t\\tflex-wrap: wrap;\\n\\n\\t\\t& .ck-labeled-field-view {\\n\\t\\t\\tflex-basis: 100%;\\n\\t\\t}\\n\\n\\t\\t& .ck-button {\\n\\t\\t\\tflex-basis: 50%;\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@define-mixin ck-media-phone {\\n\\t@media screen and (max-width: 600px) {\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck .ck-link_selected{background:var(--ck-color-link-selected-background)}.ck .ck-link_selected span.image-inline{outline:var(--ck-widget-outline-thickness) solid var(--ck-color-link-selected-background)}.ck .ck-fake-link-selection{background:var(--ck-color-link-fake-selection)}.ck .ck-fake-link-selection_collapsed{border-right:1px solid var(--ck-color-base-text);height:100%;margin-right:-1px;outline:1px solid hsla(0,0%,100%,.5)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-link/link.css\"],\"names\":[],\"mappings\":\"AAMA,sBACC,mDAMD,CAHC,wCACC,yFACD,CAOD,4BACC,8CACD,CAGA,sCAEC,gDAAiD,CADjD,WAAY,CAEZ,iBAAkB,CAClB,oCACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/* Class added to span element surrounding currently selected link. */\\n.ck .ck-link_selected {\\n\\tbackground: var(--ck-color-link-selected-background);\\n\\n\\t/* Give linked inline images some outline to let the user know they are also part of the link. */\\n\\t& span.image-inline {\\n\\t\\toutline: var(--ck-widget-outline-thickness) solid var(--ck-color-link-selected-background);\\n\\t}\\n}\\n\\n/*\\n * Classes used by the \\\"fake visual selection\\\" displayed in the content when an input\\n * in the link UI has focus (the browser does not render the native selection in this state).\\n */\\n.ck .ck-fake-link-selection {\\n\\tbackground: var(--ck-color-link-fake-selection);\\n}\\n\\n/* A collapsed fake visual selection. */\\n.ck .ck-fake-link-selection_collapsed {\\n\\theight: 100%;\\n\\tborder-right: 1px solid var(--ck-color-base-text);\\n\\tmargin-right: -1px;\\n\\toutline: solid 1px hsla(0, 0%, 100%, .5);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-link-actions{display:flex;flex-direction:row;flex-wrap:nowrap}.ck.ck-link-actions .ck-link-actions__preview{display:inline-block}.ck.ck-link-actions .ck-link-actions__preview .ck-button__label{overflow:hidden}@media screen and (max-width:600px){.ck.ck-link-actions{flex-wrap:wrap}.ck.ck-link-actions .ck-link-actions__preview{flex-basis:100%}.ck.ck-link-actions .ck-button:not(.ck-link-actions__preview){flex-basis:50%}}.ck.ck-link-actions .ck-button.ck-link-actions__preview{padding-left:0;padding-right:0}.ck.ck-link-actions .ck-button.ck-link-actions__preview .ck-button__label{color:var(--ck-color-link-default);cursor:pointer;max-width:var(--ck-input-width);min-width:3em;padding:0 var(--ck-spacing-medium);text-align:center;text-overflow:ellipsis}.ck.ck-link-actions .ck-button.ck-link-actions__preview .ck-button__label:hover{text-decoration:underline}.ck.ck-link-actions .ck-button.ck-link-actions__preview,.ck.ck-link-actions .ck-button.ck-link-actions__preview:active,.ck.ck-link-actions .ck-button.ck-link-actions__preview:focus,.ck.ck-link-actions .ck-button.ck-link-actions__preview:hover{background:none}.ck.ck-link-actions .ck-button.ck-link-actions__preview:active{box-shadow:none}.ck.ck-link-actions .ck-button.ck-link-actions__preview:focus .ck-button__label{text-decoration:underline}[dir=ltr] .ck.ck-link-actions .ck-button:not(:first-child),[dir=rtl] .ck.ck-link-actions .ck-button:not(:last-child){margin-left:var(--ck-spacing-standard)}@media screen and (max-width:600px){.ck.ck-link-actions .ck-button.ck-link-actions__preview{margin:var(--ck-spacing-standard) var(--ck-spacing-standard) 0}.ck.ck-link-actions .ck-button.ck-link-actions__preview .ck-button__label{max-width:100%;min-width:0}[dir=ltr] .ck.ck-link-actions .ck-button:not(.ck-link-actions__preview),[dir=rtl] .ck.ck-link-actions .ck-button:not(.ck-link-actions__preview){margin-left:0}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-link/theme/linkactions.css\",\"webpack://./../ckeditor5-ui/theme/mixins/_rwd.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-link/linkactions.css\"],\"names\":[],\"mappings\":\"AAOA,oBACC,YAAa,CACb,kBAAmB,CACnB,gBAqBD,CAnBC,8CACC,oBAKD,CAHC,gEACC,eACD,CCXD,oCDCD,oBAcE,cAUF,CARE,8CACC,eACD,CAEA,8DACC,cACD,CCrBD,CCIA,wDACC,cAAe,CACf,eAmCD,CAjCC,0EAEC,kCAAmC,CAEnC,cAAe,CAIf,+BAAgC,CAChC,aAAc,CARd,kCAAmC,CASnC,iBAAkB,CAPlB,sBAYD,CAHC,gFACC,yBACD,CAGD,mPAIC,eACD,CAEA,+DACC,eACD,CAGC,gFACC,yBACD,CAWD,qHACC,sCACD,CDtDD,oCC0DC,wDACC,8DAMD,CAJC,0EAEC,cAAe,CADf,WAED,CAGD,gJAME,aAEF,CDzED\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_rwd.css\\\";\\n\\n.ck.ck-link-actions {\\n\\tdisplay: flex;\\n\\tflex-direction: row;\\n\\tflex-wrap: nowrap;\\n\\n\\t& .ck-link-actions__preview {\\n\\t\\tdisplay: inline-block;\\n\\n\\t\\t& .ck-button__label {\\n\\t\\t\\toverflow: hidden;\\n\\t\\t}\\n\\t}\\n\\n\\t@mixin ck-media-phone {\\n\\t\\tflex-wrap: wrap;\\n\\n\\t\\t& .ck-link-actions__preview {\\n\\t\\t\\tflex-basis: 100%;\\n\\t\\t}\\n\\n\\t\\t& .ck-button:not(.ck-link-actions__preview) {\\n\\t\\t\\tflex-basis: 50%;\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@define-mixin ck-media-phone {\\n\\t@media screen and (max-width: 600px) {\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_unselectable.css\\\";\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n@import \\\"../mixins/_focus.css\\\";\\n@import \\\"../mixins/_shadow.css\\\";\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_rwd.css\\\";\\n\\n.ck.ck-link-actions {\\n\\t& .ck-button.ck-link-actions__preview {\\n\\t\\tpadding-left: 0;\\n\\t\\tpadding-right: 0;\\n\\n\\t\\t& .ck-button__label {\\n\\t\\t\\tpadding: 0 var(--ck-spacing-medium);\\n\\t\\t\\tcolor: var(--ck-color-link-default);\\n\\t\\t\\ttext-overflow: ellipsis;\\n\\t\\t\\tcursor: pointer;\\n\\n\\t\\t\\t/* Match the box model of the link editor form's input so the balloon\\n\\t\\t\\tdoes not change width when moving between actions and the form. */\\n\\t\\t\\tmax-width: var(--ck-input-width);\\n\\t\\t\\tmin-width: 3em;\\n\\t\\t\\ttext-align: center;\\n\\n\\t\\t\\t&:hover {\\n\\t\\t\\t\\ttext-decoration: underline;\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t&,\\n\\t\\t&:hover,\\n\\t\\t&:focus,\\n\\t\\t&:active {\\n\\t\\t\\tbackground: none;\\n\\t\\t}\\n\\n\\t\\t&:active {\\n\\t\\t\\tbox-shadow: none;\\n\\t\\t}\\n\\n\\t\\t&:focus {\\n\\t\\t\\t& .ck-button__label {\\n\\t\\t\\t\\ttext-decoration: underline;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t@mixin ck-dir ltr {\\n\\t\\t& .ck-button:not(:first-child) {\\n\\t\\t\\tmargin-left: var(--ck-spacing-standard);\\n\\t\\t}\\n\\t}\\n\\n\\t@mixin ck-dir rtl {\\n\\t\\t& .ck-button:not(:last-child) {\\n\\t\\t\\tmargin-left: var(--ck-spacing-standard);\\n\\t\\t}\\n\\t}\\n\\n\\t@mixin ck-media-phone {\\n\\t\\t& .ck-button.ck-link-actions__preview {\\n\\t\\t\\tmargin: var(--ck-spacing-standard) var(--ck-spacing-standard) 0;\\n\\n\\t\\t\\t& .ck-button__label {\\n\\t\\t\\t\\tmin-width: 0;\\n\\t\\t\\t\\tmax-width: 100%;\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t& .ck-button:not(.ck-link-actions__preview) {\\n\\t\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t\\tmargin-left: 0;\\n\\t\\t\\t}\\n\\n\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\tmargin-left: 0;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-link-form{display:flex}.ck.ck-link-form .ck-label{display:none}@media screen and (max-width:600px){.ck.ck-link-form{flex-wrap:wrap}.ck.ck-link-form .ck-labeled-field-view{flex-basis:100%}.ck.ck-link-form .ck-button{flex-basis:50%}}.ck.ck-link-form_layout-vertical{display:block}.ck.ck-link-form_layout-vertical .ck-button.ck-button-cancel,.ck.ck-link-form_layout-vertical .ck-button.ck-button-save{margin-top:var(--ck-spacing-medium)}.ck.ck-link-form_layout-vertical{min-width:var(--ck-input-width);padding:0}.ck.ck-link-form_layout-vertical .ck-labeled-field-view{margin:var(--ck-spacing-large) var(--ck-spacing-large) var(--ck-spacing-small)}.ck.ck-link-form_layout-vertical .ck-labeled-field-view .ck-input-text{min-width:0;width:100%}.ck.ck-link-form_layout-vertical>.ck-button{border-radius:0;margin:0;padding:var(--ck-spacing-standard);width:50%}.ck.ck-link-form_layout-vertical>.ck-button:not(:focus){border-top:1px solid var(--ck-color-base-border)}[dir=ltr] .ck.ck-link-form_layout-vertical>.ck-button,[dir=rtl] .ck.ck-link-form_layout-vertical>.ck-button{margin-left:0}[dir=rtl] .ck.ck-link-form_layout-vertical>.ck-button:last-of-type{border-right:1px solid var(--ck-color-base-border)}.ck.ck-link-form_layout-vertical .ck.ck-list{margin:var(--ck-spacing-standard) var(--ck-spacing-large)}.ck.ck-link-form_layout-vertical .ck.ck-list .ck-button.ck-switchbutton{padding:0;width:100%}.ck.ck-link-form_layout-vertical .ck.ck-list .ck-button.ck-switchbutton:hover{background:none}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-link/theme/linkform.css\",\"webpack://./../ckeditor5-ui/theme/mixins/_rwd.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-link/linkform.css\"],\"names\":[],\"mappings\":\"AAOA,iBACC,YAiBD,CAfC,2BACC,YACD,CCNA,oCDCD,iBAQE,cAUF,CARE,wCACC,eACD,CAEA,4BACC,cACD,CCfD,CDuBD,iCACC,aAYD,CALE,wHAEC,mCACD,CE/BF,iCAEC,+BAAgC,CADhC,SAgDD,CA7CC,wDACC,8EAMD,CAJC,uEACC,WAAY,CACZ,UACD,CAGD,4CAIC,eAAgB,CAFhB,QAAS,CADT,kCAAmC,CAEnC,SAkBD,CAfC,wDACC,gDACD,CARD,4GAeE,aAMF,CAJE,mEACC,kDACD,CAKF,6CACC,yDAUD,CARC,wEACC,SAAU,CACV,UAKD,CAHC,8EACC,eACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_rwd.css\\\";\\n\\n.ck.ck-link-form {\\n\\tdisplay: flex;\\n\\n\\t& .ck-label {\\n\\t\\tdisplay: none;\\n\\t}\\n\\n\\t@mixin ck-media-phone {\\n\\t\\tflex-wrap: wrap;\\n\\n\\t\\t& .ck-labeled-field-view {\\n\\t\\t\\tflex-basis: 100%;\\n\\t\\t}\\n\\n\\t\\t& .ck-button {\\n\\t\\t\\tflex-basis: 50%;\\n\\t\\t}\\n\\t}\\n}\\n\\n/*\\n * Style link form differently when manual decorators are available.\\n * See: https://github.com/ckeditor/ckeditor5-link/issues/186.\\n */\\n.ck.ck-link-form_layout-vertical {\\n\\tdisplay: block;\\n\\n\\t/*\\n\\t * Whether the form is in the responsive mode or not, if there are decorator buttons\\n\\t * keep the top margin of action buttons medium.\\n\\t */\\n\\t& .ck-button {\\n\\t\\t&.ck-button-save,\\n\\t\\t&.ck-button-cancel {\\n\\t\\t\\tmargin-top: var(--ck-spacing-medium);\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@define-mixin ck-media-phone {\\n\\t@media screen and (max-width: 600px) {\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n/*\\n * Style link form differently when manual decorators are available.\\n * See: https://github.com/ckeditor/ckeditor5-link/issues/186.\\n */\\n.ck.ck-link-form_layout-vertical {\\n\\tpadding: 0;\\n\\tmin-width: var(--ck-input-width);\\n\\n\\t& .ck-labeled-field-view {\\n\\t\\tmargin: var(--ck-spacing-large) var(--ck-spacing-large) var(--ck-spacing-small);\\n\\n\\t\\t& .ck-input-text {\\n\\t\\t\\tmin-width: 0;\\n\\t\\t\\twidth: 100%;\\n\\t\\t}\\n\\t}\\n\\n\\t& > .ck-button {\\n\\t\\tpadding: var(--ck-spacing-standard);\\n\\t\\tmargin: 0;\\n\\t\\twidth: 50%;\\n\\t\\tborder-radius: 0;\\n\\n\\t\\t&:not(:focus) {\\n\\t\\t\\tborder-top: 1px solid var(--ck-color-base-border);\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\tmargin-left: 0;\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\tmargin-left: 0;\\n\\n\\t\\t\\t&:last-of-type {\\n\\t\\t\\t\\tborder-right: 1px solid var(--ck-color-base-border);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t/* Using additional `.ck` class for stronger CSS specificity than `.ck.ck-link-form > :not(:first-child)`. */\\n\\t& .ck.ck-list {\\n\\t\\tmargin: var(--ck-spacing-standard) var(--ck-spacing-large);\\n\\n\\t\\t& .ck-button.ck-switchbutton {\\n\\t\\t\\tpadding: 0;\\n\\t\\t\\twidth: 100%;\\n\\n\\t\\t\\t&:hover {\\n\\t\\t\\t\\tbackground: none;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-editor__editable a span.image-inline:after,.ck.ck-editor__editable figure.image>a:after{display:block;position:absolute}:root{--ck-link-image-indicator-icon-size:20;--ck-link-image-indicator-icon-is-visible:clamp(0px,100% - 50px,1px)}.ck.ck-editor__editable a span.image-inline:after,.ck.ck-editor__editable figure.image>a:after{background-color:rgba(0,0,0,.4);background-image:url(\\\"\\\");background-position:50%;background-repeat:no-repeat;background-size:14px;border-radius:100%;content:\\\"\\\";height:calc(var(--ck-link-image-indicator-icon-is-visible)*var(--ck-link-image-indicator-icon-size));overflow:hidden;right:min(var(--ck-spacing-medium),6%);top:min(var(--ck-spacing-medium),6%);width:calc(var(--ck-link-image-indicator-icon-is-visible)*var(--ck-link-image-indicator-icon-size))}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-link/theme/linkimage.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-link/linkimage.css\"],\"names\":[],\"mappings\":\"AASE,+FACC,aAAc,CACd,iBACD,CCPF,MAEC,sCAAuC,CACvC,oEACD,CAME,+FAUC,+BAAqC,CACrC,83BAA+3B,CAG/3B,uBAA2B,CAD3B,2BAA4B,CAD5B,oBAAqB,CAGrB,kBAAmB,CAdnB,UAAW,CAsBX,oGAAuG,CAFvG,eAAgB,CAbhB,sCAAwC,CADxC,oCAAsC,CAetC,mGAED\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-editor__editable {\\n\\t/* Linked image indicator */\\n\\t& figure.image > a,\\n\\t& a span.image-inline {\\n\\t\\t&::after {\\n\\t\\t\\tdisplay: block;\\n\\t\\t\\tposition: absolute;\\n\\t\\t}\\n\\t}\\n}\\n\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t/* Match the icon size with the upload indicator brought by the image upload feature. */\\n\\t--ck-link-image-indicator-icon-size: 20;\\n\\t--ck-link-image-indicator-icon-is-visible: clamp(0px, 100% - 50px, 1px);\\n}\\n\\n.ck.ck-editor__editable {\\n\\t/* Linked image indicator */\\n\\t& figure.image > a,\\n\\t& a span.image-inline {\\n\\t\\t&::after {\\n\\t\\t\\tcontent: \\\"\\\";\\n\\n\\t\\t\\t/*\\n\\t\\t\\t * Smaller images should have the icon closer to the border.\\n\\t\\t\\t * Match the icon position with the upload indicator brought by the image upload feature.\\n\\t\\t\\t */\\n\\t\\t\\ttop: min(var(--ck-spacing-medium), 6%);\\n\\t\\t\\tright: min(var(--ck-spacing-medium), 6%);\\n\\n\\t\\t\\tbackground-color: hsla(0, 0%, 0%, .4);\\n\\t\\t\\tbackground-image: url(\\\"\\\");\\n\\t\\t\\tbackground-size: 14px;\\n\\t\\t\\tbackground-repeat: no-repeat;\\n\\t\\t\\tbackground-position: center;\\n\\t\\t\\tborder-radius: 100%;\\n\\n\\t\\t\\t/*\\n\\t\\t\\t* Use CSS math to simulate container queries.\\n\\t\\t\\t* https://css-tricks.com/the-raven-technique-one-step-closer-to-container-queries/#what-about-showing-and-hiding-things\\n\\t\\t\\t*/\\n\\t\\t\\toverflow: hidden;\\n\\t\\t\\twidth: calc(var(--ck-link-image-indicator-icon-is-visible) * var(--ck-link-image-indicator-icon-size));\\n\\t\\t\\theight: calc(var(--ck-link-image-indicator-icon-is-visible) * var(--ck-link-image-indicator-icon-size));\\n\\t\\t}\\n\\t}\\n}\\n\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-collapsible.ck-collapsible_collapsed>.ck-collapsible__children{display:none}:root{--ck-collapsible-arrow-size:calc(var(--ck-icon-size)*0.5)}.ck.ck-collapsible>.ck.ck-button{border-radius:0;color:inherit;font-weight:700;padding:var(--ck-spacing-medium) var(--ck-spacing-large);width:100%}.ck.ck-collapsible>.ck.ck-button:focus{background:transparent}.ck.ck-collapsible>.ck.ck-button:active,.ck.ck-collapsible>.ck.ck-button:hover:not(:focus),.ck.ck-collapsible>.ck.ck-button:not(:focus){background:transparent;border-color:transparent;box-shadow:none}.ck.ck-collapsible>.ck.ck-button>.ck-icon{margin-right:var(--ck-spacing-medium);width:var(--ck-collapsible-arrow-size)}.ck.ck-collapsible>.ck-collapsible__children{padding:0 var(--ck-spacing-large) var(--ck-spacing-large)}.ck.ck-collapsible.ck-collapsible_collapsed>.ck.ck-button .ck-icon{transform:rotate(-90deg)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-list/theme/collapsible.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-list/collapsible.css\"],\"names\":[],\"mappings\":\"AAMC,sEACC,YACD,CCHD,MACC,yDACD,CAGC,iCAIC,eAAgB,CAChB,aAAc,CAHd,eAAiB,CACjB,wDAAyD,CAFzD,UAoBD,CAdC,uCACC,sBACD,CAEA,wIACC,sBAAuB,CACvB,wBAAyB,CACzB,eACD,CAEA,0CACC,qCAAsC,CACtC,sCACD,CAGD,6CACC,yDACD,CAGC,mEACC,wBACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-collapsible.ck-collapsible_collapsed {\\n\\t& > .ck-collapsible__children {\\n\\t\\tdisplay: none;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-collapsible-arrow-size: calc(0.5 * var(--ck-icon-size));\\n}\\n\\n.ck.ck-collapsible {\\n\\t& > .ck.ck-button {\\n\\t\\twidth: 100%;\\n\\t\\tfont-weight: bold;\\n\\t\\tpadding: var(--ck-spacing-medium) var(--ck-spacing-large);\\n\\t\\tborder-radius: 0;\\n\\t\\tcolor: inherit;\\n\\n\\t\\t&:focus {\\n\\t\\t\\tbackground: transparent;\\n\\t\\t}\\n\\n\\t\\t&:active, &:not(:focus), &:hover:not(:focus) {\\n\\t\\t\\tbackground: transparent;\\n\\t\\t\\tborder-color: transparent;\\n\\t\\t\\tbox-shadow: none;\\n\\t\\t}\\n\\n\\t\\t& > .ck-icon {\\n\\t\\t\\tmargin-right: var(--ck-spacing-medium);\\n\\t\\t\\twidth: var(--ck-collapsible-arrow-size);\\n\\t\\t}\\n\\t}\\n\\n\\t& > .ck-collapsible__children {\\n\\t\\tpadding: 0 var(--ck-spacing-large) var(--ck-spacing-large);\\n\\t}\\n\\n\\t&.ck-collapsible_collapsed {\\n\\t\\t& > .ck.ck-button .ck-icon {\\n\\t\\t\\ttransform: rotate(-90deg);\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-editor__editable .ck-list-bogus-paragraph{display:block}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-list/theme/documentlist.css\"],\"names\":[],\"mappings\":\"AAKA,8CACC,aACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-editor__editable .ck-list-bogus-paragraph {\\n\\tdisplay: block;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-content ol{list-style-type:decimal}.ck-content ol ol{list-style-type:lower-latin}.ck-content ol ol ol{list-style-type:lower-roman}.ck-content ol ol ol ol{list-style-type:upper-latin}.ck-content ol ol ol ol ol{list-style-type:upper-roman}.ck-content ul{list-style-type:disc}.ck-content ul ul{list-style-type:circle}.ck-content ul ul ul,.ck-content ul ul ul ul{list-style-type:square}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-list/theme/list.css\"],\"names\":[],\"mappings\":\"AAKA,eACC,uBAiBD,CAfC,kBACC,2BAaD,CAXC,qBACC,2BASD,CAPC,wBACC,2BAKD,CAHC,2BACC,2BACD,CAMJ,eACC,oBAaD,CAXC,kBACC,sBASD,CAJE,6CACC,sBACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-content ol {\\n\\tlist-style-type: decimal;\\n\\n\\t& ol {\\n\\t\\tlist-style-type: lower-latin;\\n\\n\\t\\t& ol {\\n\\t\\t\\tlist-style-type: lower-roman;\\n\\n\\t\\t\\t& ol {\\n\\t\\t\\t\\tlist-style-type: upper-latin;\\n\\n\\t\\t\\t\\t& ol {\\n\\t\\t\\t\\t\\tlist-style-type: upper-roman;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\\n.ck-content ul {\\n\\tlist-style-type: disc;\\n\\n\\t& ul {\\n\\t\\tlist-style-type: circle;\\n\\n\\t\\t& ul {\\n\\t\\t\\tlist-style-type: square;\\n\\n\\t\\t\\t& ul {\\n\\t\\t\\t\\tlist-style-type: square;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-list-properties.ck-list-properties_without-styles{padding:var(--ck-spacing-large)}.ck.ck-list-properties.ck-list-properties_without-styles>*{min-width:14em}.ck.ck-list-properties.ck-list-properties_without-styles>*+*{margin-top:var(--ck-spacing-standard)}.ck.ck-list-properties.ck-list-properties_with-numbered-properties>.ck-list-styles-list{grid-template-columns:repeat(4,auto)}.ck.ck-list-properties.ck-list-properties_with-numbered-properties>.ck-collapsible{border-top:1px solid var(--ck-color-base-border)}.ck.ck-list-properties.ck-list-properties_with-numbered-properties>.ck-collapsible>.ck-collapsible__children>*{width:100%}.ck.ck-list-properties.ck-list-properties_with-numbered-properties>.ck-collapsible>.ck-collapsible__children>*+*{margin-top:var(--ck-spacing-standard)}.ck.ck-list-properties .ck.ck-numbered-list-properties__start-index .ck-input{min-width:auto;width:100%}.ck.ck-list-properties .ck.ck-numbered-list-properties__reversed-order{background:transparent;margin-bottom:calc(var(--ck-spacing-tiny)*-1);padding-left:0;padding-right:0}.ck.ck-list-properties .ck.ck-numbered-list-properties__reversed-order:active,.ck.ck-list-properties .ck.ck-numbered-list-properties__reversed-order:hover{background:none;border-color:transparent;box-shadow:none}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-list/listproperties.css\"],\"names\":[],\"mappings\":\"AAOC,yDACC,+BASD,CAPC,2DACC,cAKD,CAHC,6DACC,qCACD,CASD,wFACC,oCACD,CAGA,mFACC,gDAWD,CARE,+GACC,UAKD,CAHC,iHACC,qCACD,CAMJ,8EACC,cAAe,CACf,UACD,CAEA,uEACC,sBAAuB,CAGvB,6CAAgD,CAFhD,cAAe,CACf,eAQD,CALC,2JAGC,eAAgB,CADhB,wBAAyB,CADzB,eAGD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-list-properties {\\n\\t/* When there are no list styles and there is no collapsible. */\\n\\t&.ck-list-properties_without-styles {\\n\\t\\tpadding: var(--ck-spacing-large);\\n\\n\\t\\t& > * {\\n\\t\\t\\tmin-width: 14em;\\n\\n\\t\\t\\t& + * {\\n\\t\\t\\t\\tmargin-top: var(--ck-spacing-standard);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t/*\\n\\t * When the numbered list property fields (start at, reversed) should be displayed,\\n\\t * more horizontal space is needed. Reconfigure the style grid to create that space.\\n\\t */\\n\\t&.ck-list-properties_with-numbered-properties {\\n\\t\\t& > .ck-list-styles-list {\\n\\t\\t\\tgrid-template-columns: repeat( 4, auto );\\n\\t\\t}\\n\\n\\t\\t/* When list styles are rendered and property fields are in a collapsible. */\\n\\t\\t& > .ck-collapsible {\\n\\t\\t\\tborder-top: 1px solid var(--ck-color-base-border);\\n\\n\\t\\t\\t& > .ck-collapsible__children {\\n\\t\\t\\t\\t& > * {\\n\\t\\t\\t\\t\\twidth: 100%;\\n\\n\\t\\t\\t\\t\\t& + * {\\n\\t\\t\\t\\t\\t\\tmargin-top: var(--ck-spacing-standard);\\n\\t\\t\\t\\t\\t}\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck.ck-numbered-list-properties__start-index .ck-input {\\n\\t\\tmin-width: auto;\\n\\t\\twidth: 100%;\\n\\t}\\n\\n\\t& .ck.ck-numbered-list-properties__reversed-order {\\n\\t\\tbackground: transparent;\\n\\t\\tpadding-left: 0;\\n\\t\\tpadding-right: 0;\\n\\t\\tmargin-bottom: calc(-1 * var(--ck-spacing-tiny));\\n\\n\\t\\t&:active, &:hover {\\n\\t\\t\\tbox-shadow: none;\\n\\t\\t\\tborder-color: transparent;\\n\\t\\t\\tbackground: none;\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-list-styles-list{display:grid}:root{--ck-list-style-button-size:44px}.ck.ck-list-styles-list{column-gap:var(--ck-spacing-medium);grid-template-columns:repeat(3,auto);padding:var(--ck-spacing-large);row-gap:var(--ck-spacing-medium)}.ck.ck-list-styles-list .ck-button{box-sizing:content-box;margin:0;padding:0}.ck.ck-list-styles-list .ck-button,.ck.ck-list-styles-list .ck-button .ck-icon{height:var(--ck-list-style-button-size);width:var(--ck-list-style-button-size)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-list/theme/liststyles.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-list/liststyles.css\"],\"names\":[],\"mappings\":\"AAKA,wBACC,YACD,CCFA,MACC,gCACD,CAEA,wBAGC,mCAAoC,CAFpC,oCAAwC,CAGxC,+BAAgC,CAFhC,gCA4BD,CAxBC,mCAiBC,sBAAuB,CAPvB,QAAS,CANT,SAmBD,CAJC,+EAhBA,uCAAwC,CADxC,sCAoBA\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-list-styles-list {\\n\\tdisplay: grid;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-list-style-button-size: 44px;\\n}\\n\\n.ck.ck-list-styles-list {\\n\\tgrid-template-columns: repeat( 3, auto );\\n\\trow-gap: var(--ck-spacing-medium);\\n\\tcolumn-gap: var(--ck-spacing-medium);\\n\\tpadding: var(--ck-spacing-large);\\n\\n\\t& .ck-button {\\n\\t\\t/* Make the button look like a thumbnail (the icon \\\"takes it all\\\"). */\\n\\t\\twidth: var(--ck-list-style-button-size);\\n\\t\\theight: var(--ck-list-style-button-size);\\n\\t\\tpadding: 0;\\n\\n\\t\\t/*\\n\\t\\t * Buttons are aligned by the grid so disable default button margins to not collide with the\\n\\t\\t * gaps in the grid.\\n\\t\\t */\\n\\t\\tmargin: 0;\\n\\n\\t\\t/*\\n\\t\\t * Make sure the button border (which is displayed on focus, BTW) does not steal pixels\\n\\t\\t * from the button dimensions and, as a result, decrease the size of the icon\\n\\t\\t * (which becomes blurry as it scales down).\\n\\t\\t */\\n\\t\\tbox-sizing: content-box;\\n\\n\\t\\t& .ck-icon {\\n\\t\\t\\twidth: var(--ck-list-style-button-size);\\n\\t\\t\\theight: var(--ck-list-style-button-size);\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-todo-list-checkmark-size:16px}.ck-content .todo-list{list-style:none}.ck-content .todo-list li{margin-bottom:5px}.ck-content .todo-list li .todo-list{margin-top:5px}.ck-content .todo-list .todo-list__label>input{-webkit-appearance:none;border:0;display:inline-block;height:var(--ck-todo-list-checkmark-size);left:-25px;margin-left:0;margin-right:-15px;position:relative;right:0;vertical-align:middle;width:var(--ck-todo-list-checkmark-size)}.ck-content .todo-list .todo-list__label>input:before{border:1px solid #333;border-radius:2px;box-sizing:border-box;content:\\\"\\\";display:block;height:100%;position:absolute;transition:box-shadow .25s ease-in-out,background .25s ease-in-out,border .25s ease-in-out;width:100%}.ck-content .todo-list .todo-list__label>input:after{border-color:transparent;border-style:solid;border-width:0 calc(var(--ck-todo-list-checkmark-size)/8) calc(var(--ck-todo-list-checkmark-size)/8) 0;box-sizing:content-box;content:\\\"\\\";display:block;height:calc(var(--ck-todo-list-checkmark-size)/2.6);left:calc(var(--ck-todo-list-checkmark-size)/3);pointer-events:none;position:absolute;top:calc(var(--ck-todo-list-checkmark-size)/5.3);transform:rotate(45deg);width:calc(var(--ck-todo-list-checkmark-size)/5.3)}.ck-content .todo-list .todo-list__label>input[checked]:before{background:#26ab33;border-color:#26ab33}.ck-content .todo-list .todo-list__label>input[checked]:after{border-color:#fff}.ck-content .todo-list .todo-list__label .todo-list__label__description{vertical-align:middle}[dir=rtl] .todo-list .todo-list__label>input{left:0;margin-left:-15px;margin-right:0;right:-25px}.ck-editor__editable .todo-list .todo-list__label>input{cursor:pointer}.ck-editor__editable .todo-list .todo-list__label>input:hover:before{box-shadow:0 0 0 5px rgba(0,0,0,.1)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-list/theme/todolist.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,kCACD,CAEA,uBACC,eA0ED,CAxEC,0BACC,iBAKD,CAHC,qCACC,cACD,CAIA,+CACC,uBAAwB,CAQxB,QAAS,CAPT,oBAAqB,CAGrB,yCAA0C,CAO1C,UAAW,CAGX,aAAc,CAFd,kBAAmB,CAVnB,iBAAkB,CAWlB,OAAQ,CARR,qBAAsB,CAFtB,wCAqDD,CAxCC,sDAOC,qBAAiC,CACjC,iBAAkB,CALlB,qBAAsB,CACtB,UAAW,CAHX,aAAc,CAKd,WAAY,CAJZ,iBAAkB,CAOlB,0FAAgG,CAJhG,UAKD,CAEA,qDAaC,wBAAyB,CADzB,kBAAmB,CAEnB,sGAA+G,CAX/G,sBAAuB,CAEvB,UAAW,CAJX,aAAc,CAUd,mDAAwD,CAHxD,+CAAoD,CAJpD,mBAAoB,CAFpB,iBAAkB,CAOlB,gDAAqD,CAMrD,uBAAwB,CALxB,kDAMD,CAGC,+DACC,kBAA8B,CAC9B,oBACD,CAEA,8DACC,iBACD,CAIF,wEACC,qBACD,CAKF,6CACC,MAAO,CAGP,iBAAkB,CAFlB,cAAe,CACf,WAED,CAMA,wDACC,cAKD,CAHC,qEACC,mCACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-todo-list-checkmark-size: 16px;\\n}\\n\\n.ck-content .todo-list {\\n\\tlist-style: none;\\n\\n\\t& li {\\n\\t\\tmargin-bottom: 5px;\\n\\n\\t\\t& .todo-list {\\n\\t\\t\\tmargin-top: 5px;\\n\\t\\t}\\n\\t}\\n\\n\\t& .todo-list__label {\\n\\t\\t& > input {\\n\\t\\t\\t-webkit-appearance: none;\\n\\t\\t\\tdisplay: inline-block;\\n\\t\\t\\tposition: relative;\\n\\t\\t\\twidth: var(--ck-todo-list-checkmark-size);\\n\\t\\t\\theight: var(--ck-todo-list-checkmark-size);\\n\\t\\t\\tvertical-align: middle;\\n\\n\\t\\t\\t/* Needed on iOS */\\n\\t\\t\\tborder: 0;\\n\\n\\t\\t\\t/* LTR styles */\\n\\t\\t\\tleft: -25px;\\n\\t\\t\\tmargin-right: -15px;\\n\\t\\t\\tright: 0;\\n\\t\\t\\tmargin-left: 0;\\n\\n\\t\\t\\t&::before {\\n\\t\\t\\t\\tdisplay: block;\\n\\t\\t\\t\\tposition: absolute;\\n\\t\\t\\t\\tbox-sizing: border-box;\\n\\t\\t\\t\\tcontent: '';\\n\\t\\t\\t\\twidth: 100%;\\n\\t\\t\\t\\theight: 100%;\\n\\t\\t\\t\\tborder: 1px solid hsl(0, 0%, 20%);\\n\\t\\t\\t\\tborder-radius: 2px;\\n\\t\\t\\t\\ttransition: 250ms ease-in-out box-shadow, 250ms ease-in-out background, 250ms ease-in-out border;\\n\\t\\t\\t}\\n\\n\\t\\t\\t&::after {\\n\\t\\t\\t\\tdisplay: block;\\n\\t\\t\\t\\tposition: absolute;\\n\\t\\t\\t\\tbox-sizing: content-box;\\n\\t\\t\\t\\tpointer-events: none;\\n\\t\\t\\t\\tcontent: '';\\n\\n\\t\\t\\t\\t/* Calculate tick position, size and border-width proportional to the checkmark size. */\\n\\t\\t\\t\\tleft: calc( var(--ck-todo-list-checkmark-size) / 3 );\\n\\t\\t\\t\\ttop: calc( var(--ck-todo-list-checkmark-size) / 5.3 );\\n\\t\\t\\t\\twidth: calc( var(--ck-todo-list-checkmark-size) / 5.3 );\\n\\t\\t\\t\\theight: calc( var(--ck-todo-list-checkmark-size) / 2.6 );\\n\\t\\t\\t\\tborder-style: solid;\\n\\t\\t\\t\\tborder-color: transparent;\\n\\t\\t\\t\\tborder-width: 0 calc( var(--ck-todo-list-checkmark-size) / 8 ) calc( var(--ck-todo-list-checkmark-size) / 8 ) 0;\\n\\t\\t\\t\\ttransform: rotate(45deg);\\n\\t\\t\\t}\\n\\n\\t\\t\\t&[checked] {\\n\\t\\t\\t\\t&::before {\\n\\t\\t\\t\\t\\tbackground: hsl(126, 64%, 41%);\\n\\t\\t\\t\\t\\tborder-color: hsl(126, 64%, 41%);\\n\\t\\t\\t\\t}\\n\\n\\t\\t\\t\\t&::after {\\n\\t\\t\\t\\t\\tborder-color: hsl(0, 0%, 100%);\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t& .todo-list__label__description {\\n\\t\\t\\tvertical-align: middle;\\n\\t\\t}\\n\\t}\\n}\\n\\n/* RTL styles */\\n[dir=\\\"rtl\\\"] .todo-list .todo-list__label > input {\\n\\tleft: 0;\\n\\tmargin-right: 0;\\n\\tright: -25px;\\n\\tmargin-left: -15px;\\n}\\n\\n/*\\n * To-do list should be interactive only during the editing\\n * (https://github.com/ckeditor/ckeditor5/issues/2090).\\n */\\n.ck-editor__editable .todo-list .todo-list__label > input {\\n\\tcursor: pointer;\\n\\n\\t&:hover::before {\\n\\t\\tbox-shadow: 0 0 0 5px hsla(0, 0%, 0%, 0.1);\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-content .media{clear:both;display:block;margin:.9em 0;min-width:15em}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-media-embed/theme/mediaembed.css\"],\"names\":[],\"mappings\":\"AAKA,mBAGC,UAAW,CASX,aAAc,CAJd,aAAe,CAQf,cACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-content .media {\\n\\t/* Don't allow floated content overlap the media.\\n\\thttps://github.com/ckeditor/ckeditor5-media-embed/issues/53 */\\n\\tclear: both;\\n\\n\\t/* Make sure there is some space between the content and the media. */\\n\\t/* The first value should be equal to --ck-spacing-large variable if used in the editor context\\n\\tto avoid the content jumping (See https://github.com/ckeditor/ckeditor5/issues/9825). */\\n\\tmargin: 0.9em 0;\\n\\n\\t/* Make sure media is not overriden with Bootstrap default `flex` value.\\n\\tSee: https://github.com/ckeditor/ckeditor5/issues/1373. */\\n\\tdisplay: block;\\n\\n\\t/* Give the media some minimal width in the content to prevent them\\n\\tfrom being \\\"squashed\\\" in tight spaces, e.g. in table cells (#44) */\\n\\tmin-width: 15em;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-media__wrapper .ck-media__placeholder{align-items:center;display:flex;flex-direction:column}.ck-media__wrapper .ck-media__placeholder .ck-media__placeholder__url{max-width:100%;position:relative}.ck-media__wrapper .ck-media__placeholder .ck-media__placeholder__url .ck-media__placeholder__url__text{display:block;overflow:hidden}.ck-media__wrapper[data-oembed-url*=\\\"facebook.com\\\"] .ck-media__placeholder__icon *,.ck-media__wrapper[data-oembed-url*=\\\"goo.gl/maps\\\"] .ck-media__placeholder__icon *,.ck-media__wrapper[data-oembed-url*=\\\"google.com/maps\\\"] .ck-media__placeholder__icon *,.ck-media__wrapper[data-oembed-url*=\\\"instagram.com\\\"] .ck-media__placeholder__icon *,.ck-media__wrapper[data-oembed-url*=\\\"maps.app.goo.gl\\\"] .ck-media__placeholder__icon *,.ck-media__wrapper[data-oembed-url*=\\\"maps.google.com\\\"] .ck-media__placeholder__icon *,.ck-media__wrapper[data-oembed-url*=\\\"twitter.com\\\"] .ck-media__placeholder__icon *{display:none}.ck-editor__editable:not(.ck-read-only) .ck-media__wrapper>:not(.ck-media__placeholder),.ck-editor__editable:not(.ck-read-only) .ck-widget:not(.ck-widget_selected) .ck-media__placeholder{pointer-events:none}:root{--ck-media-embed-placeholder-icon-size:3em;--ck-color-media-embed-placeholder-url-text:#757575;--ck-color-media-embed-placeholder-url-text-hover:var(--ck-color-base-text)}.ck-media__wrapper{margin:0 auto}.ck-media__wrapper .ck-media__placeholder{background:var(--ck-color-base-foreground);padding:calc(var(--ck-spacing-standard)*3)}.ck-media__wrapper .ck-media__placeholder .ck-media__placeholder__icon{background-position:50%;background-size:cover;height:var(--ck-media-embed-placeholder-icon-size);margin-bottom:var(--ck-spacing-large);min-width:var(--ck-media-embed-placeholder-icon-size)}.ck-media__wrapper .ck-media__placeholder .ck-media__placeholder__icon .ck-icon{height:100%;width:100%}.ck-media__wrapper .ck-media__placeholder .ck-media__placeholder__url__text{color:var(--ck-color-media-embed-placeholder-url-text);font-style:italic;text-align:center;text-overflow:ellipsis;white-space:nowrap}.ck-media__wrapper .ck-media__placeholder .ck-media__placeholder__url__text:hover{color:var(--ck-color-media-embed-placeholder-url-text-hover);cursor:pointer;text-decoration:underline}.ck-media__wrapper[data-oembed-url*=\\\"open.spotify.com\\\"]{max-height:380px;max-width:300px}.ck-media__wrapper[data-oembed-url*=\\\"goo.gl/maps\\\"] .ck-media__placeholder__icon,.ck-media__wrapper[data-oembed-url*=\\\"google.com/maps\\\"] .ck-media__placeholder__icon,.ck-media__wrapper[data-oembed-url*=\\\"maps.app.goo.gl\\\"] .ck-media__placeholder__icon,.ck-media__wrapper[data-oembed-url*=\\\"maps.google.com\\\"] .ck-media__placeholder__icon{background-image:url()}.ck-media__wrapper[data-oembed-url*=\\\"facebook.com\\\"] .ck-media__placeholder{background:#4268b3}.ck-media__wrapper[data-oembed-url*=\\\"facebook.com\\\"] .ck-media__placeholder .ck-media__placeholder__icon{background-image:url()}.ck-media__wrapper[data-oembed-url*=\\\"facebook.com\\\"] .ck-media__placeholder .ck-media__placeholder__url__text{color:#cdf}.ck-media__wrapper[data-oembed-url*=\\\"facebook.com\\\"] .ck-media__placeholder .ck-media__placeholder__url__text:hover{color:#fff}.ck-media__wrapper[data-oembed-url*=\\\"instagram.com\\\"] .ck-media__placeholder{background:linear-gradient(-135deg,#1400c7,#b800b1,#f50000)}.ck-media__wrapper[data-oembed-url*=\\\"instagram.com\\\"] .ck-media__placeholder .ck-media__placeholder__icon{background-image:url()}.ck-media__wrapper[data-oembed-url*=\\\"instagram.com\\\"] .ck-media__placeholder .ck-media__placeholder__url__text{color:#ffe0fe}.ck-media__wrapper[data-oembed-url*=\\\"instagram.com\\\"] .ck-media__placeholder .ck-media__placeholder__url__text:hover{color:#fff}.ck-media__wrapper[data-oembed-url*=\\\"twitter.com\\\"] .ck.ck-media__placeholder{background:linear-gradient(90deg,#71c6f4,#0d70a5)}.ck-media__wrapper[data-oembed-url*=\\\"twitter.com\\\"] .ck.ck-media__placeholder .ck-media__placeholder__icon{background-image:url()}.ck-media__wrapper[data-oembed-url*=\\\"twitter.com\\\"] .ck.ck-media__placeholder .ck-media__placeholder__url__text{color:#b8e6ff}.ck-media__wrapper[data-oembed-url*=\\\"twitter.com\\\"] .ck.ck-media__placeholder .ck-media__placeholder__url__text:hover{color:#fff}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-media-embed/theme/mediaembedediting.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-media-embed/mediaembedediting.css\"],\"names\":[],\"mappings\":\"AAMC,0CAGC,kBAAmB,CAFnB,YAAa,CACb,qBAcD,CAXC,sEAEC,cAAe,CAEf,iBAMD,CAJC,wGAEC,aAAc,CADd,eAED,CAWD,6kBACC,YACD,CAYF,2LACC,mBACD,CC1CA,MACC,0CAA2C,CAE3C,mDAA4D,CAC5D,2EACD,CAEA,mBACC,aA+FD,CA7FC,0CAEC,0CAA2C,CAD3C,0CA6BD,CA1BC,uEAIC,uBAA2B,CAC3B,qBAAsB,CAHtB,kDAAmD,CACnD,qCAAsC,CAFtC,qDAUD,CAJC,gFAEC,WAAY,CADZ,UAED,CAGD,4EACC,sDAAuD,CAGvD,iBAAkB,CADlB,iBAAkB,CAElB,sBAAuB,CAHvB,kBAUD,CALC,kFACC,4DAA6D,CAC7D,cAAe,CACf,yBACD,CAIF,wDAEC,gBAAiB,CADjB,eAED,CAEA,4UAIC,wvGACD,CAEA,2EACC,kBAaD,CAXC,wGACC,orBACD,CAEA,6GACC,UAKD,CAHC,mHACC,UACD,CAIF,4EACC,2DAcD,CAZC,yGACC,4jHACD,CAGA,8GACC,aAKD,CAHC,oHACC,UACD,CAIF,6EAEC,iDAaD,CAXC,0GACC,wiCACD,CAEA,+GACC,aAKD,CAHC,qHACC,UACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-media__wrapper {\\n\\t& .ck-media__placeholder {\\n\\t\\tdisplay: flex;\\n\\t\\tflex-direction: column;\\n\\t\\talign-items: center;\\n\\n\\t\\t& .ck-media__placeholder__url {\\n\\t\\t\\t/* Otherwise the URL will overflow when the content is very narrow. */\\n\\t\\t\\tmax-width: 100%;\\n\\n\\t\\t\\tposition: relative;\\n\\n\\t\\t\\t& .ck-media__placeholder__url__text {\\n\\t\\t\\t\\toverflow: hidden;\\n\\t\\t\\t\\tdisplay: block;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t&[data-oembed-url*=\\\"twitter.com\\\"],\\n\\t&[data-oembed-url*=\\\"google.com/maps\\\"],\\n\\t&[data-oembed-url*=\\\"goo.gl/maps\\\"],\\n\\t&[data-oembed-url*=\\\"maps.google.com\\\"],\\n\\t&[data-oembed-url*=\\\"maps.app.goo.gl\\\"],\\n\\t&[data-oembed-url*=\\\"facebook.com\\\"],\\n\\t&[data-oembed-url*=\\\"instagram.com\\\"] {\\n\\t\\t& .ck-media__placeholder__icon * {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\t}\\n}\\n\\n/* Disable all mouse interaction as long as the editor is not readonly.\\n https://github.com/ckeditor/ckeditor5-media-embed/issues/58 */\\n.ck-editor__editable:not(.ck-read-only) .ck-media__wrapper > *:not(.ck-media__placeholder) {\\n\\tpointer-events: none;\\n}\\n\\n/* Disable all mouse interaction when the widget is not selected (e.g. to avoid opening links by accident).\\n https://github.com/ckeditor/ckeditor5-media-embed/issues/18 */\\n.ck-editor__editable:not(.ck-read-only) .ck-widget:not(.ck-widget_selected) .ck-media__placeholder {\\n\\tpointer-events: none;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-media-embed-placeholder-icon-size: 3em;\\n\\n\\t--ck-color-media-embed-placeholder-url-text: hsl(0, 0%, 46%);\\n\\t--ck-color-media-embed-placeholder-url-text-hover: var(--ck-color-base-text);\\n}\\n\\n.ck-media__wrapper {\\n\\tmargin: 0 auto;\\n\\n\\t& .ck-media__placeholder {\\n\\t\\tpadding: calc( 3 * var(--ck-spacing-standard) );\\n\\t\\tbackground: var(--ck-color-base-foreground);\\n\\n\\t\\t& .ck-media__placeholder__icon {\\n\\t\\t\\tmin-width: var(--ck-media-embed-placeholder-icon-size);\\n\\t\\t\\theight: var(--ck-media-embed-placeholder-icon-size);\\n\\t\\t\\tmargin-bottom: var(--ck-spacing-large);\\n\\t\\t\\tbackground-position: center;\\n\\t\\t\\tbackground-size: cover;\\n\\n\\t\\t\\t& .ck-icon {\\n\\t\\t\\t\\twidth: 100%;\\n\\t\\t\\t\\theight: 100%;\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t& .ck-media__placeholder__url__text {\\n\\t\\t\\tcolor: var(--ck-color-media-embed-placeholder-url-text);\\n\\t\\t\\twhite-space: nowrap;\\n\\t\\t\\ttext-align: center;\\n\\t\\t\\tfont-style: italic;\\n\\t\\t\\ttext-overflow: ellipsis;\\n\\n\\t\\t\\t&:hover {\\n\\t\\t\\t\\tcolor: var(--ck-color-media-embed-placeholder-url-text-hover);\\n\\t\\t\\t\\tcursor: pointer;\\n\\t\\t\\t\\ttext-decoration: underline;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t&[data-oembed-url*=\\\"open.spotify.com\\\"] {\\n\\t\\tmax-width: 300px;\\n\\t\\tmax-height: 380px;\\n\\t}\\n\\n\\t&[data-oembed-url*=\\\"google.com/maps\\\"] .ck-media__placeholder__icon,\\n\\t&[data-oembed-url*=\\\"goo.gl/maps\\\"] .ck-media__placeholder__icon,\\n\\t&[data-oembed-url*=\\\"maps.google.com\\\"] .ck-media__placeholder__icon,\\n\\t&[data-oembed-url*=\\\"maps.app.goo.gl\\\"] .ck-media__placeholder__icon {\\n\\t\\tbackground-image: url();\\n\\t}\\n\\n\\t&[data-oembed-url*=\\\"facebook.com\\\"] .ck-media__placeholder {\\n\\t\\tbackground: hsl(220, 46%, 48%);\\n\\n\\t\\t& .ck-media__placeholder__icon {\\n\\t\\t\\tbackground-image: url();\\n\\t\\t}\\n\\n\\t\\t& .ck-media__placeholder__url__text {\\n\\t\\t\\tcolor: hsl(220, 100%, 90%);\\n\\n\\t\\t\\t&:hover {\\n\\t\\t\\t\\tcolor: hsl(0, 0%, 100%);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t&[data-oembed-url*=\\\"instagram.com\\\"] .ck-media__placeholder {\\n\\t\\tbackground: linear-gradient(-135deg,hsl(246, 100%, 39%),hsl(302, 100%, 36%),hsl(0, 100%, 48%));\\n\\n\\t\\t& .ck-media__placeholder__icon {\\n\\t\\t\\tbackground-image: url();\\n\\t\\t}\\n\\n\\t\\t/* stylelint-disable-next-line no-descending-specificity */\\n\\t\\t& .ck-media__placeholder__url__text {\\n\\t\\t\\tcolor: hsl(302, 100%, 94%);\\n\\n\\t\\t\\t&:hover {\\n\\t\\t\\t\\tcolor: hsl(0, 0%, 100%);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t&[data-oembed-url*=\\\"twitter.com\\\"] .ck.ck-media__placeholder {\\n\\t\\t/* Use gradient to contrast with focused widget (ckeditor/ckeditor5-media-embed#22). */\\n\\t\\tbackground: linear-gradient( to right, hsl(201, 85%, 70%), hsl(201, 85%, 35%) );\\n\\n\\t\\t& .ck-media__placeholder__icon {\\n\\t\\t\\tbackground-image: url();\\n\\t\\t}\\n\\n\\t\\t& .ck-media__placeholder__url__text {\\n\\t\\t\\tcolor: hsl(201, 100%, 86%);\\n\\n\\t\\t\\t&:hover {\\n\\t\\t\\t\\tcolor: hsl(0, 0%, 100%);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-media-form{align-items:flex-start;display:flex;flex-direction:row;flex-wrap:nowrap}.ck.ck-media-form .ck-labeled-field-view{display:inline-block}.ck.ck-media-form .ck-label{display:none}@media screen and (max-width:600px){.ck.ck-media-form{flex-wrap:wrap}.ck.ck-media-form .ck-labeled-field-view{flex-basis:100%}.ck.ck-media-form .ck-button{flex-basis:50%}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-media-embed/theme/mediaform.css\",\"webpack://./../ckeditor5-ui/theme/mixins/_rwd.css\"],\"names\":[],\"mappings\":\"AAOA,kBAEC,sBAAuB,CADvB,YAAa,CAEb,kBAAmB,CACnB,gBAqBD,CAnBC,yCACC,oBACD,CAEA,4BACC,YACD,CCbA,oCDCD,kBAeE,cAUF,CARE,yCACC,eACD,CAEA,6BACC,cACD,CCtBD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_rwd.css\\\";\\n\\n.ck.ck-media-form {\\n\\tdisplay: flex;\\n\\talign-items: flex-start;\\n\\tflex-direction: row;\\n\\tflex-wrap: nowrap;\\n\\n\\t& .ck-labeled-field-view {\\n\\t\\tdisplay: inline-block;\\n\\t}\\n\\n\\t& .ck-label {\\n\\t\\tdisplay: none;\\n\\t}\\n\\n\\t@mixin ck-media-phone {\\n\\t\\tflex-wrap: wrap;\\n\\n\\t\\t& .ck-labeled-field-view {\\n\\t\\t\\tflex-basis: 100%;\\n\\t\\t}\\n\\n\\t\\t& .ck-button {\\n\\t\\t\\tflex-basis: 50%;\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@define-mixin ck-media-phone {\\n\\t@media screen and (max-width: 600px) {\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-input-color{display:flex;flex-direction:row-reverse;width:100%}.ck.ck-input-color>input.ck.ck-input-text{flex-grow:1;min-width:auto}.ck.ck-input-color>div.ck.ck-dropdown{min-width:auto}.ck.ck-input-color>div.ck.ck-dropdown>.ck-input-color__button .ck-dropdown__arrow{display:none}.ck.ck-input-color .ck.ck-input-color__button{display:flex}.ck.ck-input-color .ck.ck-input-color__button .ck.ck-input-color__button__preview{overflow:hidden;position:relative}.ck.ck-input-color .ck.ck-input-color__button .ck.ck-input-color__button__preview>.ck.ck-input-color__button__preview__no-color-indicator{display:block;position:absolute}[dir=ltr] .ck.ck-input-color>.ck.ck-input-text{border-bottom-right-radius:0;border-top-right-radius:0}[dir=rtl] .ck.ck-input-color>.ck.ck-input-text{border-bottom-left-radius:0;border-top-left-radius:0}.ck.ck-input-color>.ck.ck-input-text:focus{z-index:0}.ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button{padding:0}[dir=ltr] .ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button{border-bottom-left-radius:0;border-top-left-radius:0}[dir=ltr] .ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button:not(:focus){border-left:1px solid transparent}[dir=rtl] .ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button{border-bottom-right-radius:0;border-top-right-radius:0}[dir=rtl] .ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button:not(:focus){border-right:1px solid transparent}.ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button.ck-disabled{background:var(--ck-color-input-disabled-background)}.ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button>.ck.ck-input-color__button__preview{border-radius:0}.ck-rounded-corners .ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button>.ck.ck-input-color__button__preview,.ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button>.ck.ck-input-color__button__preview.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button>.ck.ck-input-color__button__preview{border:1px solid var(--ck-color-input-border);height:20px;width:20px}.ck.ck-input-color>.ck.ck-dropdown>.ck.ck-button.ck-input-color__button>.ck.ck-input-color__button__preview>.ck.ck-input-color__button__preview__no-color-indicator{background:red;border-radius:2px;height:150%;left:50%;top:-30%;transform:rotate(45deg);transform-origin:50%;width:8%}.ck.ck-input-color .ck.ck-input-color__remove-color{border-bottom-left-radius:0;border-bottom-right-radius:0;padding:calc(var(--ck-spacing-standard)/2) var(--ck-spacing-standard);width:100%}.ck.ck-input-color .ck.ck-input-color__remove-color:not(:focus){border-bottom:1px solid var(--ck-color-input-border)}[dir=ltr] .ck.ck-input-color .ck.ck-input-color__remove-color{border-top-right-radius:0}[dir=rtl] .ck.ck-input-color .ck.ck-input-color__remove-color{border-top-left-radius:0}.ck.ck-input-color .ck.ck-input-color__remove-color .ck.ck-icon{margin-right:var(--ck-spacing-standard)}[dir=rtl] .ck.ck-input-color .ck.ck-input-color__remove-color .ck.ck-icon{margin-left:var(--ck-spacing-standard);margin-right:0}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/colorinput.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/colorinput.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\"],\"names\":[],\"mappings\":\"AAKA,mBAEC,YAAa,CACb,0BAA2B,CAF3B,UAgCD,CA5BC,0CAEC,WAAY,CADZ,cAED,CAEA,sCACC,cAMD,CAHC,kFACC,YACD,CAGD,8CAEC,YAWD,CATC,kFAEC,eAAgB,CADhB,iBAOD,CAJC,0IAEC,aAAc,CADd,iBAED,CC1BF,+CAGE,4BAA6B,CAD7B,yBAcF,CAhBA,+CAQE,2BAA4B,CAD5B,wBASF,CAHC,2CACC,SACD,CAIA,wEACC,SA0CD,CA3CA,kFAKE,2BAA4B,CAD5B,wBAuCF,CApCE,8FACC,iCACD,CATF,kFAcE,4BAA6B,CAD7B,yBA8BF,CA3BE,8FACC,kCACD,CAGD,oFACC,oDACD,CAEA,4GC1CF,eD2DE,CAjBA,+PCtCD,qCDuDC,CAjBA,4GAKC,6CAA8C,CAD9C,WAAY,CADZ,UAcD,CAVC,oKAKC,cAA6B,CAC7B,iBAAkB,CAHlB,WAAY,CADZ,QAAS,CADT,QAAS,CAMT,uBAAwB,CACxB,oBAAqB,CAJrB,QAKD,CAKH,oDAIC,2BAA4B,CAC5B,4BAA6B,CAH7B,qEAAwE,CADxE,UA0BD,CApBC,gEACC,oDACD,CATD,8DAYE,yBAeF,CA3BA,8DAgBE,wBAWF,CARC,gEACC,uCAMD,CAPA,0EAKE,sCAAuC,CADvC,cAGF\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-input-color {\\n\\twidth: 100%;\\n\\tdisplay: flex;\\n\\tflex-direction: row-reverse;\\n\\n\\t& > input.ck.ck-input-text {\\n\\t\\tmin-width: auto;\\n\\t\\tflex-grow: 1;\\n\\t}\\n\\n\\t& > div.ck.ck-dropdown {\\n\\t\\tmin-width: auto;\\n\\n\\t\\t/* This dropdown has no arrow but a color preview instead. */\\n\\t\\t& > .ck-input-color__button .ck-dropdown__arrow {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck.ck-input-color__button {\\n\\t\\t/* Resolving issue with misaligned buttons on Safari (see #10589) */\\n\\t\\tdisplay: flex;\\n\\n\\t\\t& .ck.ck-input-color__button__preview {\\n\\t\\t\\tposition: relative;\\n\\t\\t\\toverflow: hidden;\\n\\n\\t\\t\\t& > .ck.ck-input-color__button__preview__no-color-indicator {\\n\\t\\t\\t\\tposition: absolute;\\n\\t\\t\\t\\tdisplay: block;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n@import \\\"../mixins/_rounded.css\\\";\\n\\n.ck.ck-input-color {\\n\\t& > .ck.ck-input-text {\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\tborder-top-right-radius: 0;\\n\\t\\t\\tborder-bottom-right-radius: 0;\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\tborder-top-left-radius: 0;\\n\\t\\t\\tborder-bottom-left-radius: 0;\\n\\t\\t}\\n\\n\\t\\t/* Make sure the focused input is always on top of the dropdown button so its\\n\\t\\t outline and border are never cropped (also when the input is read-only). */\\n\\t\\t&:focus {\\n\\t\\t\\tz-index: 0;\\n\\t\\t}\\n\\t}\\n\\n\\t& > .ck.ck-dropdown {\\n\\t\\t& > .ck.ck-button.ck-input-color__button {\\n\\t\\t\\tpadding: 0;\\n\\n\\t\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t\\tborder-top-left-radius: 0;\\n\\t\\t\\t\\tborder-bottom-left-radius: 0;\\n\\n\\t\\t\\t\\t&:not(:focus) {\\n\\t\\t\\t\\t\\tborder-left: 1px solid transparent;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\n\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\tborder-top-right-radius: 0;\\n\\t\\t\\t\\tborder-bottom-right-radius: 0;\\n\\n\\t\\t\\t\\t&:not(:focus) {\\n\\t\\t\\t\\t\\tborder-right: 1px solid transparent;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\n\\t\\t\\t&.ck-disabled {\\n\\t\\t\\t\\tbackground: var(--ck-color-input-disabled-background);\\n\\t\\t\\t}\\n\\n\\t\\t\\t& > .ck.ck-input-color__button__preview {\\n\\t\\t\\t\\t@mixin ck-rounded-corners;\\n\\n\\t\\t\\t\\twidth: 20px;\\n\\t\\t\\t\\theight: 20px;\\n\\t\\t\\t\\tborder: 1px solid var(--ck-color-input-border);\\n\\n\\t\\t\\t\\t& > .ck.ck-input-color__button__preview__no-color-indicator {\\n\\t\\t\\t\\t\\ttop: -30%;\\n\\t\\t\\t\\t\\tleft: 50%;\\n\\t\\t\\t\\t\\theight: 150%;\\n\\t\\t\\t\\t\\twidth: 8%;\\n\\t\\t\\t\\t\\tbackground: hsl(0, 100%, 50%);\\n\\t\\t\\t\\t\\tborder-radius: 2px;\\n\\t\\t\\t\\t\\ttransform: rotate(45deg);\\n\\t\\t\\t\\t\\ttransform-origin: 50%;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck.ck-input-color__remove-color {\\n\\t\\twidth: 100%;\\n\\t\\tpadding: calc(var(--ck-spacing-standard) / 2) var(--ck-spacing-standard);\\n\\n\\t\\tborder-bottom-left-radius: 0;\\n\\t\\tborder-bottom-right-radius: 0;\\n\\n\\t\\t&:not(:focus) {\\n\\t\\t\\tborder-bottom: 1px solid var(--ck-color-input-border);\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\tborder-top-right-radius: 0;\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\tborder-top-left-radius: 0;\\n\\t\\t}\\n\\n\\t\\t& .ck.ck-icon {\\n\\t\\t\\tmargin-right: var(--ck-spacing-standard);\\n\\n\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\tmargin-right: 0;\\n\\t\\t\\t\\tmargin-left: var(--ck-spacing-standard);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-form{padding:0 0 var(--ck-spacing-large)}.ck.ck-form:focus{outline:none}.ck.ck-form .ck.ck-input-text{min-width:100%;width:0}.ck.ck-form .ck.ck-dropdown{min-width:100%}.ck.ck-form .ck.ck-dropdown .ck-dropdown__button:not(:focus){border:1px solid var(--ck-color-base-border)}.ck.ck-form .ck.ck-dropdown .ck-dropdown__button .ck-button__label{width:100%}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/form.css\"],\"names\":[],\"mappings\":\"AAKA,YACC,mCAyBD,CAvBC,kBAEC,YACD,CAEA,8BACC,cAAe,CACf,OACD,CAEA,4BACC,cAWD,CARE,6DACC,4CACD,CAEA,mEACC,UACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-form {\\n\\tpadding: 0 0 var(--ck-spacing-large);\\n\\n\\t&:focus {\\n\\t\\t/* See: https://github.com/ckeditor/ckeditor5/issues/4773 */\\n\\t\\toutline: none;\\n\\t}\\n\\n\\t& .ck.ck-input-text {\\n\\t\\tmin-width: 100%;\\n\\t\\twidth: 0;\\n\\t}\\n\\n\\t& .ck.ck-dropdown {\\n\\t\\tmin-width: 100%;\\n\\n\\t\\t& .ck-dropdown__button {\\n\\t\\t\\t&:not(:focus) {\\n\\t\\t\\t\\tborder: 1px solid var(--ck-color-base-border);\\n\\t\\t\\t}\\n\\n\\t\\t\\t& .ck-button__label {\\n\\t\\t\\t\\twidth: 100%;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-form__row{display:flex;flex-direction:row;flex-wrap:nowrap;justify-content:space-between}.ck.ck-form__row>:not(.ck-label){flex-grow:1}.ck.ck-form__row.ck-table-form__action-row .ck-button-cancel,.ck.ck-form__row.ck-table-form__action-row .ck-button-save{justify-content:center}.ck.ck-form__row{padding:var(--ck-spacing-standard) var(--ck-spacing-large) 0}[dir=ltr] .ck.ck-form__row>:not(.ck-label)+*{margin-left:var(--ck-spacing-large)}[dir=rtl] .ck.ck-form__row>:not(.ck-label)+*{margin-right:var(--ck-spacing-large)}.ck.ck-form__row>.ck-label{min-width:100%;width:100%}.ck.ck-form__row.ck-table-form__action-row{margin-top:var(--ck-spacing-large)}.ck.ck-form__row.ck-table-form__action-row .ck-button .ck-button__label{color:var(--ck-color-text)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/formrow.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/formrow.css\"],\"names\":[],\"mappings\":\"AAKA,iBACC,YAAa,CACb,kBAAmB,CACnB,gBAAiB,CACjB,6BAaD,CAVC,iCACC,WACD,CAGC,wHAEC,sBACD,CCbF,iBACC,4DA2BD,CAvBE,6CAEE,mCAMF,CARA,6CAME,oCAEF,CAGD,2BAEC,cAAe,CADf,UAED,CAEA,2CACC,kCAKD,CAHC,wEACC,0BACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-form__row {\\n\\tdisplay: flex;\\n\\tflex-direction: row;\\n\\tflex-wrap: nowrap;\\n\\tjustify-content: space-between;\\n\\n\\t/* Ignore labels that work as fieldset legends */\\n\\t& > *:not(.ck-label) {\\n\\t\\tflex-grow: 1;\\n\\t}\\n\\n\\t&.ck-table-form__action-row {\\n\\t\\t& .ck-button-save,\\n\\t\\t& .ck-button-cancel {\\n\\t\\t\\tjustify-content: center;\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n.ck.ck-form__row {\\n\\tpadding: var(--ck-spacing-standard) var(--ck-spacing-large) 0;\\n\\n\\t/* Ignore labels that work as fieldset legends */\\n\\t& > *:not(.ck-label) {\\n\\t\\t& + * {\\n\\t\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t\\tmargin-left: var(--ck-spacing-large);\\n\\t\\t\\t}\\n\\n\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\tmargin-right: var(--ck-spacing-large);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t& > .ck-label {\\n\\t\\twidth: 100%;\\n\\t\\tmin-width: 100%;\\n\\t}\\n\\n\\t&.ck-table-form__action-row {\\n\\t\\tmargin-top: var(--ck-spacing-large);\\n\\n\\t\\t& .ck-button .ck-button__label {\\n\\t\\t\\tcolor: var(--ck-color-text);\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck .ck-insert-table-dropdown__grid{display:flex;flex-direction:row;flex-wrap:wrap}:root{--ck-insert-table-dropdown-padding:10px;--ck-insert-table-dropdown-box-height:11px;--ck-insert-table-dropdown-box-width:12px;--ck-insert-table-dropdown-box-margin:1px}.ck .ck-insert-table-dropdown__grid{padding:var(--ck-insert-table-dropdown-padding) var(--ck-insert-table-dropdown-padding) 0;width:calc(var(--ck-insert-table-dropdown-box-width)*10 + var(--ck-insert-table-dropdown-box-margin)*20 + var(--ck-insert-table-dropdown-padding)*2)}.ck .ck-insert-table-dropdown__label,.ck[dir=rtl] .ck-insert-table-dropdown__label{text-align:center}.ck .ck-insert-table-dropdown-grid-box{border:1px solid var(--ck-color-base-border);border-radius:1px;margin:var(--ck-insert-table-dropdown-box-margin);min-height:var(--ck-insert-table-dropdown-box-height);min-width:var(--ck-insert-table-dropdown-box-width);outline:none;transition:none}.ck .ck-insert-table-dropdown-grid-box:focus{box-shadow:none}.ck .ck-insert-table-dropdown-grid-box.ck-on{background:var(--ck-color-focus-outer-shadow);border-color:var(--ck-color-focus-border)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/inserttable.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/inserttable.css\"],\"names\":[],\"mappings\":\"AAKA,oCACC,YAAa,CACb,kBAAmB,CACnB,cACD,CCJA,MACC,uCAAwC,CACxC,0CAA2C,CAC3C,yCAA0C,CAC1C,yCACD,CAEA,oCAGC,yFAA0F,CAD1F,oJAED,CAEA,mFAEC,iBACD,CAEA,uCAIC,4CAA6C,CAC7C,iBAAkB,CAFlB,iDAAkD,CADlD,qDAAsD,CADtD,mDAAoD,CAKpD,YAAa,CACb,eAUD,CARC,6CACC,eACD,CAEA,6CAEC,6CAA8C,CAD9C,yCAED\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck .ck-insert-table-dropdown__grid {\\n\\tdisplay: flex;\\n\\tflex-direction: row;\\n\\tflex-wrap: wrap;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-insert-table-dropdown-padding: 10px;\\n\\t--ck-insert-table-dropdown-box-height: 11px;\\n\\t--ck-insert-table-dropdown-box-width: 12px;\\n\\t--ck-insert-table-dropdown-box-margin: 1px;\\n}\\n\\n.ck .ck-insert-table-dropdown__grid {\\n\\t/* The width of a container should match 10 items in a row so there will be a 10x10 grid. */\\n\\twidth: calc(var(--ck-insert-table-dropdown-box-width) * 10 + var(--ck-insert-table-dropdown-box-margin) * 20 + var(--ck-insert-table-dropdown-padding) * 2);\\n\\tpadding: var(--ck-insert-table-dropdown-padding) var(--ck-insert-table-dropdown-padding) 0;\\n}\\n\\n.ck .ck-insert-table-dropdown__label,\\n.ck[dir=rtl] .ck-insert-table-dropdown__label {\\n\\ttext-align: center;\\n}\\n\\n.ck .ck-insert-table-dropdown-grid-box {\\n\\tmin-width: var(--ck-insert-table-dropdown-box-width);\\n\\tmin-height: var(--ck-insert-table-dropdown-box-height);\\n\\tmargin: var(--ck-insert-table-dropdown-box-margin);\\n\\tborder: 1px solid var(--ck-color-base-border);\\n\\tborder-radius: 1px;\\n\\toutline: none;\\n\\ttransition: none;\\n\\n\\t&:focus {\\n\\t\\tbox-shadow: none;\\n\\t}\\n\\n\\t&.ck-on {\\n\\t\\tborder-color: var(--ck-color-focus-border);\\n\\t\\tbackground: var(--ck-color-focus-outer-shadow);\\n\\t}\\n}\\n\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-content .table{display:table;margin:.9em auto}.ck-content .table table{border:1px double #b3b3b3;border-collapse:collapse;border-spacing:0;height:100%;width:100%}.ck-content .table table td,.ck-content .table table th{border:1px solid #bfbfbf;min-width:2em;padding:.4em}.ck-content .table table th{background:rgba(0,0,0,.05);font-weight:700}.ck-content[dir=rtl] .table th{text-align:right}.ck-content[dir=ltr] .table th{text-align:left}.ck-editor__editable .ck-table-bogus-paragraph{display:inline-block;width:100%}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/table.css\"],\"names\":[],\"mappings\":\"AAKA,mBAKC,aAAc,CADd,gBAiCD,CA9BC,yBAYC,yBAAkC,CAVlC,wBAAyB,CACzB,gBAAiB,CAKjB,WAAY,CADZ,UAsBD,CAfC,wDAQC,wBAAiC,CANjC,aAAc,CACd,YAMD,CAEA,4BAEC,0BAA+B,CAD/B,eAED,CAMF,+BACC,gBACD,CAEA,+BACC,eACD,CAEA,+CAKC,oBAAqB,CAMrB,UACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck-content .table {\\n\\t/* Give the table widget some air and center it horizontally */\\n\\t/* The first value should be equal to --ck-spacing-large variable if used in the editor context\\n\\tto avoid the content jumping (See https://github.com/ckeditor/ckeditor5/issues/9825). */\\n\\tmargin: 0.9em auto;\\n\\tdisplay: table;\\n\\n\\t& table {\\n\\t\\t/* The table cells should have slight borders */\\n\\t\\tborder-collapse: collapse;\\n\\t\\tborder-spacing: 0;\\n\\n\\t\\t/* Table width and height are set on the parent <figure>. Make sure the table inside stretches\\n\\t\\tto the full dimensions of the container (https://github.com/ckeditor/ckeditor5/issues/6186). */\\n\\t\\twidth: 100%;\\n\\t\\theight: 100%;\\n\\n\\t\\t/* The outer border of the table should be slightly darker than the inner lines.\\n\\t\\tAlso see https://github.com/ckeditor/ckeditor5-table/issues/50. */\\n\\t\\tborder: 1px double hsl(0, 0%, 70%);\\n\\n\\t\\t& td,\\n\\t\\t& th {\\n\\t\\t\\tmin-width: 2em;\\n\\t\\t\\tpadding: .4em;\\n\\n\\t\\t\\t/* The border is inherited from .ck-editor__nested-editable styles, so theoretically it's not necessary here.\\n\\t\\t\\tHowever, the border is a content style, so it should use .ck-content (so it works outside the editor).\\n\\t\\t\\tHence, the duplication. See https://github.com/ckeditor/ckeditor5/issues/6314 */\\n\\t\\t\\tborder: 1px solid hsl(0, 0%, 75%);\\n\\t\\t}\\n\\n\\t\\t& th {\\n\\t\\t\\tfont-weight: bold;\\n\\t\\t\\tbackground: hsla(0, 0%, 0%, 5%);\\n\\t\\t}\\n\\t}\\n}\\n\\n/* Text alignment of the table header should match the editor settings and override the native browser styling,\\nwhen content is available outside the editor. See https://github.com/ckeditor/ckeditor5/issues/6638 */\\n.ck-content[dir=\\\"rtl\\\"] .table th {\\n\\ttext-align: right;\\n}\\n\\n.ck-content[dir=\\\"ltr\\\"] .table th {\\n\\ttext-align: left;\\n}\\n\\n.ck-editor__editable .ck-table-bogus-paragraph {\\n\\t/*\\n\\t * Use display:inline-block to force Chrome/Safari to limit text mutations to this element.\\n\\t * See https://github.com/ckeditor/ckeditor5/issues/6062.\\n\\t */\\n\\tdisplay: inline-block;\\n\\n\\t/*\\n\\t * Inline HTML elements nested in the span should always be dimensioned in relation to the whole cell width.\\n\\t * See https://github.com/ckeditor/ckeditor5/issues/9117.\\n\\t */\\n\\twidth: 100%;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-color-selector-caption-background:#f7f7f7;--ck-color-selector-caption-text:#333;--ck-color-selector-caption-highlighted-background:#fd0}.ck-content .table>figcaption{background-color:var(--ck-color-selector-caption-background);caption-side:top;color:var(--ck-color-selector-caption-text);display:table-caption;font-size:.75em;outline-offset:-1px;padding:.6em;text-align:center;word-break:break-word}.ck.ck-editor__editable .table>figcaption.table__caption_highlighted{animation:ck-table-caption-highlight .6s ease-out}.ck.ck-editor__editable .table>figcaption.ck-placeholder:before{overflow:hidden;padding-left:inherit;padding-right:inherit;text-overflow:ellipsis;white-space:nowrap}@keyframes ck-table-caption-highlight{0%{background-color:var(--ck-color-selector-caption-highlighted-background)}to{background-color:var(--ck-color-selector-caption-background)}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/tablecaption.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,8CAAuD,CACvD,qCAAiD,CACjD,uDACD,CAGA,8BAMC,4DAA6D,CAJ7D,gBAAiB,CAGjB,2CAA4C,CAJ5C,qBAAsB,CAOtB,eAAgB,CAChB,mBAAoB,CAFpB,YAAa,CAHb,iBAAkB,CADlB,qBAOD,CAIC,qEACC,iDACD,CAEA,gEASC,eAAgB,CARhB,oBAAqB,CACrB,qBAAsB,CAQtB,sBAAuB,CAFvB,kBAGD,CAGD,sCACC,GACC,wEACD,CAEA,GACC,4DACD,CACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-selector-caption-background: hsl(0, 0%, 97%);\\n\\t--ck-color-selector-caption-text: hsl(0, 0%, 20%);\\n\\t--ck-color-selector-caption-highlighted-background: hsl(52deg 100% 50%);\\n}\\n\\n/* Content styles */\\n.ck-content .table > figcaption {\\n\\tdisplay: table-caption;\\n\\tcaption-side: top;\\n\\tword-break: break-word;\\n\\ttext-align: center;\\n\\tcolor: var(--ck-color-selector-caption-text);\\n\\tbackground-color: var(--ck-color-selector-caption-background);\\n\\tpadding: .6em;\\n\\tfont-size: .75em;\\n\\toutline-offset: -1px;\\n}\\n\\n/* Editing styles */\\n.ck.ck-editor__editable .table > figcaption {\\n\\t&.table__caption_highlighted {\\n\\t\\tanimation: ck-table-caption-highlight .6s ease-out;\\n\\t}\\n\\n\\t&.ck-placeholder::before {\\n\\t\\tpadding-left: inherit;\\n\\t\\tpadding-right: inherit;\\n\\n\\t\\t/*\\n\\t\\t * Make sure the table caption placeholder doesn't overflow the placeholder area.\\n\\t\\t * See https://github.com/ckeditor/ckeditor5/issues/9162.\\n\\t\\t */\\n\\t\\twhite-space: nowrap;\\n\\t\\toverflow: hidden;\\n\\t\\ttext-overflow: ellipsis;\\n\\t}\\n}\\n\\n@keyframes ck-table-caption-highlight {\\n\\t0% {\\n\\t\\tbackground-color: var(--ck-color-selector-caption-highlighted-background);\\n\\t}\\n\\n\\t100% {\\n\\t\\tbackground-color: var(--ck-color-selector-caption-background);\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-table-cell-properties-form .ck-form__row.ck-table-cell-properties-form__alignment-row{flex-wrap:wrap}.ck.ck-table-cell-properties-form .ck-form__row.ck-table-cell-properties-form__alignment-row .ck.ck-toolbar:first-of-type{flex-grow:0.57}.ck.ck-table-cell-properties-form .ck-form__row.ck-table-cell-properties-form__alignment-row .ck.ck-toolbar:last-of-type{flex-grow:0.43}.ck.ck-table-cell-properties-form .ck-form__row.ck-table-cell-properties-form__alignment-row .ck.ck-toolbar .ck-button{flex-grow:1}.ck.ck-table-cell-properties-form{width:320px}.ck.ck-table-cell-properties-form .ck-form__row.ck-table-cell-properties-form__padding-row{align-self:flex-end;padding:0;width:25%}.ck.ck-table-cell-properties-form .ck-form__row.ck-table-cell-properties-form__alignment-row .ck.ck-toolbar{background:none;margin-top:var(--ck-spacing-standard)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/tablecellproperties.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/tablecellproperties.css\"],\"names\":[],\"mappings\":\"AAOE,6FACC,cAiBD,CAdE,0HAEC,cACD,CAEA,yHAEC,cACD,CAEA,uHACC,WACD,CClBJ,kCACC,WAkBD,CAfE,2FACC,mBAAoB,CACpB,SAAU,CACV,SACD,CAGC,4GACC,eAAgB,CAGhB,qCACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-table-cell-properties-form {\\n\\t& .ck-form__row {\\n\\t\\t&.ck-table-cell-properties-form__alignment-row {\\n\\t\\t\\tflex-wrap: wrap;\\n\\n\\t\\t\\t& .ck.ck-toolbar {\\n\\t\\t\\t\\t&:first-of-type {\\n\\t\\t\\t\\t\\t/* 4 buttons out of 7 (h-alignment + v-alignment) = 0.57 */\\n\\t\\t\\t\\t\\tflex-grow: 0.57;\\n\\t\\t\\t\\t}\\n\\n\\t\\t\\t\\t&:last-of-type {\\n\\t\\t\\t\\t\\t/* 3 buttons out of 7 (h-alignment + v-alignment) = 0.43 */\\n\\t\\t\\t\\t\\tflex-grow: 0.43;\\n\\t\\t\\t\\t}\\n\\n\\t\\t\\t\\t& .ck-button {\\n\\t\\t\\t\\t\\tflex-grow: 1;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-table-cell-properties-form {\\n\\twidth: 320px;\\n\\n\\t& .ck-form__row {\\n\\t\\t&.ck-table-cell-properties-form__padding-row {\\n\\t\\t\\talign-self: flex-end;\\n\\t\\t\\tpadding: 0;\\n\\t\\t\\twidth: 25%;\\n\\t\\t}\\n\\n\\t\\t&.ck-table-cell-properties-form__alignment-row {\\n\\t\\t\\t& .ck.ck-toolbar {\\n\\t\\t\\t\\tbackground: none;\\n\\n\\t\\t\\t\\t/* Compensate for missing input label that would push the margin (toolbar has no inputs). */\\n\\t\\t\\t\\tmargin-top: var(--ck-spacing-standard);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-color-selector-column-resizer-hover:var(--ck-color-base-active);--ck-table-column-resizer-width:7px;--ck-table-column-resizer-position-offset:calc(var(--ck-table-column-resizer-width)*-0.5 - 0.5px)}.ck-content .table .ck-table-resized{table-layout:fixed}.ck-content .table table{overflow:hidden}.ck-content .table td,.ck-content .table th{overflow-wrap:break-word;position:relative}.ck.ck-editor__editable .table .ck-table-column-resizer{bottom:-999999px;cursor:col-resize;position:absolute;right:var(--ck-table-column-resizer-position-offset);top:-999999px;user-select:none;width:var(--ck-table-column-resizer-width);z-index:var(--ck-z-default)}.ck.ck-editor__editable .table[draggable] .ck-table-column-resizer,.ck.ck-editor__editable.ck-column-resize_disabled .table .ck-table-column-resizer{display:none}.ck.ck-editor__editable .table .ck-table-column-resizer:hover,.ck.ck-editor__editable .table .ck-table-column-resizer__active{background-color:var(--ck-color-selector-column-resizer-hover);opacity:.25}.ck.ck-editor__editable[dir=rtl] .table .ck-table-column-resizer{left:var(--ck-table-column-resizer-position-offset);right:unset}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/tablecolumnresize.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,oEAAqE,CACrE,mCAAoC,CAIpC,iGACD,CAEA,qCACC,kBACD,CAEA,yBACC,eACD,CAEA,4CAIC,wBAAyB,CACzB,iBACD,CAEA,wDAOC,gBAAiB,CAGjB,iBAAkB,CATlB,iBAAkB,CAOlB,oDAAqD,CAFrD,aAAc,CAKd,gBAAiB,CAFjB,0CAA2C,CAG3C,2BACD,CAQA,qJACC,YACD,CAEA,8HAEC,8DAA+D,CAC/D,WACD,CAEA,iEACC,mDAAoD,CACpD,WACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-selector-column-resizer-hover: var(--ck-color-base-active);\\n\\t--ck-table-column-resizer-width: 7px;\\n\\n\\t/* The offset used for absolute positioning of the resizer element, so that it is placed exactly above the cell border.\\n\\t The value is: minus half the width of the resizer decreased additionaly by the half the width of the border (0.5px). */\\n\\t--ck-table-column-resizer-position-offset: calc(var(--ck-table-column-resizer-width) * -0.5 - 0.5px);\\n}\\n\\n.ck-content .table .ck-table-resized {\\n\\ttable-layout: fixed;\\n}\\n\\n.ck-content .table table {\\n\\toverflow: hidden;\\n}\\n\\n.ck-content .table td,\\n.ck-content .table th {\\n\\t/* To prevent text overflowing beyond its cell when columns are resized by resize handler\\n\\t(https://github.com/ckeditor/ckeditor5/pull/14379#issuecomment-1589460978). */\\n\\toverflow-wrap: break-word;\\n\\tposition: relative;\\n}\\n\\n.ck.ck-editor__editable .table .ck-table-column-resizer {\\n\\tposition: absolute;\\n\\t/* The resizer element resides in each cell so to occupy the entire height of the table, which is unknown from a CSS point of view,\\n\\t it is extended to an extremely high height. Even for screens with a very high pixel density, the resizer will fulfill its role as\\n\\t it should, i.e. for a screen of 476 ppi the total height of the resizer will take over 350 sheets of A4 format, which is totally\\n\\t unrealistic height for a single table. */\\n\\ttop: -999999px;\\n\\tbottom: -999999px;\\n\\tright: var(--ck-table-column-resizer-position-offset);\\n\\twidth: var(--ck-table-column-resizer-width);\\n\\tcursor: col-resize;\\n\\tuser-select: none;\\n\\tz-index: var(--ck-z-default);\\n}\\n\\n.ck.ck-editor__editable.ck-column-resize_disabled .table .ck-table-column-resizer {\\n\\tdisplay: none;\\n}\\n\\n/* The resizer elements, which are extended to an extremely high height, break the drag & drop feature in Chrome. To make it work again,\\n all resizers must be hidden while the table is dragged. */\\n.ck.ck-editor__editable .table[draggable] .ck-table-column-resizer {\\n\\tdisplay: none;\\n}\\n\\n.ck.ck-editor__editable .table .ck-table-column-resizer:hover,\\n.ck.ck-editor__editable .table .ck-table-column-resizer__active {\\n\\tbackground-color: var(--ck-color-selector-column-resizer-hover);\\n\\topacity: 0.25;\\n}\\n\\n.ck.ck-editor__editable[dir=rtl] .table .ck-table-column-resizer {\\n\\tleft: var(--ck-table-column-resizer-position-offset);\\n\\tright: unset;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-color-selector-focused-cell-background:rgba(158,201,250,.3)}.ck-widget.table td.ck-editor__nested-editable.ck-editor__nested-editable_focused,.ck-widget.table td.ck-editor__nested-editable:focus,.ck-widget.table th.ck-editor__nested-editable.ck-editor__nested-editable_focused,.ck-widget.table th.ck-editor__nested-editable:focus{background:var(--ck-color-selector-focused-cell-background);border-style:none;outline:1px solid var(--ck-color-focus-border);outline-offset:-1px}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/tableediting.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,gEACD,CAKE,8QAGC,2DAA4D,CAK5D,iBAAkB,CAClB,8CAA+C,CAC/C,mBACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-selector-focused-cell-background: hsla(212, 90%, 80%, .3);\\n}\\n\\n.ck-widget.table {\\n\\t& td,\\n\\t& th {\\n\\t\\t&.ck-editor__nested-editable.ck-editor__nested-editable_focused,\\n\\t\\t&.ck-editor__nested-editable:focus {\\n\\t\\t\\t/* A very slight background to highlight the focused cell */\\n\\t\\t\\tbackground: var(--ck-color-selector-focused-cell-background);\\n\\n\\t\\t\\t/* Fixes the problem where surrounding cells cover the focused cell's border.\\n\\t\\t\\tIt does not fix the problem in all places but the UX is improved.\\n\\t\\t\\tSee https://github.com/ckeditor/ckeditor5-table/issues/29. */\\n\\t\\t\\tborder-style: none;\\n\\t\\t\\toutline: 1px solid var(--ck-color-focus-border);\\n\\t\\t\\toutline-offset: -1px; /* progressive enhancement - no IE support */\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-table-form .ck-form__row.ck-table-form__background-row,.ck.ck-table-form .ck-form__row.ck-table-form__border-row{flex-wrap:wrap}.ck.ck-table-form .ck-form__row.ck-table-form__dimensions-row{align-items:center;flex-wrap:wrap}.ck.ck-table-form .ck-form__row.ck-table-form__dimensions-row .ck-labeled-field-view{align-items:center;display:flex;flex-direction:column-reverse}.ck.ck-table-form .ck-form__row.ck-table-form__dimensions-row .ck-labeled-field-view .ck.ck-dropdown,.ck.ck-table-form .ck-form__row.ck-table-form__dimensions-row .ck-table-form__dimension-operator{flex-grow:0}.ck.ck-table-form .ck.ck-labeled-field-view{position:relative}.ck.ck-table-form .ck.ck-labeled-field-view .ck.ck-labeled-field-view__status{bottom:calc(var(--ck-table-properties-error-arrow-size)*-1);left:50%;position:absolute;transform:translate(-50%,100%);z-index:1}.ck.ck-table-form .ck.ck-labeled-field-view .ck.ck-labeled-field-view__status:after{content:\\\"\\\";left:50%;position:absolute;top:calc(var(--ck-table-properties-error-arrow-size)*-1);transform:translateX(-50%)}:root{--ck-table-properties-error-arrow-size:6px;--ck-table-properties-min-error-width:150px}.ck.ck-table-form .ck-form__row.ck-table-form__border-row .ck-labeled-field-view>.ck-label{font-size:var(--ck-font-size-tiny);text-align:center}.ck.ck-table-form .ck-form__row.ck-table-form__border-row .ck-table-form__border-style,.ck.ck-table-form .ck-form__row.ck-table-form__border-row .ck-table-form__border-width{max-width:80px;min-width:80px;width:80px}.ck.ck-table-form .ck-form__row.ck-table-form__dimensions-row{padding:0}.ck.ck-table-form .ck-form__row.ck-table-form__dimensions-row .ck-table-form__dimensions-row__height,.ck.ck-table-form .ck-form__row.ck-table-form__dimensions-row .ck-table-form__dimensions-row__width{margin:0}.ck.ck-table-form .ck-form__row.ck-table-form__dimensions-row .ck-table-form__dimension-operator{align-self:flex-end;display:inline-block;height:var(--ck-ui-component-min-height);line-height:var(--ck-ui-component-min-height);margin:0 var(--ck-spacing-small)}.ck.ck-table-form .ck.ck-labeled-field-view{padding-top:var(--ck-spacing-standard)}.ck.ck-table-form .ck.ck-labeled-field-view .ck.ck-labeled-field-view__status{border-radius:0}.ck-rounded-corners .ck.ck-table-form .ck.ck-labeled-field-view .ck.ck-labeled-field-view__status,.ck.ck-table-form .ck.ck-labeled-field-view .ck.ck-labeled-field-view__status.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-table-form .ck.ck-labeled-field-view .ck.ck-labeled-field-view__status{background:var(--ck-color-base-error);color:var(--ck-color-base-background);min-width:var(--ck-table-properties-min-error-width);padding:var(--ck-spacing-small) var(--ck-spacing-medium);text-align:center}.ck.ck-table-form .ck.ck-labeled-field-view .ck.ck-labeled-field-view__status:after{border-color:transparent transparent var(--ck-color-base-error) transparent;border-style:solid;border-width:0 var(--ck-table-properties-error-arrow-size) var(--ck-table-properties-error-arrow-size) var(--ck-table-properties-error-arrow-size)}.ck.ck-table-form .ck.ck-labeled-field-view .ck.ck-labeled-field-view__status{animation:ck-table-form-labeled-view-status-appear .15s ease both}.ck.ck-table-form .ck.ck-labeled-field-view .ck-input.ck-error:not(:focus)+.ck.ck-labeled-field-view__status{display:none}@keyframes ck-table-form-labeled-view-status-appear{0%{opacity:0}to{opacity:1}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/tableform.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/tableform.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\"],\"names\":[],\"mappings\":\"AAWE,wHACC,cACD,CAEA,8DAEC,kBAAmB,CADnB,cAgBD,CAbC,qFAGC,kBAAmB,CAFnB,YAAa,CACb,6BAMD,CAEA,sMACC,WACD,CAIF,4CAEC,iBAoBD,CAlBC,8EAGC,2DAAgE,CADhE,QAAS,CADT,iBAAkB,CAGlB,8BAA+B,CAG/B,SAUD,CAPC,oFACC,UAAW,CAGX,QAAS,CAFT,iBAAkB,CAClB,wDAA6D,CAE7D,0BACD,CChDH,MACC,0CAA2C,CAC3C,2CACD,CAMI,2FACC,kCAAmC,CACnC,iBACD,CAGD,8KAIC,cAAe,CADf,cAAe,CADf,UAGD,CAGD,8DACC,SAcD,CAZC,yMAEC,QACD,CAEA,iGACC,mBAAoB,CACpB,oBAAqB,CACrB,wCAAyC,CACzC,6CAA8C,CAC9C,gCACD,CAIF,4CACC,sCAyBD,CAvBC,8ECxCD,eDyDC,CAjBA,mMCpCA,qCDqDA,CAjBA,8EAGC,qCAAsC,CACtC,qCAAsC,CAEtC,oDAAqD,CADrD,wDAAyD,CAEzD,iBAUD,CAPC,oFACC,2EAA4E,CAE5E,kBAAmB,CADnB,kJAED,CAdD,8EAgBC,iEACD,CAGA,6GACC,YACD,CAIF,oDACC,GACC,SACD,CAEA,GACC,SACD,CACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-table-form {\\n\\t& .ck-form__row {\\n\\t\\t&.ck-table-form__border-row {\\n\\t\\t\\tflex-wrap: wrap;\\n\\t\\t}\\n\\n\\t\\t&.ck-table-form__background-row {\\n\\t\\t\\tflex-wrap: wrap;\\n\\t\\t}\\n\\n\\t\\t&.ck-table-form__dimensions-row {\\n\\t\\t\\tflex-wrap: wrap;\\n\\t\\t\\talign-items: center;\\n\\n\\t\\t\\t& .ck-labeled-field-view {\\n\\t\\t\\t\\tdisplay: flex;\\n\\t\\t\\t\\tflex-direction: column-reverse;\\n\\t\\t\\t\\talign-items: center;\\n\\n\\t\\t\\t\\t& .ck.ck-dropdown {\\n\\t\\t\\t\\t\\tflex-grow: 0;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\n\\t\\t\\t& .ck-table-form__dimension-operator {\\n\\t\\t\\t\\tflex-grow: 0;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck.ck-labeled-field-view {\\n\\t\\t/* Allow absolute positioning of the status (error) balloons. */\\n\\t\\tposition: relative;\\n\\n\\t\\t& .ck.ck-labeled-field-view__status {\\n\\t\\t\\tposition: absolute;\\n\\t\\t\\tleft: 50%;\\n\\t\\t\\tbottom: calc( -1 * var(--ck-table-properties-error-arrow-size) );\\n\\t\\t\\ttransform: translate(-50%,100%);\\n\\n\\t\\t\\t/* Make sure the balloon status stays on top of other form elements. */\\n\\t\\t\\tz-index: 1;\\n\\n\\t\\t\\t/* The arrow pointing towards the field. */\\n\\t\\t\\t&::after {\\n\\t\\t\\t\\tcontent: \\\"\\\";\\n\\t\\t\\t\\tposition: absolute;\\n\\t\\t\\t\\ttop: calc( -1 * var(--ck-table-properties-error-arrow-size) );\\n\\t\\t\\t\\tleft: 50%;\\n\\t\\t\\t\\ttransform: translateX( -50% );\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../mixins/_rounded.css\\\";\\n\\n:root {\\n\\t--ck-table-properties-error-arrow-size: 6px;\\n\\t--ck-table-properties-min-error-width: 150px;\\n}\\n\\n.ck.ck-table-form {\\n\\t& .ck-form__row {\\n\\t\\t&.ck-table-form__border-row {\\n\\t\\t\\t& .ck-labeled-field-view {\\n\\t\\t\\t\\t& > .ck-label {\\n\\t\\t\\t\\t\\tfont-size: var(--ck-font-size-tiny);\\n\\t\\t\\t\\t\\ttext-align: center;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\n\\t\\t\\t& .ck-table-form__border-style,\\n\\t\\t\\t& .ck-table-form__border-width {\\n\\t\\t\\t\\twidth: 80px;\\n\\t\\t\\t\\tmin-width: 80px;\\n\\t\\t\\t\\tmax-width: 80px;\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t&.ck-table-form__dimensions-row {\\n\\t\\t\\tpadding: 0;\\n\\n\\t\\t\\t& .ck-table-form__dimensions-row__width,\\n\\t\\t\\t& .ck-table-form__dimensions-row__height {\\n\\t\\t\\t\\tmargin: 0\\n\\t\\t\\t}\\n\\n\\t\\t\\t& .ck-table-form__dimension-operator {\\n\\t\\t\\t\\talign-self: flex-end;\\n\\t\\t\\t\\tdisplay: inline-block;\\n\\t\\t\\t\\theight: var(--ck-ui-component-min-height);\\n\\t\\t\\t\\tline-height: var(--ck-ui-component-min-height);\\n\\t\\t\\t\\tmargin: 0 var(--ck-spacing-small);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck.ck-labeled-field-view {\\n\\t\\tpadding-top: var(--ck-spacing-standard);\\n\\n\\t\\t& .ck.ck-labeled-field-view__status {\\n\\t\\t\\t@mixin ck-rounded-corners;\\n\\n\\t\\t\\tbackground: var(--ck-color-base-error);\\n\\t\\t\\tcolor: var(--ck-color-base-background);\\n\\t\\t\\tpadding: var(--ck-spacing-small) var(--ck-spacing-medium);\\n\\t\\t\\tmin-width: var(--ck-table-properties-min-error-width);\\n\\t\\t\\ttext-align: center;\\n\\n\\t\\t\\t/* The arrow pointing towards the field. */\\n\\t\\t\\t&::after {\\n\\t\\t\\t\\tborder-color: transparent transparent var(--ck-color-base-error) transparent;\\n\\t\\t\\t\\tborder-width: 0 var(--ck-table-properties-error-arrow-size) var(--ck-table-properties-error-arrow-size) var(--ck-table-properties-error-arrow-size);\\n\\t\\t\\t\\tborder-style: solid;\\n\\t\\t\\t}\\n\\n\\t\\t\\tanimation: ck-table-form-labeled-view-status-appear .15s ease both;\\n\\t\\t}\\n\\n\\t\\t/* Hide the error balloon when the field is blurred. Makes the experience much more clear. */\\n\\t\\t& .ck-input.ck-error:not(:focus) + .ck.ck-labeled-field-view__status {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\t}\\n}\\n\\n@keyframes ck-table-form-labeled-view-status-appear {\\n\\t0% {\\n\\t\\topacity: 0;\\n\\t}\\n\\n\\t100% {\\n\\t\\topacity: 1;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-table-properties-form .ck-form__row.ck-table-properties-form__alignment-row{align-content:baseline;flex-basis:0;flex-wrap:wrap}.ck.ck-table-properties-form .ck-form__row.ck-table-properties-form__alignment-row .ck.ck-toolbar .ck-toolbar__items{flex-wrap:nowrap}.ck.ck-table-properties-form{width:320px}.ck.ck-table-properties-form .ck-form__row.ck-table-properties-form__alignment-row{align-self:flex-end;padding:0}.ck.ck-table-properties-form .ck-form__row.ck-table-properties-form__alignment-row .ck.ck-toolbar{background:none;margin-top:var(--ck-spacing-standard)}.ck.ck-table-properties-form .ck-form__row.ck-table-properties-form__alignment-row .ck.ck-toolbar .ck-toolbar__items>*{width:40px}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-table/theme/tableproperties.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/tableproperties.css\"],\"names\":[],\"mappings\":\"AAOE,mFAGC,sBAAuB,CADvB,YAAa,CADb,cAOD,CAHC,qHACC,gBACD,CCTH,6BACC,WAmBD,CAhBE,mFACC,mBAAoB,CACpB,SAYD,CAVC,kGACC,eAAgB,CAGhB,qCAKD,CAHC,uHACC,UACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-table-properties-form {\\n\\t& .ck-form__row {\\n\\t\\t&.ck-table-properties-form__alignment-row {\\n\\t\\t\\tflex-wrap: wrap;\\n\\t\\t\\tflex-basis: 0;\\n\\t\\t\\talign-content: baseline;\\n\\n\\t\\t\\t& .ck.ck-toolbar .ck-toolbar__items {\\n\\t\\t\\t\\tflex-wrap: nowrap;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-table-properties-form {\\n\\twidth: 320px;\\n\\n\\t& .ck-form__row {\\n\\t\\t&.ck-table-properties-form__alignment-row {\\n\\t\\t\\talign-self: flex-end;\\n\\t\\t\\tpadding: 0;\\n\\n\\t\\t\\t& .ck.ck-toolbar {\\n\\t\\t\\t\\tbackground: none;\\n\\n\\t\\t\\t\\t/* Compensate for missing input label that would push the margin (toolbar has no inputs). */\\n\\t\\t\\t\\tmargin-top: var(--ck-spacing-standard);\\n\\n\\t\\t\\t\\t& .ck-toolbar__items > * {\\n\\t\\t\\t\\t\\twidth: 40px;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-table-selected-cell-background:rgba(158,207,250,.3)}.ck.ck-editor__editable .table table td.ck-editor__editable_selected,.ck.ck-editor__editable .table table th.ck-editor__editable_selected{box-shadow:unset;caret-color:transparent;outline:unset;position:relative}.ck.ck-editor__editable .table table td.ck-editor__editable_selected:after,.ck.ck-editor__editable .table table th.ck-editor__editable_selected:after{background-color:var(--ck-table-selected-cell-background);bottom:0;content:\\\"\\\";left:0;pointer-events:none;position:absolute;right:0;top:0}.ck.ck-editor__editable .table table td.ck-editor__editable_selected ::selection,.ck.ck-editor__editable .table table td.ck-editor__editable_selected:focus,.ck.ck-editor__editable .table table th.ck-editor__editable_selected ::selection,.ck.ck-editor__editable .table table th.ck-editor__editable_selected:focus{background-color:transparent}.ck.ck-editor__editable .table table td.ck-editor__editable_selected .ck-widget,.ck.ck-editor__editable .table table th.ck-editor__editable_selected .ck-widget{outline:unset}.ck.ck-editor__editable .table table td.ck-editor__editable_selected .ck-widget>.ck-widget__selection-handle,.ck.ck-editor__editable .table table th.ck-editor__editable_selected .ck-widget>.ck-widget__selection-handle{display:none}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-table/tableselection.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,wDACD,CAGC,0IAKC,gBAAiB,CAFjB,uBAAwB,CACxB,aAAc,CAFd,iBAiCD,CA3BC,sJAGC,yDAA0D,CAK1D,QAAS,CAPT,UAAW,CAKX,MAAO,CAJP,mBAAoB,CAEpB,iBAAkB,CAGlB,OAAQ,CAFR,KAID,CAEA,wTAEC,4BACD,CAMA,gKACC,aAKD,CAHC,0NACC,YACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-table-selected-cell-background: hsla(208, 90%, 80%, .3);\\n}\\n\\n.ck.ck-editor__editable .table table {\\n\\t& td.ck-editor__editable_selected,\\n\\t& th.ck-editor__editable_selected {\\n\\t\\tposition: relative;\\n\\t\\tcaret-color: transparent;\\n\\t\\toutline: unset;\\n\\t\\tbox-shadow: unset;\\n\\n\\t\\t/* https://github.com/ckeditor/ckeditor5/issues/6446 */\\n\\t\\t&:after {\\n\\t\\t\\tcontent: '';\\n\\t\\t\\tpointer-events: none;\\n\\t\\t\\tbackground-color: var(--ck-table-selected-cell-background);\\n\\t\\t\\tposition: absolute;\\n\\t\\t\\ttop: 0;\\n\\t\\t\\tleft: 0;\\n\\t\\t\\tright: 0;\\n\\t\\t\\tbottom: 0;\\n\\t\\t}\\n\\n\\t\\t& ::selection,\\n\\t\\t&:focus {\\n\\t\\t\\tbackground-color: transparent;\\n\\t\\t}\\n\\n\\t\\t/*\\n\\t\\t * To reduce the amount of noise, all widgets in the table selection have no outline and no selection handle.\\n\\t\\t * See https://github.com/ckeditor/ckeditor5/issues/9491.\\n\\t\\t */\\n\\t\\t& .ck-widget {\\n\\t\\t\\toutline: unset;\\n\\n\\t\\t\\t& > .ck-widget__selection-handle {\\n\\t\\t\\t\\tdisplay: none;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-button,a.ck.ck-button{align-items:center;display:inline-flex;justify-content:left;position:relative;-moz-user-select:none;-webkit-user-select:none;-ms-user-select:none;user-select:none}.ck.ck-button .ck-button__label,a.ck.ck-button .ck-button__label{display:none}.ck.ck-button.ck-button_with-text .ck-button__label,a.ck.ck-button.ck-button_with-text .ck-button__label{display:inline-block}.ck.ck-button:not(.ck-button_with-text),a.ck.ck-button:not(.ck-button_with-text){justify-content:center}.ck.ck-button,a.ck.ck-button{background:var(--ck-color-button-default-background)}.ck.ck-button:not(.ck-disabled):hover,a.ck.ck-button:not(.ck-disabled):hover{background:var(--ck-color-button-default-hover-background)}.ck.ck-button:not(.ck-disabled):active,a.ck.ck-button:not(.ck-disabled):active{background:var(--ck-color-button-default-active-background)}.ck.ck-button.ck-disabled,a.ck.ck-button.ck-disabled{background:var(--ck-color-button-default-disabled-background)}.ck.ck-button,a.ck.ck-button{border-radius:0}.ck-rounded-corners .ck.ck-button,.ck-rounded-corners a.ck.ck-button,.ck.ck-button.ck-rounded-corners,a.ck.ck-button.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-button,a.ck.ck-button{-webkit-appearance:none;border:1px solid transparent;cursor:default;font-size:inherit;line-height:1;min-height:var(--ck-ui-component-min-height);min-width:var(--ck-ui-component-min-height);padding:var(--ck-spacing-tiny);text-align:center;transition:box-shadow .2s ease-in-out,border .2s ease-in-out;vertical-align:middle;white-space:nowrap}.ck.ck-button:active,.ck.ck-button:focus,a.ck.ck-button:active,a.ck.ck-button:focus{border:var(--ck-focus-ring);box-shadow:var(--ck-focus-outer-shadow),0 0;outline:none}.ck.ck-button .ck-button__icon use,.ck.ck-button .ck-button__icon use *,a.ck.ck-button .ck-button__icon use,a.ck.ck-button .ck-button__icon use *{color:inherit}.ck.ck-button .ck-button__label,a.ck.ck-button .ck-button__label{color:inherit;cursor:inherit;font-size:inherit;font-weight:inherit;vertical-align:middle}[dir=ltr] .ck.ck-button .ck-button__label,[dir=ltr] a.ck.ck-button .ck-button__label{text-align:left}[dir=rtl] .ck.ck-button .ck-button__label,[dir=rtl] a.ck.ck-button .ck-button__label{text-align:right}.ck.ck-button .ck-button__keystroke,a.ck.ck-button .ck-button__keystroke{color:inherit}[dir=ltr] .ck.ck-button .ck-button__keystroke,[dir=ltr] a.ck.ck-button .ck-button__keystroke{margin-left:var(--ck-spacing-large)}[dir=rtl] .ck.ck-button .ck-button__keystroke,[dir=rtl] a.ck.ck-button .ck-button__keystroke{margin-right:var(--ck-spacing-large)}.ck.ck-button .ck-button__keystroke,a.ck.ck-button .ck-button__keystroke{font-weight:700;opacity:.7}.ck.ck-button.ck-disabled:active,.ck.ck-button.ck-disabled:focus,a.ck.ck-button.ck-disabled:active,a.ck.ck-button.ck-disabled:focus{box-shadow:var(--ck-focus-disabled-outer-shadow),0 0}.ck.ck-button.ck-disabled .ck-button__icon,.ck.ck-button.ck-disabled .ck-button__label,a.ck.ck-button.ck-disabled .ck-button__icon,a.ck.ck-button.ck-disabled .ck-button__label{opacity:var(--ck-disabled-opacity)}.ck.ck-button.ck-disabled .ck-button__keystroke,a.ck.ck-button.ck-disabled .ck-button__keystroke{opacity:.3}.ck.ck-button.ck-button_with-text,a.ck.ck-button.ck-button_with-text{padding:var(--ck-spacing-tiny) var(--ck-spacing-standard)}[dir=ltr] .ck.ck-button.ck-button_with-text .ck-button__icon,[dir=ltr] a.ck.ck-button.ck-button_with-text .ck-button__icon{margin-left:calc(var(--ck-spacing-small)*-1);margin-right:var(--ck-spacing-small)}[dir=rtl] .ck.ck-button.ck-button_with-text .ck-button__icon,[dir=rtl] a.ck.ck-button.ck-button_with-text .ck-button__icon{margin-left:var(--ck-spacing-small);margin-right:calc(var(--ck-spacing-small)*-1)}.ck.ck-button.ck-button_with-keystroke .ck-button__label,a.ck.ck-button.ck-button_with-keystroke .ck-button__label{flex-grow:1}.ck.ck-button.ck-on,a.ck.ck-button.ck-on{background:var(--ck-color-button-on-background)}.ck.ck-button.ck-on:not(.ck-disabled):hover,a.ck.ck-button.ck-on:not(.ck-disabled):hover{background:var(--ck-color-button-on-hover-background)}.ck.ck-button.ck-on:not(.ck-disabled):active,a.ck.ck-button.ck-on:not(.ck-disabled):active{background:var(--ck-color-button-on-active-background)}.ck.ck-button.ck-on.ck-disabled,a.ck.ck-button.ck-on.ck-disabled{background:var(--ck-color-button-on-disabled-background)}.ck.ck-button.ck-on,a.ck.ck-button.ck-on{color:var(--ck-color-button-on-color)}.ck.ck-button.ck-button-save,a.ck.ck-button.ck-button-save{color:var(--ck-color-button-save)}.ck.ck-button.ck-button-cancel,a.ck.ck-button.ck-button-cancel{color:var(--ck-color-button-cancel)}.ck.ck-button-action,a.ck.ck-button-action{background:var(--ck-color-button-action-background)}.ck.ck-button-action:not(.ck-disabled):hover,a.ck.ck-button-action:not(.ck-disabled):hover{background:var(--ck-color-button-action-hover-background)}.ck.ck-button-action:not(.ck-disabled):active,a.ck.ck-button-action:not(.ck-disabled):active{background:var(--ck-color-button-action-active-background)}.ck.ck-button-action.ck-disabled,a.ck.ck-button-action.ck-disabled{background:var(--ck-color-button-action-disabled-background)}.ck.ck-button-action,a.ck.ck-button-action{color:var(--ck-color-button-action-text)}.ck.ck-button-bold,a.ck.ck-button-bold{font-weight:700}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/button/button.css\",\"webpack://./../ckeditor5-ui/theme/mixins/_unselectable.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/button/button.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/mixins/_button.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_focus.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_shadow.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_disabled.css\"],\"names\":[],\"mappings\":\"AAOA,6BAMC,kBAAmB,CADnB,mBAAoB,CAEpB,oBAAqB,CAHrB,iBAAkB,CCFlB,qBAAsB,CACtB,wBAAyB,CACzB,oBAAqB,CACrB,gBDkBD,CAdC,iEACC,YACD,CAGC,yGACC,oBACD,CAID,iFACC,sBACD,CEjBD,6BCAC,oDD4ID,CCzIE,6EACC,0DACD,CAEA,+EACC,2DACD,CAID,qDACC,6DACD,CDfD,6BEDC,eF6ID,CA5IA,wIEGE,qCFyIF,CA5IA,6BA6BC,uBAAwB,CANxB,4BAA6B,CAjB7B,cAAe,CAcf,iBAAkB,CAHlB,aAAc,CAJd,4CAA6C,CAD7C,2CAA4C,CAJ5C,8BAA+B,CAC/B,iBAAkB,CAiBlB,4DAA8D,CAnB9D,qBAAsB,CAFtB,kBAuID,CA7GC,oFGhCA,2BAA2B,CCF3B,2CAA8B,CDC9B,YHqCA,CAIC,kJAEC,aACD,CAGD,iEAIC,aAAc,CACd,cAAe,CAHf,iBAAkB,CAClB,mBAAoB,CAMpB,qBASD,CAlBA,qFAYE,eAMF,CAlBA,qFAgBE,gBAEF,CAEA,yEACC,aAYD,CAbA,6FAIE,mCASF,CAbA,6FAQE,oCAKF,CAbA,yEAWC,eAAiB,CACjB,UACD,CAIC,oIIrFD,oDJyFC,CAOA,gLKhGD,kCLkGC,CAEA,iGACC,UACD,CAGD,qEACC,yDAcD,CAXC,2HAEE,4CAA+C,CAC/C,oCAOF,CAVA,2HAQE,mCAAoC,CADpC,6CAGF,CAKA,mHACC,WACD,CAID,yCC/HA,+CDmIA,CChIC,yFACC,qDACD,CAEA,2FACC,sDACD,CAID,iEACC,wDACD,CDgHA,yCAGC,qCACD,CAEA,2DACC,iCACD,CAEA,+DACC,mCACD,CAID,2CC/IC,mDDoJD,CCjJE,2FACC,yDACD,CAEA,6FACC,0DACD,CAID,mEACC,4DACD,CDgID,2CAIC,wCACD,CAEA,uCAEC,eACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../mixins/_unselectable.css\\\";\\n\\n.ck.ck-button,\\na.ck.ck-button {\\n\\t@mixin ck-unselectable;\\n\\n\\tposition: relative;\\n\\tdisplay: inline-flex;\\n\\talign-items: center;\\n\\tjustify-content: left;\\n\\n\\t& .ck-button__label {\\n\\t\\tdisplay: none;\\n\\t}\\n\\n\\t&.ck-button_with-text {\\n\\t\\t& .ck-button__label {\\n\\t\\t\\tdisplay: inline-block;\\n\\t\\t}\\n\\t}\\n\\n\\t/* Center the icon horizontally in a button without text. */\\n\\t&:not(.ck-button_with-text) {\\n\\t\\tjustify-content: center;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Makes element unselectable.\\n */\\n@define-mixin ck-unselectable {\\n\\t-moz-user-select: none;\\n\\t-webkit-user-select: none;\\n\\t-ms-user-select: none;\\n\\tuser-select: none\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_focus.css\\\";\\n@import \\\"../../../mixins/_shadow.css\\\";\\n@import \\\"../../../mixins/_disabled.css\\\";\\n@import \\\"../../../mixins/_rounded.css\\\";\\n@import \\\"../../mixins/_button.css\\\";\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n.ck.ck-button,\\na.ck.ck-button {\\n\\t@mixin ck-button-colors --ck-color-button-default;\\n\\t@mixin ck-rounded-corners;\\n\\n\\twhite-space: nowrap;\\n\\tcursor: default;\\n\\tvertical-align: middle;\\n\\tpadding: var(--ck-spacing-tiny);\\n\\ttext-align: center;\\n\\n\\t/* A very important piece of styling. Go to variable declaration to learn more. */\\n\\tmin-width: var(--ck-ui-component-min-height);\\n\\tmin-height: var(--ck-ui-component-min-height);\\n\\n\\t/* Normalize the height of the line. Removing this will break consistent height\\n\\tamong text and text-less buttons (with icons). */\\n\\tline-height: 1;\\n\\n\\t/* Enable font size inheritance, which allows fluid UI scaling. */\\n\\tfont-size: inherit;\\n\\n\\t/* Avoid flickering when the foucs border shows up. */\\n\\tborder: 1px solid transparent;\\n\\n\\t/* Apply some smooth transition to the box-shadow and border. */\\n\\ttransition: box-shadow .2s ease-in-out, border .2s ease-in-out;\\n\\n\\t/* https://github.com/ckeditor/ckeditor5-theme-lark/issues/189 */\\n\\t-webkit-appearance: none;\\n\\n\\t&:active,\\n\\t&:focus {\\n\\t\\t@mixin ck-focus-ring;\\n\\t\\t@mixin ck-box-shadow var(--ck-focus-outer-shadow);\\n\\t}\\n\\n\\t/* Allow icon coloring using the text \\\"color\\\" property. */\\n\\t& .ck-button__icon {\\n\\t\\t& use,\\n\\t\\t& use * {\\n\\t\\t\\tcolor: inherit;\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck-button__label {\\n\\t\\t/* Enable font size inheritance, which allows fluid UI scaling. */\\n\\t\\tfont-size: inherit;\\n\\t\\tfont-weight: inherit;\\n\\t\\tcolor: inherit;\\n\\t\\tcursor: inherit;\\n\\n\\t\\t/* Must be consistent with .ck-icon's vertical align. Otherwise, buttons with and\\n\\t\\twithout labels (but with icons) have different sizes in Chrome */\\n\\t\\tvertical-align: middle;\\n\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\ttext-align: left;\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\ttext-align: right;\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck-button__keystroke {\\n\\t\\tcolor: inherit;\\n\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\tmargin-left: var(--ck-spacing-large);\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\tmargin-right: var(--ck-spacing-large);\\n\\t\\t}\\n\\n\\t\\tfont-weight: bold;\\n\\t\\topacity: .7;\\n\\t}\\n\\n\\t/* https://github.com/ckeditor/ckeditor5-theme-lark/issues/70 */\\n\\t&.ck-disabled {\\n\\t\\t&:active,\\n\\t\\t&:focus {\\n\\t\\t\\t/* The disabled button should have a slightly less visible shadow when focused. */\\n\\t\\t\\t@mixin ck-box-shadow var(--ck-focus-disabled-outer-shadow);\\n\\t\\t}\\n\\n\\t\\t& .ck-button__icon {\\n\\t\\t\\t@mixin ck-disabled;\\n\\t\\t}\\n\\n\\t\\t/* https://github.com/ckeditor/ckeditor5-theme-lark/issues/98 */\\n\\t\\t& .ck-button__label {\\n\\t\\t\\t@mixin ck-disabled;\\n\\t\\t}\\n\\n\\t\\t& .ck-button__keystroke {\\n\\t\\t\\topacity: .3;\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-button_with-text {\\n\\t\\tpadding: var(--ck-spacing-tiny) var(--ck-spacing-standard);\\n\\n\\t\\t/* stylelint-disable-next-line no-descending-specificity */\\n\\t\\t& .ck-button__icon {\\n\\t\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t\\tmargin-left: calc(-1 * var(--ck-spacing-small));\\n\\t\\t\\t\\tmargin-right: var(--ck-spacing-small);\\n\\t\\t\\t}\\n\\n\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\tmargin-right: calc(-1 * var(--ck-spacing-small));\\n\\t\\t\\t\\tmargin-left: var(--ck-spacing-small);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-button_with-keystroke {\\n\\t\\t/* stylelint-disable-next-line no-descending-specificity */\\n\\t\\t& .ck-button__label {\\n\\t\\t\\tflex-grow: 1;\\n\\t\\t}\\n\\t}\\n\\n\\t/* A style of the button which is currently on, e.g. its feature is active. */\\n\\t&.ck-on {\\n\\t\\t@mixin ck-button-colors --ck-color-button-on;\\n\\n\\t\\tcolor: var(--ck-color-button-on-color);\\n\\t}\\n\\n\\t&.ck-button-save {\\n\\t\\tcolor: var(--ck-color-button-save);\\n\\t}\\n\\n\\t&.ck-button-cancel {\\n\\t\\tcolor: var(--ck-color-button-cancel);\\n\\t}\\n}\\n\\n/* A style of the button which handles the primary action. */\\n.ck.ck-button-action,\\na.ck.ck-button-action {\\n\\t@mixin ck-button-colors --ck-color-button-action;\\n\\n\\tcolor: var(--ck-color-button-action-text);\\n}\\n\\n.ck.ck-button-bold,\\na.ck.ck-button-bold {\\n\\tfont-weight: bold;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements a button of given background color.\\n *\\n * @param {String} $background - Background color of the button.\\n * @param {String} $border - Border color of the button.\\n */\\n@define-mixin ck-button-colors $prefix {\\n\\tbackground: var($(prefix)-background);\\n\\n\\t&:not(.ck-disabled) {\\n\\t\\t&:hover {\\n\\t\\t\\tbackground: var($(prefix)-hover-background);\\n\\t\\t}\\n\\n\\t\\t&:active {\\n\\t\\t\\tbackground: var($(prefix)-active-background);\\n\\t\\t}\\n\\t}\\n\\n\\t/* https://github.com/ckeditor/ckeditor5-theme-lark/issues/98 */\\n\\t&.ck-disabled {\\n\\t\\tbackground: var($(prefix)-disabled-background);\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A visual style of focused element's border.\\n */\\n@define-mixin ck-focus-ring {\\n\\t/* Disable native outline. */\\n\\toutline: none;\\n\\tborder: var(--ck-focus-ring)\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A helper to combine multiple shadows.\\n */\\n@define-mixin ck-box-shadow $shadowA, $shadowB: 0 0 {\\n\\tbox-shadow: $shadowA, $shadowB;\\n}\\n\\n/**\\n * Gives an element a drop shadow so it looks like a floating panel.\\n */\\n@define-mixin ck-drop-shadow {\\n\\t@mixin ck-box-shadow var(--ck-drop-shadow);\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A class which indicates that an element holding it is disabled.\\n */\\n@define-mixin ck-disabled {\\n\\topacity: var(--ck-disabled-opacity);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-button.ck-switchbutton .ck-button__toggle,.ck.ck-button.ck-switchbutton .ck-button__toggle .ck-button__toggle__inner{display:block}:root{--ck-switch-button-toggle-width:2.6153846154em;--ck-switch-button-toggle-inner-size:calc(1.07692em + 1px);--ck-switch-button-translation:calc(var(--ck-switch-button-toggle-width) - var(--ck-switch-button-toggle-inner-size) - 2px);--ck-switch-button-inner-hover-shadow:0 0 0 5px var(--ck-color-switch-button-inner-shadow)}.ck.ck-button.ck-switchbutton,.ck.ck-button.ck-switchbutton.ck-on:active,.ck.ck-button.ck-switchbutton.ck-on:focus,.ck.ck-button.ck-switchbutton.ck-on:hover,.ck.ck-button.ck-switchbutton:active,.ck.ck-button.ck-switchbutton:focus,.ck.ck-button.ck-switchbutton:hover{background:transparent;color:inherit}[dir=ltr] .ck.ck-button.ck-switchbutton .ck-button__label{margin-right:calc(var(--ck-spacing-large)*2)}[dir=rtl] .ck.ck-button.ck-switchbutton .ck-button__label{margin-left:calc(var(--ck-spacing-large)*2)}.ck.ck-button.ck-switchbutton .ck-button__toggle{border-radius:0}.ck-rounded-corners .ck.ck-button.ck-switchbutton .ck-button__toggle,.ck.ck-button.ck-switchbutton .ck-button__toggle.ck-rounded-corners{border-radius:var(--ck-border-radius)}[dir=ltr] .ck.ck-button.ck-switchbutton .ck-button__toggle{margin-left:auto}[dir=rtl] .ck.ck-button.ck-switchbutton .ck-button__toggle{margin-right:auto}.ck.ck-button.ck-switchbutton .ck-button__toggle{background:var(--ck-color-switch-button-off-background);border:1px solid transparent;transition:background .4s ease,box-shadow .2s ease-in-out,outline .2s ease-in-out;width:var(--ck-switch-button-toggle-width)}.ck.ck-button.ck-switchbutton .ck-button__toggle .ck-button__toggle__inner{border-radius:0}.ck-rounded-corners .ck.ck-button.ck-switchbutton .ck-button__toggle .ck-button__toggle__inner,.ck.ck-button.ck-switchbutton .ck-button__toggle .ck-button__toggle__inner.ck-rounded-corners{border-radius:var(--ck-border-radius);border-radius:calc(var(--ck-border-radius)*.5)}.ck.ck-button.ck-switchbutton .ck-button__toggle .ck-button__toggle__inner{background:var(--ck-color-switch-button-inner-background);height:var(--ck-switch-button-toggle-inner-size);transition:all .3s ease;width:var(--ck-switch-button-toggle-inner-size)}.ck.ck-button.ck-switchbutton .ck-button__toggle:hover{background:var(--ck-color-switch-button-off-hover-background)}.ck.ck-button.ck-switchbutton .ck-button__toggle:hover .ck-button__toggle__inner{box-shadow:var(--ck-switch-button-inner-hover-shadow)}.ck.ck-button.ck-switchbutton.ck-disabled .ck-button__toggle{opacity:var(--ck-disabled-opacity)}.ck.ck-button.ck-switchbutton:focus{border-color:transparent;box-shadow:none;outline:none}.ck.ck-button.ck-switchbutton:focus .ck-button__toggle{box-shadow:0 0 0 1px var(--ck-color-base-background),0 0 0 5px var(--ck-color-focus-outer-shadow);outline:var(--ck-focus-ring);outline-offset:1px}.ck.ck-button.ck-switchbutton.ck-on .ck-button__toggle{background:var(--ck-color-switch-button-on-background)}.ck.ck-button.ck-switchbutton.ck-on .ck-button__toggle:hover{background:var(--ck-color-switch-button-on-hover-background)}[dir=ltr] .ck.ck-button.ck-switchbutton.ck-on .ck-button__toggle .ck-button__toggle__inner{transform:translateX(var( --ck-switch-button-translation ))}[dir=rtl] .ck.ck-button.ck-switchbutton.ck-on .ck-button__toggle .ck-button__toggle__inner{transform:translateX(calc(var( --ck-switch-button-translation )*-1))}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/button/switchbutton.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/button/switchbutton.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_disabled.css\"],\"names\":[],\"mappings\":\"AASE,4HACC,aACD,CCCF,MAEC,8CAA+C,CAE/C,0DAAgE,CAChE,2HAIC,CACD,0FACD,CAOC,0QAEC,sBAAuB,CADvB,aAED,CAEA,0DAGE,4CAOF,CAVA,0DAQE,2CAEF,CAEA,iDCpCA,eD4EA,CAxCA,yIChCC,qCDwED,CAxCA,2DAKE,gBAmCF,CAxCA,2DAUE,iBA8BF,CAxCA,iDAkBC,uDAAwD,CAFxD,4BAA6B,CAD7B,iFAAsF,CAEtF,0CAuBD,CApBC,2ECxDD,eDmEC,CAXA,6LCpDA,qCAAsC,CDsDpC,8CASF,CAXA,2EAOC,yDAA0D,CAD1D,gDAAiD,CAIjD,uBAA0B,CAL1B,+CAMD,CAEA,uDACC,6DAKD,CAHC,iFACC,qDACD,CAIF,6DEhFA,kCFkFA,CAGA,oCACC,wBAAyB,CAEzB,eAAgB,CADhB,YAQD,CALC,uDACC,iGAAmG,CAEnG,4BAA6B,CAD7B,kBAED,CAKA,uDACC,sDAkBD,CAhBC,6DACC,4DACD,CAEA,2FAKE,2DAMF,CAXA,2FASE,oEAEF\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-button.ck-switchbutton {\\n\\t& .ck-button__toggle {\\n\\t\\tdisplay: block;\\n\\n\\t\\t& .ck-button__toggle__inner {\\n\\t\\t\\tdisplay: block;\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n@import \\\"../../../mixins/_disabled.css\\\";\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n/* Note: To avoid rendering issues (aliasing) but to preserve the responsive nature\\nof the component, floatingpoint numbers have been used which, for the default font size\\n(see: --ck-font-size-base), will generate simple integers. */\\n:root {\\n\\t/* 34px at 13px font-size */\\n\\t--ck-switch-button-toggle-width: 2.6153846154em;\\n\\t/* 14px at 13px font-size */\\n\\t--ck-switch-button-toggle-inner-size: calc(1.0769230769em + 1px);\\n\\t--ck-switch-button-translation: calc(\\n\\t\\tvar(--ck-switch-button-toggle-width) -\\n\\t\\tvar(--ck-switch-button-toggle-inner-size) -\\n\\t\\t2px /* Border */\\n\\t);\\n\\t--ck-switch-button-inner-hover-shadow: 0 0 0 5px var(--ck-color-switch-button-inner-shadow);\\n}\\n\\n.ck.ck-button.ck-switchbutton {\\n\\t/* Unlike a regular button, the switch button text color and background should never change.\\n\\t * Changing toggle switch (background, outline) is enough to carry the information about the\\n\\t * state of the entire component (https://github.com/ckeditor/ckeditor5/issues/12519)\\n\\t */\\n\\t&, &:hover, &:focus, &:active, &.ck-on:hover, &.ck-on:focus, &.ck-on:active {\\n\\t\\tcolor: inherit;\\n\\t\\tbackground: transparent;\\n\\t}\\n\\n\\t& .ck-button__label {\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t/* Separate the label from the switch */\\n\\t\\t\\tmargin-right: calc(2 * var(--ck-spacing-large));\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t/* Separate the label from the switch */\\n\\t\\t\\tmargin-left: calc(2 * var(--ck-spacing-large));\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck-button__toggle {\\n\\t\\t@mixin ck-rounded-corners;\\n\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t/* Make sure the toggle is always to the right as far as possible. */\\n\\t\\t\\tmargin-left: auto;\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t/* Make sure the toggle is always to the left as far as possible. */\\n\\t\\t\\tmargin-right: auto;\\n\\t\\t}\\n\\n\\t\\t/* Apply some smooth transition to the box-shadow and border. */\\n\\t\\t/* Gently animate the background color of the toggle switch */\\n\\t\\ttransition: background 400ms ease, box-shadow .2s ease-in-out, outline .2s ease-in-out;\\n\\t\\tborder: 1px solid transparent;\\n\\t\\twidth: var(--ck-switch-button-toggle-width);\\n\\t\\tbackground: var(--ck-color-switch-button-off-background);\\n\\n\\t\\t& .ck-button__toggle__inner {\\n\\t\\t\\t@mixin ck-rounded-corners {\\n\\t\\t\\t\\tborder-radius: calc(.5 * var(--ck-border-radius));\\n\\t\\t\\t}\\n\\n\\t\\t\\twidth: var(--ck-switch-button-toggle-inner-size);\\n\\t\\t\\theight: var(--ck-switch-button-toggle-inner-size);\\n\\t\\t\\tbackground: var(--ck-color-switch-button-inner-background);\\n\\n\\t\\t\\t/* Gently animate the inner part of the toggle switch */\\n\\t\\t\\ttransition: all 300ms ease;\\n\\t\\t}\\n\\n\\t\\t&:hover {\\n\\t\\t\\tbackground: var(--ck-color-switch-button-off-hover-background);\\n\\n\\t\\t\\t& .ck-button__toggle__inner {\\n\\t\\t\\t\\tbox-shadow: var(--ck-switch-button-inner-hover-shadow);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-disabled .ck-button__toggle {\\n\\t\\t@mixin ck-disabled;\\n\\t}\\n\\n\\t/* Overriding default .ck-button:focus styles + an outline around the toogle */\\n\\t&:focus {\\n\\t\\tborder-color: transparent;\\n\\t\\toutline: none;\\n\\t\\tbox-shadow: none;\\n\\n\\t\\t& .ck-button__toggle {\\n\\t\\t\\tbox-shadow: 0 0 0 1px var(--ck-color-base-background), 0 0 0 5px var(--ck-color-focus-outer-shadow);\\n\\t\\t\\toutline-offset: 1px;\\n\\t\\t\\toutline: var(--ck-focus-ring);\\n\\t\\t}\\n\\t}\\n\\n\\t/* stylelint-disable-next-line no-descending-specificity */\\n\\t&.ck-on {\\n\\t\\t& .ck-button__toggle {\\n\\t\\t\\tbackground: var(--ck-color-switch-button-on-background);\\n\\n\\t\\t\\t&:hover {\\n\\t\\t\\t\\tbackground: var(--ck-color-switch-button-on-hover-background);\\n\\t\\t\\t}\\n\\n\\t\\t\\t& .ck-button__toggle__inner {\\n\\t\\t\\t\\t/*\\n\\t\\t\\t\\t* Move the toggle switch to the right. It will be animated.\\n\\t\\t\\t\\t*/\\n\\t\\t\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t\\t\\ttransform: translateX( var( --ck-switch-button-translation ) );\\n\\t\\t\\t\\t}\\n\\n\\t\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\t\\ttransform: translateX( calc( -1 * var( --ck-switch-button-translation ) ) );\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A class which indicates that an element holding it is disabled.\\n */\\n@define-mixin ck-disabled {\\n\\topacity: var(--ck-disabled-opacity);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-color-grid{display:grid}:root{--ck-color-grid-tile-size:24px;--ck-color-color-grid-check-icon:#166fd4}.ck.ck-color-grid{grid-gap:5px;padding:8px}.ck.ck-color-grid__tile{border:0;height:var(--ck-color-grid-tile-size);min-height:var(--ck-color-grid-tile-size);min-width:var(--ck-color-grid-tile-size);padding:0;transition:box-shadow .2s ease;width:var(--ck-color-grid-tile-size)}.ck.ck-color-grid__tile.ck-disabled{cursor:unset;transition:unset}.ck.ck-color-grid__tile.ck-color-selector__color-tile_bordered{box-shadow:0 0 0 1px var(--ck-color-base-border)}.ck.ck-color-grid__tile .ck.ck-icon{color:var(--ck-color-color-grid-check-icon);display:none}.ck.ck-color-grid__tile.ck-on{box-shadow:inset 0 0 0 1px var(--ck-color-base-background),0 0 0 2px var(--ck-color-base-text)}.ck.ck-color-grid__tile.ck-on .ck.ck-icon{display:block}.ck.ck-color-grid__tile.ck-on,.ck.ck-color-grid__tile:focus:not(.ck-disabled),.ck.ck-color-grid__tile:hover:not(.ck-disabled){border:0}.ck.ck-color-grid__tile:focus:not(.ck-disabled),.ck.ck-color-grid__tile:hover:not(.ck-disabled){box-shadow:inset 0 0 0 1px var(--ck-color-base-background),0 0 0 2px var(--ck-color-focus-border)}.ck.ck-color-grid__label{padding:0 var(--ck-spacing-standard)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/colorgrid/colorgrid.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/colorgrid/colorgrid.css\"],\"names\":[],\"mappings\":\"AAKA,kBACC,YACD,CCAA,MACC,8BAA+B,CAK/B,wCACD,CAEA,kBACC,YAAa,CACb,WACD,CAEA,wBAOC,QAAS,CALT,qCAAsC,CAEtC,yCAA0C,CAD1C,wCAAyC,CAEzC,SAAU,CACV,8BAA+B,CAL/B,oCAyCD,CAjCC,oCACC,YAAa,CACb,gBACD,CAEA,+DACC,gDACD,CAEA,oCAEC,2CAA4C,CAD5C,YAED,CAEA,8BACC,8FAKD,CAHC,0CACC,aACD,CAGD,8HAIC,QACD,CAEA,gGAEC,iGACD,CAGD,yBACC,oCACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-color-grid {\\n\\tdisplay: grid;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n\\n:root {\\n\\t--ck-color-grid-tile-size: 24px;\\n\\n\\t/* Not using global colors here because these may change but some colors in a pallette\\n\\t * require special treatment. For instance, this ensures no matter what the UI text color is,\\n\\t * the check icon will look good on the black color tile. */\\n\\t--ck-color-color-grid-check-icon: hsl(212, 81%, 46%);\\n}\\n\\n.ck.ck-color-grid {\\n\\tgrid-gap: 5px;\\n\\tpadding: 8px;\\n}\\n\\n.ck.ck-color-grid__tile {\\n\\twidth: var(--ck-color-grid-tile-size);\\n\\theight: var(--ck-color-grid-tile-size);\\n\\tmin-width: var(--ck-color-grid-tile-size);\\n\\tmin-height: var(--ck-color-grid-tile-size);\\n\\tpadding: 0;\\n\\ttransition: .2s ease box-shadow;\\n\\tborder: 0;\\n\\n\\t&.ck-disabled {\\n\\t\\tcursor: unset;\\n\\t\\ttransition: unset;\\n\\t}\\n\\n\\t&.ck-color-selector__color-tile_bordered {\\n\\t\\tbox-shadow: 0 0 0 1px var(--ck-color-base-border);\\n\\t}\\n\\n\\t& .ck.ck-icon {\\n\\t\\tdisplay: none;\\n\\t\\tcolor: var(--ck-color-color-grid-check-icon);\\n\\t}\\n\\n\\t&.ck-on {\\n\\t\\tbox-shadow: inset 0 0 0 1px var(--ck-color-base-background), 0 0 0 2px var(--ck-color-base-text);\\n\\n\\t\\t& .ck.ck-icon {\\n\\t\\t\\tdisplay: block;\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-on,\\n\\t&:focus:not( .ck-disabled ),\\n\\t&:hover:not( .ck-disabled ) {\\n\\t\\t/* Disable the default .ck-button's border ring. */\\n\\t\\tborder: 0;\\n\\t}\\n\\n\\t&:focus:not( .ck-disabled ),\\n\\t&:hover:not( .ck-disabled ) {\\n\\t\\tbox-shadow: inset 0 0 0 1px var(--ck-color-base-background), 0 0 0 2px var(--ck-color-focus-border);\\n\\t}\\n}\\n\\n.ck.ck-color-grid__label {\\n\\tpadding: 0 var(--ck-spacing-standard);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".color-picker-hex-input{width:max-content}.color-picker-hex-input .ck.ck-input{min-width:unset}.ck.ck-color-picker__row{display:flex;flex-direction:row;flex-wrap:nowrap;justify-content:space-between;margin:var(--ck-spacing-large) 0 0;width:unset}.ck.ck-color-picker__row .ck.ck-labeled-field-view{padding-top:unset}.ck.ck-color-picker__row .ck.ck-input-text{width:unset}.ck.ck-color-picker__row .ck-color-picker__hash-view{padding-right:var(--ck-spacing-medium);padding-top:var(--ck-spacing-tiny)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/colorpicker/colorpicker.css\"],\"names\":[],\"mappings\":\"AAKA,wBACC,iBAKD,CAHC,qCACC,eACD,CAGD,yBACC,YAAa,CACb,kBAAmB,CACnB,gBAAiB,CACjB,6BAA8B,CAC9B,kCAAmC,CACnC,WAcD,CAZC,mDACC,iBACD,CAEA,2CACC,WACD,CAEA,qDAEC,sCAAuC,CADvC,kCAED\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.color-picker-hex-input {\\n\\twidth: max-content;\\n\\n\\t& .ck.ck-input {\\n\\t\\tmin-width: unset;\\n\\t}\\n}\\n\\n.ck.ck-color-picker__row {\\n\\tdisplay: flex;\\n\\tflex-direction: row;\\n\\tflex-wrap: nowrap;\\n\\tjustify-content: space-between;\\n\\tmargin: var(--ck-spacing-large) 0 0;\\n\\twidth: unset;\\n\\n\\t& .ck.ck-labeled-field-view {\\n\\t\\tpadding-top: unset;\\n\\t}\\n\\n\\t& .ck.ck-input-text {\\n\\t\\twidth: unset;\\n\\t}\\n\\n\\t& .ck-color-picker__hash-view {\\n\\t\\tpadding-top: var(--ck-spacing-tiny);\\n\\t\\tpadding-right: var(--ck-spacing-medium);\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__color-picker,.ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__remove-color{align-items:center;display:flex}[dir=rtl] .ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__color-picker,[dir=rtl] .ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__remove-color{justify-content:flex-start}.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-selector_action-bar{display:flex;flex-direction:row;justify-content:space-around}.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-selector_action-bar .ck-button-cancel,.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-selector_action-bar .ck-button-save{flex:1}.ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__color-picker,.ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__remove-color{width:100%}.ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__color-picker{border-bottom-left-radius:0;border-bottom-right-radius:0;padding:calc(var(--ck-spacing-standard)/2) var(--ck-spacing-standard)}.ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__color-picker:not(:focus){border-top:1px solid var(--ck-color-base-border)}[dir=ltr] .ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__color-picker .ck.ck-icon{margin-right:var(--ck-spacing-standard)}[dir=rtl] .ck.ck-color-selector .ck-color-grids-fragment .ck-button.ck-color-selector__color-picker .ck.ck-icon{margin-left:var(--ck-spacing-standard)}.ck.ck-color-selector .ck-color-grids-fragment label.ck.ck-color-grid__label{font-weight:unset}.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-picker{padding:8px}.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-picker .hex-color-picker{height:100px;min-width:180px}.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-picker .hex-color-picker::part(saturation){border-radius:var(--ck-border-radius) var(--ck-border-radius) 0 0}.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-picker .hex-color-picker::part(hue){border-radius:0 0 var(--ck-border-radius) var(--ck-border-radius)}.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-picker .hex-color-picker::part(hue-pointer),.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-picker .hex-color-picker::part(saturation-pointer){height:15px;width:15px}.ck.ck-color-selector .ck-color-picker-fragment .ck.ck-color-selector_action-bar{padding:0 8px 8px}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/colorselector/colorselector.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/colorselector/colorselector.css\"],\"names\":[],\"mappings\":\"AAUE,oLAGC,kBAAmB,CADnB,YAMD,CARA,wMAME,0BAEF,CAKA,iFACC,YAAa,CACb,kBAAmB,CACnB,4BAMD,CAJC,oMAEC,MACD,CCrBD,oLAEC,UACD,CAEA,0FAEC,2BAA4B,CAC5B,4BAA6B,CAF7B,qEAiBD,CAbC,sGACC,gDACD,CAEA,gHAEE,uCAMF,CARA,gHAME,sCAEF,CAGD,6EACC,iBACD,CAKA,oEACC,WAoBD,CAlBC,sFACC,YAAa,CACb,eAeD,CAbC,wGACC,iEACD,CAEA,iGACC,iEACD,CAEA,yNAGC,WAAY,CADZ,UAED,CAIF,iFACC,iBACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n.ck.ck-color-selector {\\n\\t/* View fragment with color grids. */\\n\\t& .ck-color-grids-fragment {\\n\\t\\t& .ck-button.ck-color-selector__remove-color,\\n\\t\\t& .ck-button.ck-color-selector__color-picker {\\n\\t\\t\\tdisplay: flex;\\n\\t\\t\\talign-items: center;\\n\\n\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\tjustify-content: flex-start;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t/* View fragment with a color picker. */\\n\\t& .ck-color-picker-fragment {\\n\\t\\t& .ck.ck-color-selector_action-bar {\\n\\t\\t\\tdisplay: flex;\\n\\t\\t\\tflex-direction: row;\\n\\t\\t\\tjustify-content: space-around;\\n\\n\\t\\t\\t& .ck-button-save,\\n\\t\\t\\t& .ck-button-cancel {\\n\\t\\t\\t\\tflex: 1\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n.ck.ck-color-selector {\\n\\t/* View fragment with color grids. */\\n\\t& .ck-color-grids-fragment {\\n\\t\\t& .ck-button.ck-color-selector__remove-color,\\n\\t\\t& .ck-button.ck-color-selector__color-picker {\\n\\t\\t\\twidth: 100%;\\n\\t\\t}\\n\\n\\t\\t& .ck-button.ck-color-selector__color-picker {\\n\\t\\t\\tpadding: calc(var(--ck-spacing-standard) / 2) var(--ck-spacing-standard);\\n\\t\\t\\tborder-bottom-left-radius: 0;\\n\\t\\t\\tborder-bottom-right-radius: 0;\\n\\n\\t\\t\\t&:not(:focus) {\\n\\t\\t\\t\\tborder-top: 1px solid var(--ck-color-base-border);\\n\\t\\t\\t}\\n\\n\\t\\t\\t& .ck.ck-icon {\\n\\t\\t\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t\\t\\tmargin-right: var(--ck-spacing-standard);\\n\\t\\t\\t\\t}\\n\\n\\t\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\t\\tmargin-left: var(--ck-spacing-standard);\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t& label.ck.ck-color-grid__label {\\n\\t\\t\\tfont-weight: unset;\\n\\t\\t}\\n\\t}\\n\\n\\t/* View fragment with a color picker. */\\n\\t& .ck-color-picker-fragment {\\n\\t\\t& .ck.ck-color-picker {\\n\\t\\t\\tpadding: 8px;\\n\\n\\t\\t\\t& .hex-color-picker {\\n\\t\\t\\t\\theight: 100px;\\n\\t\\t\\t\\tmin-width: 180px;\\n\\n\\t\\t\\t\\t&::part(saturation) {\\n\\t\\t\\t\\t\\tborder-radius: var(--ck-border-radius) var(--ck-border-radius) 0 0;\\n\\t\\t\\t\\t}\\n\\n\\t\\t\\t\\t&::part(hue) {\\n\\t\\t\\t\\t\\tborder-radius: 0 0 var(--ck-border-radius) var(--ck-border-radius);\\n\\t\\t\\t\\t}\\n\\n\\t\\t\\t\\t&::part(saturation-pointer),\\n\\t\\t\\t\\t&::part(hue-pointer) {\\n\\t\\t\\t\\t\\twidth: 15px;\\n\\t\\t\\t\\t\\theight: 15px;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t& .ck.ck-color-selector_action-bar {\\n\\t\\t\\tpadding: 0 8px 8px;\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-dropdown-max-width:75vw}.ck.ck-dropdown{display:inline-block;position:relative}.ck.ck-dropdown .ck-dropdown__arrow{pointer-events:none;z-index:var(--ck-z-default)}.ck.ck-dropdown .ck-button.ck-dropdown__button{width:100%}.ck.ck-dropdown .ck-dropdown__panel{display:none;max-width:var(--ck-dropdown-max-width);position:absolute;z-index:var(--ck-z-modal)}.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel-visible{display:inline-block}.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_n,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_ne,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_nme,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_nmw,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_nw{bottom:100%}.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_s,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_se,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_sme,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_smw,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_sw{bottom:auto;top:100%}.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_ne,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_se{left:0}.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_nw,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_sw{right:0}.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_n,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_s{left:50%;transform:translateX(-50%)}.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_nmw,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_smw{left:75%;transform:translateX(-75%)}.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_nme,.ck.ck-dropdown .ck-dropdown__panel.ck-dropdown__panel_sme{left:25%;transform:translateX(-25%)}.ck.ck-toolbar .ck-dropdown__panel{z-index:calc(var(--ck-z-modal) + 1)}:root{--ck-dropdown-arrow-size:calc(var(--ck-icon-size)*0.5)}.ck.ck-dropdown{font-size:inherit}.ck.ck-dropdown .ck-dropdown__arrow{width:var(--ck-dropdown-arrow-size)}[dir=ltr] .ck.ck-dropdown .ck-dropdown__arrow{margin-left:var(--ck-spacing-standard);right:var(--ck-spacing-standard)}[dir=rtl] .ck.ck-dropdown .ck-dropdown__arrow{left:var(--ck-spacing-standard);margin-right:var(--ck-spacing-small)}.ck.ck-dropdown.ck-disabled .ck-dropdown__arrow{opacity:var(--ck-disabled-opacity)}[dir=ltr] .ck.ck-dropdown .ck-button.ck-dropdown__button:not(.ck-button_with-text){padding-left:var(--ck-spacing-small)}[dir=rtl] .ck.ck-dropdown .ck-button.ck-dropdown__button:not(.ck-button_with-text){padding-right:var(--ck-spacing-small)}.ck.ck-dropdown .ck-button.ck-dropdown__button .ck-button__label{overflow:hidden;text-overflow:ellipsis;width:7em}.ck.ck-dropdown .ck-button.ck-dropdown__button.ck-disabled .ck-button__label{opacity:var(--ck-disabled-opacity)}.ck.ck-dropdown .ck-button.ck-dropdown__button.ck-on{border-bottom-left-radius:0;border-bottom-right-radius:0}.ck.ck-dropdown .ck-button.ck-dropdown__button.ck-dropdown__button_label-width_auto .ck-button__label{width:auto}.ck.ck-dropdown .ck-button.ck-dropdown__button.ck-off:active,.ck.ck-dropdown .ck-button.ck-dropdown__button.ck-on:active{box-shadow:none}.ck.ck-dropdown .ck-button.ck-dropdown__button.ck-off:active:focus,.ck.ck-dropdown .ck-button.ck-dropdown__button.ck-on:active:focus{box-shadow:var(--ck-focus-outer-shadow),0 0}.ck.ck-dropdown__panel{border-radius:0}.ck-rounded-corners .ck.ck-dropdown__panel,.ck.ck-dropdown__panel.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-dropdown__panel{background:var(--ck-color-dropdown-panel-background);border:1px solid var(--ck-color-dropdown-panel-border);bottom:0;box-shadow:var(--ck-drop-shadow),0 0;min-width:100%}.ck.ck-dropdown__panel.ck-dropdown__panel_se{border-top-left-radius:0}.ck.ck-dropdown__panel.ck-dropdown__panel_sw{border-top-right-radius:0}.ck.ck-dropdown__panel.ck-dropdown__panel_ne{border-bottom-left-radius:0}.ck.ck-dropdown__panel.ck-dropdown__panel_nw{border-bottom-right-radius:0}.ck.ck-dropdown__panel:focus{outline:none}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/dropdown/dropdown.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/dropdown/dropdown.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_disabled.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_shadow.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,4BACD,CAEA,gBACC,oBAAqB,CACrB,iBA2ED,CAzEC,oCACC,mBAAoB,CACpB,2BACD,CAGA,+CACC,UACD,CAEA,oCACC,YAAa,CAEb,sCAAuC,CAEvC,iBAAkB,CAHlB,yBA4DD,CAvDC,+DACC,oBACD,CAEA,mSAKC,WACD,CAEA,mSAUC,WAAY,CADZ,QAED,CAEA,oHAEC,MACD,CAEA,oHAEC,OACD,CAEA,kHAGC,QAAS,CACT,0BACD,CAEA,sHAGC,QAAS,CACT,0BACD,CAEA,sHAGC,QAAS,CACT,0BACD,CAQF,mCACC,mCACD,CCpFA,MACC,sDACD,CAEA,gBAEC,iBA2ED,CAzEC,oCACC,mCACD,CAGC,8CAIC,sCAAuC,CAHvC,gCAID,CAIA,8CACC,+BAAgC,CAGhC,oCACD,CAGD,gDC/BA,kCDiCA,CAIE,mFAEC,oCACD,CAIA,mFAEC,qCACD,CAID,iEAEC,eAAgB,CAChB,sBAAuB,CAFvB,SAGD,CAGA,6EC1DD,kCD4DC,CAGA,qDACC,2BAA4B,CAC5B,4BACD,CAEA,sGACC,UACD,CAGA,yHAEC,eAKD,CAHC,qIE7EF,2CF+EE,CAKH,uBGlFC,eHkHD,CAhCA,qFG9EE,qCH8GF,CAhCA,uBAIC,oDAAqD,CACrD,sDAAuD,CACvD,QAAS,CE1FT,oCAA8B,CF6F9B,cAuBD,CAnBC,6CACC,wBACD,CAEA,6CACC,yBACD,CAEA,6CACC,2BACD,CAEA,6CACC,4BACD,CAEA,6BACC,YACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-dropdown-max-width: 75vw;\\n}\\n\\n.ck.ck-dropdown {\\n\\tdisplay: inline-block;\\n\\tposition: relative;\\n\\n\\t& .ck-dropdown__arrow {\\n\\t\\tpointer-events: none;\\n\\t\\tz-index: var(--ck-z-default);\\n\\t}\\n\\n\\t/* Dropdown button should span horizontally, e.g. in vertical toolbars */\\n\\t& .ck-button.ck-dropdown__button {\\n\\t\\twidth: 100%;\\n\\t}\\n\\n\\t& .ck-dropdown__panel {\\n\\t\\tdisplay: none;\\n\\t\\tz-index: var(--ck-z-modal);\\n\\t\\tmax-width: var(--ck-dropdown-max-width);\\n\\n\\t\\tposition: absolute;\\n\\n\\t\\t&.ck-dropdown__panel-visible {\\n\\t\\t\\tdisplay: inline-block;\\n\\t\\t}\\n\\n\\t\\t&.ck-dropdown__panel_ne,\\n\\t\\t&.ck-dropdown__panel_nw,\\n\\t\\t&.ck-dropdown__panel_n,\\n\\t\\t&.ck-dropdown__panel_nmw,\\n\\t\\t&.ck-dropdown__panel_nme {\\n\\t\\t\\tbottom: 100%;\\n\\t\\t}\\n\\n\\t\\t&.ck-dropdown__panel_se,\\n\\t\\t&.ck-dropdown__panel_sw,\\n\\t\\t&.ck-dropdown__panel_smw,\\n\\t\\t&.ck-dropdown__panel_sme,\\n\\t\\t&.ck-dropdown__panel_s {\\n\\t\\t\\t/*\\n\\t\\t\\t * Using transform: translate3d( 0, 100%, 0 ) causes blurry dropdown on Chrome 67-78+ on non-retina displays.\\n\\t\\t\\t * See https://github.com/ckeditor/ckeditor5/issues/1053.\\n\\t\\t\\t */\\n\\t\\t\\ttop: 100%;\\n\\t\\t\\tbottom: auto;\\n\\t\\t}\\n\\n\\t\\t&.ck-dropdown__panel_ne,\\n\\t\\t&.ck-dropdown__panel_se {\\n\\t\\t\\tleft: 0px;\\n\\t\\t}\\n\\n\\t\\t&.ck-dropdown__panel_nw,\\n\\t\\t&.ck-dropdown__panel_sw {\\n\\t\\t\\tright: 0px;\\n\\t\\t}\\n\\n\\t\\t&.ck-dropdown__panel_s,\\n\\t\\t&.ck-dropdown__panel_n {\\n\\t\\t\\t/* Positioning panels relative to the center of the button */\\n\\t\\t\\tleft: 50%;\\n\\t\\t\\ttransform: translateX(-50%);\\n\\t\\t}\\n\\n\\t\\t&.ck-dropdown__panel_nmw,\\n\\t\\t&.ck-dropdown__panel_smw {\\n\\t\\t\\t/* Positioning panels relative to the middle-west of the button */\\n\\t\\t\\tleft: 75%;\\n\\t\\t\\ttransform: translateX(-75%);\\n\\t\\t}\\n\\n\\t\\t&.ck-dropdown__panel_nme,\\n\\t\\t&.ck-dropdown__panel_sme {\\n\\t\\t\\t/* Positioning panels relative to the middle-east of the button */\\n\\t\\t\\tleft: 25%;\\n\\t\\t\\ttransform: translateX(-25%);\\n\\t\\t}\\n\\t}\\n}\\n\\n/*\\n * Toolbar dropdown panels should be always above the UI (eg. other dropdown panels) from the editor's content.\\n * See https://github.com/ckeditor/ckeditor5/issues/7874\\n */\\n.ck.ck-toolbar .ck-dropdown__panel {\\n\\tz-index: calc( var(--ck-z-modal) + 1 );\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n@import \\\"../../../mixins/_disabled.css\\\";\\n@import \\\"../../../mixins/_shadow.css\\\";\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n:root {\\n\\t--ck-dropdown-arrow-size: calc(0.5 * var(--ck-icon-size));\\n}\\n\\n.ck.ck-dropdown {\\n\\t/* Enable font size inheritance, which allows fluid UI scaling. */\\n\\tfont-size: inherit;\\n\\n\\t& .ck-dropdown__arrow {\\n\\t\\twidth: var(--ck-dropdown-arrow-size);\\n\\t}\\n\\n\\t@mixin ck-dir ltr {\\n\\t\\t& .ck-dropdown__arrow {\\n\\t\\t\\tright: var(--ck-spacing-standard);\\n\\n\\t\\t\\t/* A space to accommodate the triangle. */\\n\\t\\t\\tmargin-left: var(--ck-spacing-standard);\\n\\t\\t}\\n\\t}\\n\\n\\t@mixin ck-dir rtl {\\n\\t\\t& .ck-dropdown__arrow {\\n\\t\\t\\tleft: var(--ck-spacing-standard);\\n\\n\\t\\t\\t/* A space to accommodate the triangle. */\\n\\t\\t\\tmargin-right: var(--ck-spacing-small);\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-disabled .ck-dropdown__arrow {\\n\\t\\t@mixin ck-disabled;\\n\\t}\\n\\n\\t& .ck-button.ck-dropdown__button {\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t&:not(.ck-button_with-text) {\\n\\t\\t\\t\\t/* Make sure dropdowns with just an icon have the right inner spacing */\\n\\t\\t\\t\\tpadding-left: var(--ck-spacing-small);\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t&:not(.ck-button_with-text) {\\n\\t\\t\\t\\t/* Make sure dropdowns with just an icon have the right inner spacing */\\n\\t\\t\\t\\tpadding-right: var(--ck-spacing-small);\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t/* #23 */\\n\\t\\t& .ck-button__label {\\n\\t\\t\\twidth: 7em;\\n\\t\\t\\toverflow: hidden;\\n\\t\\t\\ttext-overflow: ellipsis;\\n\\t\\t}\\n\\n\\t\\t/* https://github.com/ckeditor/ckeditor5-theme-lark/issues/70 */\\n\\t\\t&.ck-disabled .ck-button__label {\\n\\t\\t\\t@mixin ck-disabled;\\n\\t\\t}\\n\\n\\t\\t/* https://github.com/ckeditor/ckeditor5/issues/816 */\\n\\t\\t&.ck-on {\\n\\t\\t\\tborder-bottom-left-radius: 0;\\n\\t\\t\\tborder-bottom-right-radius: 0;\\n\\t\\t}\\n\\n\\t\\t&.ck-dropdown__button_label-width_auto .ck-button__label {\\n\\t\\t\\twidth: auto;\\n\\t\\t}\\n\\n\\t\\t/* https://github.com/ckeditor/ckeditor5/issues/8699 */\\n\\t\\t&.ck-off:active,\\n\\t\\t&.ck-on:active {\\n\\t\\t\\tbox-shadow: none;\\n\\n\\t\\t\\t&:focus {\\n\\t\\t\\t\\t@mixin ck-box-shadow var(--ck-focus-outer-shadow);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\\n.ck.ck-dropdown__panel {\\n\\t@mixin ck-rounded-corners;\\n\\t@mixin ck-drop-shadow;\\n\\n\\tbackground: var(--ck-color-dropdown-panel-background);\\n\\tborder: 1px solid var(--ck-color-dropdown-panel-border);\\n\\tbottom: 0;\\n\\n\\t/* Make sure the panel is at least as wide as the drop-down's button. */\\n\\tmin-width: 100%;\\n\\n\\t/* Disabled corner border radius to be consistent with the .dropdown__button\\n\\thttps://github.com/ckeditor/ckeditor5/issues/816 */\\n\\t&.ck-dropdown__panel_se {\\n\\t\\tborder-top-left-radius: 0;\\n\\t}\\n\\n\\t&.ck-dropdown__panel_sw {\\n\\t\\tborder-top-right-radius: 0;\\n\\t}\\n\\n\\t&.ck-dropdown__panel_ne {\\n\\t\\tborder-bottom-left-radius: 0;\\n\\t}\\n\\n\\t&.ck-dropdown__panel_nw {\\n\\t\\tborder-bottom-right-radius: 0;\\n\\t}\\n\\n\\t&:focus {\\n\\t\\toutline: none;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A class which indicates that an element holding it is disabled.\\n */\\n@define-mixin ck-disabled {\\n\\topacity: var(--ck-disabled-opacity);\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A helper to combine multiple shadows.\\n */\\n@define-mixin ck-box-shadow $shadowA, $shadowB: 0 0 {\\n\\tbox-shadow: $shadowA, $shadowB;\\n}\\n\\n/**\\n * Gives an element a drop shadow so it looks like a floating panel.\\n */\\n@define-mixin ck-drop-shadow {\\n\\t@mixin ck-box-shadow var(--ck-drop-shadow);\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-dropdown .ck-dropdown__panel .ck-list{border-radius:0}.ck-rounded-corners .ck.ck-dropdown .ck-dropdown__panel .ck-list,.ck.ck-dropdown .ck-dropdown__panel .ck-list.ck-rounded-corners{border-radius:var(--ck-border-radius);border-top-left-radius:0}.ck.ck-dropdown .ck-dropdown__panel .ck-list .ck-list__item:first-child .ck-button{border-radius:0}.ck-rounded-corners .ck.ck-dropdown .ck-dropdown__panel .ck-list .ck-list__item:first-child .ck-button,.ck.ck-dropdown .ck-dropdown__panel .ck-list .ck-list__item:first-child .ck-button.ck-rounded-corners{border-radius:var(--ck-border-radius);border-bottom-left-radius:0;border-bottom-right-radius:0;border-top-left-radius:0}.ck.ck-dropdown .ck-dropdown__panel .ck-list .ck-list__item:last-child .ck-button{border-radius:0}.ck-rounded-corners .ck.ck-dropdown .ck-dropdown__panel .ck-list .ck-list__item:last-child .ck-button,.ck.ck-dropdown .ck-dropdown__panel .ck-list .ck-list__item:last-child .ck-button.ck-rounded-corners{border-radius:var(--ck-border-radius);border-top-left-radius:0;border-top-right-radius:0}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/dropdown/listdropdown.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\"],\"names\":[],\"mappings\":\"AAOA,6CCIC,eDqBD,CAzBA,iICQE,qCAAsC,CDJtC,wBAqBF,CAfE,mFCND,eDYC,CANA,6MCFA,qCAAsC,CDKpC,2BAA4B,CAC5B,4BAA6B,CAF7B,wBAIF,CAEA,kFCdD,eDmBC,CALA,2MCVA,qCAAsC,CDYpC,wBAAyB,CACzB,yBAEF\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n\\n.ck.ck-dropdown .ck-dropdown__panel .ck-list {\\n\\t/* Disabled radius of top-left border to be consistent with .dropdown__button\\n\\thttps://github.com/ckeditor/ckeditor5/issues/816 */\\n\\t@mixin ck-rounded-corners {\\n\\t\\tborder-top-left-radius: 0;\\n\\t}\\n\\n\\t/* Make sure the button belonging to the first/last child of the list goes well with the\\n\\tborder radius of the entire panel. */\\n\\t& .ck-list__item {\\n\\t\\t&:first-child .ck-button {\\n\\t\\t\\t@mixin ck-rounded-corners {\\n\\t\\t\\t\\tborder-top-left-radius: 0;\\n\\t\\t\\t\\tborder-bottom-left-radius: 0;\\n\\t\\t\\t\\tborder-bottom-right-radius: 0;\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t&:last-child .ck-button {\\n\\t\\t\\t@mixin ck-rounded-corners {\\n\\t\\t\\t\\tborder-top-left-radius: 0;\\n\\t\\t\\t\\tborder-top-right-radius: 0;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-splitbutton{font-size:inherit}.ck.ck-splitbutton .ck-splitbutton__action:focus{z-index:calc(var(--ck-z-default) + 1)}:root{--ck-color-split-button-hover-background:#ebebeb;--ck-color-split-button-hover-border:#b3b3b3}[dir=ltr] .ck.ck-splitbutton.ck-splitbutton_open>.ck-splitbutton__action,[dir=ltr] .ck.ck-splitbutton:hover>.ck-splitbutton__action{border-bottom-right-radius:unset;border-top-right-radius:unset}[dir=rtl] .ck.ck-splitbutton.ck-splitbutton_open>.ck-splitbutton__action,[dir=rtl] .ck.ck-splitbutton:hover>.ck-splitbutton__action{border-bottom-left-radius:unset;border-top-left-radius:unset}.ck.ck-splitbutton>.ck-splitbutton__arrow{min-width:unset}[dir=ltr] .ck.ck-splitbutton>.ck-splitbutton__arrow{border-bottom-left-radius:unset;border-top-left-radius:unset}[dir=rtl] .ck.ck-splitbutton>.ck-splitbutton__arrow{border-bottom-right-radius:unset;border-top-right-radius:unset}.ck.ck-splitbutton>.ck-splitbutton__arrow svg{width:var(--ck-dropdown-arrow-size)}.ck.ck-splitbutton>.ck-splitbutton__arrow:not(:focus){border-bottom-width:0;border-top-width:0}.ck.ck-splitbutton.ck-splitbutton_open>.ck-button:not(.ck-on):not(.ck-disabled):not(:hover),.ck.ck-splitbutton:hover>.ck-button:not(.ck-on):not(.ck-disabled):not(:hover){background:var(--ck-color-split-button-hover-background)}.ck.ck-splitbutton.ck-splitbutton_open>.ck-splitbutton__arrow:not(.ck-disabled):after,.ck.ck-splitbutton:hover>.ck-splitbutton__arrow:not(.ck-disabled):after{background-color:var(--ck-color-split-button-hover-border);content:\\\"\\\";height:100%;position:absolute;width:1px}.ck.ck-splitbutton.ck-splitbutton_open>.ck-splitbutton__arrow:focus:after,.ck.ck-splitbutton:hover>.ck-splitbutton__arrow:focus:after{--ck-color-split-button-hover-border:var(--ck-color-focus-border)}[dir=ltr] .ck.ck-splitbutton.ck-splitbutton_open>.ck-splitbutton__arrow:not(.ck-disabled):after,[dir=ltr] .ck.ck-splitbutton:hover>.ck-splitbutton__arrow:not(.ck-disabled):after{left:-1px}[dir=rtl] .ck.ck-splitbutton.ck-splitbutton_open>.ck-splitbutton__arrow:not(.ck-disabled):after,[dir=rtl] .ck.ck-splitbutton:hover>.ck-splitbutton__arrow:not(.ck-disabled):after{right:-1px}.ck.ck-splitbutton.ck-splitbutton_open{border-radius:0}.ck-rounded-corners .ck.ck-splitbutton.ck-splitbutton_open,.ck.ck-splitbutton.ck-splitbutton_open.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck-rounded-corners .ck.ck-splitbutton.ck-splitbutton_open>.ck-splitbutton__action,.ck.ck-splitbutton.ck-splitbutton_open.ck-rounded-corners>.ck-splitbutton__action{border-bottom-left-radius:0}.ck-rounded-corners .ck.ck-splitbutton.ck-splitbutton_open>.ck-splitbutton__arrow,.ck.ck-splitbutton.ck-splitbutton_open.ck-rounded-corners>.ck-splitbutton__arrow{border-bottom-right-radius:0}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/dropdown/splitbutton.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/dropdown/splitbutton.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\"],\"names\":[],\"mappings\":\"AAKA,mBAEC,iBAKD,CAHC,iDACC,qCACD,CCJD,MACC,gDAAyD,CACzD,4CACD,CAMC,oIAKE,gCAAiC,CADjC,6BASF,CAbA,oIAWE,+BAAgC,CADhC,4BAGF,CAEA,0CAGC,eAiBD,CApBA,oDAQE,+BAAgC,CADhC,4BAaF,CApBA,oDAcE,gCAAiC,CADjC,6BAOF,CAHC,8CACC,mCACD,CAKD,sDAEC,qBAAwB,CADxB,kBAED,CAQC,0KACC,wDACD,CAIA,8JAKC,0DAA2D,CAJ3D,UAAW,CAGX,WAAY,CAFZ,iBAAkB,CAClB,SAGD,CAGA,sIACC,iEACD,CAGC,kLACC,SACD,CAIA,kLACC,UACD,CAMF,uCCzFA,eDmGA,CAVA,qHCrFC,qCD+FD,CARE,qKACC,2BACD,CAEA,mKACC,4BACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-splitbutton {\\n\\t/* Enable font size inheritance, which allows fluid UI scaling. */\\n\\tfont-size: inherit;\\n\\n\\t& .ck-splitbutton__action:focus {\\n\\t\\tz-index: calc(var(--ck-z-default) + 1);\\n\\t}\\n}\\n\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n\\n:root {\\n\\t--ck-color-split-button-hover-background: hsl(0, 0%, 92%);\\n\\t--ck-color-split-button-hover-border: hsl(0, 0%, 70%);\\n}\\n\\n.ck.ck-splitbutton {\\n\\t/*\\n\\t * Note: ck-rounded and ck-dir mixins don't go together (because they both use @nest).\\n\\t */\\n\\t&:hover > .ck-splitbutton__action,\\n\\t&.ck-splitbutton_open > .ck-splitbutton__action {\\n\\t\\t@nest [dir=\\\"ltr\\\"] & {\\n\\t\\t\\t/* Don't round the action button on the right side */\\n\\t\\t\\tborder-top-right-radius: unset;\\n\\t\\t\\tborder-bottom-right-radius: unset;\\n\\t\\t}\\n\\n\\t\\t@nest [dir=\\\"rtl\\\"] & {\\n\\t\\t\\t/* Don't round the action button on the left side */\\n\\t\\t\\tborder-top-left-radius: unset;\\n\\t\\t\\tborder-bottom-left-radius: unset;\\n\\t\\t}\\n\\t}\\n\\n\\t& > .ck-splitbutton__arrow {\\n\\t\\t/* It's a text-less button and since the icon is positioned absolutely in such situation,\\n\\t\\tit must get some arbitrary min-width. */\\n\\t\\tmin-width: unset;\\n\\n\\t\\t@nest [dir=\\\"ltr\\\"] & {\\n\\t\\t\\t/* Don't round the arrow button on the left side */\\n\\t\\t\\tborder-top-left-radius: unset;\\n\\t\\t\\tborder-bottom-left-radius: unset;\\n\\t\\t}\\n\\n\\t\\t@nest [dir=\\\"rtl\\\"] & {\\n\\t\\t\\t/* Don't round the arrow button on the right side */\\n\\t\\t\\tborder-top-right-radius: unset;\\n\\t\\t\\tborder-bottom-right-radius: unset;\\n\\t\\t}\\n\\n\\t\\t& svg {\\n\\t\\t\\twidth: var(--ck-dropdown-arrow-size);\\n\\t\\t}\\n\\t}\\n\\n\\t/* Make sure the divider stretches 100% height of the button\\n\\thttps://github.com/ckeditor/ckeditor5/issues/10936 */\\n\\t& > .ck-splitbutton__arrow:not(:focus) {\\n\\t\\tborder-top-width: 0px;\\n\\t\\tborder-bottom-width: 0px;\\n\\t}\\n\\n\\t/* When the split button is \\\"open\\\" (the arrow is on) or being hovered, it should get some styling\\n\\tas a whole. The background of both buttons should stand out and there should be a visual\\n\\tseparation between both buttons. */\\n\\t&.ck-splitbutton_open,\\n\\t&:hover {\\n\\t\\t/* When the split button hovered as a whole, not as individual buttons. */\\n\\t\\t& > .ck-button:not(.ck-on):not(.ck-disabled):not(:hover) {\\n\\t\\t\\tbackground: var(--ck-color-split-button-hover-background);\\n\\t\\t}\\n\\n\\t\\t/* Splitbutton separator needs to be set with the ::after pseudoselector\\n\\t\\tto display properly the borders on focus */\\n\\t\\t& > .ck-splitbutton__arrow:not(.ck-disabled)::after {\\n\\t\\t\\tcontent: '';\\n\\t\\t\\tposition: absolute;\\n\\t\\t\\twidth: 1px;\\n\\t\\t\\theight: 100%;\\n\\t\\t\\tbackground-color: var(--ck-color-split-button-hover-border);\\n\\t\\t}\\n\\n\\t\\t/* Make sure the divider between the buttons looks fine when the button is focused */\\n\\t\\t& > .ck-splitbutton__arrow:focus::after {\\n\\t\\t\\t--ck-color-split-button-hover-border: var(--ck-color-focus-border);\\n\\t\\t}\\n\\n\\t\\t@nest [dir=\\\"ltr\\\"] & {\\n\\t\\t\\t& > .ck-splitbutton__arrow:not(.ck-disabled)::after {\\n\\t\\t\\t\\tleft: -1px;\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t@nest [dir=\\\"rtl\\\"] & {\\n\\t\\t\\t& > .ck-splitbutton__arrow:not(.ck-disabled)::after {\\n\\t\\t\\t\\tright: -1px;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t/* Don't round the bottom left and right corners of the buttons when \\\"open\\\"\\n\\thttps://github.com/ckeditor/ckeditor5/issues/816 */\\n\\t&.ck-splitbutton_open {\\n\\t\\t@mixin ck-rounded-corners {\\n\\t\\t\\t& > .ck-splitbutton__action {\\n\\t\\t\\t\\tborder-bottom-left-radius: 0;\\n\\t\\t\\t}\\n\\n\\t\\t\\t& > .ck-splitbutton__arrow {\\n\\t\\t\\t\\tborder-bottom-right-radius: 0;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-toolbar-dropdown-max-width:60vw}.ck.ck-toolbar-dropdown>.ck-dropdown__panel{max-width:var(--ck-toolbar-dropdown-max-width);width:max-content}.ck.ck-toolbar-dropdown>.ck-dropdown__panel .ck-button:focus{z-index:calc(var(--ck-z-default) + 1)}.ck.ck-toolbar-dropdown .ck-toolbar{border:0}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/dropdown/toolbardropdown.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/dropdown/toolbardropdown.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,oCACD,CAEA,4CAGC,8CAA+C,CAD/C,iBAQD,CAJE,6DACC,qCACD,CCZF,oCACC,QACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-toolbar-dropdown-max-width: 60vw;\\n}\\n\\n.ck.ck-toolbar-dropdown > .ck-dropdown__panel {\\n\\t/* https://github.com/ckeditor/ckeditor5/issues/5586 */\\n\\twidth: max-content;\\n\\tmax-width: var(--ck-toolbar-dropdown-max-width);\\n\\n\\t& .ck-button {\\n\\t\\t&:focus {\\n\\t\\t\\tz-index: calc(var(--ck-z-default) + 1);\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-toolbar-dropdown .ck-toolbar {\\n\\tborder: 0;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-color-editable-blur-selection:#d9d9d9}.ck.ck-editor__editable:not(.ck-editor__nested-editable){border-radius:0}.ck-rounded-corners .ck.ck-editor__editable:not(.ck-editor__nested-editable),.ck.ck-editor__editable.ck-rounded-corners:not(.ck-editor__nested-editable){border-radius:var(--ck-border-radius)}.ck.ck-editor__editable.ck-focused:not(.ck-editor__nested-editable){border:var(--ck-focus-ring);box-shadow:var(--ck-inner-shadow),0 0;outline:none}.ck.ck-editor__editable_inline{border:1px solid transparent;overflow:auto;padding:0 var(--ck-spacing-standard)}.ck.ck-editor__editable_inline[dir=ltr]{text-align:left}.ck.ck-editor__editable_inline[dir=rtl]{text-align:right}.ck.ck-editor__editable_inline>:first-child{margin-top:var(--ck-spacing-large)}.ck.ck-editor__editable_inline>:last-child{margin-bottom:var(--ck-spacing-large)}.ck.ck-editor__editable_inline.ck-blurred ::selection{background:var(--ck-color-editable-blur-selection)}.ck.ck-balloon-panel.ck-toolbar-container[class*=arrow_n]:after{border-bottom-color:var(--ck-color-panel-background)}.ck.ck-balloon-panel.ck-toolbar-container[class*=arrow_s]:after{border-top-color:var(--ck-color-panel-background)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/editorui/editorui.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_focus.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_shadow.css\"],\"names\":[],\"mappings\":\"AAWA,MACC,0CACD,CAEA,yDCJC,eDWD,CAPA,yJCAE,qCDOF,CAJC,oEEPA,2BAA2B,CCF3B,qCAA8B,CDC9B,YFWA,CAGD,+BAGC,4BAA6B,CAF7B,aAAc,CACd,oCA6BD,CA1BC,wCACC,eACD,CAEA,wCACC,gBACD,CAGA,4CACC,kCACD,CAGA,2CAKC,qCACD,CAGA,sDACC,kDACD,CAKA,gEACC,oDACD,CAIA,gEACC,iDACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n@import \\\"../../../mixins/_disabled.css\\\";\\n@import \\\"../../../mixins/_shadow.css\\\";\\n@import \\\"../../../mixins/_focus.css\\\";\\n@import \\\"../../mixins/_button.css\\\";\\n\\n:root {\\n\\t--ck-color-editable-blur-selection: hsl(0, 0%, 85%);\\n}\\n\\n.ck.ck-editor__editable:not(.ck-editor__nested-editable) {\\n\\t@mixin ck-rounded-corners;\\n\\n\\t&.ck-focused {\\n\\t\\t@mixin ck-focus-ring;\\n\\t\\t@mixin ck-box-shadow var(--ck-inner-shadow);\\n\\t}\\n}\\n\\n.ck.ck-editor__editable_inline {\\n\\toverflow: auto;\\n\\tpadding: 0 var(--ck-spacing-standard);\\n\\tborder: 1px solid transparent;\\n\\n\\t&[dir=\\\"ltr\\\"] {\\n\\t\\ttext-align: left;\\n\\t}\\n\\n\\t&[dir=\\\"rtl\\\"] {\\n\\t\\ttext-align: right;\\n\\t}\\n\\n\\t/* https://github.com/ckeditor/ckeditor5-theme-lark/issues/116 */\\n\\t& > *:first-child {\\n\\t\\tmargin-top: var(--ck-spacing-large);\\n\\t}\\n\\n\\t/* https://github.com/ckeditor/ckeditor5/issues/847 */\\n\\t& > *:last-child {\\n\\t\\t/*\\n\\t\\t * This value should match with the default margins of the block elements (like .media or .image)\\n\\t\\t * to avoid a content jumping when the fake selection container shows up (See https://github.com/ckeditor/ckeditor5/issues/9825).\\n\\t\\t */\\n\\t\\tmargin-bottom: var(--ck-spacing-large);\\n\\t}\\n\\n\\t/* https://github.com/ckeditor/ckeditor5/issues/6517 */\\n\\t&.ck-blurred ::selection {\\n\\t\\tbackground: var(--ck-color-editable-blur-selection);\\n\\t}\\n}\\n\\n/* https://github.com/ckeditor/ckeditor5-theme-lark/issues/111 */\\n.ck.ck-balloon-panel.ck-toolbar-container[class*=\\\"arrow_n\\\"] {\\n\\t&::after {\\n\\t\\tborder-bottom-color: var(--ck-color-panel-background);\\n\\t}\\n}\\n\\n.ck.ck-balloon-panel.ck-toolbar-container[class*=\\\"arrow_s\\\"] {\\n\\t&::after {\\n\\t\\tborder-top-color: var(--ck-color-panel-background);\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A visual style of focused element's border.\\n */\\n@define-mixin ck-focus-ring {\\n\\t/* Disable native outline. */\\n\\toutline: none;\\n\\tborder: var(--ck-focus-ring)\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A helper to combine multiple shadows.\\n */\\n@define-mixin ck-box-shadow $shadowA, $shadowB: 0 0 {\\n\\tbox-shadow: $shadowA, $shadowB;\\n}\\n\\n/**\\n * Gives an element a drop shadow so it looks like a floating panel.\\n */\\n@define-mixin ck-drop-shadow {\\n\\t@mixin ck-box-shadow var(--ck-drop-shadow);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-form__header{align-items:center;display:flex;flex-direction:row;flex-wrap:nowrap;justify-content:space-between}:root{--ck-form-header-height:38px}.ck.ck-form__header{border-bottom:1px solid var(--ck-color-base-border);height:var(--ck-form-header-height);line-height:var(--ck-form-header-height);padding:var(--ck-spacing-small) var(--ck-spacing-large)}.ck.ck-form__header .ck-form__header__label{font-weight:700}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/formheader/formheader.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/formheader/formheader.css\"],\"names\":[],\"mappings\":\"AAKA,oBAIC,kBAAmB,CAHnB,YAAa,CACb,kBAAmB,CACnB,gBAAiB,CAEjB,6BACD,CCNA,MACC,4BACD,CAEA,oBAIC,mDAAoD,CAFpD,mCAAoC,CACpC,wCAAyC,CAFzC,uDAQD,CAHC,4CACC,eACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-form__header {\\n\\tdisplay: flex;\\n\\tflex-direction: row;\\n\\tflex-wrap: nowrap;\\n\\talign-items: center;\\n\\tjustify-content: space-between;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-form-header-height: 38px;\\n}\\n\\n.ck.ck-form__header {\\n\\tpadding: var(--ck-spacing-small) var(--ck-spacing-large);\\n\\theight: var(--ck-form-header-height);\\n\\tline-height: var(--ck-form-header-height);\\n\\tborder-bottom: 1px solid var(--ck-color-base-border);\\n\\n\\t& .ck-form__header__label {\\n\\t\\tfont-weight: bold;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-icon{vertical-align:middle}:root{--ck-icon-size:calc(var(--ck-line-height-base)*var(--ck-font-size-normal))}.ck.ck-icon{font-size:.8333350694em;height:var(--ck-icon-size);width:var(--ck-icon-size);will-change:transform}.ck.ck-icon,.ck.ck-icon *{cursor:inherit}.ck.ck-icon.ck-icon_inherit-color,.ck.ck-icon.ck-icon_inherit-color *{color:inherit}.ck.ck-icon.ck-icon_inherit-color :not([fill]){fill:currentColor}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/icon/icon.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/icon/icon.css\"],\"names\":[],\"mappings\":\"AAKA,YACC,qBACD,CCFA,MACC,0EACD,CAEA,YAKC,uBAAwB,CAHxB,0BAA2B,CAD3B,yBAA0B,CAU1B,qBAoBD,CAlBC,0BALA,cAQA,CAMC,sEACC,aAMD,CAJC,+CAEC,iBACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-icon {\\n\\tvertical-align: middle;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-icon-size: calc(var(--ck-line-height-base) * var(--ck-font-size-normal));\\n}\\n\\n.ck.ck-icon {\\n\\twidth: var(--ck-icon-size);\\n\\theight: var(--ck-icon-size);\\n\\n\\t/* Multiplied by the height of the line in \\\"px\\\" should give SVG \\\"viewport\\\" dimensions */\\n\\tfont-size: .8333350694em;\\n\\n\\t/* Inherit cursor style (#5). */\\n\\tcursor: inherit;\\n\\n\\t/* This will prevent blurry icons on Firefox. See #340. */\\n\\twill-change: transform;\\n\\n\\t& * {\\n\\t\\t/* Inherit cursor style (#5). */\\n\\t\\tcursor: inherit;\\n\\t}\\n\\n\\t/* Allows dynamic coloring of an icon by inheriting its color from the parent. */\\n\\t&.ck-icon_inherit-color {\\n\\t\\tcolor: inherit;\\n\\n\\t\\t& * {\\n\\t\\t\\tcolor: inherit;\\n\\n\\t\\t\\t&:not([fill]) {\\n\\t\\t\\t\\t/* Needed by FF. */\\n\\t\\t\\t\\tfill: currentColor;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-input-width:18em;--ck-input-text-width:var(--ck-input-width)}.ck.ck-input{border-radius:0}.ck-rounded-corners .ck.ck-input,.ck.ck-input.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-input{background:var(--ck-color-input-background);border:1px solid var(--ck-color-input-border);min-height:var(--ck-ui-component-min-height);min-width:var(--ck-input-width);padding:var(--ck-spacing-extra-tiny) var(--ck-spacing-medium);transition:box-shadow .1s ease-in-out,border .1s ease-in-out}.ck.ck-input:focus{border:var(--ck-focus-ring);box-shadow:var(--ck-focus-outer-shadow),0 0;outline:none}.ck.ck-input[readonly]{background:var(--ck-color-input-disabled-background);border:1px solid var(--ck-color-input-disabled-border);color:var(--ck-color-input-disabled-text)}.ck.ck-input[readonly]:focus{box-shadow:var(--ck-focus-disabled-outer-shadow),0 0}.ck.ck-input.ck-error{animation:ck-input-shake .3s ease both;border-color:var(--ck-color-input-error-border)}.ck.ck-input.ck-error:focus{box-shadow:var(--ck-focus-error-outer-shadow),0 0}@keyframes ck-input-shake{20%{transform:translateX(-2px)}40%{transform:translateX(2px)}60%{transform:translateX(-1px)}80%{transform:translateX(1px)}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/input/input.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_focus.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_shadow.css\"],\"names\":[],\"mappings\":\"AASA,MACC,qBAAsB,CAGtB,2CACD,CAEA,aCLC,eD2CD,CAtCA,iECDE,qCDuCF,CAtCA,aAGC,2CAA4C,CAC5C,6CAA8C,CAK9C,4CAA6C,CAH7C,+BAAgC,CADhC,6DAA8D,CAO9D,4DA0BD,CAxBC,mBEnBA,2BAA2B,CCF3B,2CAA8B,CDC9B,YFuBA,CAEA,uBAEC,oDAAqD,CADrD,sDAAuD,CAEvD,yCAMD,CAJC,6BG/BD,oDHkCC,CAGD,sBAEC,sCAAuC,CADvC,+CAMD,CAHC,4BGzCD,iDH2CC,CAIF,0BACC,IACC,0BACD,CAEA,IACC,yBACD,CAEA,IACC,0BACD,CAEA,IACC,yBACD,CACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n@import \\\"../../../mixins/_focus.css\\\";\\n@import \\\"../../../mixins/_shadow.css\\\";\\n\\n:root {\\n\\t--ck-input-width: 18em;\\n\\n\\t/* Backward compatibility. */\\n\\t--ck-input-text-width: var(--ck-input-width);\\n}\\n\\n.ck.ck-input {\\n\\t@mixin ck-rounded-corners;\\n\\n\\tbackground: var(--ck-color-input-background);\\n\\tborder: 1px solid var(--ck-color-input-border);\\n\\tpadding: var(--ck-spacing-extra-tiny) var(--ck-spacing-medium);\\n\\tmin-width: var(--ck-input-width);\\n\\n\\t/* This is important to stay of the same height as surrounding buttons */\\n\\tmin-height: var(--ck-ui-component-min-height);\\n\\n\\t/* Apply some smooth transition to the box-shadow and border. */\\n\\ttransition: box-shadow .1s ease-in-out, border .1s ease-in-out;\\n\\n\\t&:focus {\\n\\t\\t@mixin ck-focus-ring;\\n\\t\\t@mixin ck-box-shadow var(--ck-focus-outer-shadow);\\n\\t}\\n\\n\\t&[readonly] {\\n\\t\\tborder: 1px solid var(--ck-color-input-disabled-border);\\n\\t\\tbackground: var(--ck-color-input-disabled-background);\\n\\t\\tcolor: var(--ck-color-input-disabled-text);\\n\\n\\t\\t&:focus {\\n\\t\\t\\t/* The read-only input should have a slightly less visible shadow when focused. */\\n\\t\\t\\t@mixin ck-box-shadow var(--ck-focus-disabled-outer-shadow);\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-error {\\n\\t\\tborder-color: var(--ck-color-input-error-border);\\n\\t\\tanimation: ck-input-shake .3s ease both;\\n\\n\\t\\t&:focus {\\n\\t\\t\\t@mixin ck-box-shadow var(--ck-focus-error-outer-shadow);\\n\\t\\t}\\n\\t}\\n}\\n\\n@keyframes ck-input-shake {\\n\\t20% {\\n\\t\\ttransform: translateX(-2px);\\n\\t}\\n\\n\\t40% {\\n\\t\\ttransform: translateX(2px);\\n\\t}\\n\\n\\t60% {\\n\\t\\ttransform: translateX(-1px);\\n\\t}\\n\\n\\t80% {\\n\\t\\ttransform: translateX(1px);\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A visual style of focused element's border.\\n */\\n@define-mixin ck-focus-ring {\\n\\t/* Disable native outline. */\\n\\toutline: none;\\n\\tborder: var(--ck-focus-ring)\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A helper to combine multiple shadows.\\n */\\n@define-mixin ck-box-shadow $shadowA, $shadowB: 0 0 {\\n\\tbox-shadow: $shadowA, $shadowB;\\n}\\n\\n/**\\n * Gives an element a drop shadow so it looks like a floating panel.\\n */\\n@define-mixin ck-drop-shadow {\\n\\t@mixin ck-box-shadow var(--ck-drop-shadow);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-label{display:block}.ck.ck-voice-label{display:none}.ck.ck-label{font-weight:700}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/label/label.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/label/label.css\"],\"names\":[],\"mappings\":\"AAKA,aACC,aACD,CAEA,mBACC,YACD,CCNA,aACC,eACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-label {\\n\\tdisplay: block;\\n}\\n\\n.ck.ck-voice-label {\\n\\tdisplay: none;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-label {\\n\\tfont-weight: bold;\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-labeled-field-view>.ck.ck-labeled-field-view__input-wrapper{display:flex;position:relative}.ck.ck-labeled-field-view .ck.ck-label{display:block;position:absolute}:root{--ck-labeled-field-view-transition:.1s cubic-bezier(0,0,0.24,0.95);--ck-labeled-field-empty-unfocused-max-width:100% - 2 * var(--ck-spacing-medium);--ck-labeled-field-label-default-position-x:var(--ck-spacing-medium);--ck-labeled-field-label-default-position-y:calc(var(--ck-font-size-base)*0.6);--ck-color-labeled-field-label-background:var(--ck-color-base-background)}.ck.ck-labeled-field-view{border-radius:0}.ck-rounded-corners .ck.ck-labeled-field-view,.ck.ck-labeled-field-view.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-labeled-field-view>.ck.ck-labeled-field-view__input-wrapper{width:100%}.ck.ck-labeled-field-view>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{top:0}[dir=ltr] .ck.ck-labeled-field-view>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{left:0}[dir=rtl] .ck.ck-labeled-field-view>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{right:0}.ck.ck-labeled-field-view>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{background:var(--ck-color-labeled-field-label-background);font-weight:400;line-height:normal;max-width:100%;overflow:hidden;padding:0 calc(var(--ck-font-size-tiny)*.5);pointer-events:none;text-overflow:ellipsis;transform:translate(var(--ck-spacing-medium),-6px) scale(.75);transform-origin:0 0;transition:transform var(--ck-labeled-field-view-transition),padding var(--ck-labeled-field-view-transition),background var(--ck-labeled-field-view-transition)}.ck.ck-labeled-field-view.ck-error .ck-input:not([readonly])+.ck.ck-label,.ck.ck-labeled-field-view.ck-error>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{color:var(--ck-color-base-error)}.ck.ck-labeled-field-view .ck-labeled-field-view__status{font-size:var(--ck-font-size-small);margin-top:var(--ck-spacing-small);white-space:normal}.ck.ck-labeled-field-view .ck-labeled-field-view__status.ck-labeled-field-view__status_error{color:var(--ck-color-base-error)}.ck.ck-labeled-field-view.ck-disabled>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label,.ck.ck-labeled-field-view.ck-labeled-field-view_empty:not(.ck-labeled-field-view_focused)>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{color:var(--ck-color-input-disabled-text)}[dir=ltr] .ck.ck-labeled-field-view.ck-disabled.ck-labeled-field-view_empty>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label,[dir=ltr] .ck.ck-labeled-field-view.ck-labeled-field-view_empty:not(.ck-labeled-field-view_focused):not(.ck-labeled-field-view_placeholder)>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{transform:translate(var(--ck-labeled-field-label-default-position-x),var(--ck-labeled-field-label-default-position-y)) scale(1)}[dir=rtl] .ck.ck-labeled-field-view.ck-disabled.ck-labeled-field-view_empty>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label,[dir=rtl] .ck.ck-labeled-field-view.ck-labeled-field-view_empty:not(.ck-labeled-field-view_focused):not(.ck-labeled-field-view_placeholder)>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{transform:translate(calc(var(--ck-labeled-field-label-default-position-x)*-1),var(--ck-labeled-field-label-default-position-y)) scale(1)}.ck.ck-labeled-field-view.ck-disabled.ck-labeled-field-view_empty>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label,.ck.ck-labeled-field-view.ck-labeled-field-view_empty:not(.ck-labeled-field-view_focused):not(.ck-labeled-field-view_placeholder)>.ck.ck-labeled-field-view__input-wrapper>.ck.ck-label{background:transparent;max-width:calc(var(--ck-labeled-field-empty-unfocused-max-width));padding:0}.ck.ck-labeled-field-view>.ck.ck-labeled-field-view__input-wrapper>.ck-dropdown>.ck.ck-button{background:transparent}.ck.ck-labeled-field-view.ck-labeled-field-view_empty>.ck.ck-labeled-field-view__input-wrapper>.ck-dropdown>.ck-button>.ck-button__label{opacity:0}.ck.ck-labeled-field-view.ck-labeled-field-view_empty:not(.ck-labeled-field-view_focused):not(.ck-labeled-field-view_placeholder)>.ck.ck-labeled-field-view__input-wrapper>.ck-dropdown+.ck-label{max-width:calc(var(--ck-labeled-field-empty-unfocused-max-width) - var(--ck-dropdown-arrow-size) - var(--ck-spacing-standard))}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/labeledfield/labeledfieldview.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/labeledfield/labeledfieldview.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\"],\"names\":[],\"mappings\":\"AAMC,mEACC,YAAa,CACb,iBACD,CAEA,uCACC,aAAc,CACd,iBACD,CCND,MACC,kEAAsE,CACtE,gFAAiF,CACjF,oEAAqE,CACrE,8EAAiF,CACjF,yEACD,CAEA,0BCLC,eD8GD,CAzGA,2FCDE,qCD0GF,CAtGC,mEACC,UAmCD,CAjCC,gFACC,KA+BD,CAhCA,0FAIE,MA4BF,CAhCA,0FAQE,OAwBF,CAhCA,gFAiBC,yDAA0D,CAG1D,eAAmB,CADnB,kBAAoB,CAOpB,cAAe,CAFf,eAAgB,CANhB,2CAA8C,CAP9C,mBAAoB,CAYpB,sBAAuB,CARvB,6DAA+D,CAH/D,oBAAqB,CAgBrB,+JAID,CAQA,mKACC,gCACD,CAGD,yDACC,mCAAoC,CACpC,kCAAmC,CAInC,kBAKD,CAHC,6FACC,gCACD,CAID,4OAEC,yCACD,CAIA,oUAGE,+HAYF,CAfA,oUAOE,wIAQF,CAfA,gTAaC,sBAAuB,CAFvB,iEAAkE,CAGlE,SACD,CAKA,8FACC,sBACD,CAGA,yIACC,SACD,CAGA,kMACC,8HACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-labeled-field-view {\\n\\t& > .ck.ck-labeled-field-view__input-wrapper {\\n\\t\\tdisplay: flex;\\n\\t\\tposition: relative;\\n\\t}\\n\\n\\t& .ck.ck-label {\\n\\t\\tdisplay: block;\\n\\t\\tposition: absolute;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n@import \\\"../../../mixins/_rounded.css\\\";\\n\\n:root {\\n\\t--ck-labeled-field-view-transition: .1s cubic-bezier(0, 0, 0.24, 0.95);\\n\\t--ck-labeled-field-empty-unfocused-max-width: 100% - 2 * var(--ck-spacing-medium);\\n\\t--ck-labeled-field-label-default-position-x: var(--ck-spacing-medium);\\n\\t--ck-labeled-field-label-default-position-y: calc(0.6 * var(--ck-font-size-base));\\n\\t--ck-color-labeled-field-label-background: var(--ck-color-base-background);\\n}\\n\\n.ck.ck-labeled-field-view {\\n\\t@mixin ck-rounded-corners;\\n\\n\\t& > .ck.ck-labeled-field-view__input-wrapper {\\n\\t\\twidth: 100%;\\n\\n\\t\\t& > .ck.ck-label {\\n\\t\\t\\ttop: 0px;\\n\\n\\t\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t\\tleft: 0px;\\n\\t\\t\\t}\\n\\n\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\tright: 0px;\\n\\t\\t\\t}\\n\\n\\t\\t\\tpointer-events: none;\\n\\t\\t\\ttransform-origin: 0 0;\\n\\n\\t\\t\\t/* By default, display the label scaled down above the field. */\\n\\t\\t\\ttransform: translate(var(--ck-spacing-medium), -6px) scale(.75);\\n\\n\\t\\t\\tbackground: var(--ck-color-labeled-field-label-background);\\n\\t\\t\\tpadding: 0 calc(.5 * var(--ck-font-size-tiny));\\n\\t\\t\\tline-height: initial;\\n\\t\\t\\tfont-weight: normal;\\n\\n\\t\\t\\t/* Prevent overflow when the label is longer than the input */\\n\\t\\t\\ttext-overflow: ellipsis;\\n\\t\\t\\toverflow: hidden;\\n\\n\\t\\t\\tmax-width: 100%;\\n\\n\\t\\t\\ttransition:\\n\\t\\t\\t\\ttransform var(--ck-labeled-field-view-transition),\\n\\t\\t\\t\\tpadding var(--ck-labeled-field-view-transition),\\n\\t\\t\\t\\tbackground var(--ck-labeled-field-view-transition);\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-error {\\n\\t\\t& > .ck.ck-labeled-field-view__input-wrapper > .ck.ck-label {\\n\\t\\t\\tcolor: var(--ck-color-base-error);\\n\\t\\t}\\n\\n\\t\\t& .ck-input:not([readonly]) + .ck.ck-label {\\n\\t\\t\\tcolor: var(--ck-color-base-error);\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck-labeled-field-view__status {\\n\\t\\tfont-size: var(--ck-font-size-small);\\n\\t\\tmargin-top: var(--ck-spacing-small);\\n\\n\\t\\t/* Let the info wrap to the next line to avoid stretching the layout horizontally.\\n\\t\\tThe status could be very long. */\\n\\t\\twhite-space: normal;\\n\\n\\t\\t&.ck-labeled-field-view__status_error {\\n\\t\\t\\tcolor: var(--ck-color-base-error);\\n\\t\\t}\\n\\t}\\n\\n\\t/* Disabled fields and fields that have no focus should fade out. */\\n\\t&.ck-disabled > .ck.ck-labeled-field-view__input-wrapper > .ck.ck-label,\\n\\t&.ck-labeled-field-view_empty:not(.ck-labeled-field-view_focused) > .ck.ck-labeled-field-view__input-wrapper > .ck.ck-label {\\n\\t\\tcolor: var(--ck-color-input-disabled-text);\\n\\t}\\n\\n\\t/* Fields that are disabled or not focused and without a placeholder should have full-sized labels. */\\n\\t/* stylelint-disable-next-line no-descending-specificity */\\n\\t&.ck-disabled.ck-labeled-field-view_empty > .ck.ck-labeled-field-view__input-wrapper > .ck.ck-label,\\n\\t&.ck-labeled-field-view_empty:not(.ck-labeled-field-view_focused):not(.ck-labeled-field-view_placeholder) > .ck.ck-labeled-field-view__input-wrapper > .ck.ck-label {\\n\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\ttransform: translate(var(--ck-labeled-field-label-default-position-x), var(--ck-labeled-field-label-default-position-y)) scale(1);\\n\\t\\t}\\n\\n\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\ttransform: translate(calc(-1 * var(--ck-labeled-field-label-default-position-x)), var(--ck-labeled-field-label-default-position-y)) scale(1);\\n\\t\\t}\\n\\n\\t\\t/* Compensate for the default translate position. */\\n\\t\\tmax-width: calc(var(--ck-labeled-field-empty-unfocused-max-width));\\n\\n\\t\\tbackground: transparent;\\n\\t\\tpadding: 0;\\n\\t}\\n\\n\\t/*------ DropdownView integration ----------------------------------------------------------------------------------- */\\n\\n\\t/* Make sure dropdown' background color in any of dropdown's state does not collide with labeled field. */\\n\\t& > .ck.ck-labeled-field-view__input-wrapper > .ck-dropdown > .ck.ck-button {\\n\\t\\tbackground: transparent;\\n\\t}\\n\\n\\t/* When the dropdown is \\\"empty\\\", the labeled field label replaces its label. */\\n\\t&.ck-labeled-field-view_empty > .ck.ck-labeled-field-view__input-wrapper > .ck-dropdown > .ck-button > .ck-button__label {\\n\\t\\topacity: 0;\\n\\t}\\n\\n\\t/* Make sure the label of the empty, unfocused input does not cover the dropdown arrow. */\\n\\t&.ck-labeled-field-view_empty:not(.ck-labeled-field-view_focused):not(.ck-labeled-field-view_placeholder) > .ck.ck-labeled-field-view__input-wrapper > .ck-dropdown + .ck-label {\\n\\t\\tmax-width: calc(var(--ck-labeled-field-empty-unfocused-max-width) - var(--ck-dropdown-arrow-size) - var(--ck-spacing-standard));\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-list{display:flex;flex-direction:column;-moz-user-select:none;-webkit-user-select:none;-ms-user-select:none;user-select:none}.ck.ck-list .ck-list__item,.ck.ck-list .ck-list__separator{display:block}.ck.ck-list .ck-list__item>:focus{position:relative;z-index:var(--ck-z-default)}.ck.ck-list{border-radius:0}.ck-rounded-corners .ck.ck-list,.ck.ck-list.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-list{background:var(--ck-color-list-background);list-style-type:none}.ck.ck-list__item{cursor:default;min-width:12em}.ck.ck-list__item .ck-button{border-radius:0;min-height:unset;padding:calc(var(--ck-line-height-base)*.2*var(--ck-font-size-base)) calc(var(--ck-line-height-base)*.4*var(--ck-font-size-base));text-align:left;width:100%}.ck.ck-list__item .ck-button .ck-button__label{line-height:calc(var(--ck-line-height-base)*1.2*var(--ck-font-size-base))}.ck.ck-list__item .ck-button:active{box-shadow:none}.ck.ck-list__item .ck-button.ck-on{background:var(--ck-color-list-button-on-background);color:var(--ck-color-list-button-on-text)}.ck.ck-list__item .ck-button.ck-on:active{box-shadow:none}.ck.ck-list__item .ck-button.ck-on:hover:not(.ck-disabled){background:var(--ck-color-list-button-on-background-focus)}.ck.ck-list__item .ck-button.ck-on:focus:not(.ck-switchbutton):not(.ck-disabled){border-color:var(--ck-color-base-background)}.ck.ck-list__item .ck-button:hover:not(.ck-disabled){background:var(--ck-color-list-button-hover-background)}.ck.ck-list__item .ck-switchbutton.ck-on{background:var(--ck-color-list-background);color:inherit}.ck.ck-list__item .ck-switchbutton.ck-on:hover:not(.ck-disabled){background:var(--ck-color-list-button-hover-background);color:inherit}.ck.ck-list__separator{background:var(--ck-color-base-border);height:1px;width:100%}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/list/list.css\",\"webpack://./../ckeditor5-ui/theme/mixins/_unselectable.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/list/list.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\"],\"names\":[],\"mappings\":\"AAOA,YAGC,YAAa,CACb,qBAAsB,CCFtB,qBAAsB,CACtB,wBAAyB,CACzB,oBAAqB,CACrB,gBDaD,CAZC,2DAEC,aACD,CAKA,kCACC,iBAAkB,CAClB,2BACD,CEfD,YCEC,eDGD,CALA,+DCME,qCDDF,CALA,YAIC,0CAA2C,CAD3C,oBAED,CAEA,kBACC,cAAe,CACf,cA2DD,CAzDC,6BAIC,eAAgB,CAHhB,gBAAiB,CAQjB,iIAEiE,CARjE,eAAgB,CADhB,UAwCD,CA7BC,+CAEC,yEACD,CAEA,oCACC,eACD,CAEA,mCACC,oDAAqD,CACrD,yCAaD,CAXC,0CACC,eACD,CAEA,2DACC,0DACD,CAEA,iFACC,4CACD,CAGD,qDACC,uDACD,CAMA,yCACC,0CAA2C,CAC3C,aAMD,CAJC,iEACC,uDAAwD,CACxD,aACD,CAKH,uBAGC,sCAAuC,CAFvC,UAAW,CACX,UAED\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../mixins/_unselectable.css\\\";\\n\\n.ck.ck-list {\\n\\t@mixin ck-unselectable;\\n\\n\\tdisplay: flex;\\n\\tflex-direction: column;\\n\\n\\t& .ck-list__item,\\n\\t& .ck-list__separator {\\n\\t\\tdisplay: block;\\n\\t}\\n\\n\\t/* Make sure that whatever child of the list item gets focus, it remains on the\\n\\ttop. Thanks to that, styles like box-shadow, outline, etc. are not masked by\\n\\tadjacent list items. */\\n\\t& .ck-list__item > *:focus {\\n\\t\\tposition: relative;\\n\\t\\tz-index: var(--ck-z-default);\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Makes element unselectable.\\n */\\n@define-mixin ck-unselectable {\\n\\t-moz-user-select: none;\\n\\t-webkit-user-select: none;\\n\\t-ms-user-select: none;\\n\\tuser-select: none\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_disabled.css\\\";\\n@import \\\"../../../mixins/_rounded.css\\\";\\n@import \\\"../../../mixins/_shadow.css\\\";\\n\\n.ck.ck-list {\\n\\t@mixin ck-rounded-corners;\\n\\n\\tlist-style-type: none;\\n\\tbackground: var(--ck-color-list-background);\\n}\\n\\n.ck.ck-list__item {\\n\\tcursor: default;\\n\\tmin-width: 12em;\\n\\n\\t& .ck-button {\\n\\t\\tmin-height: unset;\\n\\t\\twidth: 100%;\\n\\t\\ttext-align: left;\\n\\t\\tborder-radius: 0;\\n\\n\\t\\t/* List items should have the same height. Use absolute units to make sure it is so\\n\\t\\t because e.g. different heading styles may have different height\\n\\t\\t https://github.com/ckeditor/ckeditor5-heading/issues/63 */\\n\\t\\tpadding:\\n\\t\\t\\tcalc(.2 * var(--ck-line-height-base) * var(--ck-font-size-base))\\n\\t\\t\\tcalc(.4 * var(--ck-line-height-base) * var(--ck-font-size-base));\\n\\n\\t\\t& .ck-button__label {\\n\\t\\t\\t/* https://github.com/ckeditor/ckeditor5-heading/issues/63 */\\n\\t\\t\\tline-height: calc(1.2 * var(--ck-line-height-base) * var(--ck-font-size-base));\\n\\t\\t}\\n\\n\\t\\t&:active {\\n\\t\\t\\tbox-shadow: none;\\n\\t\\t}\\n\\n\\t\\t&.ck-on {\\n\\t\\t\\tbackground: var(--ck-color-list-button-on-background);\\n\\t\\t\\tcolor: var(--ck-color-list-button-on-text);\\n\\n\\t\\t\\t&:active {\\n\\t\\t\\t\\tbox-shadow: none;\\n\\t\\t\\t}\\n\\n\\t\\t\\t&:hover:not(.ck-disabled) {\\n\\t\\t\\t\\tbackground: var(--ck-color-list-button-on-background-focus);\\n\\t\\t\\t}\\n\\n\\t\\t\\t&:focus:not(.ck-switchbutton):not(.ck-disabled) {\\n\\t\\t\\t\\tborder-color: var(--ck-color-base-background);\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t&:hover:not(.ck-disabled) {\\n\\t\\t\\tbackground: var(--ck-color-list-button-hover-background);\\n\\t\\t}\\n\\t}\\n\\n\\t/* It's unnecessary to change the background/text of a switch toggle; it has different ways\\n\\tof conveying its state (like the switcher) */\\n\\t& .ck-switchbutton {\\n\\t\\t&.ck-on {\\n\\t\\t\\tbackground: var(--ck-color-list-background);\\n\\t\\t\\tcolor: inherit;\\n\\n\\t\\t\\t&:hover:not(.ck-disabled) {\\n\\t\\t\\t\\tbackground: var(--ck-color-list-button-hover-background);\\n\\t\\t\\t\\tcolor: inherit;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\\n.ck.ck-list__separator {\\n\\theight: 1px;\\n\\twidth: 100%;\\n\\tbackground: var(--ck-color-base-border);\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-balloon-panel-arrow-z-index:calc(var(--ck-z-default) - 3)}.ck.ck-balloon-panel{display:none;position:absolute;z-index:var(--ck-z-modal)}.ck.ck-balloon-panel.ck-balloon-panel_with-arrow:after,.ck.ck-balloon-panel.ck-balloon-panel_with-arrow:before{content:\\\"\\\";position:absolute}.ck.ck-balloon-panel.ck-balloon-panel_with-arrow:before{z-index:var(--ck-balloon-panel-arrow-z-index)}.ck.ck-balloon-panel.ck-balloon-panel_with-arrow:after{z-index:calc(var(--ck-balloon-panel-arrow-z-index) + 1)}.ck.ck-balloon-panel[class*=arrow_n]:before{z-index:var(--ck-balloon-panel-arrow-z-index)}.ck.ck-balloon-panel[class*=arrow_n]:after{z-index:calc(var(--ck-balloon-panel-arrow-z-index) + 1)}.ck.ck-balloon-panel[class*=arrow_s]:before{z-index:var(--ck-balloon-panel-arrow-z-index)}.ck.ck-balloon-panel[class*=arrow_s]:after{z-index:calc(var(--ck-balloon-panel-arrow-z-index) + 1)}.ck.ck-balloon-panel.ck-balloon-panel_visible{display:block}:root{--ck-balloon-border-width:1px;--ck-balloon-arrow-offset:2px;--ck-balloon-arrow-height:10px;--ck-balloon-arrow-half-width:8px;--ck-balloon-arrow-drop-shadow:0 2px 2px var(--ck-color-shadow-drop)}.ck.ck-balloon-panel{border-radius:0}.ck-rounded-corners .ck.ck-balloon-panel,.ck.ck-balloon-panel.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-balloon-panel{background:var(--ck-color-panel-background);border:var(--ck-balloon-border-width) solid var(--ck-color-panel-border);box-shadow:var(--ck-drop-shadow),0 0;min-height:15px}.ck.ck-balloon-panel.ck-balloon-panel_with-arrow:after,.ck.ck-balloon-panel.ck-balloon-panel_with-arrow:before{border-style:solid;height:0;width:0}.ck.ck-balloon-panel[class*=arrow_n]:after,.ck.ck-balloon-panel[class*=arrow_n]:before{border-width:0 var(--ck-balloon-arrow-half-width) var(--ck-balloon-arrow-height) var(--ck-balloon-arrow-half-width)}.ck.ck-balloon-panel[class*=arrow_n]:before{border-color:transparent transparent var(--ck-color-panel-border) transparent;margin-top:calc(var(--ck-balloon-border-width)*-1)}.ck.ck-balloon-panel[class*=arrow_n]:after{border-color:transparent transparent var(--ck-color-panel-background) transparent;margin-top:calc(var(--ck-balloon-arrow-offset) - var(--ck-balloon-border-width))}.ck.ck-balloon-panel[class*=arrow_s]:after,.ck.ck-balloon-panel[class*=arrow_s]:before{border-width:var(--ck-balloon-arrow-height) var(--ck-balloon-arrow-half-width) 0 var(--ck-balloon-arrow-half-width)}.ck.ck-balloon-panel[class*=arrow_s]:before{border-color:var(--ck-color-panel-border) transparent transparent;filter:drop-shadow(var(--ck-balloon-arrow-drop-shadow));margin-bottom:calc(var(--ck-balloon-border-width)*-1)}.ck.ck-balloon-panel[class*=arrow_s]:after{border-color:var(--ck-color-panel-background) transparent transparent transparent;margin-bottom:calc(var(--ck-balloon-arrow-offset) - var(--ck-balloon-border-width))}.ck.ck-balloon-panel[class*=arrow_e]:after,.ck.ck-balloon-panel[class*=arrow_e]:before{border-width:var(--ck-balloon-arrow-half-width) 0 var(--ck-balloon-arrow-half-width) var(--ck-balloon-arrow-height)}.ck.ck-balloon-panel[class*=arrow_e]:before{border-color:transparent transparent transparent var(--ck-color-panel-border);margin-right:calc(var(--ck-balloon-border-width)*-1)}.ck.ck-balloon-panel[class*=arrow_e]:after{border-color:transparent transparent transparent var(--ck-color-panel-background);margin-right:calc(var(--ck-balloon-arrow-offset) - var(--ck-balloon-border-width))}.ck.ck-balloon-panel[class*=arrow_w]:after,.ck.ck-balloon-panel[class*=arrow_w]:before{border-width:var(--ck-balloon-arrow-half-width) var(--ck-balloon-arrow-height) var(--ck-balloon-arrow-half-width) 0}.ck.ck-balloon-panel[class*=arrow_w]:before{border-color:transparent var(--ck-color-panel-border) transparent transparent;margin-left:calc(var(--ck-balloon-border-width)*-1)}.ck.ck-balloon-panel[class*=arrow_w]:after{border-color:transparent var(--ck-color-panel-background) transparent transparent;margin-left:calc(var(--ck-balloon-arrow-offset) - var(--ck-balloon-border-width))}.ck.ck-balloon-panel.ck-balloon-panel_arrow_n:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_n:before{left:50%;margin-left:calc(var(--ck-balloon-arrow-half-width)*-1);top:calc(var(--ck-balloon-arrow-height)*-1)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_nw:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_nw:before{left:calc(var(--ck-balloon-arrow-half-width)*2);top:calc(var(--ck-balloon-arrow-height)*-1)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_ne:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_ne:before{right:calc(var(--ck-balloon-arrow-half-width)*2);top:calc(var(--ck-balloon-arrow-height)*-1)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_s:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_s:before{bottom:calc(var(--ck-balloon-arrow-height)*-1);left:50%;margin-left:calc(var(--ck-balloon-arrow-half-width)*-1)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_sw:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_sw:before{bottom:calc(var(--ck-balloon-arrow-height)*-1);left:calc(var(--ck-balloon-arrow-half-width)*2)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_se:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_se:before{bottom:calc(var(--ck-balloon-arrow-height)*-1);right:calc(var(--ck-balloon-arrow-half-width)*2)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_sme:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_sme:before{bottom:calc(var(--ck-balloon-arrow-height)*-1);margin-right:calc(var(--ck-balloon-arrow-half-width)*2);right:25%}.ck.ck-balloon-panel.ck-balloon-panel_arrow_smw:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_smw:before{bottom:calc(var(--ck-balloon-arrow-height)*-1);left:25%;margin-left:calc(var(--ck-balloon-arrow-half-width)*2)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_nme:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_nme:before{margin-right:calc(var(--ck-balloon-arrow-half-width)*2);right:25%;top:calc(var(--ck-balloon-arrow-height)*-1)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_nmw:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_nmw:before{left:25%;margin-left:calc(var(--ck-balloon-arrow-half-width)*2);top:calc(var(--ck-balloon-arrow-height)*-1)}.ck.ck-balloon-panel.ck-balloon-panel_arrow_e:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_e:before{margin-top:calc(var(--ck-balloon-arrow-half-width)*-1);right:calc(var(--ck-balloon-arrow-height)*-1);top:50%}.ck.ck-balloon-panel.ck-balloon-panel_arrow_w:after,.ck.ck-balloon-panel.ck-balloon-panel_arrow_w:before{left:calc(var(--ck-balloon-arrow-height)*-1);margin-top:calc(var(--ck-balloon-arrow-half-width)*-1);top:50%}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/panel/balloonpanel.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/panel/balloonpanel.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_shadow.css\"],\"names\":[],\"mappings\":\"AAKA,MAEC,8DACD,CAEA,qBACC,YAAa,CACb,iBAAkB,CAElB,yBAyCD,CAtCE,+GAEC,UAAW,CACX,iBACD,CAEA,wDACC,6CACD,CAEA,uDACC,uDACD,CAIA,4CACC,6CACD,CAEA,2CACC,uDACD,CAIA,4CACC,6CACD,CAEA,2CACC,uDACD,CAGD,8CACC,aACD,CC9CD,MACC,6BAA8B,CAC9B,6BAA8B,CAC9B,8BAA+B,CAC/B,iCAAkC,CAClC,oEACD,CAEA,qBCLC,eDmMD,CA9LA,iFCDE,qCD+LF,CA9LA,qBAMC,2CAA4C,CAC5C,wEAAyE,CEdzE,oCAA8B,CFW9B,eA0LD,CApLE,+GAIC,kBAAmB,CADnB,QAAS,CADT,OAGD,CAIA,uFAEC,mHACD,CAEA,4CACC,6EAA8E,CAC9E,kDACD,CAEA,2CACC,iFAAkF,CAClF,gFACD,CAIA,uFAEC,mHACD,CAEA,4CACC,iEAAkE,CAClE,uDAAwD,CACxD,qDACD,CAEA,2CACC,iFAAkF,CAClF,mFACD,CAIA,uFAEC,mHACD,CAEA,4CACC,6EAA8E,CAC9E,oDACD,CAEA,2CACC,iFAAkF,CAClF,kFACD,CAIA,uFAEC,mHACD,CAEA,4CACC,6EAA8E,CAC9E,mDACD,CAEA,2CACC,iFAAkF,CAClF,iFACD,CAIA,yGAEC,QAAS,CACT,uDAA0D,CAC1D,2CACD,CAIA,2GAEC,+CAAkD,CAClD,2CACD,CAIA,2GAEC,gDAAmD,CACnD,2CACD,CAIA,yGAIC,8CAAiD,CAFjD,QAAS,CACT,uDAED,CAIA,2GAGC,8CAAiD,CADjD,+CAED,CAIA,2GAGC,8CAAiD,CADjD,gDAED,CAIA,6GAIC,8CAAiD,CADjD,uDAA0D,CAD1D,SAGD,CAIA,6GAIC,8CAAiD,CAFjD,QAAS,CACT,sDAED,CAIA,6GAGC,uDAA0D,CAD1D,SAAU,CAEV,2CACD,CAIA,6GAEC,QAAS,CACT,sDAAyD,CACzD,2CACD,CAIA,yGAGC,sDAAyD,CADzD,6CAAgD,CAEhD,OACD,CAIA,yGAEC,4CAA+C,CAC/C,sDAAyD,CACzD,OACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t/* Make sure the balloon arrow does not float over its children. */\\n\\t--ck-balloon-panel-arrow-z-index: calc(var(--ck-z-default) - 3);\\n}\\n\\n.ck.ck-balloon-panel {\\n\\tdisplay: none;\\n\\tposition: absolute;\\n\\n\\tz-index: var(--ck-z-modal);\\n\\n\\t&.ck-balloon-panel_with-arrow {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tcontent: \\\"\\\";\\n\\t\\t\\tposition: absolute;\\n\\t\\t}\\n\\n\\t\\t&::before {\\n\\t\\t\\tz-index: var(--ck-balloon-panel-arrow-z-index);\\n\\t\\t}\\n\\n\\t\\t&::after {\\n\\t\\t\\tz-index: calc(var(--ck-balloon-panel-arrow-z-index) + 1);\\n\\t\\t}\\n\\t}\\n\\n\\t&[class*=\\\"arrow_n\\\"] {\\n\\t\\t&::before {\\n\\t\\t\\tz-index: var(--ck-balloon-panel-arrow-z-index);\\n\\t\\t}\\n\\n\\t\\t&::after {\\n\\t\\t\\tz-index: calc(var(--ck-balloon-panel-arrow-z-index) + 1);\\n\\t\\t}\\n\\t}\\n\\n\\t&[class*=\\\"arrow_s\\\"] {\\n\\t\\t&::before {\\n\\t\\t\\tz-index: var(--ck-balloon-panel-arrow-z-index);\\n\\t\\t}\\n\\n\\t\\t&::after {\\n\\t\\t\\tz-index: calc(var(--ck-balloon-panel-arrow-z-index) + 1);\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_visible {\\n\\t\\tdisplay: block;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n@import \\\"../../../mixins/_shadow.css\\\";\\n\\n:root {\\n\\t--ck-balloon-border-width: 1px;\\n\\t--ck-balloon-arrow-offset: 2px;\\n\\t--ck-balloon-arrow-height: 10px;\\n\\t--ck-balloon-arrow-half-width: 8px;\\n\\t--ck-balloon-arrow-drop-shadow: 0 2px 2px var(--ck-color-shadow-drop);\\n}\\n\\n.ck.ck-balloon-panel {\\n\\t@mixin ck-rounded-corners;\\n\\t@mixin ck-drop-shadow;\\n\\n\\tmin-height: 15px;\\n\\n\\tbackground: var(--ck-color-panel-background);\\n\\tborder: var(--ck-balloon-border-width) solid var(--ck-color-panel-border);\\n\\n\\t&.ck-balloon-panel_with-arrow {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\twidth: 0;\\n\\t\\t\\theight: 0;\\n\\t\\t\\tborder-style: solid;\\n\\t\\t}\\n\\t}\\n\\n\\t&[class*=\\\"arrow_n\\\"] {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tborder-width: 0 var(--ck-balloon-arrow-half-width) var(--ck-balloon-arrow-height) var(--ck-balloon-arrow-half-width);\\n\\t\\t}\\n\\n\\t\\t&::before {\\n\\t\\t\\tborder-color: transparent transparent var(--ck-color-panel-border) transparent;\\n\\t\\t\\tmargin-top: calc( -1 * var(--ck-balloon-border-width) );\\n\\t\\t}\\n\\n\\t\\t&::after {\\n\\t\\t\\tborder-color: transparent transparent var(--ck-color-panel-background) transparent;\\n\\t\\t\\tmargin-top: calc( var(--ck-balloon-arrow-offset) - var(--ck-balloon-border-width) );\\n\\t\\t}\\n\\t}\\n\\n\\t&[class*=\\\"arrow_s\\\"] {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tborder-width: var(--ck-balloon-arrow-height) var(--ck-balloon-arrow-half-width) 0 var(--ck-balloon-arrow-half-width);\\n\\t\\t}\\n\\n\\t\\t&::before {\\n\\t\\t\\tborder-color: var(--ck-color-panel-border) transparent transparent;\\n\\t\\t\\tfilter: drop-shadow(var(--ck-balloon-arrow-drop-shadow));\\n\\t\\t\\tmargin-bottom: calc( -1 * var(--ck-balloon-border-width) );\\n\\t\\t}\\n\\n\\t\\t&::after {\\n\\t\\t\\tborder-color: var(--ck-color-panel-background) transparent transparent transparent;\\n\\t\\t\\tmargin-bottom: calc( var(--ck-balloon-arrow-offset) - var(--ck-balloon-border-width) );\\n\\t\\t}\\n\\t}\\n\\n\\t&[class*=\\\"arrow_e\\\"] {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tborder-width: var(--ck-balloon-arrow-half-width) 0 var(--ck-balloon-arrow-half-width) var(--ck-balloon-arrow-height);\\n\\t\\t}\\n\\n\\t\\t&::before {\\n\\t\\t\\tborder-color: transparent transparent transparent var(--ck-color-panel-border);\\n\\t\\t\\tmargin-right: calc( -1 * var(--ck-balloon-border-width) );\\n\\t\\t}\\n\\n\\t\\t&::after {\\n\\t\\t\\tborder-color: transparent transparent transparent var(--ck-color-panel-background);\\n\\t\\t\\tmargin-right: calc( var(--ck-balloon-arrow-offset) - var(--ck-balloon-border-width) );\\n\\t\\t}\\n\\t}\\n\\n\\t&[class*=\\\"arrow_w\\\"] {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tborder-width: var(--ck-balloon-arrow-half-width) var(--ck-balloon-arrow-height) var(--ck-balloon-arrow-half-width) 0;\\n\\t\\t}\\n\\n\\t\\t&::before {\\n\\t\\t\\tborder-color: transparent var(--ck-color-panel-border) transparent transparent;\\n\\t\\t\\tmargin-left: calc( -1 * var(--ck-balloon-border-width) );\\n\\t\\t}\\n\\n\\t\\t&::after {\\n\\t\\t\\tborder-color: transparent var(--ck-color-panel-background) transparent transparent;\\n\\t\\t\\tmargin-left: calc( var(--ck-balloon-arrow-offset) - var(--ck-balloon-border-width) );\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_n {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tleft: 50%;\\n\\t\\t\\tmargin-left: calc(-1 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\ttop: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_nw {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tleft: calc(2 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\ttop: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_ne {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tright: calc(2 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\ttop: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_s {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tleft: 50%;\\n\\t\\t\\tmargin-left: calc(-1 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\tbottom: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_sw {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tleft: calc(2 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\tbottom: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_se {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tright: calc(2 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\tbottom: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_sme {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tright: 25%;\\n\\t\\t\\tmargin-right: calc(2 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\tbottom: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_smw {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tleft: 25%;\\n\\t\\t\\tmargin-left: calc(2 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\tbottom: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_nme {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tright: 25%;\\n\\t\\t\\tmargin-right: calc(2 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\ttop: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_nmw {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tleft: 25%;\\n\\t\\t\\tmargin-left: calc(2 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\ttop: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_e {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tright: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t\\tmargin-top: calc(-1 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\ttop: 50%;\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-balloon-panel_arrow_w {\\n\\t\\t&::before,\\n\\t\\t&::after {\\n\\t\\t\\tleft: calc(-1 * var(--ck-balloon-arrow-height));\\n\\t\\t\\tmargin-top: calc(-1 * var(--ck-balloon-arrow-half-width));\\n\\t\\t\\ttop: 50%;\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A helper to combine multiple shadows.\\n */\\n@define-mixin ck-box-shadow $shadowA, $shadowB: 0 0 {\\n\\tbox-shadow: $shadowA, $shadowB;\\n}\\n\\n/**\\n * Gives an element a drop shadow so it looks like a floating panel.\\n */\\n@define-mixin ck-drop-shadow {\\n\\t@mixin ck-box-shadow var(--ck-drop-shadow);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck .ck-balloon-rotator__navigation{align-items:center;display:flex;justify-content:center}.ck .ck-balloon-rotator__content .ck-toolbar{justify-content:center}.ck .ck-balloon-rotator__navigation{background:var(--ck-color-toolbar-background);border-bottom:1px solid var(--ck-color-toolbar-border);padding:0 var(--ck-spacing-small)}.ck .ck-balloon-rotator__navigation>*{margin-bottom:var(--ck-spacing-small);margin-right:var(--ck-spacing-small);margin-top:var(--ck-spacing-small)}.ck .ck-balloon-rotator__navigation .ck-balloon-rotator__counter{margin-left:var(--ck-spacing-small);margin-right:var(--ck-spacing-standard)}.ck .ck-balloon-rotator__content .ck.ck-annotation-wrapper{box-shadow:none}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/panel/balloonrotator.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/panel/balloonrotator.css\"],\"names\":[],\"mappings\":\"AAKA,oCAEC,kBAAmB,CADnB,YAAa,CAEb,sBACD,CAKA,6CACC,sBACD,CCXA,oCACC,6CAA8C,CAC9C,sDAAuD,CACvD,iCAgBD,CAbC,sCAGC,qCAAsC,CAFtC,oCAAqC,CACrC,kCAED,CAGA,iEAIC,mCAAoC,CAHpC,uCAID,CAMA,2DACC,eACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck .ck-balloon-rotator__navigation {\\n\\tdisplay: flex;\\n\\talign-items: center;\\n\\tjustify-content: center;\\n}\\n\\n/* Buttons inside a toolbar should be centered when rotator bar is wider.\\n * See: https://github.com/ckeditor/ckeditor5-ui/issues/495\\n */\\n.ck .ck-balloon-rotator__content .ck-toolbar {\\n\\tjustify-content: center;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck .ck-balloon-rotator__navigation {\\n\\tbackground: var(--ck-color-toolbar-background);\\n\\tborder-bottom: 1px solid var(--ck-color-toolbar-border);\\n\\tpadding: 0 var(--ck-spacing-small);\\n\\n\\t/* Let's keep similar appearance to `ck-toolbar`. */\\n\\t& > * {\\n\\t\\tmargin-right: var(--ck-spacing-small);\\n\\t\\tmargin-top: var(--ck-spacing-small);\\n\\t\\tmargin-bottom: var(--ck-spacing-small);\\n\\t}\\n\\n\\t/* Gives counter more breath than buttons. */\\n\\t& .ck-balloon-rotator__counter {\\n\\t\\tmargin-right: var(--ck-spacing-standard);\\n\\n\\t\\t/* We need to use smaller margin because of previous button's right margin. */\\n\\t\\tmargin-left: var(--ck-spacing-small);\\n\\t}\\n}\\n\\n.ck .ck-balloon-rotator__content {\\n\\n\\t/* Disable default annotation shadow inside rotator with fake panels. */\\n\\t& .ck.ck-annotation-wrapper {\\n\\t\\tbox-shadow: none;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck .ck-fake-panel{position:absolute;z-index:calc(var(--ck-z-modal) - 1)}.ck .ck-fake-panel div{position:absolute}.ck .ck-fake-panel div:first-child{z-index:2}.ck .ck-fake-panel div:nth-child(2){z-index:1}:root{--ck-balloon-fake-panel-offset-horizontal:6px;--ck-balloon-fake-panel-offset-vertical:6px}.ck .ck-fake-panel div{background:var(--ck-color-panel-background);border:1px solid var(--ck-color-panel-border);border-radius:var(--ck-border-radius);box-shadow:var(--ck-drop-shadow),0 0;height:100%;min-height:15px;width:100%}.ck .ck-fake-panel div:first-child{margin-left:var(--ck-balloon-fake-panel-offset-horizontal);margin-top:var(--ck-balloon-fake-panel-offset-vertical)}.ck .ck-fake-panel div:nth-child(2){margin-left:calc(var(--ck-balloon-fake-panel-offset-horizontal)*2);margin-top:calc(var(--ck-balloon-fake-panel-offset-vertical)*2)}.ck .ck-fake-panel div:nth-child(3){margin-left:calc(var(--ck-balloon-fake-panel-offset-horizontal)*3);margin-top:calc(var(--ck-balloon-fake-panel-offset-vertical)*3)}.ck .ck-balloon-panel_arrow_s+.ck-fake-panel,.ck .ck-balloon-panel_arrow_se+.ck-fake-panel,.ck .ck-balloon-panel_arrow_sw+.ck-fake-panel{--ck-balloon-fake-panel-offset-vertical:-6px}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/panel/fakepanel.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/panel/fakepanel.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_shadow.css\"],\"names\":[],\"mappings\":\"AAKA,mBACC,iBAAkB,CAGlB,mCACD,CAEA,uBACC,iBACD,CAEA,mCACC,SACD,CAEA,oCACC,SACD,CCfA,MACC,6CAA8C,CAC9C,2CACD,CAGA,uBAKC,2CAA4C,CAC5C,6CAA8C,CAC9C,qCAAsC,CCXtC,oCAA8B,CDc9B,WAAY,CAPZ,eAAgB,CAMhB,UAED,CAEA,mCACC,0DAA2D,CAC3D,uDACD,CAEA,oCACC,kEAAqE,CACrE,+DACD,CACA,oCACC,kEAAqE,CACrE,+DACD,CAGA,yIAGC,4CACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck .ck-fake-panel {\\n\\tposition: absolute;\\n\\n\\t/* Fake panels should be placed under main balloon content. */\\n\\tz-index: calc(var(--ck-z-modal) - 1);\\n}\\n\\n.ck .ck-fake-panel div {\\n\\tposition: absolute;\\n}\\n\\n.ck .ck-fake-panel div:nth-child( 1 ) {\\n\\tz-index: 2;\\n}\\n\\n.ck .ck-fake-panel div:nth-child( 2 ) {\\n\\tz-index: 1;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_shadow.css\\\";\\n\\n:root {\\n\\t--ck-balloon-fake-panel-offset-horizontal: 6px;\\n\\t--ck-balloon-fake-panel-offset-vertical: 6px;\\n}\\n\\n/* Let's use `.ck-balloon-panel` appearance. See: balloonpanel.css. */\\n.ck .ck-fake-panel div {\\n\\t@mixin ck-drop-shadow;\\n\\n\\tmin-height: 15px;\\n\\n\\tbackground: var(--ck-color-panel-background);\\n\\tborder: 1px solid var(--ck-color-panel-border);\\n\\tborder-radius: var(--ck-border-radius);\\n\\n\\twidth: 100%;\\n\\theight: 100%;\\n}\\n\\n.ck .ck-fake-panel div:nth-child( 1 ) {\\n\\tmargin-left: var(--ck-balloon-fake-panel-offset-horizontal);\\n\\tmargin-top: var(--ck-balloon-fake-panel-offset-vertical);\\n}\\n\\n.ck .ck-fake-panel div:nth-child( 2 ) {\\n\\tmargin-left: calc(var(--ck-balloon-fake-panel-offset-horizontal) * 2);\\n\\tmargin-top: calc(var(--ck-balloon-fake-panel-offset-vertical) * 2);\\n}\\n.ck .ck-fake-panel div:nth-child( 3 ) {\\n\\tmargin-left: calc(var(--ck-balloon-fake-panel-offset-horizontal) * 3);\\n\\tmargin-top: calc(var(--ck-balloon-fake-panel-offset-vertical) * 3);\\n}\\n\\n/* If balloon is positioned above element, we need to move fake panel to the top. */\\n.ck .ck-balloon-panel_arrow_s + .ck-fake-panel,\\n.ck .ck-balloon-panel_arrow_se + .ck-fake-panel,\\n.ck .ck-balloon-panel_arrow_sw + .ck-fake-panel {\\n\\t--ck-balloon-fake-panel-offset-vertical: -6px;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A helper to combine multiple shadows.\\n */\\n@define-mixin ck-box-shadow $shadowA, $shadowB: 0 0 {\\n\\tbox-shadow: $shadowA, $shadowB;\\n}\\n\\n/**\\n * Gives an element a drop shadow so it looks like a floating panel.\\n */\\n@define-mixin ck-drop-shadow {\\n\\t@mixin ck-box-shadow var(--ck-drop-shadow);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-sticky-panel .ck-sticky-panel__content_sticky{position:fixed;top:0;z-index:var(--ck-z-modal)}.ck.ck-sticky-panel .ck-sticky-panel__content_sticky_bottom-limit{position:absolute;top:auto}.ck.ck-sticky-panel .ck-sticky-panel__content_sticky{border-top-left-radius:0;border-top-right-radius:0;border-width:0 1px 1px;box-shadow:var(--ck-drop-shadow),0 0}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/panel/stickypanel.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/panel/stickypanel.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_shadow.css\"],\"names\":[],\"mappings\":\"AAMC,qDAEC,cAAe,CACf,KAAM,CAFN,yBAGD,CAEA,kEAEC,iBAAkB,CADlB,QAED,CCPA,qDAIC,wBAAyB,CACzB,yBAA0B,CAF1B,sBAAuB,CCFxB,oCDKA\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-sticky-panel {\\n\\t& .ck-sticky-panel__content_sticky {\\n\\t\\tz-index: var(--ck-z-modal); /* #315 */\\n\\t\\tposition: fixed;\\n\\t\\ttop: 0;\\n\\t}\\n\\n\\t& .ck-sticky-panel__content_sticky_bottom-limit {\\n\\t\\ttop: auto;\\n\\t\\tposition: absolute;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_shadow.css\\\";\\n\\n.ck.ck-sticky-panel {\\n\\t& .ck-sticky-panel__content_sticky {\\n\\t\\t@mixin ck-drop-shadow;\\n\\n\\t\\tborder-width: 0 1px 1px;\\n\\t\\tborder-top-left-radius: 0;\\n\\t\\tborder-top-right-radius: 0;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A helper to combine multiple shadows.\\n */\\n@define-mixin ck-box-shadow $shadowA, $shadowB: 0 0 {\\n\\tbox-shadow: $shadowA, $shadowB;\\n}\\n\\n/**\\n * Gives an element a drop shadow so it looks like a floating panel.\\n */\\n@define-mixin ck-drop-shadow {\\n\\t@mixin ck-box-shadow var(--ck-drop-shadow);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-vertical-form .ck-button:after{bottom:-1px;content:\\\"\\\";position:absolute;right:-1px;top:-1px;width:0;z-index:1}.ck-vertical-form .ck-button:focus:after{display:none}@media screen and (max-width:600px){.ck.ck-responsive-form .ck-button:after{bottom:-1px;content:\\\"\\\";position:absolute;right:-1px;top:-1px;width:0;z-index:1}.ck.ck-responsive-form .ck-button:focus:after{display:none}}.ck-vertical-form>.ck-button:nth-last-child(2):after{border-right:1px solid var(--ck-color-base-border)}.ck.ck-responsive-form{padding:var(--ck-spacing-large)}.ck.ck-responsive-form:focus{outline:none}[dir=ltr] .ck.ck-responsive-form>:not(:first-child),[dir=rtl] .ck.ck-responsive-form>:not(:last-child){margin-left:var(--ck-spacing-standard)}@media screen and (max-width:600px){.ck.ck-responsive-form{padding:0;width:calc(var(--ck-input-width)*.8)}.ck.ck-responsive-form .ck-labeled-field-view{margin:var(--ck-spacing-large) var(--ck-spacing-large) 0}.ck.ck-responsive-form .ck-labeled-field-view .ck-input-text{min-width:0;width:100%}.ck.ck-responsive-form .ck-labeled-field-view .ck-labeled-field-view__error{white-space:normal}.ck.ck-responsive-form>.ck-button:nth-last-child(2):after{border-right:1px solid var(--ck-color-base-border)}.ck.ck-responsive-form>.ck-button:last-child,.ck.ck-responsive-form>.ck-button:nth-last-child(2){border-radius:0;margin-top:var(--ck-spacing-large);padding:var(--ck-spacing-standard)}.ck.ck-responsive-form>.ck-button:last-child:not(:focus),.ck.ck-responsive-form>.ck-button:nth-last-child(2):not(:focus){border-top:1px solid var(--ck-color-base-border)}[dir=ltr] .ck.ck-responsive-form>.ck-button:last-child,[dir=ltr] .ck.ck-responsive-form>.ck-button:nth-last-child(2),[dir=rtl] .ck.ck-responsive-form>.ck-button:last-child,[dir=rtl] .ck.ck-responsive-form>.ck-button:nth-last-child(2){margin-left:0}[dir=rtl] .ck.ck-responsive-form>.ck-button:last-child:last-of-type,[dir=rtl] .ck.ck-responsive-form>.ck-button:nth-last-child(2):last-of-type{border-right:1px solid var(--ck-color-base-border)}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/responsive-form/responsiveform.css\",\"webpack://./../ckeditor5-ui/theme/mixins/_rwd.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/responsive-form/responsiveform.css\"],\"names\":[],\"mappings\":\"AAQC,mCAMC,WAAY,CALZ,UAAW,CAEX,iBAAkB,CAClB,UAAW,CACX,QAAS,CAHT,OAAQ,CAKR,SACD,CAEA,yCACC,YACD,CCdA,oCDoBE,wCAMC,WAAY,CALZ,UAAW,CAEX,iBAAkB,CAClB,UAAW,CACX,QAAS,CAHT,OAAQ,CAKR,SACD,CAEA,8CACC,YACD,CC9BF,CCAD,qDACC,kDACD,CAEA,uBACC,+BAmED,CAjEC,6BAEC,YACD,CASC,uGACC,sCACD,CDvBD,oCCMD,uBAqBE,SAAU,CACV,oCA8CF,CA5CE,8CACC,wDAWD,CATC,6DACC,WAAY,CACZ,UACD,CAGA,4EACC,kBACD,CAKA,0DACC,kDACD,CAGD,iGAIC,eAAgB,CADhB,kCAAmC,CADnC,kCAmBD,CAfC,yHACC,gDACD,CARD,0OAeE,aAMF,CAJE,+IACC,kDACD,CDpEH\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_rwd.css\\\";\\n\\n.ck-vertical-form .ck-button {\\n\\t&::after {\\n\\t\\tcontent: \\\"\\\";\\n\\t\\twidth: 0;\\n\\t\\tposition: absolute;\\n\\t\\tright: -1px;\\n\\t\\ttop: -1px;\\n\\t\\tbottom: -1px;\\n\\t\\tz-index: 1;\\n\\t}\\n\\n\\t&:focus::after {\\n\\t\\tdisplay: none;\\n\\t}\\n}\\n\\n.ck.ck-responsive-form {\\n\\t@mixin ck-media-phone {\\n\\t\\t& .ck-button {\\n\\t\\t\\t&::after {\\n\\t\\t\\t\\tcontent: \\\"\\\";\\n\\t\\t\\t\\twidth: 0;\\n\\t\\t\\t\\tposition: absolute;\\n\\t\\t\\t\\tright: -1px;\\n\\t\\t\\t\\ttop: -1px;\\n\\t\\t\\t\\tbottom: -1px;\\n\\t\\t\\t\\tz-index: 1;\\n\\t\\t\\t}\\n\\n\\t\\t\\t&:focus::after {\\n\\t\\t\\t\\tdisplay: none;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@define-mixin ck-media-phone {\\n\\t@media screen and (max-width: 600px) {\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_rwd.css\\\";\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n.ck-vertical-form > .ck-button:nth-last-child(2)::after {\\n\\tborder-right: 1px solid var(--ck-color-base-border);\\n}\\n\\n.ck.ck-responsive-form {\\n\\tpadding: var(--ck-spacing-large);\\n\\n\\t&:focus {\\n\\t\\t/* See: https://github.com/ckeditor/ckeditor5/issues/4773 */\\n\\t\\toutline: none;\\n\\t}\\n\\n\\t@mixin ck-dir ltr {\\n\\t\\t& > :not(:first-child) {\\n\\t\\t\\tmargin-left: var(--ck-spacing-standard);\\n\\t\\t}\\n\\t}\\n\\n\\t@mixin ck-dir rtl {\\n\\t\\t& > :not(:last-child) {\\n\\t\\t\\tmargin-left: var(--ck-spacing-standard);\\n\\t\\t}\\n\\t}\\n\\n\\t@mixin ck-media-phone {\\n\\t\\tpadding: 0;\\n\\t\\twidth: calc(.8 * var(--ck-input-width));\\n\\n\\t\\t& .ck-labeled-field-view {\\n\\t\\t\\tmargin: var(--ck-spacing-large) var(--ck-spacing-large) 0;\\n\\n\\t\\t\\t& .ck-input-text {\\n\\t\\t\\t\\tmin-width: 0;\\n\\t\\t\\t\\twidth: 100%;\\n\\t\\t\\t}\\n\\n\\t\\t\\t/* Let the long error messages wrap in the narrow form. */\\n\\t\\t\\t& .ck-labeled-field-view__error {\\n\\t\\t\\t\\twhite-space: normal;\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t/* Styles for two last buttons in the form (save&cancel, edit&unlink, etc.). */\\n\\t\\t& > .ck-button:nth-last-child(2) {\\n\\t\\t\\t&::after {\\n\\t\\t\\t\\tborder-right: 1px solid var(--ck-color-base-border);\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t& > .ck-button:nth-last-child(1),\\n\\t\\t& > .ck-button:nth-last-child(2) {\\n\\t\\t\\tpadding: var(--ck-spacing-standard);\\n\\t\\t\\tmargin-top: var(--ck-spacing-large);\\n\\t\\t\\tborder-radius: 0;\\n\\n\\t\\t\\t&:not(:focus) {\\n\\t\\t\\t\\tborder-top: 1px solid var(--ck-color-base-border);\\n\\t\\t\\t}\\n\\n\\t\\t\\t@mixin ck-dir ltr {\\n\\t\\t\\t\\tmargin-left: 0;\\n\\t\\t\\t}\\n\\n\\t\\t\\t@mixin ck-dir rtl {\\n\\t\\t\\t\\tmargin-left: 0;\\n\\n\\t\\t\\t\\t&:last-of-type {\\n\\t\\t\\t\\t\\tborder-right: 1px solid var(--ck-color-base-border);\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-block-toolbar-button{position:absolute;z-index:var(--ck-z-default)}:root{--ck-color-block-toolbar-button:var(--ck-color-text);--ck-block-toolbar-button-size:var(--ck-font-size-normal)}.ck.ck-block-toolbar-button{color:var(--ck-color-block-toolbar-button);font-size:var(--ck-block-toolbar-size)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/toolbar/blocktoolbar.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/toolbar/blocktoolbar.css\"],\"names\":[],\"mappings\":\"AAKA,4BACC,iBAAkB,CAClB,2BACD,CCHA,MACC,oDAAqD,CACrD,yDACD,CAEA,4BACC,0CAA2C,CAC3C,sCACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-block-toolbar-button {\\n\\tposition: absolute;\\n\\tz-index: var(--ck-z-default);\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-block-toolbar-button: var(--ck-color-text);\\n\\t--ck-block-toolbar-button-size: var(--ck-font-size-normal);\\n}\\n\\n.ck.ck-block-toolbar-button {\\n\\tcolor: var(--ck-color-block-toolbar-button);\\n\\tfont-size: var(--ck-block-toolbar-size);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-toolbar{align-items:center;display:flex;flex-flow:row nowrap;-moz-user-select:none;-webkit-user-select:none;-ms-user-select:none;user-select:none}.ck.ck-toolbar>.ck-toolbar__items{align-items:center;display:flex;flex-flow:row wrap;flex-grow:1}.ck.ck-toolbar .ck.ck-toolbar__separator{display:inline-block}.ck.ck-toolbar .ck.ck-toolbar__separator:first-child,.ck.ck-toolbar .ck.ck-toolbar__separator:last-child{display:none}.ck.ck-toolbar .ck-toolbar__line-break{flex-basis:100%}.ck.ck-toolbar.ck-toolbar_grouping>.ck-toolbar__items{flex-wrap:nowrap}.ck.ck-toolbar.ck-toolbar_vertical>.ck-toolbar__items{flex-direction:column}.ck.ck-toolbar.ck-toolbar_floating>.ck-toolbar__items{flex-wrap:nowrap}.ck.ck-toolbar>.ck.ck-toolbar__grouped-dropdown>.ck-dropdown__button .ck-dropdown__arrow{display:none}.ck.ck-toolbar{border-radius:0}.ck-rounded-corners .ck.ck-toolbar,.ck.ck-toolbar.ck-rounded-corners{border-radius:var(--ck-border-radius)}.ck.ck-toolbar{background:var(--ck-color-toolbar-background);border:1px solid var(--ck-color-toolbar-border);padding:0 var(--ck-spacing-small)}.ck.ck-toolbar .ck.ck-toolbar__separator{align-self:stretch;background:var(--ck-color-toolbar-border);margin-bottom:var(--ck-spacing-small);margin-top:var(--ck-spacing-small);min-width:1px;width:1px}.ck.ck-toolbar .ck-toolbar__line-break{height:0}.ck.ck-toolbar>.ck-toolbar__items>:not(.ck-toolbar__line-break){margin-right:var(--ck-spacing-small)}.ck.ck-toolbar>.ck-toolbar__items:empty+.ck.ck-toolbar__separator{display:none}.ck.ck-toolbar>.ck-toolbar__items>:not(.ck-toolbar__line-break),.ck.ck-toolbar>.ck.ck-toolbar__grouped-dropdown{margin-bottom:var(--ck-spacing-small);margin-top:var(--ck-spacing-small)}.ck.ck-toolbar.ck-toolbar_vertical{padding:0}.ck.ck-toolbar.ck-toolbar_vertical>.ck-toolbar__items>.ck{border-radius:0;margin:0;width:100%}.ck.ck-toolbar.ck-toolbar_compact{padding:0}.ck.ck-toolbar.ck-toolbar_compact>.ck-toolbar__items>*{margin:0}.ck.ck-toolbar.ck-toolbar_compact>.ck-toolbar__items>:not(:first-child):not(:last-child){border-radius:0}.ck.ck-toolbar>.ck.ck-toolbar__grouped-dropdown>.ck.ck-button.ck-dropdown__button{padding-left:var(--ck-spacing-tiny)}.ck.ck-toolbar .ck-toolbar__nested-toolbar-dropdown>.ck-dropdown__panel{min-width:auto}.ck.ck-toolbar .ck-toolbar__nested-toolbar-dropdown>.ck-button>.ck-button__label{max-width:7em;width:auto}.ck.ck-toolbar:focus{outline:none}.ck-toolbar-container .ck.ck-toolbar{border:0}.ck.ck-toolbar[dir=rtl]>.ck-toolbar__items>.ck,[dir=rtl] .ck.ck-toolbar>.ck-toolbar__items>.ck{margin-right:0}.ck.ck-toolbar[dir=rtl]:not(.ck-toolbar_compact)>.ck-toolbar__items>.ck,[dir=rtl] .ck.ck-toolbar:not(.ck-toolbar_compact)>.ck-toolbar__items>.ck{margin-left:var(--ck-spacing-small)}.ck.ck-toolbar[dir=rtl]>.ck-toolbar__items>.ck:last-child,[dir=rtl] .ck.ck-toolbar>.ck-toolbar__items>.ck:last-child{margin-left:0}.ck.ck-toolbar.ck-toolbar_compact[dir=rtl]>.ck-toolbar__items>.ck:first-child,[dir=rtl] .ck.ck-toolbar.ck-toolbar_compact>.ck-toolbar__items>.ck:first-child{border-bottom-left-radius:0;border-top-left-radius:0}.ck.ck-toolbar.ck-toolbar_compact[dir=rtl]>.ck-toolbar__items>.ck:last-child,[dir=rtl] .ck.ck-toolbar.ck-toolbar_compact>.ck-toolbar__items>.ck:last-child{border-bottom-right-radius:0;border-top-right-radius:0}.ck.ck-toolbar.ck-toolbar_grouping[dir=rtl]>.ck-toolbar__items:not(:empty):not(:only-child),.ck.ck-toolbar[dir=rtl]>.ck.ck-toolbar__separator,[dir=rtl] .ck.ck-toolbar.ck-toolbar_grouping>.ck-toolbar__items:not(:empty):not(:only-child),[dir=rtl] .ck.ck-toolbar>.ck.ck-toolbar__separator{margin-left:var(--ck-spacing-small)}.ck.ck-toolbar[dir=ltr]>.ck-toolbar__items>.ck:last-child,[dir=ltr] .ck.ck-toolbar>.ck-toolbar__items>.ck:last-child{margin-right:0}.ck.ck-toolbar.ck-toolbar_compact[dir=ltr]>.ck-toolbar__items>.ck:first-child,[dir=ltr] .ck.ck-toolbar.ck-toolbar_compact>.ck-toolbar__items>.ck:first-child{border-bottom-right-radius:0;border-top-right-radius:0}.ck.ck-toolbar.ck-toolbar_compact[dir=ltr]>.ck-toolbar__items>.ck:last-child,[dir=ltr] .ck.ck-toolbar.ck-toolbar_compact>.ck-toolbar__items>.ck:last-child{border-bottom-left-radius:0;border-top-left-radius:0}.ck.ck-toolbar.ck-toolbar_grouping[dir=ltr]>.ck-toolbar__items:not(:empty):not(:only-child),.ck.ck-toolbar[dir=ltr]>.ck.ck-toolbar__separator,[dir=ltr] .ck.ck-toolbar.ck-toolbar_grouping>.ck-toolbar__items:not(:empty):not(:only-child),[dir=ltr] .ck.ck-toolbar>.ck.ck-toolbar__separator{margin-right:var(--ck-spacing-small)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/toolbar/toolbar.css\",\"webpack://./../ckeditor5-ui/theme/mixins/_unselectable.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/toolbar/toolbar.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_rounded.css\"],\"names\":[],\"mappings\":\"AAOA,eAKC,kBAAmB,CAFnB,YAAa,CACb,oBAAqB,CCFrB,qBAAsB,CACtB,wBAAyB,CACzB,oBAAqB,CACrB,gBD6CD,CA3CC,kCAGC,kBAAmB,CAFnB,YAAa,CACb,kBAAmB,CAEnB,WAED,CAEA,yCACC,oBAWD,CAJC,yGAEC,YACD,CAGD,uCACC,eACD,CAEA,sDACC,gBACD,CAEA,sDACC,qBACD,CAEA,sDACC,gBACD,CAGC,yFACC,YACD,CE/CF,eCGC,eDwGD,CA3GA,qECOE,qCDoGF,CA3GA,eAGC,6CAA8C,CAE9C,+CAAgD,CADhD,iCAuGD,CApGC,yCACC,kBAAmB,CAGnB,yCAA0C,CAO1C,qCAAsC,CADtC,kCAAmC,CAPnC,aAAc,CADd,SAUD,CAEA,uCACC,QACD,CAGC,gEAEC,oCACD,CAIA,kEACC,YACD,CAGD,gHAIC,qCAAsC,CADtC,kCAED,CAEA,mCAEC,SAaD,CAVC,0DAQC,eAAgB,CAHhB,QAAS,CAHT,UAOD,CAGD,kCAEC,SAWD,CATC,uDAEC,QAMD,CAHC,yFACC,eACD,CASD,kFACC,mCACD,CAMA,wEACC,cACD,CAEA,iFACC,aAAc,CACd,UACD,CAGD,qBACC,YACD,CAtGD,qCAyGE,QAEF,CAYC,+FACC,cACD,CAEA,iJAEC,mCACD,CAEA,qHACC,aACD,CAIC,6JAEC,2BAA4B,CAD5B,wBAED,CAGA,2JAEC,4BAA6B,CAD7B,yBAED,CASD,8RACC,mCACD,CAWA,qHACC,cACD,CAIC,6JAEC,4BAA6B,CAD7B,yBAED,CAGA,2JAEC,2BAA4B,CAD5B,wBAED,CASD,8RACC,oCACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../mixins/_unselectable.css\\\";\\n\\n.ck.ck-toolbar {\\n\\t@mixin ck-unselectable;\\n\\n\\tdisplay: flex;\\n\\tflex-flow: row nowrap;\\n\\talign-items: center;\\n\\n\\t& > .ck-toolbar__items {\\n\\t\\tdisplay: flex;\\n\\t\\tflex-flow: row wrap;\\n\\t\\talign-items: center;\\n\\t\\tflex-grow: 1;\\n\\n\\t}\\n\\n\\t& .ck.ck-toolbar__separator {\\n\\t\\tdisplay: inline-block;\\n\\n\\t\\t/*\\n\\t\\t * A leading or trailing separator makes no sense (separates from nothing on one side).\\n\\t\\t * For instance, it can happen when toolbar items (also separators) are getting grouped one by one and\\n\\t\\t * moved to another toolbar in the dropdown.\\n\\t\\t */\\n\\t\\t&:first-child,\\n\\t\\t&:last-child {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\t}\\n\\n\\t& .ck-toolbar__line-break {\\n\\t\\tflex-basis: 100%;\\n\\t}\\n\\n\\t&.ck-toolbar_grouping > .ck-toolbar__items {\\n\\t\\tflex-wrap: nowrap;\\n\\t}\\n\\n\\t&.ck-toolbar_vertical > .ck-toolbar__items {\\n\\t\\tflex-direction: column;\\n\\t}\\n\\n\\t&.ck-toolbar_floating > .ck-toolbar__items {\\n\\t\\tflex-wrap: nowrap;\\n\\t}\\n\\n\\t& > .ck.ck-toolbar__grouped-dropdown {\\n\\t\\t& > .ck-dropdown__button .ck-dropdown__arrow {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Makes element unselectable.\\n */\\n@define-mixin ck-unselectable {\\n\\t-moz-user-select: none;\\n\\t-webkit-user-select: none;\\n\\t-ms-user-select: none;\\n\\tuser-select: none\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n@import \\\"@ckeditor/ckeditor5-ui/theme/mixins/_dir.css\\\";\\n\\n.ck.ck-toolbar {\\n\\t@mixin ck-rounded-corners;\\n\\n\\tbackground: var(--ck-color-toolbar-background);\\n\\tpadding: 0 var(--ck-spacing-small);\\n\\tborder: 1px solid var(--ck-color-toolbar-border);\\n\\n\\t& .ck.ck-toolbar__separator {\\n\\t\\talign-self: stretch;\\n\\t\\twidth: 1px;\\n\\t\\tmin-width: 1px;\\n\\t\\tbackground: var(--ck-color-toolbar-border);\\n\\n\\t\\t/*\\n\\t\\t * These margins make the separators look better in balloon toolbars (when aligned with the \\\"tip\\\").\\n\\t\\t * See https://github.com/ckeditor/ckeditor5/issues/7493.\\n\\t\\t */\\n\\t\\tmargin-top: var(--ck-spacing-small);\\n\\t\\tmargin-bottom: var(--ck-spacing-small);\\n\\t}\\n\\n\\t& .ck-toolbar__line-break {\\n\\t\\theight: 0;\\n\\t}\\n\\n\\t& > .ck-toolbar__items {\\n\\t\\t& > *:not(.ck-toolbar__line-break) {\\n\\t\\t\\t/* (#11) Separate toolbar items. */\\n\\t\\t\\tmargin-right: var(--ck-spacing-small);\\n\\t\\t}\\n\\n\\t\\t/* Don't display a separator after an empty items container, for instance,\\n\\t\\twhen all items were grouped */\\n\\t\\t&:empty + .ck.ck-toolbar__separator {\\n\\t\\t\\tdisplay: none;\\n\\t\\t}\\n\\t}\\n\\n\\t& > .ck-toolbar__items > *:not(.ck-toolbar__line-break),\\n\\t& > .ck.ck-toolbar__grouped-dropdown {\\n\\t\\t/* Make sure items wrapped to the next line have v-spacing */\\n\\t\\tmargin-top: var(--ck-spacing-small);\\n\\t\\tmargin-bottom: var(--ck-spacing-small);\\n\\t}\\n\\n\\t&.ck-toolbar_vertical {\\n\\t\\t/* Items in a vertical toolbar span the entire width. */\\n\\t\\tpadding: 0;\\n\\n\\t\\t/* Specificity matters here. See https://github.com/ckeditor/ckeditor5-theme-lark/issues/168. */\\n\\t\\t& > .ck-toolbar__items > .ck {\\n\\t\\t\\t/* Items in a vertical toolbar should span the horizontal space. */\\n\\t\\t\\twidth: 100%;\\n\\n\\t\\t\\t/* Items in a vertical toolbar should have no margin. */\\n\\t\\t\\tmargin: 0;\\n\\n\\t\\t\\t/* Items in a vertical toolbar span the entire width so rounded corners are pointless. */\\n\\t\\t\\tborder-radius: 0;\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-toolbar_compact {\\n\\t\\t/* No spacing around items. */\\n\\t\\tpadding: 0;\\n\\n\\t\\t& > .ck-toolbar__items > * {\\n\\t\\t\\t/* Compact toolbar items have no spacing between them. */\\n\\t\\t\\tmargin: 0;\\n\\n\\t\\t\\t/* \\\"Middle\\\" children should have no rounded corners. */\\n\\t\\t\\t&:not(:first-child):not(:last-child) {\\n\\t\\t\\t\\tborder-radius: 0;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t& > .ck.ck-toolbar__grouped-dropdown {\\n\\t\\t/*\\n\\t\\t * Dropdown button has asymmetric padding to fit the arrow.\\n\\t\\t * This button has no arrow so let's revert that padding back to normal.\\n\\t\\t */\\n\\t\\t& > .ck.ck-button.ck-dropdown__button {\\n\\t\\t\\tpadding-left: var(--ck-spacing-tiny);\\n\\t\\t}\\n\\t}\\n\\n\\t/* A drop-down containing the nested toolbar with configured items. */\\n\\t& .ck-toolbar__nested-toolbar-dropdown {\\n\\t\\t/* Prevent empty space in the panel when the dropdown label is visible and long but the toolbar has few items. */\\n\\t\\t& > .ck-dropdown__panel {\\n\\t\\t\\tmin-width: auto;\\n\\t\\t}\\n\\n\\t\\t& > .ck-button > .ck-button__label {\\n\\t\\t\\tmax-width: 7em;\\n\\t\\t\\twidth: auto;\\n\\t\\t}\\n\\t}\\n\\n\\t&:focus {\\n\\t\\toutline: none;\\n\\t}\\n\\n\\t@nest .ck-toolbar-container & {\\n\\t\\tborder: 0;\\n\\t}\\n}\\n\\n/* stylelint-disable */\\n\\n/*\\n * Styles for RTL toolbars.\\n *\\n * Note: In some cases (e.g. a decoupled editor), the toolbar has its own \\\"dir\\\"\\n * because its parent is not controlled by the editor framework.\\n */\\n[dir=\\\"rtl\\\"] .ck.ck-toolbar,\\n.ck.ck-toolbar[dir=\\\"rtl\\\"] {\\n\\t& > .ck-toolbar__items > .ck {\\n\\t\\tmargin-right: 0;\\n\\t}\\n\\n\\t&:not(.ck-toolbar_compact) > .ck-toolbar__items > .ck {\\n\\t\\t/* (#11) Separate toolbar items. */\\n\\t\\tmargin-left: var(--ck-spacing-small);\\n\\t}\\n\\n\\t& > .ck-toolbar__items > .ck:last-child {\\n\\t\\tmargin-left: 0;\\n\\t}\\n\\n\\t&.ck-toolbar_compact > .ck-toolbar__items > .ck {\\n\\t\\t/* No rounded corners on the right side of the first child. */\\n\\t\\t&:first-child {\\n\\t\\t\\tborder-top-left-radius: 0;\\n\\t\\t\\tborder-bottom-left-radius: 0;\\n\\t\\t}\\n\\n\\t\\t/* No rounded corners on the left side of the last child. */\\n\\t\\t&:last-child {\\n\\t\\t\\tborder-top-right-radius: 0;\\n\\t\\t\\tborder-bottom-right-radius: 0;\\n\\t\\t}\\n\\t}\\n\\n\\t/* Separate the the separator form the grouping dropdown when some items are grouped. */\\n\\t& > .ck.ck-toolbar__separator {\\n\\t\\tmargin-left: var(--ck-spacing-small);\\n\\t}\\n\\n\\t/* Some spacing between the items and the separator before the grouped items dropdown. */\\n\\t&.ck-toolbar_grouping > .ck-toolbar__items:not(:empty):not(:only-child) {\\n\\t\\tmargin-left: var(--ck-spacing-small);\\n\\t}\\n}\\n\\n/*\\n * Styles for LTR toolbars.\\n *\\n * Note: In some cases (e.g. a decoupled editor), the toolbar has its own \\\"dir\\\"\\n * because its parent is not controlled by the editor framework.\\n */\\n[dir=\\\"ltr\\\"] .ck.ck-toolbar,\\n.ck.ck-toolbar[dir=\\\"ltr\\\"] {\\n\\t& > .ck-toolbar__items > .ck:last-child {\\n\\t\\tmargin-right: 0;\\n\\t}\\n\\n\\t&.ck-toolbar_compact > .ck-toolbar__items > .ck {\\n\\t\\t/* No rounded corners on the right side of the first child. */\\n\\t\\t&:first-child {\\n\\t\\t\\tborder-top-right-radius: 0;\\n\\t\\t\\tborder-bottom-right-radius: 0;\\n\\t\\t}\\n\\n\\t\\t/* No rounded corners on the left side of the last child. */\\n\\t\\t&:last-child {\\n\\t\\t\\tborder-top-left-radius: 0;\\n\\t\\t\\tborder-bottom-left-radius: 0;\\n\\t\\t}\\n\\t}\\n\\n\\t/* Separate the the separator form the grouping dropdown when some items are grouped. */\\n\\t& > .ck.ck-toolbar__separator {\\n\\t\\tmargin-right: var(--ck-spacing-small);\\n\\t}\\n\\n\\t/* Some spacing between the items and the separator before the grouped items dropdown. */\\n\\t&.ck-toolbar_grouping > .ck-toolbar__items:not(:empty):not(:only-child) {\\n\\t\\tmargin-right: var(--ck-spacing-small);\\n\\t}\\n}\\n\\n/* stylelint-enable */\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Implements rounded corner interface for .ck-rounded-corners class.\\n *\\n * @see $ck-border-radius\\n */\\n@define-mixin ck-rounded-corners {\\n\\tborder-radius: 0;\\n\\n\\t@nest .ck-rounded-corners &,\\n\\t&.ck-rounded-corners {\\n\\t\\tborder-radius: var(--ck-border-radius);\\n\\t\\t@mixin-content;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck.ck-balloon-panel.ck-tooltip{--ck-balloon-border-width:0px;--ck-balloon-arrow-offset:0px;--ck-balloon-arrow-half-width:4px;--ck-balloon-arrow-height:4px;--ck-color-panel-background:var(--ck-color-tooltip-background);padding:0 var(--ck-spacing-medium);pointer-events:none;z-index:calc(var(--ck-z-modal) + 100)}.ck.ck-balloon-panel.ck-tooltip .ck-tooltip__text{color:var(--ck-color-tooltip-text);font-size:.9em;line-height:1.5}.ck.ck-balloon-panel.ck-tooltip{box-shadow:none}.ck.ck-balloon-panel.ck-tooltip:before{display:none}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/components/tooltip/tooltip.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/components/tooltip/tooltip.css\"],\"names\":[],\"mappings\":\"AAKA,gCCGC,6BAA8B,CAC9B,6BAA8B,CAC9B,iCAAkC,CAClC,6BAA8B,CAC9B,8DAA+D,CAE/D,kCAAmC,CDPnC,mBAAoB,CAEpB,qCACD,CCMC,kDAGC,kCAAmC,CAFnC,cAAe,CACf,eAED,CAbD,gCAgBC,eAMD,CAHC,uCACC,YACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-balloon-panel.ck-tooltip {\\n\\t/* Keep tooltips transparent for any interactions. */\\n\\tpointer-events: none;\\n\\n\\tz-index: calc( var(--ck-z-modal) + 100 );\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../../../mixins/_rounded.css\\\";\\n\\n.ck.ck-balloon-panel.ck-tooltip {\\n\\t--ck-balloon-border-width: 0px;\\n\\t--ck-balloon-arrow-offset: 0px;\\n\\t--ck-balloon-arrow-half-width: 4px;\\n\\t--ck-balloon-arrow-height: 4px;\\n\\t--ck-color-panel-background: var(--ck-color-tooltip-background);\\n\\n\\tpadding: 0 var(--ck-spacing-medium);\\n\\n\\t& .ck-tooltip__text {\\n\\t\\tfont-size: .9em;\\n\\t\\tline-height: 1.5;\\n\\t\\tcolor: var(--ck-color-tooltip-text);\\n\\t}\\n\\n\\t/* Reset balloon panel styles */\\n\\tbox-shadow: none;\\n\\n\\t/* Hide the default shadow of the .ck-balloon-panel tip */\\n\\t&::before {\\n\\t\\tdisplay: none;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck-hidden{display:none!important}.ck-reset_all :not(.ck-reset_all-excluded *),.ck.ck-reset,.ck.ck-reset_all{box-sizing:border-box;height:auto;position:static;width:auto}:root{--ck-z-default:1;--ck-z-modal:calc(var(--ck-z-default) + 999)}.ck-transitions-disabled,.ck-transitions-disabled *{transition:none!important}:root{--ck-powered-by-line-height:10px;--ck-powered-by-padding-vertical:2px;--ck-powered-by-padding-horizontal:4px;--ck-powered-by-text-color:#4f4f4f;--ck-powered-by-border-radius:var(--ck-border-radius);--ck-powered-by-background:#fff;--ck-powered-by-border-color:var(--ck-color-focus-border)}.ck.ck-balloon-panel.ck-powered-by-balloon{--ck-border-radius:var(--ck-powered-by-border-radius);background:var(--ck-powered-by-background);box-shadow:none;min-height:unset;z-index:calc(var(--ck-z-modal) - 1)}.ck.ck-balloon-panel.ck-powered-by-balloon .ck.ck-powered-by{line-height:var(--ck-powered-by-line-height)}.ck.ck-balloon-panel.ck-powered-by-balloon .ck.ck-powered-by a{align-items:center;cursor:pointer;display:flex;filter:grayscale(80%);line-height:var(--ck-powered-by-line-height);opacity:.66;padding:var(--ck-powered-by-padding-vertical) var(--ck-powered-by-padding-horizontal)}.ck.ck-balloon-panel.ck-powered-by-balloon .ck.ck-powered-by .ck-powered-by__label{color:var(--ck-powered-by-text-color);cursor:pointer;font-size:7.5px;font-weight:700;letter-spacing:-.2px;line-height:normal;margin-right:4px;padding-left:2px;text-transform:uppercase}.ck.ck-balloon-panel.ck-powered-by-balloon .ck.ck-powered-by .ck-icon{cursor:pointer;display:block}.ck.ck-balloon-panel.ck-powered-by-balloon .ck.ck-powered-by:hover a{filter:grayscale(0);opacity:1}.ck.ck-balloon-panel.ck-powered-by-balloon[class*=position_inside]{border-color:transparent}.ck.ck-balloon-panel.ck-powered-by-balloon[class*=position_border]{border:var(--ck-focus-ring);border-color:var(--ck-powered-by-border-color)}:root{--ck-color-base-foreground:#fafafa;--ck-color-base-background:#fff;--ck-color-base-border:#ccced1;--ck-color-base-action:#53a336;--ck-color-base-focus:#6cb5f9;--ck-color-base-text:#333;--ck-color-base-active:#2977ff;--ck-color-base-active-focus:#0d65ff;--ck-color-base-error:#db3700;--ck-color-focus-border-coordinates:218,81.8%,56.9%;--ck-color-focus-border:hsl(var(--ck-color-focus-border-coordinates));--ck-color-focus-outer-shadow:#cae1fc;--ck-color-focus-disabled-shadow:rgba(119,186,248,.3);--ck-color-focus-error-shadow:rgba(255,64,31,.3);--ck-color-text:var(--ck-color-base-text);--ck-color-shadow-drop:rgba(0,0,0,.15);--ck-color-shadow-drop-active:rgba(0,0,0,.2);--ck-color-shadow-inner:rgba(0,0,0,.1);--ck-color-button-default-background:transparent;--ck-color-button-default-hover-background:#f0f0f0;--ck-color-button-default-active-background:#f0f0f0;--ck-color-button-default-disabled-background:transparent;--ck-color-button-on-background:#f0f7ff;--ck-color-button-on-hover-background:#dbecff;--ck-color-button-on-active-background:#dbecff;--ck-color-button-on-disabled-background:#f0f2f4;--ck-color-button-on-color:#2977ff;--ck-color-button-action-background:var(--ck-color-base-action);--ck-color-button-action-hover-background:#4d9d30;--ck-color-button-action-active-background:#4d9d30;--ck-color-button-action-disabled-background:#7ec365;--ck-color-button-action-text:var(--ck-color-base-background);--ck-color-button-save:#008a00;--ck-color-button-cancel:#db3700;--ck-color-switch-button-off-background:#939393;--ck-color-switch-button-off-hover-background:#7d7d7d;--ck-color-switch-button-on-background:var(--ck-color-button-action-background);--ck-color-switch-button-on-hover-background:#4d9d30;--ck-color-switch-button-inner-background:var(--ck-color-base-background);--ck-color-switch-button-inner-shadow:rgba(0,0,0,.1);--ck-color-dropdown-panel-background:var(--ck-color-base-background);--ck-color-dropdown-panel-border:var(--ck-color-base-border);--ck-color-input-background:var(--ck-color-base-background);--ck-color-input-border:var(--ck-color-base-border);--ck-color-input-error-border:var(--ck-color-base-error);--ck-color-input-text:var(--ck-color-base-text);--ck-color-input-disabled-background:#f2f2f2;--ck-color-input-disabled-border:var(--ck-color-base-border);--ck-color-input-disabled-text:#757575;--ck-color-list-background:var(--ck-color-base-background);--ck-color-list-button-hover-background:var(--ck-color-button-default-hover-background);--ck-color-list-button-on-background:var(--ck-color-button-on-color);--ck-color-list-button-on-background-focus:var(--ck-color-button-on-color);--ck-color-list-button-on-text:var(--ck-color-base-background);--ck-color-panel-background:var(--ck-color-base-background);--ck-color-panel-border:var(--ck-color-base-border);--ck-color-toolbar-background:var(--ck-color-base-background);--ck-color-toolbar-border:var(--ck-color-base-border);--ck-color-tooltip-background:var(--ck-color-base-text);--ck-color-tooltip-text:var(--ck-color-base-background);--ck-color-engine-placeholder-text:#707070;--ck-color-upload-bar-background:#6cb5f9;--ck-color-link-default:#0000f0;--ck-color-link-selected-background:rgba(31,176,255,.1);--ck-color-link-fake-selection:rgba(31,176,255,.3);--ck-color-highlight-background:#ff0;--ck-disabled-opacity:.5;--ck-focus-outer-shadow-geometry:0 0 0 3px;--ck-focus-outer-shadow:var(--ck-focus-outer-shadow-geometry) var(--ck-color-focus-outer-shadow);--ck-focus-disabled-outer-shadow:var(--ck-focus-outer-shadow-geometry) var(--ck-color-focus-disabled-shadow);--ck-focus-error-outer-shadow:var(--ck-focus-outer-shadow-geometry) var(--ck-color-focus-error-shadow);--ck-focus-ring:1px solid var(--ck-color-focus-border);--ck-font-size-base:13px;--ck-line-height-base:1.84615;--ck-font-face:Helvetica,Arial,Tahoma,Verdana,Sans-Serif;--ck-font-size-tiny:0.7em;--ck-font-size-small:0.75em;--ck-font-size-normal:1em;--ck-font-size-big:1.4em;--ck-font-size-large:1.8em;--ck-ui-component-min-height:2.3em}.ck-reset_all :not(.ck-reset_all-excluded *),.ck.ck-reset,.ck.ck-reset_all{word-wrap:break-word;background:transparent;border:0;margin:0;padding:0;text-decoration:none;transition:none;vertical-align:middle}.ck-reset_all :not(.ck-reset_all-excluded *),.ck.ck-reset_all{border-collapse:collapse;color:var(--ck-color-text);cursor:auto;float:none;font:normal normal normal var(--ck-font-size-base)/var(--ck-line-height-base) var(--ck-font-face);text-align:left;white-space:nowrap}.ck-reset_all .ck-rtl :not(.ck-reset_all-excluded *){text-align:right}.ck-reset_all iframe:not(.ck-reset_all-excluded *){vertical-align:inherit}.ck-reset_all textarea:not(.ck-reset_all-excluded *){white-space:pre-wrap}.ck-reset_all input[type=password]:not(.ck-reset_all-excluded *),.ck-reset_all input[type=text]:not(.ck-reset_all-excluded *),.ck-reset_all textarea:not(.ck-reset_all-excluded *){cursor:text}.ck-reset_all input[type=password][disabled]:not(.ck-reset_all-excluded *),.ck-reset_all input[type=text][disabled]:not(.ck-reset_all-excluded *),.ck-reset_all textarea[disabled]:not(.ck-reset_all-excluded *){cursor:default}.ck-reset_all fieldset:not(.ck-reset_all-excluded *){border:2px groove #dfdee3;padding:10px}.ck-reset_all button:not(.ck-reset_all-excluded *)::-moz-focus-inner{border:0;padding:0}.ck[dir=rtl],.ck[dir=rtl] .ck{text-align:right}:root{--ck-border-radius:2px;--ck-inner-shadow:2px 2px 3px var(--ck-color-shadow-inner) inset;--ck-drop-shadow:0 1px 2px 1px var(--ck-color-shadow-drop);--ck-drop-shadow-active:0 3px 6px 1px var(--ck-color-shadow-drop-active);--ck-spacing-unit:0.6em;--ck-spacing-large:calc(var(--ck-spacing-unit)*1.5);--ck-spacing-standard:var(--ck-spacing-unit);--ck-spacing-medium:calc(var(--ck-spacing-unit)*0.8);--ck-spacing-small:calc(var(--ck-spacing-unit)*0.5);--ck-spacing-tiny:calc(var(--ck-spacing-unit)*0.3);--ck-spacing-extra-tiny:calc(var(--ck-spacing-unit)*0.16)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-ui/theme/globals/_hidden.css\",\"webpack://./../ckeditor5-ui/theme/globals/_reset.css\",\"webpack://./../ckeditor5-ui/theme/globals/_zindex.css\",\"webpack://./../ckeditor5-ui/theme/globals/_transition.css\",\"webpack://./../ckeditor5-ui/theme/globals/_poweredby.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/globals/_colors.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/globals/_disabled.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/globals/_focus.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/globals/_fonts.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/globals/_reset.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/globals/_rounded.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/globals/_shadow.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-ui/globals/_spacing.css\"],\"names\":[],\"mappings\":\"AAQA,WAGC,sBACD,CCPA,2EAGC,qBAAsB,CAEtB,WAAY,CACZ,eAAgB,CAFhB,UAGD,CCPA,MACC,gBAAiB,CACjB,4CACD,CCAA,oDAEC,yBACD,CCNA,MACC,gCAAiC,CACjC,oCAAqC,CACrC,sCAAuC,CACvC,kCAA2C,CAC3C,qDAAsD,CACtD,+BAA4C,CAC5C,yDACD,CAEA,2CACC,qDAAsD,CAGtD,0CAA2C,CAD3C,eAAgB,CAEhB,gBAAiB,CACjB,mCAiDD,CA/CC,6DACC,4CAoCD,CAlCC,+DAGC,kBAAmB,CAFnB,cAAe,CACf,YAAa,CAGb,qBAAsB,CACtB,4CAA6C,CAF7C,WAAY,CAGZ,qFACD,CAEA,mFASC,qCAAsC,CAFtC,cAAe,CANf,eAAgB,CAIhB,eAAiB,CAHjB,oBAAqB,CAMrB,kBAAmB,CAFnB,gBAAiB,CAHjB,gBAAiB,CACjB,wBAOD,CAEA,sEAEC,cAAe,CADf,aAED,CAGC,qEACC,mBAAqB,CACrB,SACD,CAIF,mEACC,wBACD,CAEA,mEACC,2BAA4B,CAC5B,8CACD,CChED,MACC,kCAAmD,CACnD,+BAAoD,CACpD,8BAAkD,CAClD,8BAAuD,CACvD,6BAAmD,CACnD,yBAA+C,CAC/C,8BAAsD,CACtD,oCAA4D,CAC5D,6BAAkD,CAIlD,mDAA4D,CAC5D,qEAA+E,CAC/E,qCAA4D,CAC5D,qDAA8D,CAC9D,gDAAyD,CACzD,yCAAqD,CACrD,sCAAsD,CACtD,4CAA0D,CAC1D,sCAAsD,CAItD,gDAAuD,CACvD,kDAAiE,CACjE,mDAAkE,CAClE,yDAA8D,CAE9D,uCAA6D,CAC7D,6CAAoE,CACpE,8CAAoE,CACpE,gDAAiE,CACjE,kCAAyD,CAGzD,+DAAsE,CACtE,iDAAsE,CACtE,kDAAsE,CACtE,oDAAoE,CACpE,6DAAsE,CAEtE,8BAAoD,CACpD,gCAAqD,CAErD,+CAA8D,CAC9D,qDAAiE,CACjE,+EAAqF,CACrF,oDAAuE,CACvE,yEAA8E,CAC9E,oDAAgE,CAIhE,oEAA2E,CAC3E,4DAAoE,CAIpE,2DAAoE,CACpE,mDAA6D,CAC7D,wDAAgE,CAChE,+CAA0D,CAC1D,4CAA2D,CAC3D,4DAAoE,CACpE,sCAAsD,CAItD,0DAAmE,CACnE,uFAA6F,CAC7F,oEAA2E,CAC3E,0EAA+E,CAC/E,8DAAsE,CAItE,2DAAoE,CACpE,mDAA6D,CAI7D,6DAAsE,CACtE,qDAA+D,CAI/D,uDAAgE,CAChE,uDAAiE,CAIjE,0CAAyD,CAIzD,wCAA2D,CAI3D,+BAAoD,CACpD,uDAAmE,CACnE,kDAAgE,CAIhE,oCAAwD,CCvGxD,wBAAyB,CCAzB,0CAA2C,CAK3C,gGAAiG,CAKjG,4GAA6G,CAK7G,sGAAuG,CAKvG,sDAAuD,CCvBvD,wBAAyB,CACzB,6BAA8B,CAC9B,wDAA6D,CAE7D,yBAA0B,CAC1B,2BAA4B,CAC5B,yBAA0B,CAC1B,wBAAyB,CACzB,0BAA2B,CCJ3B,kCJuGD,CIjGA,2EAaC,oBAAqB,CANrB,sBAAuB,CADvB,QAAS,CAFT,QAAS,CACT,SAAU,CAGV,oBAAqB,CAErB,eAAgB,CADhB,qBAKD,CAKA,8DAGC,wBAAyB,CAEzB,0BAA2B,CAG3B,WAAY,CACZ,UAAW,CALX,iGAAkG,CAElG,eAAgB,CAChB,kBAGD,CAGC,qDACC,gBACD,CAEA,mDAEC,sBACD,CAEA,qDACC,oBACD,CAEA,mLAGC,WACD,CAEA,iNAGC,cACD,CAEA,qDAEC,yBAAoC,CADpC,YAED,CAEA,qEAGC,QAAQ,CADR,SAED,CAMD,8BAEC,gBACD,CCnFA,MACC,sBAAuB,CCAvB,gEAAiE,CAKjE,0DAA2D,CAK3D,wEAAyE,CCbzE,uBAA8B,CAC9B,mDAA2D,CAC3D,4CAAkD,CAClD,oDAA4D,CAC5D,mDAA2D,CAC3D,kDAA2D,CAC3D,yDFFD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A class which hides an element in DOM.\\n */\\n.ck-hidden {\\n\\t/* Override selector specificity. Otherwise, all elements with some display\\n\\tstyle defined will override this one, which is not a desired result. */\\n\\tdisplay: none !important;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck.ck-reset,\\n.ck.ck-reset_all,\\n.ck-reset_all *:not(.ck-reset_all-excluded *) {\\n\\tbox-sizing: border-box;\\n\\twidth: auto;\\n\\theight: auto;\\n\\tposition: static;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-z-default: 1;\\n\\t--ck-z-modal: calc( var(--ck-z-default) + 999 );\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A class that disables all transitions of the element and its children.\\n */\\n.ck-transitions-disabled,\\n.ck-transitions-disabled * {\\n\\ttransition: none !important;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-powered-by-line-height: 10px;\\n\\t--ck-powered-by-padding-vertical: 2px;\\n\\t--ck-powered-by-padding-horizontal: 4px;\\n\\t--ck-powered-by-text-color: hsl(0, 0%, 31%);\\n\\t--ck-powered-by-border-radius: var(--ck-border-radius);\\n\\t--ck-powered-by-background: hsl(0, 0%, 100%);\\n\\t--ck-powered-by-border-color: var(--ck-color-focus-border);\\n}\\n\\n.ck.ck-balloon-panel.ck-powered-by-balloon {\\n\\t--ck-border-radius: var(--ck-powered-by-border-radius);\\n\\n\\tbox-shadow: none;\\n\\tbackground: var(--ck-powered-by-background);\\n\\tmin-height: unset;\\n\\tz-index: calc( var(--ck-z-modal) - 1 );\\n\\n\\t& .ck.ck-powered-by {\\n\\t\\tline-height: var(--ck-powered-by-line-height);\\n\\n\\t\\t& a {\\n\\t\\t\\tcursor: pointer;\\n\\t\\t\\tdisplay: flex;\\n\\t\\t\\talign-items: center;\\n\\t\\t\\topacity: .66;\\n\\t\\t\\tfilter: grayscale(80%);\\n\\t\\t\\tline-height: var(--ck-powered-by-line-height);\\n\\t\\t\\tpadding: var(--ck-powered-by-padding-vertical) var(--ck-powered-by-padding-horizontal);\\n\\t\\t}\\n\\n\\t\\t& .ck-powered-by__label {\\n\\t\\t\\tfont-size: 7.5px;\\n\\t\\t\\tletter-spacing: -.2px;\\n\\t\\t\\tpadding-left: 2px;\\n\\t\\t\\ttext-transform: uppercase;\\n\\t\\t\\tfont-weight: bold;\\n\\t\\t\\tmargin-right: 4px;\\n\\t\\t\\tcursor: pointer;\\n\\t\\t\\tline-height: normal;\\n\\t\\t\\tcolor: var(--ck-powered-by-text-color);\\n\\n\\t\\t}\\n\\n\\t\\t& .ck-icon {\\n\\t\\t\\tdisplay: block;\\n\\t\\t\\tcursor: pointer;\\n\\t\\t}\\n\\n\\t\\t&:hover {\\n\\t\\t\\t& a {\\n\\t\\t\\t\\tfilter: grayscale(0%);\\n\\t\\t\\t\\topacity: 1;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t&[class*=\\\"position_inside\\\"] {\\n\\t\\tborder-color: transparent;\\n\\t}\\n\\n\\t&[class*=\\\"position_border\\\"] {\\n\\t\\tborder: var(--ck-focus-ring);\\n\\t\\tborder-color: var(--ck-powered-by-border-color);\\n\\t}\\n}\\n\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-base-foreground: \\t\\t\\t\\t\\t\\t\\t\\thsl(0, 0%, 98%);\\n\\t--ck-color-base-background: \\t\\t\\t\\t\\t\\t\\t\\thsl(0, 0%, 100%);\\n\\t--ck-color-base-border: \\t\\t\\t\\t\\t\\t\\t\\t\\thsl(220, 6%, 81%);\\n\\t--ck-color-base-action: \\t\\t\\t\\t\\t\\t\\t\\t\\thsl(104, 50.2%, 42.5%);\\n\\t--ck-color-base-focus: \\t\\t\\t\\t\\t\\t\\t\\t\\t\\thsl(209, 92%, 70%);\\n\\t--ck-color-base-text: \\t\\t\\t\\t\\t\\t\\t\\t\\t\\thsl(0, 0%, 20%);\\n\\t--ck-color-base-active: \\t\\t\\t\\t\\t\\t\\t\\t\\thsl(218.1, 100%, 58%);\\n\\t--ck-color-base-active-focus:\\t\\t\\t\\t\\t\\t\\t\\thsl(218.2, 100%, 52.5%);\\n\\t--ck-color-base-error:\\t\\t\\t\\t\\t\\t\\t\\t\\t\\thsl(15, 100%, 43%);\\n\\n\\t/* -- Generic colors ------------------------------------------------------------------------ */\\n\\n\\t--ck-color-focus-border-coordinates: \\t\\t\\t\\t\\t\\t218, 81.8%, 56.9%;\\n\\t--ck-color-focus-border: \\t\\t\\t\\t\\t\\t\\t\\t\\thsl(var(--ck-color-focus-border-coordinates));\\n\\t--ck-color-focus-outer-shadow:\\t\\t\\t\\t\\t\\t\\t\\thsl(212.4, 89.3%, 89%);\\n\\t--ck-color-focus-disabled-shadow:\\t\\t\\t\\t\\t\\t\\thsla(209, 90%, 72%,.3);\\n\\t--ck-color-focus-error-shadow:\\t\\t\\t\\t\\t\\t\\t\\thsla(9,100%,56%,.3);\\n\\t--ck-color-text: \\t\\t\\t\\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-text);\\n\\t--ck-color-shadow-drop: \\t\\t\\t\\t\\t\\t\\t\\t\\thsla(0, 0%, 0%, 0.15);\\n\\t--ck-color-shadow-drop-active:\\t\\t\\t\\t\\t\\t\\t\\thsla(0, 0%, 0%, 0.2);\\n\\t--ck-color-shadow-inner: \\t\\t\\t\\t\\t\\t\\t\\t\\thsla(0, 0%, 0%, 0.1);\\n\\n\\t/* -- Buttons ------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-button-default-background: \\t\\t\\t\\t\\t\\ttransparent;\\n\\t--ck-color-button-default-hover-background: \\t\\t\\t\\thsl(0, 0%, 94.1%);\\n\\t--ck-color-button-default-active-background: \\t\\t\\t\\thsl(0, 0%, 94.1%);\\n\\t--ck-color-button-default-disabled-background: \\t\\t\\t\\ttransparent;\\n\\n\\t--ck-color-button-on-background: \\t\\t\\t\\t\\t\\t\\thsl(212, 100%, 97.1%);\\n\\t--ck-color-button-on-hover-background: \\t\\t\\t\\t\\t\\thsl(211.7, 100%, 92.9%);\\n\\t--ck-color-button-on-active-background: \\t\\t\\t\\t\\thsl(211.7, 100%, 92.9%);\\n\\t--ck-color-button-on-disabled-background: \\t\\t\\t\\t\\thsl(211, 15%, 95%);\\n\\t--ck-color-button-on-color:\\t\\t\\t\\t\\t\\t\\t\\t\\thsl(218.1, 100%, 58%);\\n\\n\\n\\t--ck-color-button-action-background: \\t\\t\\t\\t\\t\\tvar(--ck-color-base-action);\\n\\t--ck-color-button-action-hover-background: \\t\\t\\t\\t\\thsl(104, 53.2%, 40.2%);\\n\\t--ck-color-button-action-active-background: \\t\\t\\t\\thsl(104, 53.2%, 40.2%);\\n\\t--ck-color-button-action-disabled-background: \\t\\t\\t\\thsl(104, 44%, 58%);\\n\\t--ck-color-button-action-text: \\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\n\\t--ck-color-button-save: \\t\\t\\t\\t\\t\\t\\t\\t\\thsl(120, 100%, 27%);\\n\\t--ck-color-button-cancel: \\t\\t\\t\\t\\t\\t\\t\\t\\thsl(15, 100%, 43%);\\n\\n\\t--ck-color-switch-button-off-background:\\t\\t\\t\\t\\thsl(0, 0%, 57.6%);\\n\\t--ck-color-switch-button-off-hover-background:\\t\\t\\t\\thsl(0, 0%, 49%);\\n\\t--ck-color-switch-button-on-background:\\t\\t\\t\\t\\t\\tvar(--ck-color-button-action-background);\\n\\t--ck-color-switch-button-on-hover-background:\\t\\t\\t\\thsl(104, 53.2%, 40.2%);\\n\\t--ck-color-switch-button-inner-background:\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\t--ck-color-switch-button-inner-shadow:\\t\\t\\t\\t\\t\\thsla(0, 0%, 0%, 0.1);\\n\\n\\t/* -- Dropdown ------------------------------------------------------------------------------ */\\n\\n\\t--ck-color-dropdown-panel-background: \\t\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\t--ck-color-dropdown-panel-border: \\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-border);\\n\\n\\t/* -- Input --------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-input-background: \\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\t--ck-color-input-border: \\t\\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-border);\\n\\t--ck-color-input-error-border:\\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-error);\\n\\t--ck-color-input-text: \\t\\t\\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-text);\\n\\t--ck-color-input-disabled-background: \\t\\t\\t\\t\\t\\thsl(0, 0%, 95%);\\n\\t--ck-color-input-disabled-border: \\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-border);\\n\\t--ck-color-input-disabled-text: \\t\\t\\t\\t\\t\\t\\thsl(0, 0%, 46%);\\n\\n\\t/* -- List ---------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-list-background: \\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\t--ck-color-list-button-hover-background: \\t\\t\\t\\t\\tvar(--ck-color-button-default-hover-background);\\n\\t--ck-color-list-button-on-background: \\t\\t\\t\\t\\t\\tvar(--ck-color-button-on-color);\\n\\t--ck-color-list-button-on-background-focus: \\t\\t\\t\\tvar(--ck-color-button-on-color);\\n\\t--ck-color-list-button-on-text:\\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\n\\t/* -- Panel --------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-panel-background: \\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\t--ck-color-panel-border: \\t\\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-border);\\n\\n\\t/* -- Toolbar ------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-toolbar-background: \\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\t--ck-color-toolbar-border: \\t\\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-border);\\n\\n\\t/* -- Tooltip ------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-tooltip-background: \\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-text);\\n\\t--ck-color-tooltip-text: \\t\\t\\t\\t\\t\\t\\t\\t\\tvar(--ck-color-base-background);\\n\\n\\t/* -- Engine -------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-engine-placeholder-text: \\t\\t\\t\\t\\t\\thsl(0, 0%, 44%);\\n\\n\\t/* -- Upload -------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-upload-bar-background:\\t\\t \\t\\t\\t\\t\\thsl(209, 92%, 70%);\\n\\n\\t/* -- Link -------------------------------------------------------------------------------- */\\n\\n\\t--ck-color-link-default:\\t\\t\\t\\t\\t\\t\\t\\t\\thsl(240, 100%, 47%);\\n\\t--ck-color-link-selected-background:\\t\\t\\t\\t\\t\\thsla(201, 100%, 56%, 0.1);\\n\\t--ck-color-link-fake-selection:\\t\\t\\t\\t\\t\\t\\t\\thsla(201, 100%, 56%, 0.3);\\n\\n\\t/* -- Search result highlight ---------------------------------------------------------------- */\\n\\n\\t--ck-color-highlight-background:\\t\\t\\t\\t\\t\\t\\thsl(60, 100%, 50%)\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t/**\\n\\t * An opacity value of disabled UI item.\\n\\t */\\n\\t--ck-disabled-opacity: .5;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t/**\\n\\t * The geometry of the of focused element's outer shadow.\\n\\t */\\n\\t--ck-focus-outer-shadow-geometry: 0 0 0 3px;\\n\\n\\t/**\\n\\t * A visual style of focused element's outer shadow.\\n\\t */\\n\\t--ck-focus-outer-shadow: var(--ck-focus-outer-shadow-geometry) var(--ck-color-focus-outer-shadow);\\n\\n\\t/**\\n\\t * A visual style of focused element's outer shadow (when disabled).\\n\\t */\\n\\t--ck-focus-disabled-outer-shadow: var(--ck-focus-outer-shadow-geometry) var(--ck-color-focus-disabled-shadow);\\n\\n\\t/**\\n\\t * A visual style of focused element's outer shadow (when has errors).\\n\\t */\\n\\t--ck-focus-error-outer-shadow: var(--ck-focus-outer-shadow-geometry) var(--ck-color-focus-error-shadow);\\n\\n\\t/**\\n\\t * A visual style of focused element's border or outline.\\n\\t */\\n\\t--ck-focus-ring: 1px solid var(--ck-color-focus-border);\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-font-size-base: 13px;\\n\\t--ck-line-height-base: 1.84615;\\n\\t--ck-font-face: Helvetica, Arial, Tahoma, Verdana, Sans-Serif;\\n\\n\\t--ck-font-size-tiny: 0.7em;\\n\\t--ck-font-size-small: 0.75em;\\n\\t--ck-font-size-normal: 1em;\\n\\t--ck-font-size-big: 1.4em;\\n\\t--ck-font-size-large: 1.8em;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t/* This is super-important. This is **manually** adjusted so a button without an icon\\n\\tis never smaller than a button with icon, additionally making sure that text-less buttons\\n\\tare perfect squares. The value is also shared by other components which should stay \\\"in-line\\\"\\n\\twith buttons. */\\n\\t--ck-ui-component-min-height: 2.3em;\\n}\\n\\n/**\\n * Resets an element, ignoring its children.\\n */\\n.ck.ck-reset,\\n.ck.ck-reset_all,\\n.ck-reset_all *:not(.ck-reset_all-excluded *) {\\n\\t/* Do not include inheritable rules here. */\\n\\tmargin: 0;\\n\\tpadding: 0;\\n\\tborder: 0;\\n\\tbackground: transparent;\\n\\ttext-decoration: none;\\n\\tvertical-align: middle;\\n\\ttransition: none;\\n\\n\\t/* https://github.com/ckeditor/ckeditor5-theme-lark/issues/105 */\\n\\tword-wrap: break-word;\\n}\\n\\n/**\\n * Resets an element AND its children.\\n */\\n.ck.ck-reset_all,\\n.ck-reset_all *:not(.ck-reset_all-excluded *) {\\n\\t/* These are rule inherited by all children elements. */\\n\\tborder-collapse: collapse;\\n\\tfont: normal normal normal var(--ck-font-size-base)/var(--ck-line-height-base) var(--ck-font-face);\\n\\tcolor: var(--ck-color-text);\\n\\ttext-align: left;\\n\\twhite-space: nowrap;\\n\\tcursor: auto;\\n\\tfloat: none;\\n}\\n\\n.ck-reset_all {\\n\\t& .ck-rtl *:not(.ck-reset_all-excluded *) {\\n\\t\\ttext-align: right;\\n\\t}\\n\\n\\t& iframe:not(.ck-reset_all-excluded *) {\\n\\t\\t/* For IE */\\n\\t\\tvertical-align: inherit;\\n\\t}\\n\\n\\t& textarea:not(.ck-reset_all-excluded *) {\\n\\t\\twhite-space: pre-wrap;\\n\\t}\\n\\n\\t& textarea:not(.ck-reset_all-excluded *),\\n\\t& input[type=\\\"text\\\"]:not(.ck-reset_all-excluded *),\\n\\t& input[type=\\\"password\\\"]:not(.ck-reset_all-excluded *) {\\n\\t\\tcursor: text;\\n\\t}\\n\\n\\t& textarea[disabled]:not(.ck-reset_all-excluded *),\\n\\t& input[type=\\\"text\\\"][disabled]:not(.ck-reset_all-excluded *),\\n\\t& input[type=\\\"password\\\"][disabled]:not(.ck-reset_all-excluded *) {\\n\\t\\tcursor: default;\\n\\t}\\n\\n\\t& fieldset:not(.ck-reset_all-excluded *) {\\n\\t\\tpadding: 10px;\\n\\t\\tborder: 2px groove hsl(255, 7%, 88%);\\n\\t}\\n\\n\\t& button:not(.ck-reset_all-excluded *)::-moz-focus-inner {\\n\\t\\t/* See http://stackoverflow.com/questions/5517744/remove-extra-button-spacing-padding-in-firefox */\\n\\t\\tpadding: 0;\\n\\t\\tborder: 0\\n\\t}\\n}\\n\\n/**\\n * Default UI rules for RTL languages.\\n */\\n.ck[dir=\\\"rtl\\\"],\\n.ck[dir=\\\"rtl\\\"] .ck {\\n\\ttext-align: right;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * Default border-radius value.\\n */\\n:root{\\n\\t--ck-border-radius: 2px;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t/**\\n\\t * A visual style of element's inner shadow (i.e. input).\\n\\t */\\n\\t--ck-inner-shadow: 2px 2px 3px var(--ck-color-shadow-inner) inset;\\n\\n\\t/**\\n\\t * A visual style of element's drop shadow (i.e. panel).\\n\\t */\\n\\t--ck-drop-shadow: 0 1px 2px 1px var(--ck-color-shadow-drop);\\n\\n\\t/**\\n\\t * A visual style of element's active shadow (i.e. comment or suggestion).\\n\\t */\\n\\t--ck-drop-shadow-active: 0 3px 6px 1px var(--ck-color-shadow-drop-active);\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-spacing-unit: \\t\\t\\t\\t\\t\\t0.6em;\\n\\t--ck-spacing-large: \\t\\t\\t\\t\\tcalc(var(--ck-spacing-unit) * 1.5);\\n\\t--ck-spacing-standard: \\t\\t\\t\\t\\tvar(--ck-spacing-unit);\\n\\t--ck-spacing-medium: \\t\\t\\t\\t\\tcalc(var(--ck-spacing-unit) * 0.8);\\n\\t--ck-spacing-small: \\t\\t\\t\\t\\tcalc(var(--ck-spacing-unit) * 0.5);\\n\\t--ck-spacing-tiny: \\t\\t\\t\\t\\t\\tcalc(var(--ck-spacing-unit) * 0.3);\\n\\t--ck-spacing-extra-tiny: \\t\\t\\t\\tcalc(var(--ck-spacing-unit) * 0.16);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \":root{--ck-color-resizer:var(--ck-color-focus-border);--ck-color-resizer-tooltip-background:#262626;--ck-color-resizer-tooltip-text:#f2f2f2;--ck-resizer-border-radius:var(--ck-border-radius);--ck-resizer-tooltip-offset:10px;--ck-resizer-tooltip-height:calc(var(--ck-spacing-small)*2 + 10px)}.ck .ck-widget,.ck .ck-widget.ck-widget_with-selection-handle{position:relative}.ck .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle{position:absolute}.ck .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle .ck-icon{display:block}.ck .ck-widget.ck-widget_with-selection-handle.ck-widget_selected>.ck-widget__selection-handle,.ck .ck-widget.ck-widget_with-selection-handle:hover>.ck-widget__selection-handle{visibility:visible}.ck .ck-size-view{background:var(--ck-color-resizer-tooltip-background);border:1px solid var(--ck-color-resizer-tooltip-text);border-radius:var(--ck-resizer-border-radius);color:var(--ck-color-resizer-tooltip-text);display:block;font-size:var(--ck-font-size-tiny);height:var(--ck-resizer-tooltip-height);line-height:var(--ck-resizer-tooltip-height);padding:0 var(--ck-spacing-small)}.ck .ck-size-view.ck-orientation-above-center,.ck .ck-size-view.ck-orientation-bottom-left,.ck .ck-size-view.ck-orientation-bottom-right,.ck .ck-size-view.ck-orientation-top-left,.ck .ck-size-view.ck-orientation-top-right{position:absolute}.ck .ck-size-view.ck-orientation-top-left{left:var(--ck-resizer-tooltip-offset);top:var(--ck-resizer-tooltip-offset)}.ck .ck-size-view.ck-orientation-top-right{right:var(--ck-resizer-tooltip-offset);top:var(--ck-resizer-tooltip-offset)}.ck .ck-size-view.ck-orientation-bottom-right{bottom:var(--ck-resizer-tooltip-offset);right:var(--ck-resizer-tooltip-offset)}.ck .ck-size-view.ck-orientation-bottom-left{bottom:var(--ck-resizer-tooltip-offset);left:var(--ck-resizer-tooltip-offset)}.ck .ck-size-view.ck-orientation-above-center{left:50%;top:calc(var(--ck-resizer-tooltip-height)*-1);transform:translate(-50%)}:root{--ck-widget-outline-thickness:3px;--ck-widget-handler-icon-size:16px;--ck-widget-handler-animation-duration:200ms;--ck-widget-handler-animation-curve:ease;--ck-color-widget-blurred-border:#dedede;--ck-color-widget-hover-border:#ffc83d;--ck-color-widget-editable-focus-background:var(--ck-color-base-background);--ck-color-widget-drag-handler-icon-color:var(--ck-color-base-background)}.ck .ck-widget{outline-color:transparent;outline-style:solid;outline-width:var(--ck-widget-outline-thickness);transition:outline-color var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve)}.ck .ck-widget.ck-widget_selected,.ck .ck-widget.ck-widget_selected:hover{outline:var(--ck-widget-outline-thickness) solid var(--ck-color-focus-border)}.ck .ck-widget:hover{outline-color:var(--ck-color-widget-hover-border)}.ck .ck-editor__nested-editable{border:1px solid transparent}.ck .ck-editor__nested-editable.ck-editor__nested-editable_focused,.ck .ck-editor__nested-editable:focus{background-color:var(--ck-color-widget-editable-focus-background);border:var(--ck-focus-ring);box-shadow:var(--ck-inner-shadow),0 0;outline:none}.ck .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle{background-color:transparent;border-radius:var(--ck-border-radius) var(--ck-border-radius) 0 0;box-sizing:border-box;left:calc(0px - var(--ck-widget-outline-thickness));opacity:0;padding:4px;top:0;transform:translateY(-100%);transition:background-color var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve),visibility var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve),opacity var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve)}.ck .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle .ck-icon{color:var(--ck-color-widget-drag-handler-icon-color);height:var(--ck-widget-handler-icon-size);width:var(--ck-widget-handler-icon-size)}.ck .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle .ck-icon .ck-icon__selected-indicator{opacity:0;transition:opacity .3s var(--ck-widget-handler-animation-curve)}.ck .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle:hover .ck-icon .ck-icon__selected-indicator{opacity:1}.ck .ck-widget.ck-widget_with-selection-handle:hover>.ck-widget__selection-handle{background-color:var(--ck-color-widget-hover-border);opacity:1}.ck .ck-widget.ck-widget_with-selection-handle.ck-widget_selected:hover>.ck-widget__selection-handle,.ck .ck-widget.ck-widget_with-selection-handle.ck-widget_selected>.ck-widget__selection-handle{background-color:var(--ck-color-focus-border);opacity:1}.ck .ck-widget.ck-widget_with-selection-handle.ck-widget_selected:hover>.ck-widget__selection-handle .ck-icon .ck-icon__selected-indicator,.ck .ck-widget.ck-widget_with-selection-handle.ck-widget_selected>.ck-widget__selection-handle .ck-icon .ck-icon__selected-indicator{opacity:1}.ck[dir=rtl] .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle{left:auto;right:calc(0px - var(--ck-widget-outline-thickness))}.ck.ck-editor__editable.ck-read-only .ck-widget{transition:none}.ck.ck-editor__editable.ck-read-only .ck-widget:not(.ck-widget_selected){--ck-widget-outline-thickness:0px}.ck.ck-editor__editable.ck-read-only .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle,.ck.ck-editor__editable.ck-read-only .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle:hover{background:var(--ck-color-widget-blurred-border)}.ck.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected,.ck.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected:hover{outline-color:var(--ck-color-widget-blurred-border)}.ck.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected.ck-widget_with-selection-handle:hover>.ck-widget__selection-handle,.ck.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected.ck-widget_with-selection-handle:hover>.ck-widget__selection-handle:hover,.ck.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected.ck-widget_with-selection-handle>.ck-widget__selection-handle,.ck.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected.ck-widget_with-selection-handle>.ck-widget__selection-handle:hover{background:var(--ck-color-widget-blurred-border)}.ck.ck-editor__editable blockquote>.ck-widget.ck-widget_with-selection-handle:first-child,.ck.ck-editor__editable>.ck-widget.ck-widget_with-selection-handle:first-child{margin-top:calc(1em + var(--ck-widget-handler-icon-size))}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-widget/theme/widget.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-widget/widget.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_focus.css\",\"webpack://./../ckeditor5-theme-lark/theme/mixins/_shadow.css\"],\"names\":[],\"mappings\":\"AAKA,MACC,+CAAgD,CAChD,6CAAsD,CACtD,uCAAgD,CAEhD,kDAAmD,CACnD,gCAAiC,CACjC,kEACD,CAOA,8DAEC,iBAqBD,CAnBC,4EACC,iBAOD,CALC,qFAGC,aACD,CASD,iLACC,kBACD,CAGD,kBACC,qDAAsD,CAEtD,qDAAsD,CACtD,6CAA8C,CAF9C,0CAA2C,CAI3C,aAAc,CADd,kCAAmC,CAGnC,uCAAwC,CACxC,4CAA6C,CAF7C,iCAsCD,CAlCC,8NAKC,iBACD,CAEA,0CAEC,qCAAsC,CADtC,oCAED,CAEA,2CAEC,sCAAuC,CADvC,oCAED,CAEA,8CACC,uCAAwC,CACxC,sCACD,CAEA,6CACC,uCAAwC,CACxC,qCACD,CAGA,8CAEC,QAAS,CADT,6CAAgD,CAEhD,yBACD,CCjFD,MACC,iCAAkC,CAClC,kCAAmC,CACnC,4CAA6C,CAC7C,wCAAyC,CAEzC,wCAAiD,CACjD,sCAAkD,CAClD,2EAA4E,CAC5E,yEACD,CAEA,eAGC,yBAA0B,CAD1B,mBAAoB,CADpB,gDAAiD,CAGjD,6GAUD,CARC,0EAEC,6EACD,CAEA,qBACC,iDACD,CAGD,gCACC,4BAWD,CAPC,yGAKC,iEAAkE,CCnCnE,2BAA2B,CCF3B,qCAA8B,CDC9B,YDqCA,CAIA,4EAKC,4BAA6B,CAa7B,iEAAkE,CAhBlE,qBAAsB,CAoBtB,mDAAoD,CAhBpD,SAAU,CALV,WAAY,CAsBZ,KAAM,CAFN,2BAA4B,CAT5B,6SAgCD,CAnBC,qFAIC,oDAAqD,CADrD,yCAA0C,CAD1C,wCAWD,CANC,kHACC,SAAU,CAGV,+DACD,CAID,wHACC,SACD,CAID,kFAEC,oDAAqD,CADrD,SAED,CAKC,oMAEC,6CAA8C,CAD9C,SAOD,CAHC,gRACC,SACD,CAOH,qFACC,SAAU,CACV,oDACD,CAGA,gDAEC,eAkBD,CAhBC,yEAOC,iCACD,CAGC,gOAEC,gDACD,CAOD,wIAEC,mDAQD,CALE,ghBAEC,gDACD,CAKH,yKAOC,yDACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-color-resizer: var(--ck-color-focus-border);\\n\\t--ck-color-resizer-tooltip-background: hsl(0, 0%, 15%);\\n\\t--ck-color-resizer-tooltip-text: hsl(0, 0%, 95%);\\n\\n\\t--ck-resizer-border-radius: var(--ck-border-radius);\\n\\t--ck-resizer-tooltip-offset: 10px;\\n\\t--ck-resizer-tooltip-height: calc(var(--ck-spacing-small) * 2 + 10px);\\n}\\n\\n.ck .ck-widget {\\n\\t/* This is neccessary for type around UI to be positioned properly. */\\n\\tposition: relative;\\n}\\n\\n.ck .ck-widget.ck-widget_with-selection-handle {\\n\\t/* Make the widget wrapper a relative positioning container for the drag handle. */\\n\\tposition: relative;\\n\\n\\t& .ck-widget__selection-handle {\\n\\t\\tposition: absolute;\\n\\n\\t\\t& .ck-icon {\\n\\t\\t\\t/* Make sure the icon in not a subject to font-size or line-height to avoid\\n\\t\\t\\tunnecessary spacing around it. */\\n\\t\\t\\tdisplay: block;\\n\\t\\t}\\n\\t}\\n\\n\\t/* Show the selection handle on mouse hover over the widget, but not for nested widgets. */\\n\\t&:hover > .ck-widget__selection-handle {\\n\\t\\tvisibility: visible;\\n\\t}\\n\\n\\t/* Show the selection handle when the widget is selected, but not for nested widgets. */\\n\\t&.ck-widget_selected > .ck-widget__selection-handle {\\n\\t\\tvisibility: visible;\\n\\t}\\n}\\n\\n.ck .ck-size-view {\\n\\tbackground: var(--ck-color-resizer-tooltip-background);\\n\\tcolor: var(--ck-color-resizer-tooltip-text);\\n\\tborder: 1px solid var(--ck-color-resizer-tooltip-text);\\n\\tborder-radius: var(--ck-resizer-border-radius);\\n\\tfont-size: var(--ck-font-size-tiny);\\n\\tdisplay: block;\\n\\tpadding: 0 var(--ck-spacing-small);\\n\\theight: var(--ck-resizer-tooltip-height);\\n\\tline-height: var(--ck-resizer-tooltip-height);\\n\\n\\t&.ck-orientation-top-left,\\n\\t&.ck-orientation-top-right,\\n\\t&.ck-orientation-bottom-right,\\n\\t&.ck-orientation-bottom-left,\\n\\t&.ck-orientation-above-center {\\n\\t\\tposition: absolute;\\n\\t}\\n\\n\\t&.ck-orientation-top-left {\\n\\t\\ttop: var(--ck-resizer-tooltip-offset);\\n\\t\\tleft: var(--ck-resizer-tooltip-offset);\\n\\t}\\n\\n\\t&.ck-orientation-top-right {\\n\\t\\ttop: var(--ck-resizer-tooltip-offset);\\n\\t\\tright: var(--ck-resizer-tooltip-offset);\\n\\t}\\n\\n\\t&.ck-orientation-bottom-right {\\n\\t\\tbottom: var(--ck-resizer-tooltip-offset);\\n\\t\\tright: var(--ck-resizer-tooltip-offset);\\n\\t}\\n\\n\\t&.ck-orientation-bottom-left {\\n\\t\\tbottom: var(--ck-resizer-tooltip-offset);\\n\\t\\tleft: var(--ck-resizer-tooltip-offset);\\n\\t}\\n\\n\\t/* Class applied if the widget is too small to contain the size label */\\n\\t&.ck-orientation-above-center {\\n\\t\\ttop: calc(var(--ck-resizer-tooltip-height) * -1);\\n\\t\\tleft: 50%;\\n\\t\\ttransform: translate(-50%);\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n@import \\\"../mixins/_focus.css\\\";\\n@import \\\"../mixins/_shadow.css\\\";\\n\\n:root {\\n\\t--ck-widget-outline-thickness: 3px;\\n\\t--ck-widget-handler-icon-size: 16px;\\n\\t--ck-widget-handler-animation-duration: 200ms;\\n\\t--ck-widget-handler-animation-curve: ease;\\n\\n\\t--ck-color-widget-blurred-border: hsl(0, 0%, 87%);\\n\\t--ck-color-widget-hover-border: hsl(43, 100%, 62%);\\n\\t--ck-color-widget-editable-focus-background: var(--ck-color-base-background);\\n\\t--ck-color-widget-drag-handler-icon-color: var(--ck-color-base-background);\\n}\\n\\n.ck .ck-widget {\\n\\toutline-width: var(--ck-widget-outline-thickness);\\n\\toutline-style: solid;\\n\\toutline-color: transparent;\\n\\ttransition: outline-color var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve);\\n\\n\\t&.ck-widget_selected,\\n\\t&.ck-widget_selected:hover {\\n\\t\\toutline: var(--ck-widget-outline-thickness) solid var(--ck-color-focus-border);\\n\\t}\\n\\n\\t&:hover {\\n\\t\\toutline-color: var(--ck-color-widget-hover-border);\\n\\t}\\n}\\n\\n.ck .ck-editor__nested-editable {\\n\\tborder: 1px solid transparent;\\n\\n\\t/* The :focus style is applied before .ck-editor__nested-editable_focused class is rendered in the view.\\n\\tThese styles show a different border for a blink of an eye, so `:focus` need to have same styles applied. */\\n\\t&.ck-editor__nested-editable_focused,\\n\\t&:focus {\\n\\t\\t@mixin ck-focus-ring;\\n\\t\\t@mixin ck-box-shadow var(--ck-inner-shadow);\\n\\n\\t\\tbackground-color: var(--ck-color-widget-editable-focus-background);\\n\\t}\\n}\\n\\n.ck .ck-widget.ck-widget_with-selection-handle {\\n\\t& .ck-widget__selection-handle {\\n\\t\\tpadding: 4px;\\n\\t\\tbox-sizing: border-box;\\n\\n\\t\\t/* Background and opacity will be animated as the handler shows up or the widget gets selected. */\\n\\t\\tbackground-color: transparent;\\n\\t\\topacity: 0;\\n\\n\\t\\t/* Transition:\\n\\t\\t * background-color for the .ck-widget_selected state change,\\n\\t\\t * visibility for hiding the handler,\\n\\t\\t * opacity for the proper look of the icon when the handler disappears. */\\n\\t\\ttransition:\\n\\t\\t\\tbackground-color var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve),\\n\\t\\t\\tvisibility var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve),\\n\\t\\t\\topacity var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve);\\n\\n\\t\\t/* Make only top corners round. */\\n\\t\\tborder-radius: var(--ck-border-radius) var(--ck-border-radius) 0 0;\\n\\n\\t\\t/* Place the drag handler outside the widget wrapper. */\\n\\t\\ttransform: translateY(-100%);\\n\\t\\tleft: calc(0px - var(--ck-widget-outline-thickness));\\n\\t\\ttop: 0;\\n\\n\\t\\t& .ck-icon {\\n\\t\\t\\t/* Make sure the dimensions of the icon are independent of the fon-size of the content. */\\n\\t\\t\\twidth: var(--ck-widget-handler-icon-size);\\n\\t\\t\\theight: var(--ck-widget-handler-icon-size);\\n\\t\\t\\tcolor: var(--ck-color-widget-drag-handler-icon-color);\\n\\n\\t\\t\\t/* The \\\"selected\\\" part of the icon is invisible by default */\\n\\t\\t\\t& .ck-icon__selected-indicator {\\n\\t\\t\\t\\topacity: 0;\\n\\n\\t\\t\\t\\t/* Note: The animation is longer on purpose. Simply feels better. */\\n\\t\\t\\t\\ttransition: opacity 300ms var(--ck-widget-handler-animation-curve);\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t/* Advertise using the look of the icon that once clicked the handler, the widget will be selected. */\\n\\t\\t&:hover .ck-icon .ck-icon__selected-indicator {\\n\\t\\t\\topacity: 1;\\n\\t\\t}\\n\\t}\\n\\n\\t/* Show the selection handler on mouse hover over the widget, but not for nested widgets. */\\n\\t&:hover > .ck-widget__selection-handle {\\n\\t\\topacity: 1;\\n\\t\\tbackground-color: var(--ck-color-widget-hover-border);\\n\\t}\\n\\n\\t/* Show the selection handler when the widget is selected, but not for nested widgets. */\\n\\t&.ck-widget_selected,\\n\\t&.ck-widget_selected:hover {\\n\\t\\t& > .ck-widget__selection-handle {\\n\\t\\t\\topacity: 1;\\n\\t\\t\\tbackground-color: var(--ck-color-focus-border);\\n\\n\\t\\t\\t/* When the widget is selected, notify the user using the proper look of the icon. */\\n\\t\\t\\t& .ck-icon .ck-icon__selected-indicator {\\n\\t\\t\\t\\topacity: 1;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\\n/* In a RTL environment, align the selection handler to the right side of the widget */\\n/* stylelint-disable-next-line no-descending-specificity */\\n.ck[dir=\\\"rtl\\\"] .ck-widget.ck-widget_with-selection-handle .ck-widget__selection-handle {\\n\\tleft: auto;\\n\\tright: calc(0px - var(--ck-widget-outline-thickness));\\n}\\n\\n/* https://github.com/ckeditor/ckeditor5/issues/6415 */\\n.ck.ck-editor__editable.ck-read-only .ck-widget {\\n\\t/* Prevent the :hover outline from showing up because of the used outline-color transition. */\\n\\ttransition: none;\\n\\n\\t&:not(.ck-widget_selected) {\\n\\t\\t/* Disable visual effects of hover/active widget when CKEditor is in readOnly mode.\\n\\t\\t * See: https://github.com/ckeditor/ckeditor5/issues/1261\\n\\t\\t *\\n\\t\\t * Leave the unit because this custom property is used in calc() by other features.\\n\\t\\t * See: https://github.com/ckeditor/ckeditor5/issues/6775\\n\\t\\t */\\n\\t\\t--ck-widget-outline-thickness: 0px;\\n\\t}\\n\\n\\t&.ck-widget_with-selection-handle {\\n\\t\\t& .ck-widget__selection-handle,\\n\\t\\t& .ck-widget__selection-handle:hover {\\n\\t\\t\\tbackground: var(--ck-color-widget-blurred-border);\\n\\t\\t}\\n\\t}\\n}\\n\\n/* Style the widget when it's selected but the editable it belongs to lost focus. */\\n/* stylelint-disable-next-line no-descending-specificity */\\n.ck.ck-editor__editable.ck-blurred .ck-widget {\\n\\t&.ck-widget_selected,\\n\\t&.ck-widget_selected:hover {\\n\\t\\toutline-color: var(--ck-color-widget-blurred-border);\\n\\n\\t\\t&.ck-widget_with-selection-handle {\\n\\t\\t\\t& > .ck-widget__selection-handle,\\n\\t\\t\\t& > .ck-widget__selection-handle:hover {\\n\\t\\t\\t\\tbackground: var(--ck-color-widget-blurred-border);\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\\n.ck.ck-editor__editable > .ck-widget.ck-widget_with-selection-handle:first-child,\\n.ck.ck-editor__editable blockquote > .ck-widget.ck-widget_with-selection-handle:first-child {\\n\\t/* Do not crop selection handler if a widget is a first-child in the blockquote or in the root editable.\\n\\tIn fact, anything with overflow: hidden.\\n\\thttps://github.com/ckeditor/ckeditor5-block-quote/issues/28\\n\\thttps://github.com/ckeditor/ckeditor5-widget/issues/44\\n\\thttps://github.com/ckeditor/ckeditor5-widget/issues/66 */\\n\\tmargin-top: calc(1em + var(--ck-widget-handler-icon-size));\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A visual style of focused element's border.\\n */\\n@define-mixin ck-focus-ring {\\n\\t/* Disable native outline. */\\n\\toutline: none;\\n\\tborder: var(--ck-focus-ring)\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n/**\\n * A helper to combine multiple shadows.\\n */\\n@define-mixin ck-box-shadow $shadowA, $shadowB: 0 0 {\\n\\tbox-shadow: $shadowA, $shadowB;\\n}\\n\\n/**\\n * Gives an element a drop shadow so it looks like a floating panel.\\n */\\n@define-mixin ck-drop-shadow {\\n\\t@mixin ck-box-shadow var(--ck-drop-shadow);\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck .ck-widget_with-resizer{position:relative}.ck .ck-widget__resizer{display:none;left:0;pointer-events:none;position:absolute;top:0}.ck-focused .ck-widget_with-resizer.ck-widget_selected>.ck-widget__resizer{display:block}.ck .ck-widget__resizer__handle{pointer-events:all;position:absolute}.ck .ck-widget__resizer__handle.ck-widget__resizer__handle-bottom-right,.ck .ck-widget__resizer__handle.ck-widget__resizer__handle-top-left{cursor:nwse-resize}.ck .ck-widget__resizer__handle.ck-widget__resizer__handle-bottom-left,.ck .ck-widget__resizer__handle.ck-widget__resizer__handle-top-right{cursor:nesw-resize}:root{--ck-resizer-size:10px;--ck-resizer-offset:calc(var(--ck-resizer-size)/-2 - 2px);--ck-resizer-border-width:1px}.ck .ck-widget__resizer{outline:1px solid var(--ck-color-resizer)}.ck .ck-widget__resizer__handle{background:var(--ck-color-focus-border);border:var(--ck-resizer-border-width) solid #fff;border-radius:var(--ck-resizer-border-radius);height:var(--ck-resizer-size);width:var(--ck-resizer-size)}.ck .ck-widget__resizer__handle.ck-widget__resizer__handle-top-left{left:var(--ck-resizer-offset);top:var(--ck-resizer-offset)}.ck .ck-widget__resizer__handle.ck-widget__resizer__handle-top-right{right:var(--ck-resizer-offset);top:var(--ck-resizer-offset)}.ck .ck-widget__resizer__handle.ck-widget__resizer__handle-bottom-right{bottom:var(--ck-resizer-offset);right:var(--ck-resizer-offset)}.ck .ck-widget__resizer__handle.ck-widget__resizer__handle-bottom-left{bottom:var(--ck-resizer-offset);left:var(--ck-resizer-offset)}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-widget/theme/widgetresize.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-widget/widgetresize.css\"],\"names\":[],\"mappings\":\"AAKA,4BAEC,iBACD,CAEA,wBACC,YAAa,CAMb,MAAO,CAFP,mBAAoB,CAHpB,iBAAkB,CAMlB,KACD,CAGC,2EACC,aACD,CAGD,gCAIC,kBAAmB,CAHnB,iBAcD,CATC,4IAEC,kBACD,CAEA,4IAEC,kBACD,CCpCD,MACC,sBAAuB,CAGvB,yDAAiE,CACjE,6BACD,CAEA,wBACC,yCACD,CAEA,gCAGC,uCAAwC,CACxC,gDAA6D,CAC7D,6CAA8C,CAH9C,6BAA8B,CAD9B,4BAyBD,CAnBC,oEAEC,6BAA8B,CAD9B,4BAED,CAEA,qEAEC,8BAA+B,CAD/B,4BAED,CAEA,wEACC,+BAAgC,CAChC,8BACD,CAEA,uEACC,+BAAgC,CAChC,6BACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck .ck-widget_with-resizer {\\n\\t/* Make the widget wrapper a relative positioning container for the drag handle. */\\n\\tposition: relative;\\n}\\n\\n.ck .ck-widget__resizer {\\n\\tdisplay: none;\\n\\tposition: absolute;\\n\\n\\t/* The wrapper itself should not interfere with the pointer device, only the handles should. */\\n\\tpointer-events: none;\\n\\n\\tleft: 0;\\n\\ttop: 0;\\n}\\n\\n.ck-focused .ck-widget_with-resizer.ck-widget_selected {\\n\\t& > .ck-widget__resizer {\\n\\t\\tdisplay: block;\\n\\t}\\n}\\n\\n.ck .ck-widget__resizer__handle {\\n\\tposition: absolute;\\n\\n\\t/* Resizers are the only UI elements that should interfere with a pointer device. */\\n\\tpointer-events: all;\\n\\n\\t&.ck-widget__resizer__handle-top-left,\\n\\t&.ck-widget__resizer__handle-bottom-right {\\n\\t\\tcursor: nwse-resize;\\n\\t}\\n\\n\\t&.ck-widget__resizer__handle-top-right,\\n\\t&.ck-widget__resizer__handle-bottom-left {\\n\\t\\tcursor: nesw-resize;\\n\\t}\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-resizer-size: 10px;\\n\\n\\t/* Set the resizer with a 50% offset. */\\n\\t--ck-resizer-offset: calc( ( var(--ck-resizer-size) / -2 ) - 2px);\\n\\t--ck-resizer-border-width: 1px;\\n}\\n\\n.ck .ck-widget__resizer {\\n\\toutline: 1px solid var(--ck-color-resizer);\\n}\\n\\n.ck .ck-widget__resizer__handle {\\n\\twidth: var(--ck-resizer-size);\\n\\theight: var(--ck-resizer-size);\\n\\tbackground: var(--ck-color-focus-border);\\n\\tborder: var(--ck-resizer-border-width) solid hsl(0, 0%, 100%);\\n\\tborder-radius: var(--ck-resizer-border-radius);\\n\\n\\t&.ck-widget__resizer__handle-top-left {\\n\\t\\ttop: var(--ck-resizer-offset);\\n\\t\\tleft: var(--ck-resizer-offset);\\n\\t}\\n\\n\\t&.ck-widget__resizer__handle-top-right {\\n\\t\\ttop: var(--ck-resizer-offset);\\n\\t\\tright: var(--ck-resizer-offset);\\n\\t}\\n\\n\\t&.ck-widget__resizer__handle-bottom-right {\\n\\t\\tbottom: var(--ck-resizer-offset);\\n\\t\\tright: var(--ck-resizer-offset);\\n\\t}\\n\\n\\t&.ck-widget__resizer__handle-bottom-left {\\n\\t\\tbottom: var(--ck-resizer-offset);\\n\\t\\tleft: var(--ck-resizer-offset);\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","// Imports\nimport ___CSS_LOADER_API_SOURCEMAP_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/cssWithMappingToString.js\";\nimport ___CSS_LOADER_API_IMPORT___ from \"../../../node_modules/css-loader/dist/runtime/api.js\";\nvar ___CSS_LOADER_EXPORT___ = ___CSS_LOADER_API_IMPORT___(___CSS_LOADER_API_SOURCEMAP_IMPORT___);\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".ck .ck-widget .ck-widget__type-around__button{display:block;overflow:hidden;position:absolute;z-index:var(--ck-z-default)}.ck .ck-widget .ck-widget__type-around__button svg{left:50%;position:absolute;top:50%;z-index:calc(var(--ck-z-default) + 2)}.ck .ck-widget .ck-widget__type-around__button.ck-widget__type-around__button_before{left:min(10%,30px);top:calc(var(--ck-widget-outline-thickness)*-.5);transform:translateY(-50%)}.ck .ck-widget .ck-widget__type-around__button.ck-widget__type-around__button_after{bottom:calc(var(--ck-widget-outline-thickness)*-.5);right:min(10%,30px);transform:translateY(50%)}.ck .ck-widget.ck-widget_selected>.ck-widget__type-around>.ck-widget__type-around__button:after,.ck .ck-widget>.ck-widget__type-around>.ck-widget__type-around__button:hover:after{content:\\\"\\\";display:block;left:1px;position:absolute;top:1px;z-index:calc(var(--ck-z-default) + 1)}.ck .ck-widget>.ck-widget__type-around>.ck-widget__type-around__fake-caret{display:none;left:0;position:absolute;right:0}.ck .ck-widget:hover>.ck-widget__type-around>.ck-widget__type-around__fake-caret{left:calc(var(--ck-widget-outline-thickness)*-1);right:calc(var(--ck-widget-outline-thickness)*-1)}.ck .ck-widget.ck-widget_type-around_show-fake-caret_before>.ck-widget__type-around>.ck-widget__type-around__fake-caret{display:block;top:calc(var(--ck-widget-outline-thickness)*-1 - 1px)}.ck .ck-widget.ck-widget_type-around_show-fake-caret_after>.ck-widget__type-around>.ck-widget__type-around__fake-caret{bottom:calc(var(--ck-widget-outline-thickness)*-1 - 1px);display:block}.ck.ck-editor__editable.ck-read-only .ck-widget__type-around,.ck.ck-editor__editable.ck-restricted-editing_mode_restricted .ck-widget__type-around,.ck.ck-editor__editable.ck-widget__type-around_disabled .ck-widget__type-around{display:none}:root{--ck-widget-type-around-button-size:20px;--ck-color-widget-type-around-button-active:var(--ck-color-focus-border);--ck-color-widget-type-around-button-hover:var(--ck-color-widget-hover-border);--ck-color-widget-type-around-button-blurred-editable:var(--ck-color-widget-blurred-border);--ck-color-widget-type-around-button-radar-start-alpha:0;--ck-color-widget-type-around-button-radar-end-alpha:.3;--ck-color-widget-type-around-button-icon:var(--ck-color-base-background)}.ck .ck-widget .ck-widget__type-around__button{background:var(--ck-color-widget-type-around-button);border-radius:100px;height:var(--ck-widget-type-around-button-size);opacity:0;pointer-events:none;transition:opacity var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve),background var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve);width:var(--ck-widget-type-around-button-size)}.ck .ck-widget .ck-widget__type-around__button svg{height:8px;margin-top:1px;transform:translate(-50%,-50%);transition:transform .5s ease;width:10px}.ck .ck-widget .ck-widget__type-around__button svg *{stroke-dasharray:10;stroke-dashoffset:0;fill:none;stroke:var(--ck-color-widget-type-around-button-icon);stroke-width:1.5px;stroke-linecap:round;stroke-linejoin:round}.ck .ck-widget .ck-widget__type-around__button svg line{stroke-dasharray:7}.ck .ck-widget .ck-widget__type-around__button:hover{animation:ck-widget-type-around-button-sonar 1s ease infinite}.ck .ck-widget .ck-widget__type-around__button:hover svg polyline{animation:ck-widget-type-around-arrow-dash 2s linear}.ck .ck-widget .ck-widget__type-around__button:hover svg line{animation:ck-widget-type-around-arrow-tip-dash 2s linear}.ck .ck-widget.ck-widget_selected>.ck-widget__type-around>.ck-widget__type-around__button,.ck .ck-widget:hover>.ck-widget__type-around>.ck-widget__type-around__button{opacity:1;pointer-events:auto}.ck .ck-widget:not(.ck-widget_selected)>.ck-widget__type-around>.ck-widget__type-around__button{background:var(--ck-color-widget-type-around-button-hover)}.ck .ck-widget.ck-widget_selected>.ck-widget__type-around>.ck-widget__type-around__button,.ck .ck-widget>.ck-widget__type-around>.ck-widget__type-around__button:hover{background:var(--ck-color-widget-type-around-button-active)}.ck .ck-widget.ck-widget_selected>.ck-widget__type-around>.ck-widget__type-around__button:after,.ck .ck-widget>.ck-widget__type-around>.ck-widget__type-around__button:hover:after{background:linear-gradient(135deg,hsla(0,0%,100%,0),hsla(0,0%,100%,.3));border-radius:100px;height:calc(var(--ck-widget-type-around-button-size) - 2px);width:calc(var(--ck-widget-type-around-button-size) - 2px)}.ck .ck-widget.ck-widget_with-selection-handle>.ck-widget__type-around>.ck-widget__type-around__button_before{margin-left:20px}.ck .ck-widget .ck-widget__type-around__fake-caret{animation:ck-widget-type-around-fake-caret-pulse 1s linear infinite normal forwards;background:var(--ck-color-base-text);height:1px;outline:1px solid hsla(0,0%,100%,.5);pointer-events:none}.ck .ck-widget.ck-widget_selected.ck-widget_type-around_show-fake-caret_after,.ck .ck-widget.ck-widget_selected.ck-widget_type-around_show-fake-caret_before{outline-color:transparent}.ck .ck-widget.ck-widget_type-around_show-fake-caret_after.ck-widget_selected:hover,.ck .ck-widget.ck-widget_type-around_show-fake-caret_before.ck-widget_selected:hover{outline-color:var(--ck-color-widget-hover-border)}.ck .ck-widget.ck-widget_type-around_show-fake-caret_after>.ck-widget__type-around>.ck-widget__type-around__button,.ck .ck-widget.ck-widget_type-around_show-fake-caret_before>.ck-widget__type-around>.ck-widget__type-around__button{opacity:0;pointer-events:none}.ck .ck-widget.ck-widget_type-around_show-fake-caret_after.ck-widget_selected.ck-widget_with-resizer>.ck-widget__resizer,.ck .ck-widget.ck-widget_type-around_show-fake-caret_after.ck-widget_with-selection-handle.ck-widget_selected:hover>.ck-widget__selection-handle,.ck .ck-widget.ck-widget_type-around_show-fake-caret_after.ck-widget_with-selection-handle.ck-widget_selected>.ck-widget__selection-handle,.ck .ck-widget.ck-widget_type-around_show-fake-caret_before.ck-widget_selected.ck-widget_with-resizer>.ck-widget__resizer,.ck .ck-widget.ck-widget_type-around_show-fake-caret_before.ck-widget_with-selection-handle.ck-widget_selected:hover>.ck-widget__selection-handle,.ck .ck-widget.ck-widget_type-around_show-fake-caret_before.ck-widget_with-selection-handle.ck-widget_selected>.ck-widget__selection-handle{opacity:0}.ck[dir=rtl] .ck-widget.ck-widget_with-selection-handle .ck-widget__type-around>.ck-widget__type-around__button_before{margin-left:0;margin-right:20px}.ck-editor__nested-editable.ck-editor__editable_selected .ck-widget.ck-widget_selected>.ck-widget__type-around>.ck-widget__type-around__button,.ck-editor__nested-editable.ck-editor__editable_selected .ck-widget:hover>.ck-widget__type-around>.ck-widget__type-around__button{opacity:0;pointer-events:none}.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected>.ck-widget__type-around>.ck-widget__type-around__button:not(:hover){background:var(--ck-color-widget-type-around-button-blurred-editable)}.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected>.ck-widget__type-around>.ck-widget__type-around__button:not(:hover) svg *{stroke:#999}@keyframes ck-widget-type-around-arrow-dash{0%{stroke-dashoffset:10}20%,to{stroke-dashoffset:0}}@keyframes ck-widget-type-around-arrow-tip-dash{0%,20%{stroke-dashoffset:7}40%,to{stroke-dashoffset:0}}@keyframes ck-widget-type-around-button-sonar{0%{box-shadow:0 0 0 0 hsla(var(--ck-color-focus-border-coordinates),var(--ck-color-widget-type-around-button-radar-start-alpha))}50%{box-shadow:0 0 0 5px hsla(var(--ck-color-focus-border-coordinates),var(--ck-color-widget-type-around-button-radar-end-alpha))}to{box-shadow:0 0 0 5px hsla(var(--ck-color-focus-border-coordinates),var(--ck-color-widget-type-around-button-radar-start-alpha))}}@keyframes ck-widget-type-around-fake-caret-pulse{0%{opacity:1}49%{opacity:1}50%{opacity:0}99%{opacity:0}to{opacity:1}}\", \"\",{\"version\":3,\"sources\":[\"webpack://./../ckeditor5-widget/theme/widgettypearound.css\",\"webpack://./../ckeditor5-theme-lark/theme/ckeditor5-widget/widgettypearound.css\"],\"names\":[],\"mappings\":\"AASC,+CACC,aAAc,CAEd,eAAgB,CADhB,iBAAkB,CAElB,2BAwBD,CAtBC,mDAGC,QAAS,CAFT,iBAAkB,CAClB,OAAQ,CAER,qCACD,CAEA,qFAGC,kBAAoB,CADpB,gDAAoD,CAGpD,0BACD,CAEA,oFAEC,mDAAuD,CACvD,mBAAqB,CAErB,yBACD,CAUA,mLACC,UAAW,CACX,aAAc,CAGd,QAAS,CAFT,iBAAkB,CAClB,OAAQ,CAER,qCACD,CAMD,2EACC,YAAa,CAEb,MAAO,CADP,iBAAkB,CAElB,OACD,CAOA,iFACC,gDAAqD,CACrD,iDACD,CAKA,wHAEC,aAAc,CADd,qDAED,CAKA,uHACC,wDAA6D,CAC7D,aACD,CAoBD,mOACC,YACD,CC3GA,MACC,wCAAyC,CACzC,wEAAyE,CACzE,8EAA+E,CAC/E,2FAA4F,CAC5F,wDAAyD,CACzD,uDAAwD,CACxD,yEACD,CAgBC,+CAGC,oDAAqD,CACrD,mBAAoB,CAFpB,+CAAgD,CAVjD,SAAU,CACV,mBAAoB,CAYnB,uMAAyM,CAJzM,8CAkDD,CA1CC,mDAEC,UAAW,CAGX,cAAe,CAFf,8BAA+B,CAC/B,6BAA8B,CAH9B,UAoBD,CAdC,qDACC,mBAAoB,CACpB,mBAAoB,CAEpB,SAAU,CACV,qDAAsD,CACtD,kBAAmB,CACnB,oBAAqB,CACrB,qBACD,CAEA,wDACC,kBACD,CAGD,qDAIC,6DAcD,CARE,kEACC,oDACD,CAEA,8DACC,wDACD,CAUF,uKAvED,SAAU,CACV,mBAwEC,CAOD,gGACC,0DACD,CAOA,uKAEC,2DAQD,CANC,mLAIC,uEAAkF,CADlF,mBAAoB,CADpB,2DAA4D,CAD5D,0DAID,CAOD,8GACC,gBACD,CAKA,mDAGC,mFAAoF,CAOpF,oCAAqC,CARrC,UAAW,CAOX,oCAAwC,CARxC,mBAUD,CAOC,6JAEC,yBACD,CAUA,yKACC,iDACD,CAMA,uOAlJD,SAAU,CACV,mBAmJC,CAoBA,6yBACC,SACD,CASF,uHACC,aAAc,CACd,iBACD,CAYG,iRAlMF,SAAU,CACV,mBAmME,CAQH,kIACC,qEAKD,CAHC,wIACC,WACD,CAGD,4CACC,GACC,oBACD,CACA,OACC,mBACD,CACD,CAEA,gDACC,OACC,mBACD,CACA,OACC,mBACD,CACD,CAEA,8CACC,GACC,6HACD,CACA,IACC,6HACD,CACA,GACC,+HACD,CACD,CAEA,kDACC,GACC,SACD,CACA,IACC,SACD,CACA,IACC,SACD,CACA,IACC,SACD,CACA,GACC,SACD,CACD\",\"sourcesContent\":[\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n.ck .ck-widget {\\n\\t/*\\n\\t * Styles of the type around buttons\\n\\t */\\n\\t& .ck-widget__type-around__button {\\n\\t\\tdisplay: block;\\n\\t\\tposition: absolute;\\n\\t\\toverflow: hidden;\\n\\t\\tz-index: var(--ck-z-default);\\n\\n\\t\\t& svg {\\n\\t\\t\\tposition: absolute;\\n\\t\\t\\ttop: 50%;\\n\\t\\t\\tleft: 50%;\\n\\t\\t\\tz-index: calc(var(--ck-z-default) + 2);\\n\\t\\t}\\n\\n\\t\\t&.ck-widget__type-around__button_before {\\n\\t\\t\\t/* Place it in the middle of the outline */\\n\\t\\t\\ttop: calc(-0.5 * var(--ck-widget-outline-thickness));\\n\\t\\t\\tleft: min(10%, 30px);\\n\\n\\t\\t\\ttransform: translateY(-50%);\\n\\t\\t}\\n\\n\\t\\t&.ck-widget__type-around__button_after {\\n\\t\\t\\t/* Place it in the middle of the outline */\\n\\t\\t\\tbottom: calc(-0.5 * var(--ck-widget-outline-thickness));\\n\\t\\t\\tright: min(10%, 30px);\\n\\n\\t\\t\\ttransform: translateY(50%);\\n\\t\\t}\\n\\t}\\n\\n\\t/*\\n\\t * Styles for the buttons when:\\n\\t * - the widget is selected,\\n\\t * - or the button is being hovered (regardless of the widget state).\\n\\t */\\n\\t&.ck-widget_selected > .ck-widget__type-around > .ck-widget__type-around__button,\\n\\t& > .ck-widget__type-around > .ck-widget__type-around__button:hover {\\n\\t\\t&::after {\\n\\t\\t\\tcontent: \\\"\\\";\\n\\t\\t\\tdisplay: block;\\n\\t\\t\\tposition: absolute;\\n\\t\\t\\ttop: 1px;\\n\\t\\t\\tleft: 1px;\\n\\t\\t\\tz-index: calc(var(--ck-z-default) + 1);\\n\\t\\t}\\n\\t}\\n\\n\\t/*\\n\\t * Styles for the horizontal \\\"fake caret\\\" which is displayed when the user navigates using the keyboard.\\n\\t */\\n\\t& > .ck-widget__type-around > .ck-widget__type-around__fake-caret {\\n\\t\\tdisplay: none;\\n\\t\\tposition: absolute;\\n\\t\\tleft: 0;\\n\\t\\tright: 0;\\n\\t}\\n\\n\\t/*\\n\\t * When the widget is hovered the \\\"fake caret\\\" would normally be narrower than the\\n\\t * extra outline displayed around the widget. Let's extend the \\\"fake caret\\\" to match\\n\\t * the full width of the widget.\\n\\t */\\n\\t&:hover > .ck-widget__type-around > .ck-widget__type-around__fake-caret {\\n\\t\\tleft: calc( -1 * var(--ck-widget-outline-thickness) );\\n\\t\\tright: calc( -1 * var(--ck-widget-outline-thickness) );\\n\\t}\\n\\n\\t/*\\n\\t * Styles for the horizontal \\\"fake caret\\\" when it should be displayed before the widget (backward keyboard navigation).\\n\\t */\\n\\t&.ck-widget_type-around_show-fake-caret_before > .ck-widget__type-around > .ck-widget__type-around__fake-caret {\\n\\t\\ttop: calc( -1 * var(--ck-widget-outline-thickness) - 1px );\\n\\t\\tdisplay: block;\\n\\t}\\n\\n\\t/*\\n\\t * Styles for the horizontal \\\"fake caret\\\" when it should be displayed after the widget (forward keyboard navigation).\\n\\t */\\n\\t&.ck-widget_type-around_show-fake-caret_after > .ck-widget__type-around > .ck-widget__type-around__fake-caret {\\n\\t\\tbottom: calc( -1 * var(--ck-widget-outline-thickness) - 1px );\\n\\t\\tdisplay: block;\\n\\t}\\n}\\n\\n/*\\n * Integration with the read-only mode of the editor.\\n */\\n.ck.ck-editor__editable.ck-read-only .ck-widget__type-around {\\n\\tdisplay: none;\\n}\\n\\n/*\\n * Integration with the restricted editing mode (feature) of the editor.\\n */\\n.ck.ck-editor__editable.ck-restricted-editing_mode_restricted .ck-widget__type-around {\\n\\tdisplay: none;\\n}\\n\\n/*\\n * Integration with the #isEnabled property of the WidgetTypeAround plugin.\\n */\\n.ck.ck-editor__editable.ck-widget__type-around_disabled .ck-widget__type-around {\\n\\tdisplay: none;\\n}\\n\",\"/*\\n * Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\\n */\\n\\n:root {\\n\\t--ck-widget-type-around-button-size: 20px;\\n\\t--ck-color-widget-type-around-button-active: var(--ck-color-focus-border);\\n\\t--ck-color-widget-type-around-button-hover: var(--ck-color-widget-hover-border);\\n\\t--ck-color-widget-type-around-button-blurred-editable: var(--ck-color-widget-blurred-border);\\n\\t--ck-color-widget-type-around-button-radar-start-alpha: 0;\\n\\t--ck-color-widget-type-around-button-radar-end-alpha: .3;\\n\\t--ck-color-widget-type-around-button-icon: var(--ck-color-base-background);\\n}\\n\\n@define-mixin ck-widget-type-around-button-visible {\\n\\topacity: 1;\\n\\tpointer-events: auto;\\n}\\n\\n@define-mixin ck-widget-type-around-button-hidden {\\n\\topacity: 0;\\n\\tpointer-events: none;\\n}\\n\\n.ck .ck-widget {\\n\\t/*\\n\\t * Styles of the type around buttons\\n\\t */\\n\\t& .ck-widget__type-around__button {\\n\\t\\twidth: var(--ck-widget-type-around-button-size);\\n\\t\\theight: var(--ck-widget-type-around-button-size);\\n\\t\\tbackground: var(--ck-color-widget-type-around-button);\\n\\t\\tborder-radius: 100px;\\n\\t\\ttransition: opacity var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve), background var(--ck-widget-handler-animation-duration) var(--ck-widget-handler-animation-curve);\\n\\n\\t\\t@mixin ck-widget-type-around-button-hidden;\\n\\n\\t\\t& svg {\\n\\t\\t\\twidth: 10px;\\n\\t\\t\\theight: 8px;\\n\\t\\t\\ttransform: translate(-50%,-50%);\\n\\t\\t\\ttransition: transform .5s ease;\\n\\t\\t\\tmargin-top: 1px;\\n\\n\\t\\t\\t& * {\\n\\t\\t\\t\\tstroke-dasharray: 10;\\n\\t\\t\\t\\tstroke-dashoffset: 0;\\n\\n\\t\\t\\t\\tfill: none;\\n\\t\\t\\t\\tstroke: var(--ck-color-widget-type-around-button-icon);\\n\\t\\t\\t\\tstroke-width: 1.5px;\\n\\t\\t\\t\\tstroke-linecap: round;\\n\\t\\t\\t\\tstroke-linejoin: round;\\n\\t\\t\\t}\\n\\n\\t\\t\\t& line {\\n\\t\\t\\t\\tstroke-dasharray: 7;\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t&:hover {\\n\\t\\t\\t/*\\n\\t\\t\\t * Display the \\\"sonar\\\" around the button when hovered.\\n\\t\\t\\t */\\n\\t\\t\\tanimation: ck-widget-type-around-button-sonar 1s ease infinite;\\n\\n\\t\\t\\t/*\\n\\t\\t\\t * Animate active button's icon.\\n\\t\\t\\t */\\n\\t\\t\\t& svg {\\n\\t\\t\\t\\t& polyline {\\n\\t\\t\\t\\t\\tanimation: ck-widget-type-around-arrow-dash 2s linear;\\n\\t\\t\\t\\t}\\n\\n\\t\\t\\t\\t& line {\\n\\t\\t\\t\\t\\tanimation: ck-widget-type-around-arrow-tip-dash 2s linear;\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n\\n\\t/*\\n\\t * Show type around buttons when the widget gets selected or being hovered.\\n\\t */\\n\\t&.ck-widget_selected,\\n\\t&:hover {\\n\\t\\t& > .ck-widget__type-around > .ck-widget__type-around__button {\\n\\t\\t\\t@mixin ck-widget-type-around-button-visible;\\n\\t\\t}\\n\\t}\\n\\n\\t/*\\n\\t * Styles for the buttons when the widget is NOT selected (but the buttons are visible\\n\\t * and still can be hovered).\\n\\t */\\n\\t&:not(.ck-widget_selected) > .ck-widget__type-around > .ck-widget__type-around__button {\\n\\t\\tbackground: var(--ck-color-widget-type-around-button-hover);\\n\\t}\\n\\n\\t/*\\n\\t * Styles for the buttons when:\\n\\t * - the widget is selected,\\n\\t * - or the button is being hovered (regardless of the widget state).\\n\\t */\\n\\t&.ck-widget_selected > .ck-widget__type-around > .ck-widget__type-around__button,\\n\\t& > .ck-widget__type-around > .ck-widget__type-around__button:hover {\\n\\t\\tbackground: var(--ck-color-widget-type-around-button-active);\\n\\n\\t\\t&::after {\\n\\t\\t\\twidth: calc(var(--ck-widget-type-around-button-size) - 2px);\\n\\t\\t\\theight: calc(var(--ck-widget-type-around-button-size) - 2px);\\n\\t\\t\\tborder-radius: 100px;\\n\\t\\t\\tbackground: linear-gradient(135deg, hsla(0,0%,100%,0) 0%, hsla(0,0%,100%,.3) 100%);\\n\\t\\t}\\n\\t}\\n\\n\\t/*\\n\\t * Styles for the \\\"before\\\" button when the widget has a selection handle. Because some space\\n\\t * is consumed by the handle, the button must be moved slightly to the right to let it breathe.\\n\\t */\\n\\t&.ck-widget_with-selection-handle > .ck-widget__type-around > .ck-widget__type-around__button_before {\\n\\t\\tmargin-left: 20px;\\n\\t}\\n\\n\\t/*\\n\\t * Styles for the horizontal \\\"fake caret\\\" which is displayed when the user navigates using the keyboard.\\n\\t */\\n\\t& .ck-widget__type-around__fake-caret {\\n\\t\\tpointer-events: none;\\n\\t\\theight: 1px;\\n\\t\\tanimation: ck-widget-type-around-fake-caret-pulse linear 1s infinite normal forwards;\\n\\n\\t\\t/*\\n\\t\\t * The semi-transparent-outline+background combo improves the contrast\\n\\t\\t * when the background underneath the fake caret is dark.\\n\\t\\t */\\n\\t\\toutline: solid 1px hsla(0, 0%, 100%, .5);\\n\\t\\tbackground: var(--ck-color-base-text);\\n\\t}\\n\\n\\t/*\\n\\t * Styles of the widget when the \\\"fake caret\\\" is blinking (e.g. upon keyboard navigation).\\n\\t * Despite the widget being physically selected in the model, its outline should disappear.\\n\\t */\\n\\t&.ck-widget_selected {\\n\\t\\t&.ck-widget_type-around_show-fake-caret_before,\\n\\t\\t&.ck-widget_type-around_show-fake-caret_after {\\n\\t\\t\\toutline-color: transparent;\\n\\t\\t}\\n\\t}\\n\\n\\t&.ck-widget_type-around_show-fake-caret_before,\\n\\t&.ck-widget_type-around_show-fake-caret_after {\\n\\t\\t/*\\n\\t\\t * When the \\\"fake caret\\\" is visible we simulate that the widget is not selected\\n\\t\\t * (despite being physically selected), so the outline color should be for the\\n\\t\\t * unselected widget.\\n\\t\\t */\\n\\t\\t&.ck-widget_selected:hover {\\n\\t\\t\\toutline-color: var(--ck-color-widget-hover-border);\\n\\t\\t}\\n\\n\\t\\t/*\\n\\t\\t * Styles of the type around buttons when the \\\"fake caret\\\" is blinking (e.g. upon keyboard navigation).\\n\\t\\t * In this state, the type around buttons would collide with the fake carets so they should disappear.\\n\\t\\t */\\n\\t\\t& > .ck-widget__type-around > .ck-widget__type-around__button {\\n\\t\\t\\t@mixin ck-widget-type-around-button-hidden;\\n\\t\\t}\\n\\n\\t\\t/*\\n\\t\\t * Fake horizontal caret integration with the selection handle. When the caret is visible, simply\\n\\t\\t * hide the handle because it intersects with the caret (and does not make much sense anyway).\\n\\t\\t */\\n\\t\\t&.ck-widget_with-selection-handle {\\n\\t\\t\\t&.ck-widget_selected,\\n\\t\\t\\t&.ck-widget_selected:hover {\\n\\t\\t\\t\\t& > .ck-widget__selection-handle {\\n\\t\\t\\t\\t\\topacity: 0\\n\\t\\t\\t\\t}\\n\\t\\t\\t}\\n\\t\\t}\\n\\n\\t\\t/*\\n\\t\\t * Fake horizontal caret integration with the resize UI. When the caret is visible, simply\\n\\t\\t * hide the resize UI because it creates too much noise. It can be visible when the user\\n\\t\\t * hovers the widget, though.\\n\\t\\t */\\n\\t\\t&.ck-widget_selected.ck-widget_with-resizer > .ck-widget__resizer {\\n\\t\\t\\topacity: 0\\n\\t\\t}\\n\\t}\\n}\\n\\n/*\\n * Styles for the \\\"before\\\" button when the widget has a selection handle in an RTL environment.\\n * The selection handler is aligned to the right side of the widget so there is no need to create\\n * additional space for it next to the \\\"before\\\" button.\\n */\\n.ck[dir=\\\"rtl\\\"] .ck-widget.ck-widget_with-selection-handle .ck-widget__type-around > .ck-widget__type-around__button_before {\\n\\tmargin-left: 0;\\n\\tmargin-right: 20px;\\n}\\n\\n/*\\n * Hide type around buttons when the widget is selected as a child of a selected\\n * nested editable (e.g. mulit-cell table selection).\\n *\\n * See https://github.com/ckeditor/ckeditor5/issues/7263.\\n */\\n.ck-editor__nested-editable.ck-editor__editable_selected {\\n\\t& .ck-widget {\\n\\t\\t&.ck-widget_selected,\\n\\t\\t&:hover {\\n\\t\\t\\t& > .ck-widget__type-around > .ck-widget__type-around__button {\\n\\t\\t\\t\\t@mixin ck-widget-type-around-button-hidden;\\n\\t\\t\\t}\\n\\t\\t}\\n\\t}\\n}\\n\\n/*\\n * Styles for the buttons when the widget is selected but the user clicked outside of the editor (blurred the editor).\\n */\\n.ck-editor__editable.ck-blurred .ck-widget.ck-widget_selected > .ck-widget__type-around > .ck-widget__type-around__button:not(:hover) {\\n\\tbackground: var(--ck-color-widget-type-around-button-blurred-editable);\\n\\n\\t& svg * {\\n\\t\\tstroke: hsl(0,0%,60%);\\n\\t}\\n}\\n\\n@keyframes ck-widget-type-around-arrow-dash {\\n\\t0% {\\n\\t\\tstroke-dashoffset: 10;\\n\\t}\\n\\t20%, 100% {\\n\\t\\tstroke-dashoffset: 0;\\n\\t}\\n}\\n\\n@keyframes ck-widget-type-around-arrow-tip-dash {\\n\\t0%, 20% {\\n\\t\\tstroke-dashoffset: 7;\\n\\t}\\n\\t40%, 100% {\\n\\t\\tstroke-dashoffset: 0;\\n\\t}\\n}\\n\\n@keyframes ck-widget-type-around-button-sonar {\\n\\t0% {\\n\\t\\tbox-shadow: 0 0 0 0 hsla(var(--ck-color-focus-border-coordinates), var(--ck-color-widget-type-around-button-radar-start-alpha));\\n\\t}\\n\\t50% {\\n\\t\\tbox-shadow: 0 0 0 5px hsla(var(--ck-color-focus-border-coordinates), var(--ck-color-widget-type-around-button-radar-end-alpha));\\n\\t}\\n\\t100% {\\n\\t\\tbox-shadow: 0 0 0 5px hsla(var(--ck-color-focus-border-coordinates), var(--ck-color-widget-type-around-button-radar-start-alpha));\\n\\t}\\n}\\n\\n@keyframes ck-widget-type-around-fake-caret-pulse {\\n\\t0% {\\n\\t\\topacity: 1;\\n\\t}\\n\\t49% {\\n\\t\\topacity: 1;\\n\\t}\\n\\t50% {\\n\\t\\topacity: 0;\\n\\t}\\n\\t99% {\\n\\t\\topacity: 0;\\n\\t}\\n\\t100% {\\n\\t\\topacity: 1;\\n\\t}\\n}\\n\"],\"sourceRoot\":\"\"}]);\n// Exports\nexport default ___CSS_LOADER_EXPORT___;\n","\"use strict\";\n\n/*\n MIT License http://www.opensource.org/licenses/mit-license.php\n Author Tobias Koppers @sokra\n*/\n// css base code, injected by the css-loader\n// eslint-disable-next-line func-names\nmodule.exports = function (cssWithMappingToString) {\n var list = []; // return the list of modules as css string\n\n list.toString = function toString() {\n return this.map(function (item) {\n var content = cssWithMappingToString(item);\n\n if (item[2]) {\n return \"@media \".concat(item[2], \" {\").concat(content, \"}\");\n }\n\n return content;\n }).join(\"\");\n }; // import a list of modules into the list\n // eslint-disable-next-line func-names\n\n\n list.i = function (modules, mediaQuery, dedupe) {\n if (typeof modules === \"string\") {\n // eslint-disable-next-line no-param-reassign\n modules = [[null, modules, \"\"]];\n }\n\n var alreadyImportedModules = {};\n\n if (dedupe) {\n for (var i = 0; i < this.length; i++) {\n // eslint-disable-next-line prefer-destructuring\n var id = this[i][0];\n\n if (id != null) {\n alreadyImportedModules[id] = true;\n }\n }\n }\n\n for (var _i = 0; _i < modules.length; _i++) {\n var item = [].concat(modules[_i]);\n\n if (dedupe && alreadyImportedModules[item[0]]) {\n // eslint-disable-next-line no-continue\n continue;\n }\n\n if (mediaQuery) {\n if (!item[2]) {\n item[2] = mediaQuery;\n } else {\n item[2] = \"\".concat(mediaQuery, \" and \").concat(item[2]);\n }\n }\n\n list.push(item);\n }\n };\n\n return list;\n};","\"use strict\";\n\nfunction _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }\n\nfunction _nonIterableRest() { throw new TypeError(\"Invalid attempt to destructure non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\"); }\n\nfunction _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === \"string\") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === \"Object\" && o.constructor) n = o.constructor.name; if (n === \"Map\" || n === \"Set\") return Array.from(o); if (n === \"Arguments\" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }\n\nfunction _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }\n\nfunction _iterableToArrayLimit(arr, i) { var _i = arr && (typeof Symbol !== \"undefined\" && arr[Symbol.iterator] || arr[\"@@iterator\"]); if (_i == null) return; var _arr = []; var _n = true; var _d = false; var _s, _e; try { for (_i = _i.call(arr); !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i[\"return\"] != null) _i[\"return\"](); } finally { if (_d) throw _e; } } return _arr; }\n\nfunction _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }\n\nmodule.exports = function cssWithMappingToString(item) {\n var _item = _slicedToArray(item, 4),\n content = _item[1],\n cssMapping = _item[3];\n\n if (!cssMapping) {\n return content;\n }\n\n if (typeof btoa === \"function\") {\n // eslint-disable-next-line no-undef\n var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(cssMapping))));\n var data = \"sourceMappingURL=data:application/json;charset=utf-8;base64,\".concat(base64);\n var sourceMapping = \"/*# \".concat(data, \" */\");\n var sourceURLs = cssMapping.sources.map(function (source) {\n return \"/*# sourceURL=\".concat(cssMapping.sourceRoot || \"\").concat(source, \" */\");\n });\n return [content].concat(sourceURLs).concat([sourceMapping]).join(\"\\n\");\n }\n\n return [content].join(\"\\n\");\n};","\"use strict\";\n\nvar isOldIE = function isOldIE() {\n var memo;\n return function memorize() {\n if (typeof memo === 'undefined') {\n // Test for IE <= 9 as proposed by Browserhacks\n // @see http://browserhacks.com/#hack-e71d8692f65334173fee715c222cb805\n // Tests for existence of standard globals is to allow style-loader\n // to operate correctly into non-standard environments\n // @see https://github.com/webpack-contrib/style-loader/issues/177\n memo = Boolean(window && document && document.all && !window.atob);\n }\n\n return memo;\n };\n}();\n\nvar getTarget = function getTarget() {\n var memo = {};\n return function memorize(target) {\n if (typeof memo[target] === 'undefined') {\n var styleTarget = document.querySelector(target); // Special case to return head of iframe instead of iframe itself\n\n if (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) {\n try {\n // This will throw an exception if access to iframe is blocked\n // due to cross-origin restrictions\n styleTarget = styleTarget.contentDocument.head;\n } catch (e) {\n // istanbul ignore next\n styleTarget = null;\n }\n }\n\n memo[target] = styleTarget;\n }\n\n return memo[target];\n };\n}();\n\nvar stylesInDom = [];\n\nfunction getIndexByIdentifier(identifier) {\n var result = -1;\n\n for (var i = 0; i < stylesInDom.length; i++) {\n if (stylesInDom[i].identifier === identifier) {\n result = i;\n break;\n }\n }\n\n return result;\n}\n\nfunction modulesToDom(list, options) {\n var idCountMap = {};\n var identifiers = [];\n\n for (var i = 0; i < list.length; i++) {\n var item = list[i];\n var id = options.base ? item[0] + options.base : item[0];\n var count = idCountMap[id] || 0;\n var identifier = \"\".concat(id, \" \").concat(count);\n idCountMap[id] = count + 1;\n var index = getIndexByIdentifier(identifier);\n var obj = {\n css: item[1],\n media: item[2],\n sourceMap: item[3]\n };\n\n if (index !== -1) {\n stylesInDom[index].references++;\n stylesInDom[index].updater(obj);\n } else {\n stylesInDom.push({\n identifier: identifier,\n updater: addStyle(obj, options),\n references: 1\n });\n }\n\n identifiers.push(identifier);\n }\n\n return identifiers;\n}\n\nfunction insertStyleElement(options) {\n var style = document.createElement('style');\n var attributes = options.attributes || {};\n\n if (typeof attributes.nonce === 'undefined') {\n var nonce = typeof __webpack_nonce__ !== 'undefined' ? __webpack_nonce__ : null;\n\n if (nonce) {\n attributes.nonce = nonce;\n }\n }\n\n Object.keys(attributes).forEach(function (key) {\n style.setAttribute(key, attributes[key]);\n });\n\n if (typeof options.insert === 'function') {\n options.insert(style);\n } else {\n var target = getTarget(options.insert || 'head');\n\n if (!target) {\n throw new Error(\"Couldn't find a style target. This probably means that the value for the 'insert' parameter is invalid.\");\n }\n\n target.appendChild(style);\n }\n\n return style;\n}\n\nfunction removeStyleElement(style) {\n // istanbul ignore if\n if (style.parentNode === null) {\n return false;\n }\n\n style.parentNode.removeChild(style);\n}\n/* istanbul ignore next */\n\n\nvar replaceText = function replaceText() {\n var textStore = [];\n return function replace(index, replacement) {\n textStore[index] = replacement;\n return textStore.filter(Boolean).join('\\n');\n };\n}();\n\nfunction applyToSingletonTag(style, index, remove, obj) {\n var css = remove ? '' : obj.media ? \"@media \".concat(obj.media, \" {\").concat(obj.css, \"}\") : obj.css; // For old IE\n\n /* istanbul ignore if */\n\n if (style.styleSheet) {\n style.styleSheet.cssText = replaceText(index, css);\n } else {\n var cssNode = document.createTextNode(css);\n var childNodes = style.childNodes;\n\n if (childNodes[index]) {\n style.removeChild(childNodes[index]);\n }\n\n if (childNodes.length) {\n style.insertBefore(cssNode, childNodes[index]);\n } else {\n style.appendChild(cssNode);\n }\n }\n}\n\nfunction applyToTag(style, options, obj) {\n var css = obj.css;\n var media = obj.media;\n var sourceMap = obj.sourceMap;\n\n if (media) {\n style.setAttribute('media', media);\n } else {\n style.removeAttribute('media');\n }\n\n if (sourceMap && typeof btoa !== 'undefined') {\n css += \"\\n/*# sourceMappingURL=data:application/json;base64,\".concat(btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))), \" */\");\n } // For old IE\n\n /* istanbul ignore if */\n\n\n if (style.styleSheet) {\n style.styleSheet.cssText = css;\n } else {\n while (style.firstChild) {\n style.removeChild(style.firstChild);\n }\n\n style.appendChild(document.createTextNode(css));\n }\n}\n\nvar singleton = null;\nvar singletonCounter = 0;\n\nfunction addStyle(obj, options) {\n var style;\n var update;\n var remove;\n\n if (options.singleton) {\n var styleIndex = singletonCounter++;\n style = singleton || (singleton = insertStyleElement(options));\n update = applyToSingletonTag.bind(null, style, styleIndex, false);\n remove = applyToSingletonTag.bind(null, style, styleIndex, true);\n } else {\n style = insertStyleElement(options);\n update = applyToTag.bind(null, style, options);\n\n remove = function remove() {\n removeStyleElement(style);\n };\n }\n\n update(obj);\n return function updateStyle(newObj) {\n if (newObj) {\n if (newObj.css === obj.css && newObj.media === obj.media && newObj.sourceMap === obj.sourceMap) {\n return;\n }\n\n update(obj = newObj);\n } else {\n remove();\n }\n };\n}\n\nmodule.exports = function (list, options) {\n options = options || {}; // Force single-tag solution on IE6-9, which has a hard limit on the # of <style>\n // tags it will allow on a page\n\n if (!options.singleton && typeof options.singleton !== 'boolean') {\n options.singleton = isOldIE();\n }\n\n list = list || [];\n var lastIdentifiers = modulesToDom(list, options);\n return function update(newList) {\n newList = newList || [];\n\n if (Object.prototype.toString.call(newList) !== '[object Array]') {\n return;\n }\n\n for (var i = 0; i < lastIdentifiers.length; i++) {\n var identifier = lastIdentifiers[i];\n var index = getIndexByIdentifier(identifier);\n stylesInDom[index].references--;\n }\n\n var newLastIdentifiers = modulesToDom(newList, options);\n\n for (var _i = 0; _i < lastIdentifiers.length; _i++) {\n var _identifier = lastIdentifiers[_i];\n\n var _index = getIndexByIdentifier(_identifier);\n\n if (stylesInDom[_index].references === 0) {\n stylesInDom[_index].updater();\n\n stylesInDom.splice(_index, 1);\n }\n }\n\n lastIdentifiers = newLastIdentifiers;\n };\n};","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\tid: moduleId,\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","__webpack_require__.nc = undefined;","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals navigator:false */\n\n/**\n * @module utils/env\n */\n\n/**\n * Safely returns `userAgent` from browser's navigator API in a lower case.\n * If navigator API is not available it will return an empty string.\n */\nexport function getUserAgent( ): string {\n\t// In some environments navigator API might not be available.\n\ttry {\n\t\treturn navigator.userAgent.toLowerCase();\n\t} catch ( e ) {\n\t\treturn '';\n\t}\n}\n\nconst userAgent = getUserAgent();\n\n// This interface exists to make our API pages more readable.\n/**\n * A namespace containing environment and browser information.\n */\nexport interface EnvType {\n\n\t/**\n\t * Indicates that the application is running on Macintosh.\n\t */\n\treadonly isMac: boolean;\n\n\t/**\n\t * Indicates that the application is running on Windows.\n\t */\n\treadonly isWindows: boolean;\n\n\t/**\n\t * Indicates that the application is running in Firefox (Gecko).\n\t */\n\treadonly isGecko: boolean;\n\n\t/**\n\t * Indicates that the application is running in Safari.\n\t */\n\treadonly isSafari: boolean;\n\n\t/**\n\t * Indicates the the application is running in iOS.\n\t */\n\treadonly isiOS: boolean;\n\n\t/**\n\t * Indicates that the application is running on Android mobile device.\n\t */\n\treadonly isAndroid: boolean;\n\n\t/**\n\t * Indicates that the application is running in a browser using the Blink engine.\n\t */\n\treadonly isBlink: boolean;\n\n\t/**\n\t * Environment features information.\n\t */\n\treadonly features: EnvFeaturesType;\n}\n\nexport interface EnvFeaturesType {\n\n\t/**\n\t * Indicates that the environment supports ES2018 Unicode property escapes — like `\\p{P}` or `\\p{L}`.\n\t * More information about unicode properties might be found\n\t * [in Unicode Standard Annex #44](https://www.unicode.org/reports/tr44/#GC_Values_Table).\n\t */\n\treadonly isRegExpUnicodePropertySupported: boolean;\n}\n\n/**\n * A namespace containing environment and browser information.\n */\nconst env: EnvType = {\n\tisMac: isMac( userAgent ),\n\n\tisWindows: isWindows( userAgent ),\n\n\tisGecko: isGecko( userAgent ),\n\n\tisSafari: isSafari( userAgent ),\n\n\tisiOS: isiOS( userAgent ),\n\n\tisAndroid: isAndroid( userAgent ),\n\n\tisBlink: isBlink( userAgent ),\n\n\tfeatures: {\n\t\tisRegExpUnicodePropertySupported: isRegExpUnicodePropertySupported()\n\t}\n};\n\nexport default env;\n\n/**\n * Checks if User Agent represented by the string is running on Macintosh.\n *\n * @param userAgent **Lowercase** `navigator.userAgent` string.\n * @returns Whether User Agent is running on Macintosh or not.\n */\nexport function isMac( userAgent: string ): boolean {\n\treturn userAgent.indexOf( 'macintosh' ) > -1;\n}\n\n/**\n * Checks if User Agent represented by the string is running on Windows.\n *\n * @param userAgent **Lowercase** `navigator.userAgent` string.\n * @returns Whether User Agent is running on Windows or not.\n */\nexport function isWindows( userAgent: string ): boolean {\n\treturn userAgent.indexOf( 'windows' ) > -1;\n}\n\n/**\n * Checks if User Agent represented by the string is Firefox (Gecko).\n *\n * @param userAgent **Lowercase** `navigator.userAgent` string.\n * @returns Whether User Agent is Firefox or not.\n */\nexport function isGecko( userAgent: string ): boolean {\n\treturn !!userAgent.match( /gecko\\/\\d+/ );\n}\n\n/**\n * Checks if User Agent represented by the string is Safari.\n *\n * @param userAgent **Lowercase** `navigator.userAgent` string.\n * @returns Whether User Agent is Safari or not.\n */\nexport function isSafari( userAgent: string ): boolean {\n\treturn userAgent.indexOf( ' applewebkit/' ) > -1 && userAgent.indexOf( 'chrome' ) === -1;\n}\n\n/**\n * Checks if User Agent represented by the string is running in iOS.\n *\n * @param userAgent **Lowercase** `navigator.userAgent` string.\n * @returns Whether User Agent is running in iOS or not.\n */\nexport function isiOS( userAgent: string ): boolean {\n\t// \"Request mobile site\" || \"Request desktop site\".\n\treturn !!userAgent.match( /iphone|ipad/i ) || ( isMac( userAgent ) && navigator.maxTouchPoints > 0 );\n}\n\n/**\n * Checks if User Agent represented by the string is Android mobile device.\n *\n * @param userAgent **Lowercase** `navigator.userAgent` string.\n * @returns Whether User Agent is Safari or not.\n */\nexport function isAndroid( userAgent: string ): boolean {\n\treturn userAgent.indexOf( 'android' ) > -1;\n}\n\n/**\n * Checks if User Agent represented by the string is Blink engine.\n *\n * @param userAgent **Lowercase** `navigator.userAgent` string.\n * @returns Whether User Agent is Blink engine or not.\n */\nexport function isBlink( userAgent: string ): boolean {\n\t// The Edge browser before switching to the Blink engine used to report itself as Chrome (and \"Edge/\")\n\t// but after switching to the Blink it replaced \"Edge/\" with \"Edg/\".\n\treturn userAgent.indexOf( 'chrome/' ) > -1 && userAgent.indexOf( 'edge/' ) < 0;\n}\n\n/**\n * Checks if the current environment supports ES2018 Unicode properties like `\\p{P}` or `\\p{L}`.\n * More information about unicode properties might be found\n * [in Unicode Standard Annex #44](https://www.unicode.org/reports/tr44/#GC_Values_Table).\n */\nexport function isRegExpUnicodePropertySupported(): boolean {\n\tlet isSupported = false;\n\n\t// Feature detection for Unicode properties. Added in ES2018. Currently Firefox does not support it.\n\t// See https://github.com/ckeditor/ckeditor5-mention/issues/44#issuecomment-487002174.\n\n\ttry {\n\t\t// Usage of regular expression literal cause error during build (ckeditor/ckeditor5-dev#534).\n\t\tisSupported = 'ć'.search( new RegExp( '[\\\\p{L}]', 'u' ) ) === 0;\n\t} catch ( error ) {\n\t\t// Firefox throws a SyntaxError when the group is unsupported.\n\t}\n\n\treturn isSupported;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport type { DiffResult } from './diff';\nimport type { Change } from './difftochanges';\n\n/**\n * @module utils/fastdiff\n */\n\n/**\n * Finds positions of the first and last change in the given string/array and generates a set of changes:\n *\n * ```ts\n * fastDiff( '12a', '12xyza' );\n * // [ { index: 2, type: 'insert', values: [ 'x', 'y', 'z' ] } ]\n *\n * fastDiff( '12a', '12aa' );\n * // [ { index: 3, type: 'insert', values: [ 'a' ] } ]\n *\n * fastDiff( '12xyza', '12a' );\n * // [ { index: 2, type: 'delete', howMany: 3 } ]\n *\n * fastDiff( [ '1', '2', 'a', 'a' ], [ '1', '2', 'a' ] );\n * // [ { index: 3, type: 'delete', howMany: 1 } ]\n *\n * fastDiff( [ '1', '2', 'a', 'b', 'c', '3' ], [ '2', 'a', 'b' ] );\n * // [ { index: 0, type: 'insert', values: [ '2', 'a', 'b' ] }, { index: 3, type: 'delete', howMany: 6 } ]\n * ```\n *\n * Passed arrays can contain any type of data, however to compare them correctly custom comparator function\n * should be passed as a third parameter:\n *\n * ```ts\n * fastDiff( [ { value: 1 }, { value: 2 } ], [ { value: 1 }, { value: 3 } ], ( a, b ) => {\n * \treturn a.value === b.value;\n * } );\n * // [ { index: 1, type: 'insert', values: [ { value: 3 } ] }, { index: 2, type: 'delete', howMany: 1 } ]\n * ```\n *\n * The resulted set of changes can be applied to the input in order to transform it into the output, for example:\n *\n * ```ts\n * let input = '12abc3';\n * const output = '2ab';\n * const changes = fastDiff( input, output );\n *\n * changes.forEach( change => {\n * \tif ( change.type == 'insert' ) {\n * \t\tinput = input.substring( 0, change.index ) + change.values.join( '' ) + input.substring( change.index );\n * \t} else if ( change.type == 'delete' ) {\n * \t\tinput = input.substring( 0, change.index ) + input.substring( change.index + change.howMany );\n * \t}\n * } );\n *\n * // input equals output now\n * ```\n *\n * or in case of arrays:\n *\n * ```ts\n * let input = [ '1', '2', 'a', 'b', 'c', '3' ];\n * const output = [ '2', 'a', 'b' ];\n * const changes = fastDiff( input, output );\n *\n * changes.forEach( change => {\n * \tif ( change.type == 'insert' ) {\n * \t\tinput = input.slice( 0, change.index ).concat( change.values, input.slice( change.index ) );\n * \t} else if ( change.type == 'delete' ) {\n * \t\tinput = input.slice( 0, change.index ).concat( input.slice( change.index + change.howMany ) );\n * \t}\n * } );\n *\n * // input equals output now\n * ```\n *\n * By passing `true` as the fourth parameter (`atomicChanges`) the output of this function will become compatible with\n * the {@link module:utils/diff~diff `diff()`} function:\n *\n * ```ts\n * fastDiff( '12a', '12xyza', undefined, true );\n * // [ 'equal', 'equal', 'insert', 'insert', 'insert', 'equal' ]\n * ```\n *\n * The default output format of this function is compatible with the output format of\n * {@link module:utils/difftochanges~diffToChanges `diffToChanges()`}. The `diffToChanges()` input format is, in turn,\n * compatible with the output of {@link module:utils/diff~diff `diff()`}:\n *\n * ```ts\n * const a = '1234';\n * const b = '12xyz34';\n *\n * // Both calls will return the same results (grouped changes format).\n * fastDiff( a, b );\n * diffToChanges( diff( a, b ) );\n *\n * // Again, both calls will return the same results (atomic changes format).\n * fastDiff( a, b, undefined, true );\n * diff( a, b );\n * ```\n *\n * @typeParam T The type of array elements.\n * @typeParam AtomicChanges The type of `atomicChanges` parameter (selects the result type).\n * @param a Input array or string.\n * @param b Input array or string.\n * @param cmp Optional function used to compare array values, by default `===` (strict equal operator) is used.\n * @param atomicChanges Whether an array of `inset|delete|equal` operations should\n * be returned instead of changes set. This makes this function compatible with {@link module:utils/diff~diff `diff()`}.\n * Defaults to `false`.\n * @returns Array of changes. The elements are either {@link module:utils/diff~DiffResult} or {@link module:utils/difftochanges~Change},\n * depending on `atomicChanges` parameter.\n */\nexport default function fastDiff<T, AtomicChanges extends boolean = false>(\n\ta: ArrayLike<T>,\n\tb: ArrayLike<T>,\n\tcmp?: ( a: T, b: T ) => boolean,\n\tatomicChanges?: AtomicChanges\n): Array<AtomicChanges extends true ? DiffResult : Change<T>> {\n\t// Set the comparator function.\n\tcmp = cmp || function( a, b ) {\n\t\treturn a === b;\n\t};\n\n\t// Convert the string (or any array-like object - eg. NodeList) to an array by using the slice() method because,\n\t// unlike Array.from(), it returns array of UTF-16 code units instead of the code points of a string.\n\t// One code point might be a surrogate pair of two code units. All text offsets are expected to be in code units.\n\t// See ckeditor/ckeditor5#3147.\n\t//\n\t// We need to make sure here that fastDiff() works identical to diff().\n\tconst arrayA: Array<T> = Array.isArray( a ) ? a : Array.prototype.slice.call( a );\n\tconst arrayB: Array<T> = Array.isArray( b ) ? b : Array.prototype.slice.call( b );\n\n\t// Find first and last change.\n\tconst changeIndexes = findChangeBoundaryIndexes( arrayA, arrayB, cmp );\n\n\t// Transform into changes array.\n\tconst result = atomicChanges ?\n\t\tchangeIndexesToAtomicChanges( changeIndexes, arrayB.length ) :\n\t\tchangeIndexesToChanges( arrayB, changeIndexes );\n\n\treturn result as any;\n}\n\n/**\n * Finds position of the first and last change in the given arrays. For example:\n *\n * ```ts\n * const indexes = findChangeBoundaryIndexes( [ '1', '2', '3', '4' ], [ '1', '3', '4', '2', '4' ] );\n * console.log( indexes ); // { firstIndex: 1, lastIndexOld: 3, lastIndexNew: 4 }\n * ```\n *\n * The above indexes means that in the first array the modified part is `1[23]4` and in the second array it is `1[342]4`.\n * Based on such indexes, array with `insert`/`delete` operations which allows transforming first value into the second one\n * can be generated.\n */\nfunction findChangeBoundaryIndexes<T>( arr1: ReadonlyArray<T>, arr2: ReadonlyArray<T>, cmp: ( a: T, b: T ) => boolean ): ChangeIndexes {\n\t// Find the first difference between passed values.\n\tconst firstIndex = findFirstDifferenceIndex( arr1, arr2, cmp );\n\n\t// If arrays are equal return -1 indexes object.\n\tif ( firstIndex === -1 ) {\n\t\treturn { firstIndex: -1, lastIndexOld: -1, lastIndexNew: -1 };\n\t}\n\n\t// Remove the common part of each value and reverse them to make it simpler to find the last difference between them.\n\tconst oldArrayReversed = cutAndReverse( arr1, firstIndex );\n\tconst newArrayReversed = cutAndReverse( arr2, firstIndex );\n\n\t// Find the first difference between reversed values.\n\t// It should be treated as \"how many elements from the end the last difference occurred\".\n\t//\n\t// For example:\n\t//\n\t// \t\t\t\tinitial\t->\tafter cut\t-> reversed:\n\t// oldValue:\t'321ba'\t->\t'21ba'\t\t-> 'ab12'\n\t// newValue:\t'31xba'\t->\t'1xba'\t\t-> 'abx1'\n\t// lastIndex:\t\t\t\t\t\t\t-> 2\n\t//\n\t// So the last change occurred two characters from the end of the arrays.\n\tconst lastIndex = findFirstDifferenceIndex( oldArrayReversed, newArrayReversed, cmp );\n\n\t// Use `lastIndex` to calculate proper offset, starting from the beginning (`lastIndex` kind of starts from the end).\n\tconst lastIndexOld = arr1.length - lastIndex;\n\tconst lastIndexNew = arr2.length - lastIndex;\n\n\treturn { firstIndex, lastIndexOld, lastIndexNew };\n}\n\n/**\n * Returns a first index on which given arrays differ. If both arrays are the same, -1 is returned.\n */\nfunction findFirstDifferenceIndex<T>( arr1: ReadonlyArray<T>, arr2: ReadonlyArray<T>, cmp: ( a: T, b: T ) => boolean ): number {\n\tfor ( let i = 0; i < Math.max( arr1.length, arr2.length ); i++ ) {\n\t\tif ( arr1[ i ] === undefined || arr2[ i ] === undefined || !cmp( arr1[ i ], arr2[ i ] ) ) {\n\t\t\treturn i;\n\t\t}\n\t}\n\n\treturn -1; // Return -1 if arrays are equal.\n}\n\n/**\n * Returns a copy of the given array with `howMany` elements removed starting from the beginning and in reversed order.\n *\n * @param arr Array to be processed.\n * @param howMany How many elements from array beginning to remove.\n * @returns Shortened and reversed array.\n */\nfunction cutAndReverse<T>( arr: ReadonlyArray<T>, howMany: number ): Array<T> {\n\treturn arr.slice( howMany ).reverse();\n}\n\n/**\n * Generates changes array based on change indexes from `findChangeBoundaryIndexes` function. This function will\n * generate array with 0 (no changes), 1 (deletion or insertion) or 2 records (insertion and deletion).\n *\n * @param newArray New array for which change indexes were calculated.\n * @param changeIndexes Change indexes object from `findChangeBoundaryIndexes` function.\n * @returns Array of changes compatible with {@link module:utils/difftochanges~diffToChanges} format.\n */\nfunction changeIndexesToChanges<T>( newArray: ReadonlyArray<T>, changeIndexes: ChangeIndexes ): Array<Change<T>> {\n\tconst result: Array<Change<T>> = [];\n\tconst { firstIndex, lastIndexOld, lastIndexNew } = changeIndexes;\n\n\t// Order operations as 'insert', 'delete' array to keep compatibility with {@link module:utils/difftochanges~diffToChanges}\n\t// in most cases. However, 'diffToChanges' does not stick to any order so in some cases\n\t// (for example replacing '12345' with 'abcd') it will generate 'delete', 'insert' order.\n\tif ( lastIndexNew - firstIndex > 0 ) {\n\t\tresult.push( {\n\t\t\tindex: firstIndex,\n\t\t\ttype: 'insert',\n\t\t\tvalues: newArray.slice( firstIndex, lastIndexNew )\n\t\t} );\n\t}\n\n\tif ( lastIndexOld - firstIndex > 0 ) {\n\t\tresult.push( {\n\t\t\tindex: firstIndex + ( lastIndexNew - firstIndex ), // Increase index of what was inserted.\n\t\t\ttype: 'delete',\n\t\t\thowMany: lastIndexOld - firstIndex\n\t\t} );\n\t}\n\n\treturn result;\n}\n\n/**\n * Generates array with set `equal|insert|delete` operations based on change indexes from `findChangeBoundaryIndexes` function.\n *\n * @param changeIndexes Change indexes object from `findChangeBoundaryIndexes` function.\n * @param newLength Length of the new array on which `findChangeBoundaryIndexes` calculated change indexes.\n * @returns Array of changes compatible with {@link module:utils/diff~diff} format.\n */\nfunction changeIndexesToAtomicChanges( changeIndexes: ChangeIndexes, newLength: number ): Array<DiffResult> {\n\tconst { firstIndex, lastIndexOld, lastIndexNew } = changeIndexes;\n\n\t// No changes.\n\tif ( firstIndex === -1 ) {\n\t\treturn Array( newLength ).fill( 'equal' );\n\t}\n\n\tlet result: Array<DiffResult> = [];\n\n\tif ( firstIndex > 0 ) {\n\t\tresult = result.concat( Array( firstIndex ).fill( 'equal' ) );\n\t}\n\n\tif ( lastIndexNew - firstIndex > 0 ) {\n\t\tresult = result.concat( Array( lastIndexNew - firstIndex ).fill( 'insert' ) );\n\t}\n\n\tif ( lastIndexOld - firstIndex > 0 ) {\n\t\tresult = result.concat( Array( lastIndexOld - firstIndex ).fill( 'delete' ) );\n\t}\n\n\tif ( lastIndexNew < newLength ) {\n\t\tresult = result.concat( Array( newLength - lastIndexNew ).fill( 'equal' ) );\n\t}\n\n\treturn result;\n}\n\n/**\n * Indexes of the first and the last change in the given arrays.\n */\ninterface ChangeIndexes {\n\n\t/**\n\t * Index of the first change in both values (always the same for both).\n\t */\n\tfirstIndex: number;\n\n\t/**\n\t * Index of the last common value in `arr1`.\n\t */\n\tlastIndexOld: number;\n\n\t/**\n\t * Index of the last common value in `arr2`.\n\t */\n\tlastIndexNew: number;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/diff\n */\n\nimport fastDiff from './fastdiff';\n\n// The following code is based on the \"O(NP) Sequence Comparison Algorithm\"\n// by Sun Wu, Udi Manber, Gene Myers, Webb Miller.\n\n/**\n * Calculates the difference between two arrays or strings producing an array containing a list of changes\n * necessary to transform input into output.\n *\n * ```ts\n * diff( 'aba', 'acca' ); // [ 'equal', 'insert', 'insert', 'delete', 'equal' ]\n * ```\n *\n * This function is based on the \"O(NP) Sequence Comparison Algorithm\" by Sun Wu, Udi Manber, Gene Myers, Webb Miller.\n * Unfortunately, while it gives the most precise results, its to complex for longer strings/arrow (above 200 items).\n * Therefore, `diff()` automatically switches to {@link module:utils/fastdiff~fastDiff `fastDiff()`} when detecting\n * such a scenario. The return formats of both functions are identical.\n *\n * @param a Input array or string.\n * @param b Output array or string.\n * @param cmp Optional function used to compare array values, by default === is used.\n * @returns Array of changes.\n */\nexport default function diff<T>(\n\ta: ArrayLike<T>,\n\tb: ArrayLike<T>,\n\tcmp?: ( a: T, b: T ) => boolean\n): Array<DiffResult> {\n\t// Set the comparator function.\n\tcmp = cmp || function( a, b ) {\n\t\treturn a === b;\n\t};\n\n\tconst aLength = a.length;\n\tconst bLength = b.length;\n\n\t// Perform `fastDiff` for longer strings/arrays (see #269).\n\tif ( aLength > 200 || bLength > 200 || aLength + bLength > 300 ) {\n\t\treturn diff.fastDiff( a, b, cmp, true );\n\t}\n\n\t// Temporary action type statics.\n\tlet _insert: DiffResult, _delete: DiffResult;\n\n\t// Swapped the arrays to use the shorter one as the first one.\n\tif ( bLength < aLength ) {\n\t\tconst tmp = a;\n\n\t\ta = b;\n\t\tb = tmp;\n\n\t\t// We swap the action types as well.\n\t\t_insert = 'delete';\n\t\t_delete = 'insert';\n\t} else {\n\t\t_insert = 'insert';\n\t\t_delete = 'delete';\n\t}\n\n\tconst m = a.length;\n\tconst n = b.length;\n\tconst delta = n - m;\n\n\t// Edit scripts, for each diagonal.\n\tconst es: { [ k: number ]: Array<DiffResult> } = {};\n\t// Furthest points, the furthest y we can get on each diagonal.\n\tconst fp: { [ k: number]: number } = {};\n\n\tfunction snake( k: number ): number {\n\t\t// We use -1 as an alternative below to handle initial values ( instead of filling the fp with -1 first ).\n\t\t// Furthest points (y) on the diagonal below k.\n\t\tconst y1 = ( fp[ k - 1 ] !== undefined ? fp[ k - 1 ] : -1 ) + 1;\n\t\t// Furthest points (y) on the diagonal above k.\n\t\tconst y2 = fp[ k + 1 ] !== undefined ? fp[ k + 1 ] : -1;\n\t\t// The way we should go to get further.\n\t\tconst dir = y1 > y2 ? -1 : 1;\n\n\t\t// Clone previous changes array (if any).\n\t\tif ( es[ k + dir ] ) {\n\t\t\tes[ k ] = es[ k + dir ].slice( 0 );\n\t\t}\n\n\t\t// Create changes array.\n\t\tif ( !es[ k ] ) {\n\t\t\tes[ k ] = [];\n\t\t}\n\n\t\t// Push the action.\n\t\tes[ k ].push( y1 > y2 ? _insert : _delete );\n\n\t\t// Set the beginning coordinates.\n\t\tlet y = Math.max( y1, y2 );\n\t\tlet x = y - k;\n\n\t\t// Traverse the diagonal as long as the values match.\n\t\twhile ( x < m && y < n && cmp!( a[ x ], b[ y ] ) ) {\n\t\t\tx++;\n\t\t\ty++;\n\t\t\t// Push no change action.\n\t\t\tes[ k ].push( 'equal' );\n\t\t}\n\n\t\treturn y;\n\t}\n\n\tlet p = 0;\n\tlet k;\n\n\t// Traverse the graph until we reach the end of the longer string.\n\tdo {\n\t\t// Updates furthest points and edit scripts for diagonals below delta.\n\t\tfor ( k = -p; k < delta; k++ ) {\n\t\t\tfp[ k ] = snake( k );\n\t\t}\n\n\t\t// Updates furthest points and edit scripts for diagonals above delta.\n\t\tfor ( k = delta + p; k > delta; k-- ) {\n\t\t\tfp[ k ] = snake( k );\n\t\t}\n\n\t\t// Updates furthest point and edit script for the delta diagonal.\n\t\t// note that the delta diagonal is the one which goes through the sink (m, n).\n\t\tfp[ delta ] = snake( delta );\n\n\t\tp++;\n\t} while ( fp[ delta ] !== n );\n\n\t// Return the final list of edit changes.\n\t// We remove the first item that represents the action for the injected nulls.\n\treturn es[ delta ].slice( 1 );\n}\n\n// Store the API in static property to easily overwrite it in tests.\n// Too bad dependency injection does not work in Webpack + ES 6 (const) + Babel.\ndiff.fastDiff = fastDiff;\n\n/**\n * The element of the result of {@link module:utils/diff~diff} function.\n */\nexport type DiffResult = 'equal' | 'insert' | 'delete';\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/spy\n */\n\n/**\n * Creates a spy function (ala Sinon.js) that can be used to inspect call to it.\n *\n * The following are the present features:\n *\n * * spy.called: property set to `true` if the function has been called at least once.\n *\n * @returns The spy function.\n */\nfunction spy(): { (): void; called?: boolean } {\n\treturn function spy() {\n\t\t( spy as any ).called = true;\n\t};\n}\n\nexport default spy;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/eventinfo\n */\n\nimport spy from './spy';\n\n/**\n * The event object passed to event callbacks. It is used to provide information about the event as well as a tool to\n * manipulate it.\n */\nexport default class EventInfo<TName extends string = string, TReturn = unknown> {\n\t/**\n\t * The object that fired the event.\n\t */\n\tpublic readonly source: object;\n\n\t/**\n\t * The event name.\n\t */\n\tpublic readonly name: TName;\n\n\t/**\n\t * Path this event has followed. See {@link module:utils/emittermixin~Emitter#delegate}.\n\t */\n\tpublic path: Array<object>;\n\n\t/**\n\t * Stops the event emitter to call further callbacks for this event interaction.\n\t */\n\tpublic readonly stop: { (): void; called?: boolean };\n\n\t/**\n\t * Removes the current callback from future interactions of this event.\n\t */\n\tpublic readonly off: { (): void; called?: boolean };\n\n\t/**\n\t * The value which will be returned by {@link module:utils/emittermixin~Emitter#fire}.\n\t *\n\t * It's `undefined` by default and can be changed by an event listener:\n\t *\n\t * ```ts\n\t * dataController.fire( 'getSelectedContent', ( evt ) => {\n\t * \t// This listener will make `dataController.fire( 'getSelectedContent' )`\n\t * \t// always return an empty DocumentFragment.\n\t * \tevt.return = new DocumentFragment();\n\t *\n\t * \t// Make sure no other listeners are executed.\n\t * \tevt.stop();\n\t * } );\n\t * ```\n\t */\n\tpublic return: TReturn | undefined;\n\n\t/**\n\t * @param source The emitter.\n\t * @param name The event name.\n\t */\n\tconstructor( source: object, name: TName ) {\n\t\tthis.source = source;\n\t\tthis.name = name;\n\t\tthis.path = [];\n\n\t\t// The following methods are defined in the constructor because they must be re-created per instance.\n\t\tthis.stop = spy();\n\t\tthis.off = spy();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/uid\n */\n\n/**\n * A hash table of hex numbers to avoid using toString() in uid() which is costly.\n * [ '00', '01', '02', ..., 'fe', 'ff' ]\n */\nconst HEX_NUMBERS = new Array( 256 ).fill( '' )\n\t.map( ( _, index ) => ( '0' + ( index ).toString( 16 ) ).slice( -2 ) );\n\n/**\n * Returns a unique id. The id starts with an \"e\" character and a randomly generated string of\n * 32 alphanumeric characters.\n *\n * **Note**: The characters the unique id is built from correspond to the hex number notation\n * (from \"0\" to \"9\", from \"a\" to \"f\"). In other words, each id corresponds to an \"e\" followed\n * by 16 8-bit numbers next to each other.\n *\n * @returns An unique id string.\n */\nexport default function uid(): string {\n\t// Let's create some positive random 32bit integers first.\n\t//\n\t// 1. Math.random() is a float between 0 and 1.\n\t// 2. 0x100000000 is 2^32 = 4294967296.\n\t// 3. >>> 0 enforces integer (in JS all numbers are floating point).\n\t//\n\t// For instance:\n\t//\t\tMath.random() * 0x100000000 = 3366450031.853859\n\t// but\n\t//\t\tMath.random() * 0x100000000 >>> 0 = 3366450031.\n\tconst r1 = Math.random() * 0x100000000 >>> 0;\n\tconst r2 = Math.random() * 0x100000000 >>> 0;\n\tconst r3 = Math.random() * 0x100000000 >>> 0;\n\tconst r4 = Math.random() * 0x100000000 >>> 0;\n\n\t// Make sure that id does not start with number.\n\treturn 'e' +\n\t\tHEX_NUMBERS[ r1 >> 0 & 0xFF ] +\n\t\tHEX_NUMBERS[ r1 >> 8 & 0xFF ] +\n\t\tHEX_NUMBERS[ r1 >> 16 & 0xFF ] +\n\t\tHEX_NUMBERS[ r1 >> 24 & 0xFF ] +\n\t\tHEX_NUMBERS[ r2 >> 0 & 0xFF ] +\n\t\tHEX_NUMBERS[ r2 >> 8 & 0xFF ] +\n\t\tHEX_NUMBERS[ r2 >> 16 & 0xFF ] +\n\t\tHEX_NUMBERS[ r2 >> 24 & 0xFF ] +\n\t\tHEX_NUMBERS[ r3 >> 0 & 0xFF ] +\n\t\tHEX_NUMBERS[ r3 >> 8 & 0xFF ] +\n\t\tHEX_NUMBERS[ r3 >> 16 & 0xFF ] +\n\t\tHEX_NUMBERS[ r3 >> 24 & 0xFF ] +\n\t\tHEX_NUMBERS[ r4 >> 0 & 0xFF ] +\n\t\tHEX_NUMBERS[ r4 >> 8 & 0xFF ] +\n\t\tHEX_NUMBERS[ r4 >> 16 & 0xFF ] +\n\t\tHEX_NUMBERS[ r4 >> 24 & 0xFF ];\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/priorities\n */\n\n/**\n * String representing a priority value.\n */\nexport type PriorityString = 'highest' | 'high' | 'normal' | 'low' | 'lowest' | number;\n\n// This interface exists to make our API pages more readable.\n/**\n * Provides group of constants to use instead of hardcoding numeric priority values.\n */\nexport interface PrioritiesType {\n\n\t/**\n\t * Converts a string with priority name to it's numeric value. If `Number` is given, it just returns it.\n\t *\n\t * @param priority Priority to convert.\n\t * @returns Converted priority.\n\t */\n\tget( priority?: PriorityString ): number;\n\n\treadonly highest: number;\n\treadonly high: number;\n\treadonly normal: number;\n\treadonly low: number;\n\treadonly lowest: number;\n}\n\n/**\n * Provides group of constants to use instead of hardcoding numeric priority values.\n */\nconst priorities: PrioritiesType = {\n\tget( priority: PriorityString = 'normal' ): number {\n\t\tif ( typeof priority != 'number' ) {\n\t\t\treturn this[ priority ] || this.normal;\n\t\t} else {\n\t\t\treturn priority;\n\t\t}\n\t},\n\n\thighest: 100000,\n\thigh: 1000,\n\tnormal: 0,\n\tlow: -1000,\n\tlowest: -100000\n};\n\nexport default priorities;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport priorities, { type PriorityString } from './priorities';\n\n/**\n * @module utils/inserttopriorityarray\n */\n\n/**\n * The priority object descriptor.\n *\n * ```ts\n * const objectWithPriority = {\n * \tpriority: 'high'\n * }\n * ```\n */\nexport interface ObjectWithPriority {\n\n\t/**\n\t * Priority of the object.\n\t */\n\tpriority: PriorityString;\n}\n\n/**\n * Inserts any object with priority at correct index by priority so registered objects are always sorted from highest to lowest priority.\n *\n * @param objects Array of objects with priority to insert object to.\n * @param objectToInsert Object with `priority` property.\n */\nexport default function insertToPriorityArray<T extends ObjectWithPriority>( objects: Array<T>, objectToInsert: T ): void {\n\tconst priority = priorities.get( objectToInsert.priority );\n\n\tfor ( let i = 0; i < objects.length; i++ ) {\n\t\tif ( priorities.get( objects[ i ].priority ) < priority ) {\n\t\t\tobjects.splice( i, 0, objectToInsert );\n\n\t\t\treturn;\n\t\t}\n\t}\n\n\tobjects.push( objectToInsert );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/ckeditorerror\n */\n\n/* globals console */\n\n/**\n * URL to the documentation with error codes.\n */\nexport const DOCUMENTATION_URL = 'https://ckeditor.com/docs/ckeditor5/latest/support/error-codes.html';\n\n/**\n * The CKEditor error class.\n *\n * You should throw `CKEditorError` when:\n *\n * * An unexpected situation occurred and the editor (most probably) will not work properly. Such exception will be handled\n * by the {@link module:watchdog/watchdog~Watchdog watchdog} (if it is integrated),\n * * If the editor is incorrectly integrated or the editor API is used in the wrong way. This way you will give\n * feedback to the developer as soon as possible. Keep in mind that for common integration issues which should not\n * stop editor initialization (like missing upload adapter, wrong name of a toolbar component) we use\n * {@link module:utils/ckeditorerror~logWarning `logWarning()`} and\n * {@link module:utils/ckeditorerror~logError `logError()`}\n * to improve developers experience and let them see the a working editor as soon as possible.\n *\n * ```ts\n * /**\n * * Error thrown when a plugin cannot be loaded due to JavaScript errors, lack of plugins with a given name, etc.\n * *\n * * @error plugin-load\n * * @param pluginName The name of the plugin that could not be loaded.\n * * @param moduleName The name of the module which tried to load this plugin.\n * *\\/\n * throw new CKEditorError( 'plugin-load', {\n * \tpluginName: 'foo',\n * \tmoduleName: 'bar'\n * } );\n * ```\n */\nexport default class CKEditorError extends Error {\n\t/**\n\t * A context of the error by which the Watchdog is able to determine which editor crashed.\n\t */\n\tpublic readonly context: object | null | undefined;\n\n\t/**\n\t * The additional error data passed to the constructor. Undefined if none was passed.\n\t */\n\tpublic readonly data?: object;\n\n\t/**\n\t * Creates an instance of the CKEditorError class.\n\t *\n\t * @param errorName The error id in an `error-name` format. A link to this error documentation page will be added\n\t * to the thrown error's `message`.\n\t * @param context A context of the error by which the {@link module:watchdog/watchdog~Watchdog watchdog}\n\t * is able to determine which editor crashed. It should be an editor instance or a property connected to it. It can be also\n\t * a `null` value if the editor should not be restarted in case of the error (e.g. during the editor initialization).\n\t * The error context should be checked using the `areConnectedThroughProperties( editor, context )` utility\n\t * to check if the object works as the context.\n\t * @param data Additional data describing the error. A stringified version of this object\n\t * will be appended to the error message, so the data are quickly visible in the console. The original\n\t * data object will also be later available under the {@link #data} property.\n\t */\n\tconstructor( errorName: string, context?: object | null, data?: object ) {\n\t\tsuper( getErrorMessage( errorName, data ) );\n\n\t\tthis.name = 'CKEditorError';\n\t\tthis.context = context;\n\t\tthis.data = data;\n\t}\n\n\t/**\n\t * Checks if the error is of the `CKEditorError` type.\n\t */\n\tpublic is( type: string ): boolean {\n\t\treturn type === 'CKEditorError';\n\t}\n\n\t/**\n\t * A utility that ensures that the thrown error is a {@link module:utils/ckeditorerror~CKEditorError} one.\n\t * It is useful when combined with the {@link module:watchdog/watchdog~Watchdog} feature, which can restart the editor in case\n\t * of a {@link module:utils/ckeditorerror~CKEditorError} error.\n\t *\n\t * @param err The error to rethrow.\n\t * @param context An object connected through properties with the editor instance. This context will be used\n\t * by the watchdog to verify which editor should be restarted.\n\t */\n\tpublic static rethrowUnexpectedError( err: Error, context: object ): never {\n\t\tif ( ( err as any ).is && ( err as any ).is( 'CKEditorError' ) ) {\n\t\t\tthrow err;\n\t\t}\n\n\t\t/**\n\t\t * An unexpected error occurred inside the CKEditor 5 codebase. This error will look like the original one\n\t\t * to make the debugging easier.\n\t\t *\n\t\t * This error is only useful when the editor is initialized using the {@link module:watchdog/watchdog~Watchdog} feature.\n\t\t * In case of such error (or any {@link module:utils/ckeditorerror~CKEditorError} error) the watchdog should restart the editor.\n\t\t *\n\t\t * @error unexpected-error\n\t\t */\n\t\tconst error = new CKEditorError( err.message, context );\n\n\t\t// Restore the original stack trace to make the error look like the original one.\n\t\t// See https://github.com/ckeditor/ckeditor5/issues/5595 for more details.\n\t\terror.stack = err.stack;\n\n\t\tthrow error;\n\t}\n}\n\n/**\n * Logs a warning to the console with a properly formatted message and adds a link to the documentation.\n * Use whenever you want to log a warning to the console.\n *\n * ```ts\n * /**\n * * There was a problem processing the configuration of the toolbar. The item with the given\n * * name does not exist, so it was omitted when rendering the toolbar.\n * *\n * * @error toolbarview-item-unavailable\n * * @param {String} name The name of the component.\n * *\\/\n * logWarning( 'toolbarview-item-unavailable', { name } );\n * ```\n *\n * See also {@link module:utils/ckeditorerror~CKEditorError} for an explanation when to throw an error and when to log\n * a warning or an error to the console.\n *\n * @param errorName The error name to be logged.\n * @param data Additional data to be logged.\n */\nexport function logWarning( errorName: string, data?: object ): void {\n\tconsole.warn( ...formatConsoleArguments( errorName, data ) );\n}\n\n/**\n * Logs an error to the console with a properly formatted message and adds a link to the documentation.\n * Use whenever you want to log an error to the console.\n *\n * ```ts\n * /**\n * * There was a problem processing the configuration of the toolbar. The item with the given\n * * name does not exist, so it was omitted when rendering the toolbar.\n * *\n * * @error toolbarview-item-unavailable\n * * @param {String} name The name of the component.\n * *\\/\n * logError( 'toolbarview-item-unavailable', { name } );\n * ```\n *\n * **Note**: In most cases logging a warning using {@link module:utils/ckeditorerror~logWarning} is enough.\n *\n * See also {@link module:utils/ckeditorerror~CKEditorError} for an explanation when to use each method.\n *\n * @param errorName The error name to be logged.\n * @param data Additional data to be logged.\n */\nexport function logError( errorName: string, data?: object ): void {\n\tconsole.error( ...formatConsoleArguments( errorName, data ) );\n}\n\n/**\n * Returns formatted link to documentation message.\n */\nfunction getLinkToDocumentationMessage( errorName: string ): string {\n\treturn `\\nRead more: ${ DOCUMENTATION_URL }#error-${ errorName }`;\n}\n\n/**\n * Returns formatted error message.\n */\nfunction getErrorMessage( errorName: string, data?: object ): string {\n\tconst processedObjects = new WeakSet();\n\tconst circularReferencesReplacer = ( key: string, value: unknown ) => {\n\t\tif ( typeof value === 'object' && value !== null ) {\n\t\t\tif ( processedObjects.has( value ) ) {\n\t\t\t\treturn `[object ${ value.constructor.name }]`;\n\t\t\t}\n\n\t\t\tprocessedObjects.add( value );\n\t\t}\n\n\t\treturn value;\n\t};\n\n\tconst stringifiedData = data ? ` ${ JSON.stringify( data, circularReferencesReplacer ) }` : '';\n\tconst documentationLink = getLinkToDocumentationMessage( errorName );\n\n\treturn errorName + stringifiedData + documentationLink;\n}\n\n/**\n * Returns formatted console error arguments.\n */\nfunction formatConsoleArguments( errorName: string, data?: object ): Array<unknown> {\n\tconst documentationMessage = getLinkToDocumentationMessage( errorName );\n\n\treturn data ? [ errorName, data, documentationMessage ] : [ errorName, documentationMessage ];\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/version\n */\n\n/* globals window, global */\n\nimport CKEditorError from './ckeditorerror';\n\nconst version = '39.0.1';\n\nexport default version;\n\n// The second argument is not a month. It is `monthIndex` and starts from `0`.\nexport const releaseDate = new Date( 2023, 7, 10 );\n\n/* istanbul ignore next -- @preserve */\nconst windowOrGlobal = typeof window === 'object' ? window : global;\n\ndeclare global {\n\t// eslint-disable-next-line no-var\n\tvar CKEDITOR_VERSION: string;\n}\n\n/* istanbul ignore next -- @preserve */\nif ( windowOrGlobal.CKEDITOR_VERSION ) {\n\t/**\n\t * This error is thrown when due to a mistake in how CKEditor 5 was installed or initialized, some\n\t * of its modules were duplicated (evaluated and executed twice). Module duplication leads to inevitable runtime\n\t * errors.\n\t *\n\t * There are many situations in which some modules can be loaded twice. In the worst case scenario,\n\t * you may need to check your project for each of these issues and fix them all.\n\t *\n\t * # Trying to add a plugin to an existing build\n\t *\n\t * If you import an existing CKEditor 5 build and a plugin like this:\n\t *\n\t * ```ts\n\t * import ClassicEditor from '@ckeditor/ckeditor5-build-classic';\n\t * import Highlight from '@ckeditor/ckeditor5-highlight/src/highlight';\n\t * ```\n\t *\n\t * Then your project loads some CKEditor 5 packages twice. How does it happen?\n\t *\n\t * The build package contains a file which is already compiled with webpack. This means\n\t * that it contains all the necessary code from e.g. `@ckeditor/ckeditor5-engine` and `@ckeditor/ckeditor5-utils`.\n\t *\n\t * However, the `Highlight` plugin imports some of the modules from these packages, too. If you ask webpack to\n\t * build such a project, you will end up with the modules being included (and run) twice &mdash; first, because they are\n\t * included inside the build package, and second, because they are required by the `Highlight` plugin.\n\t *\n\t * Therefore, **you must never add plugins to an existing build** unless your plugin has no dependencies.\n\t *\n\t * Adding plugins to a build is done by taking the source version of this build (so, before it was built with webpack)\n\t * and adding plugins there. In this situation, webpack will know that it only needs to load each plugin once.\n\t *\n\t * Read more in the {@glink installation/plugins/installing-plugins Installing plugins} guide.\n\t *\n\t * # Confused an editor build with an editor implementation\n\t *\n\t * This scenario is very similar to the previous one, but has a different origin.\n\t *\n\t * Let's assume that you wanted to use CKEditor 5 from source, as explained in the\n\t * {@glink installation/advanced/alternative-setups/integrating-from-source-webpack \"Building from source\"} section\n\t * or in the {@glink framework/quick-start \"Quick start\"} guide of CKEditor 5 Framework.\n\t *\n\t * The correct way to do so is to import an editor and plugins and run them together like this:\n\t *\n\t * ```ts\n\t * import ClassicEditor from '@ckeditor/ckeditor5-editor-classic/src/classiceditor';\n\t * import Essentials from '@ckeditor/ckeditor5-essentials/src/essentials';\n\t * import Paragraph from '@ckeditor/ckeditor5-paragraph/src/paragraph';\n\t * import Bold from '@ckeditor/ckeditor5-basic-styles/src/bold';\n\t * import Italic from '@ckeditor/ckeditor5-basic-styles/src/italic';\n\t *\n\t * ClassicEditor\n\t * \t.create( document.querySelector( '#editor' ), {\n\t * \t\tplugins: [ Essentials, Paragraph, Bold, Italic ],\n\t * \t\ttoolbar: [ 'bold', 'italic' ]\n\t * \t} )\n\t * \t.then( editor => {\n\t * \t\tconsole.log( 'Editor was initialized', editor );\n\t * \t} )\n\t * \t.catch( error => {\n\t * \t\tconsole.error( error.stack );\n\t * \t} );\n\t * ```\n\t *\n\t * However, you might have mistakenly imported a build instead of the source `ClassicEditor`. In this case\n\t * your imports will look like this:\n\t *\n\t * ```ts\n\t * import ClassicEditor from '@ckeditor/ckeditor5-build-classic';\n\t * import Essentials from '@ckeditor/ckeditor5-essentials/src/essentials';\n\t * import Paragraph from '@ckeditor/ckeditor5-paragraph/src/paragraph';\n\t * import Bold from '@ckeditor/ckeditor5-basic-styles/src/bold';\n\t * import Italic from '@ckeditor/ckeditor5-basic-styles/src/italic';\n\t * ```\n\t *\n\t * This creates the same situation as in the previous section because you use a build together with source plugins.\n\t *\n\t * Remember: `@ckeditor/ckeditor5-build-*` packages contain editor builds and `@ckeditor/ckeditor5-editor-*` contain source editors.\n\t *\n\t * # Loading two or more builds on one page\n\t *\n\t * If you use CKEditor 5 builds, you might have loaded two (or more) `ckeditor.js` files on one web page.\n\t * Check your web page for duplicated `<script>` elements or make sure your page builder/bundler includes CKEditor only once.\n\t *\n\t * If you want to use two different types of editors at once, see the\n\t * {@glink installation/advanced/using-two-editors \"Using two different editors\"}\n\t * section.\n\t *\n\t * # Using outdated packages\n\t *\n\t * Building CKEditor 5 from source requires using multiple npm packages. These packages have their dependencies\n\t * to other packages. If you use the latest version of, for example, `@ckeditor/ckeditor5-editor-classic` with\n\t * an outdated version of `@ckeditor/ckeditor5-image`, npm or yarn will need to install two different versions of\n\t * `@ckeditor/ckeditor5-core` because `@ckeditor/ckeditor5-editor-classic` and `@ckeditor/ckeditor5-image` may require\n\t * different versions of the core package.\n\t *\n\t * The solution to this issue is to update all packages to their latest version. We recommend\n\t * using tools like [`npm-check-updates`](https://www.npmjs.com/package/npm-check-updates) which simplify this process.\n\t *\n\t * # Conflicting version of dependencies\n\t *\n\t * This is a special case of the previous scenario. If you use CKEditor 5 with some third-party plugins,\n\t * it may happen that even if you use the latest versions of the official packages and the latest version of\n\t * these third-party packages, there will be a conflict between some of their dependencies.\n\t *\n\t * Such a problem can be resolved by either downgrading CKEditor 5 packages (which we do not recommend) or\n\t * asking the author of the third-party package to upgrade its depdendencies (or forking their project and doing this yourself).\n\t *\n\t * **Note:** All official CKEditor 5 packages (excluding integrations and `ckeditor5-dev-*` packages) are released in the\n\t * same major version. This is &mdash; in the `x.y.z`, the `x` is the same for all packages. This is the simplest way to check\n\t * whether you use packages coming from the same CKEditor 5 version. You can read more about versioning in the\n\t * {@glink updating/versioning-policy Versioning policy} guide.\n\t *\n\t * # Packages were duplicated in `node_modules`\n\t *\n\t * In some situations, especially when calling `npm install` multiple times, it may happen\n\t * that npm will not correctly \"deduplicate\" packages.\n\t *\n\t * Normally, npm deduplicates all packages so, for example, `@ckeditor/ckeditor5-core` is installed only once in `node_modules/`.\n\t * However, it is known to fail to do so from time to time.\n\t *\n\t * We recommend checking if any of the steps listed below help:\n\t *\n\t * * `rm -rf node_modules && npm install` to make sure you have a clean `node_modules/` directory. This step\n\t * is known to help in most cases.\n\t * * If you use `yarn.lock` or `package-lock.json`, remove it before `npm install`.\n\t * * Check whether all CKEditor 5 packages are up to date and reinstall them\n\t * if you changed anything (`rm -rf node_modules && npm install`).\n\t *\n\t * If all packages are correct and compatible with each other, the steps above are known to help. If not, you may\n\t * try to check with `npm ls` how many times packages like `@ckeditor/ckeditor5-core`, `@ckeditor/ckeditor5-engine` and\n\t *`@ckeditor/ckeditor5-utils` are installed. If more than once, verify which package causes that.\n\t *\n\t * @error ckeditor-duplicated-modules\n\t */\n\tthrow new CKEditorError(\n\t\t'ckeditor-duplicated-modules',\n\t\tnull\n\t);\n} else {\n\twindowOrGlobal.CKEDITOR_VERSION = version;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/emittermixin\n */\n\nimport EventInfo from './eventinfo';\nimport uid from './uid';\nimport priorities, { type PriorityString } from './priorities';\nimport insertToPriorityArray from './inserttopriorityarray';\nimport type { Constructor, Mixed } from './mix';\n\n// To check if component is loaded more than once.\nimport './version';\nimport CKEditorError from './ckeditorerror';\n\nconst _listeningTo = Symbol( 'listeningTo' );\nconst _emitterId = Symbol( 'emitterId' );\nconst _delegations = Symbol( 'delegations' );\n\nconst defaultEmitterClass = EmitterMixin( Object );\n\n/**\n * Mixin that injects the {@link ~Emitter events API} into its host.\n *\n * This function creates a class that inherits from the provided `base` and implements `Emitter` interface.\n *\n * ```ts\n * class BaseClass { ... }\n *\n * class MyClass extends EmitterMixin( BaseClass ) {\n * \t// This class derives from `BaseClass` and implements the `Emitter` interface.\n * }\n * ```\n *\n * Read more about the concept of emitters in the:\n * * {@glink framework/architecture/core-editor-architecture#event-system-and-observables Event system and observables}\n * section of the {@glink framework/architecture/core-editor-architecture Core editor architecture} guide.\n * * {@glink framework/deep-dive/event-system Event system} deep-dive guide.\n *\n * @label EXTENDS\n */\nexport default function EmitterMixin<Base extends Constructor>( base: Base ): Mixed<Base, Emitter>;\n\n/**\n * Mixin that injects the {@link ~Emitter events API} into its host.\n *\n * This function creates a class that implements `Emitter` interface.\n *\n * ```ts\n * class MyClass extends EmitterMixin() {\n * \t// This class implements the `Emitter` interface.\n * }\n * ```\n *\n * Read more about the concept of emitters in the:\n * * {@glink framework/architecture/core-editor-architecture#event-system-and-observables Event system and observables}\n * section of the {@glink framework/architecture/core-editor-architecture Core editor architecture} guide.\n * * {@glink framework/deep-dive/event-system Event system} deep dive guide.\n *\n * @label NO_ARGUMENTS\n */\nexport default function EmitterMixin(): {\n\tnew (): Emitter;\n\tprototype: Emitter;\n};\n\nexport default function EmitterMixin( base?: Constructor ): unknown {\n\tif ( !base ) {\n\t\treturn defaultEmitterClass;\n\t}\n\n\tabstract class Mixin extends base implements EmitterInternal {\n\t\tpublic on<TEvent extends BaseEvent>(\n\t\t\tevent: TEvent[ 'name' ],\n\t\t\tcallback: GetCallback<TEvent>,\n\t\t\toptions?: CallbackOptions\n\t\t): void {\n\t\t\tthis.listenTo( this, event, callback, options );\n\t\t}\n\n\t\tpublic once<TEvent extends BaseEvent>(\n\t\t\tevent: TEvent[ 'name' ],\n\t\t\tcallback: GetCallback<TEvent>,\n\t\t\toptions?: CallbackOptions\n\t\t): void {\n\t\t\tlet wasFired = false;\n\n\t\t\tconst onceCallback: typeof callback = ( event, ...args ) => {\n\t\t\t\t// Ensure the callback is called only once even if the callback itself leads to re-firing the event\n\t\t\t\t// (which would call the callback again).\n\t\t\t\tif ( !wasFired ) {\n\t\t\t\t\twasFired = true;\n\n\t\t\t\t\t// Go off() at the first call.\n\t\t\t\t\tevent.off();\n\n\t\t\t\t\t// Go with the original callback.\n\t\t\t\t\tcallback.call( this, event, ...args );\n\t\t\t\t}\n\t\t\t};\n\n\t\t\t// Make a similar on() call, simply replacing the callback.\n\t\t\tthis.listenTo( this, event, onceCallback, options );\n\t\t}\n\n\t\tpublic off( event: string, callback: Function ): void {\n\t\t\tthis.stopListening( this, event, callback );\n\t\t}\n\n\t\tpublic listenTo<TEvent extends BaseEvent>(\n\t\t\temitter: Emitter,\n\t\t\tevent: TEvent[ 'name' ],\n\t\t\tcallback: GetCallback<TEvent>,\n\t\t\toptions: CallbackOptions = {}\n\t\t): void {\n\t\t\tlet emitterInfo, eventCallbacks;\n\n\t\t\t// _listeningTo contains a list of emitters that this object is listening to.\n\t\t\t// This list has the following format:\n\t\t\t//\n\t\t\t// _listeningTo: {\n\t\t\t// emitterId: {\n\t\t\t// emitter: emitter,\n\t\t\t// callbacks: {\n\t\t\t// event1: [ callback1, callback2, ... ]\n\t\t\t// ....\n\t\t\t// }\n\t\t\t// },\n\t\t\t// ...\n\t\t\t// }\n\n\t\t\tif ( !this[ _listeningTo ] ) {\n\t\t\t\tthis[ _listeningTo ] = {};\n\t\t\t}\n\n\t\t\tconst emitters = this[ _listeningTo ]!;\n\n\t\t\tif ( !_getEmitterId( emitter ) ) {\n\t\t\t\t_setEmitterId( emitter );\n\t\t\t}\n\n\t\t\tconst emitterId = _getEmitterId( emitter )!;\n\n\t\t\tif ( !( emitterInfo = emitters[ emitterId ] ) ) {\n\t\t\t\temitterInfo = emitters[ emitterId ] = {\n\t\t\t\t\temitter,\n\t\t\t\t\tcallbacks: {}\n\t\t\t\t};\n\t\t\t}\n\n\t\t\tif ( !( eventCallbacks = emitterInfo.callbacks[ event ] ) ) {\n\t\t\t\teventCallbacks = emitterInfo.callbacks[ event ] = [];\n\t\t\t}\n\n\t\t\teventCallbacks.push( callback );\n\n\t\t\t// Finally register the callback to the event.\n\t\t\taddEventListener( this, emitter, event, callback, options );\n\t\t}\n\n\t\tpublic stopListening( emitter?: Emitter, event?: string, callback?: Function ): void {\n\t\t\tconst emitters = this[ _listeningTo ];\n\t\t\tlet emitterId = emitter && _getEmitterId( emitter );\n\t\t\tconst emitterInfo = ( emitters && emitterId ) ? emitters[ emitterId ] : undefined;\n\t\t\tconst eventCallbacks = ( emitterInfo && event ) ? emitterInfo.callbacks[ event ] : undefined;\n\n\t\t\t// Stop if nothing has been listened.\n\t\t\tif ( !emitters || ( emitter && !emitterInfo ) || ( event && !eventCallbacks ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// All params provided. off() that single callback.\n\t\t\tif ( callback ) {\n\t\t\t\tremoveEventListener( this, emitter!, event!, callback );\n\n\t\t\t\t// We must remove callbacks as well in order to prevent memory leaks.\n\t\t\t\t// See https://github.com/ckeditor/ckeditor5/pull/8480\n\t\t\t\tconst index = eventCallbacks!.indexOf( callback );\n\n\t\t\t\tif ( index !== -1 ) {\n\t\t\t\t\tif ( eventCallbacks!.length === 1 ) {\n\t\t\t\t\t\tdelete emitterInfo!.callbacks[ event! ];\n\t\t\t\t\t} else {\n\t\t\t\t\t\tremoveEventListener( this, emitter!, event!, callback );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t\t// Only `emitter` and `event` provided. off() all callbacks for that event.\n\t\t\telse if ( eventCallbacks ) {\n\t\t\t\twhile ( ( callback = eventCallbacks.pop() ) ) {\n\t\t\t\t\tremoveEventListener( this, emitter!, event!, callback );\n\t\t\t\t}\n\n\t\t\t\tdelete emitterInfo!.callbacks[ event! ];\n\t\t\t}\n\t\t\t// Only `emitter` provided. off() all events for that emitter.\n\t\t\telse if ( emitterInfo ) {\n\t\t\t\tfor ( event in emitterInfo.callbacks ) {\n\t\t\t\t\tthis.stopListening( emitter!, event );\n\t\t\t\t}\n\t\t\t\tdelete emitters[ emitterId! ];\n\t\t\t}\n\t\t\t// No params provided. off() all emitters.\n\t\t\telse {\n\t\t\t\tfor ( emitterId in emitters ) {\n\t\t\t\t\tthis.stopListening( emitters[ emitterId ].emitter );\n\t\t\t\t}\n\t\t\t\tdelete this[ _listeningTo ];\n\t\t\t}\n\t\t}\n\n\t\tpublic fire<TEvent extends BaseEvent>(\n\t\t\teventOrInfo: GetNameOrEventInfo<TEvent>,\n\t\t\t...args: TEvent[ 'args' ]\n\t\t): GetEventInfo<TEvent>[ 'return' ] {\n\t\t\ttry {\n\t\t\t\tconst eventInfo = eventOrInfo instanceof EventInfo ? eventOrInfo : new EventInfo( this, eventOrInfo );\n\t\t\t\tconst event = eventInfo.name;\n\t\t\t\tlet callbacks = getCallbacksForEvent( this, event );\n\n\t\t\t\t// Record that the event passed this emitter on its path.\n\t\t\t\teventInfo.path.push( this );\n\n\t\t\t\t// Handle event listener callbacks first.\n\t\t\t\tif ( callbacks ) {\n\t\t\t\t\t// Arguments passed to each callback.\n\t\t\t\t\tconst callbackArgs = [ eventInfo, ...args ];\n\n\t\t\t\t\t// Copying callbacks array is the easiest and most secure way of preventing infinite loops, when event callbacks\n\t\t\t\t\t// are added while processing other callbacks. Previous solution involved adding counters (unique ids) but\n\t\t\t\t\t// failed if callbacks were added to the queue before currently processed callback.\n\t\t\t\t\t// If this proves to be too inefficient, another method is to change `.on()` so callbacks are stored if same\n\t\t\t\t\t// event is currently processed. Then, `.fire()` at the end, would have to add all stored events.\n\t\t\t\t\tcallbacks = Array.from( callbacks );\n\n\t\t\t\t\tfor ( let i = 0; i < callbacks.length; i++ ) {\n\t\t\t\t\t\tcallbacks[ i ].callback.apply( this, callbackArgs );\n\n\t\t\t\t\t\t// Remove the callback from future requests if off() has been called.\n\t\t\t\t\t\tif ( eventInfo.off.called ) {\n\t\t\t\t\t\t\t// Remove the called mark for the next calls.\n\t\t\t\t\t\t\tdelete eventInfo.off.called;\n\n\t\t\t\t\t\t\tthis._removeEventListener( event, callbacks[ i ].callback );\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Do not execute next callbacks if stop() was called.\n\t\t\t\t\t\tif ( eventInfo.stop.called ) {\n\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Delegate event to other emitters if needed.\n\t\t\t\tconst delegations = this[ _delegations ];\n\n\t\t\t\tif ( delegations ) {\n\t\t\t\t\tconst destinations = delegations.get( event );\n\t\t\t\t\tconst passAllDestinations = delegations.get( '*' );\n\n\t\t\t\t\tif ( destinations ) {\n\t\t\t\t\t\tfireDelegatedEvents( destinations, eventInfo, args );\n\t\t\t\t\t}\n\n\t\t\t\t\tif ( passAllDestinations ) {\n\t\t\t\t\t\tfireDelegatedEvents( passAllDestinations, eventInfo, args );\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\treturn eventInfo.return;\n\t\t\t} catch ( err ) {\n\t\t\t\t// @if CK_DEBUG // throw err;\n\t\t\t\t/* istanbul ignore next -- @preserve */\n\t\t\t\tCKEditorError.rethrowUnexpectedError( err as Error, this );\n\t\t\t}\n\t\t}\n\n\t\tpublic delegate( ...events: Array<string> ): EmitterMixinDelegateChain {\n\t\t\treturn {\n\t\t\t\tto: ( emitter, nameOrFunction ) => {\n\t\t\t\t\tif ( !this[ _delegations ] ) {\n\t\t\t\t\t\tthis[ _delegations ] = new Map();\n\t\t\t\t\t}\n\n\t\t\t\t\t// Originally there was a for..of loop which unfortunately caused an error in Babel that didn't allow\n\t\t\t\t\t// build an application. See: https://github.com/ckeditor/ckeditor5-react/issues/40.\n\t\t\t\t\tevents.forEach( eventName => {\n\t\t\t\t\t\tconst destinations = this[ _delegations ]!.get( eventName );\n\n\t\t\t\t\t\tif ( !destinations ) {\n\t\t\t\t\t\t\tthis[ _delegations ]!.set( eventName, new Map( [ [ emitter, nameOrFunction ] ] ) );\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tdestinations.set( emitter, nameOrFunction );\n\t\t\t\t\t\t}\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t};\n\t\t}\n\n\t\tpublic stopDelegating( event?: string, emitter?: Emitter ): void {\n\t\t\tif ( !this[ _delegations ] ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( !event ) {\n\t\t\t\tthis[ _delegations ]!.clear();\n\t\t\t} else if ( !emitter ) {\n\t\t\t\tthis[ _delegations ]!.delete( event );\n\t\t\t} else {\n\t\t\t\tconst destinations = this[ _delegations ]!.get( event );\n\n\t\t\t\tif ( destinations ) {\n\t\t\t\t\tdestinations.delete( emitter );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tpublic _addEventListener<TEvent extends BaseEvent>(\n\t\t\tevent: TEvent[ 'name' ],\n\t\t\tcallback: GetCallback<TEvent>,\n\t\t\toptions: CallbackOptions\n\t\t): void {\n\t\t\tcreateEventNamespace( this, event );\n\n\t\t\tconst lists = getCallbacksListsForNamespace( this, event );\n\t\t\tconst priority = priorities.get( options.priority );\n\n\t\t\tconst callbackDefinition = {\n\t\t\t\tcallback,\n\t\t\t\tpriority\n\t\t\t};\n\n\t\t\t// Add the callback to all callbacks list.\n\t\t\tfor ( const callbacks of lists ) {\n\t\t\t\t// Add the callback to the list in the right priority position.\n\t\t\t\tinsertToPriorityArray( callbacks, callbackDefinition );\n\t\t\t}\n\t\t}\n\n\t\tpublic _removeEventListener( event: string, callback: Function ): void {\n\t\t\tconst lists = getCallbacksListsForNamespace( this, event );\n\n\t\t\tfor ( const callbacks of lists ) {\n\t\t\t\tfor ( let i = 0; i < callbacks.length; i++ ) {\n\t\t\t\t\tif ( callbacks[ i ].callback == callback ) {\n\t\t\t\t\t\t// Remove the callback from the list (fixing the next index).\n\t\t\t\t\t\tcallbacks.splice( i, 1 );\n\t\t\t\t\t\ti--;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tpublic _events?: { [ eventName: string ]: EventNode };\n\n\t\tpublic [ _emitterId ]?: string;\n\n\t\tpublic [ _listeningTo ]?: {\n\t\t\t[ emitterId: string ]: {\n\t\t\t\temitter: Emitter;\n\t\t\t\tcallbacks: { [ event: string]: Array<Function> };\n\t\t\t};\n\t\t};\n\n\t\tpublic [ _delegations ]?: Map<string, Map<Emitter, string | ( ( name: string ) => string ) | undefined>>;\n\t}\n\n\treturn Mixin;\n}\n\n// Backward compatibility with `mix`\n( [\n\t'on', 'once', 'off', 'listenTo',\n\t'stopListening', 'fire', 'delegate', 'stopDelegating',\n\t'_addEventListener', '_removeEventListener'\n] ).forEach( key => {\n\t( EmitterMixin as any )[ key ] = ( defaultEmitterClass.prototype as any )[ key ];\n} );\n\n/**\n * Emitter/listener interface.\n *\n * Can be easily implemented by a class by mixing the {@link module:utils/emittermixin~Emitter} mixin.\n *\n * ```ts\n * class MyClass extends EmitterMixin() {\n * \t// This class now implements the `Emitter` interface.\n * }\n * ```\n *\n * Read more about the usage of this interface in the:\n * * {@glink framework/architecture/core-editor-architecture#event-system-and-observables Event system and observables}\n * section of the {@glink framework/architecture/core-editor-architecture Core editor architecture} guide.\n * * {@glink framework/deep-dive/event-system Event system} deep-dive guide.\n */\nexport interface Emitter {\n\n\t/**\n\t * Registers a callback function to be executed when an event is fired.\n\t *\n\t * Shorthand for {@link #listenTo `this.listenTo( this, event, callback, options )`} (it makes the emitter\n\t * listen on itself).\n\t *\n\t * @typeParam TEvent The type descibing the event. See {@link module:utils/emittermixin~BaseEvent}.\n\t * @param event The name of the event.\n\t * @param callback The function to be called on event.\n\t * @param options Additional options.\n\t */\n\ton<TEvent extends BaseEvent>(\n\t\tevent: TEvent[ 'name' ],\n\t\tcallback: GetCallback<TEvent>,\n\t\toptions?: GetCallbackOptions<TEvent>\n\t): void;\n\n\t/**\n\t * Registers a callback function to be executed on the next time the event is fired only. This is similar to\n\t * calling {@link #on} followed by {@link #off} in the callback.\n\t *\n\t * @typeParam TEvent The type descibing the event. See {@link module:utils/emittermixin~BaseEvent}.\n\t * @param event The name of the event.\n\t * @param callback The function to be called on event.\n\t * @param options Additional options.\n\t */\n\tonce<TEvent extends BaseEvent>(\n\t\tevent: TEvent[ 'name' ],\n\t\tcallback: GetCallback<TEvent>,\n\t\toptions?: GetCallbackOptions<TEvent>\n\t): void;\n\n\t/**\n\t * Stops executing the callback on the given event.\n\t * Shorthand for {@link #stopListening `this.stopListening( this, event, callback )`}.\n\t *\n\t * @param event The name of the event.\n\t * @param callback The function to stop being called.\n\t */\n\toff( event: string, callback: Function ): void;\n\n\t/**\n\t * Registers a callback function to be executed when an event is fired in a specific (emitter) object.\n\t *\n\t * Events can be grouped in namespaces using `:`.\n\t * When namespaced event is fired, it additionally fires all callbacks for that namespace.\n\t *\n\t * ```ts\n\t * // myEmitter.on( ... ) is a shorthand for myEmitter.listenTo( myEmitter, ... ).\n\t * myEmitter.on( 'myGroup', genericCallback );\n\t * myEmitter.on( 'myGroup:myEvent', specificCallback );\n\t *\n\t * // genericCallback is fired.\n\t * myEmitter.fire( 'myGroup' );\n\t * // both genericCallback and specificCallback are fired.\n\t * myEmitter.fire( 'myGroup:myEvent' );\n\t * // genericCallback is fired even though there are no callbacks for \"foo\".\n\t * myEmitter.fire( 'myGroup:foo' );\n\t * ```\n\t *\n\t * An event callback can {@link module:utils/eventinfo~EventInfo#stop stop the event} and\n\t * set the {@link module:utils/eventinfo~EventInfo#return return value} of the {@link #fire} method.\n\t *\n\t * @label BASE_EMITTER\n\t * @typeParam TEvent The type describing the event. See {@link module:utils/emittermixin~BaseEvent}.\n\t * @param emitter The object that fires the event.\n\t * @param event The name of the event.\n\t * @param callback The function to be called on event.\n\t * @param options Additional options.\n\t */\n\tlistenTo<TEvent extends BaseEvent>(\n\t\temitter: Emitter,\n\t\tevent: TEvent[ 'name' ],\n\t\tcallback: GetCallback<TEvent>,\n\t\toptions?: GetCallbackOptions<TEvent>\n\t): void;\n\n\t/**\n\t * Stops listening for events. It can be used at different levels:\n\t *\n\t * * To stop listening to a specific callback.\n\t * * To stop listening to a specific event.\n\t * * To stop listening to all events fired by a specific object.\n\t * * To stop listening to all events fired by all objects.\n\t *\n\t * @label BASE_STOP\n\t * @param emitter The object to stop listening to. If omitted, stops it for all objects.\n\t * @param event (Requires the `emitter`) The name of the event to stop listening to. If omitted, stops it\n\t * for all events from `emitter`.\n\t * @param callback (Requires the `event`) The function to be removed from the call list for the given\n\t * `event`.\n\t */\n\tstopListening( emitter?: Emitter, event?: string, callback?: Function ): void;\n\n\t/**\n\t * Fires an event, executing all callbacks registered for it.\n\t *\n\t * The first parameter passed to callbacks is an {@link module:utils/eventinfo~EventInfo} object,\n\t * followed by the optional `args` provided in the `fire()` method call.\n\t *\n\t * @typeParam TEvent The type describing the event. See {@link module:utils/emittermixin~BaseEvent}.\n\t * @param eventOrInfo The name of the event or `EventInfo` object if event is delegated.\n\t * @param args Additional arguments to be passed to the callbacks.\n\t * @returns By default the method returns `undefined`. However, the return value can be changed by listeners\n\t * through modification of the {@link module:utils/eventinfo~EventInfo#return `evt.return`}'s property (the event info\n\t * is the first param of every callback).\n\t */\n\tfire<TEvent extends BaseEvent>(\n\t\teventOrInfo: GetNameOrEventInfo<TEvent>,\n\t\t...args: TEvent[ 'args' ]\n\t): GetEventInfo<TEvent>[ 'return' ];\n\n\t/**\n\t * Delegates selected events to another {@link module:utils/emittermixin~Emitter}. For instance:\n\t *\n\t * ```ts\n\t * emitterA.delegate( 'eventX' ).to( emitterB );\n\t * emitterA.delegate( 'eventX', 'eventY' ).to( emitterC );\n\t * ```\n\t *\n\t * then `eventX` is delegated (fired by) `emitterB` and `emitterC` along with `data`:\n\t *\n\t * ```ts\n\t * emitterA.fire( 'eventX', data );\n\t * ```\n\t *\n\t * and `eventY` is delegated (fired by) `emitterC` along with `data`:\n\t *\n\t * ```ts\n\t * emitterA.fire( 'eventY', data );\n\t * ```\n\t *\n\t * @param events Event names that will be delegated to another emitter.\n\t */\n\tdelegate( ...events: Array<string> ): EmitterMixinDelegateChain;\n\n\t/**\n\t * Stops delegating events. It can be used at different levels:\n\t *\n\t * * To stop delegating all events.\n\t * * To stop delegating a specific event to all emitters.\n\t * * To stop delegating a specific event to a specific emitter.\n\t *\n\t * @param event The name of the event to stop delegating. If omitted, stops it all delegations.\n\t * @param emitter (requires `event`) The object to stop delegating a particular event to.\n\t * If omitted, stops delegation of `event` to all emitters.\n\t */\n\tstopDelegating( event?: string, emitter?: Emitter ): void;\n}\n\ninterface EmitterInternal extends Emitter {\n\n\t/**\n\t * Adds callback to emitter for given event.\n\t *\n\t * @param event The name of the event.\n\t * @param callback The function to be called on event.\n\t * @param options Additional options.\n\t */\n\t_addEventListener?: <TEvent extends BaseEvent>(\n\t\tevent: TEvent[ 'name' ],\n\t\tcallback: GetCallback<TEvent>,\n\t\toptions: CallbackOptions\n\t) => void;\n\n\t/**\n\t * Removes callback from emitter for given event.\n\t *\n\t * @param event The name of the event.\n\t * @param callback The function to stop being called.\n\t */\n\t_removeEventListener?: ( event: string, callback: Function ) => void;\n\n\t_events?: { [ eventName: string ]: EventNode };\n\n\t[ _emitterId ]?: string;\n\n\t[ _listeningTo ]?: {\n\t\t[ emitterId: string ]: {\n\t\t\temitter: Emitter;\n\t\t\tcallbacks: { [ event: string]: Array<Function> };\n\t\t};\n\t};\n\n\t[ _delegations ]?: Map<string, Map<Emitter, string | ( ( name: string ) => string ) | undefined>>;\n}\n\n/**\n * Default type describing any event.\n *\n * Every custom event has to be compatible with `BaseEvent`.\n *\n * ```ts\n * type MyEvent = {\n * \t// In `fire<MyEvent>( name )`, `on<MyEvent>( name )`, `once<MyEvent>( name )` and `listenTo<MyEvent>( name )` calls\n * \t// the `name` argument will be type-checked to ensure it's `'myEvent'` or have `'myEvent:'` prefix.\n * \t// Required.\n * \tname: 'myEvent' | `myEvent:${ string }`;\n *\n * \t// In `fire<MyEvent>( name, a, b )` call, `a` and `b` parameters will be type-checked against `number` and `string`.\n * \t// In `on<MyEvent>`, `once<MyEvent>` and `listenTo<MyEvent>` calls, the parameters of provided callback function\n * \t// will be automatically inferred as `EventInfo`, `number` and `string`.\n * \t// Required.\n * \targs: [ number, string ];\n *\n * \t// `fire<MyEvent>` will have return type `boolean | undefined`.\n * \t// Optional, unknown by default.\n * \treturn: boolean;\n *\n * \t// `fire<MyEvent>( eventInfo )` will type-check that `eventInfo` is `MyEventInfo`, not a base `EventInfo` or string.\n * \t// In `on<MyEvent>`, `once<MyEvent>` and `listenTo<MyEvent>` calls, the first callback parameter will be of this type.\n * \t// Optional.\n * \teventInfo: MyEventInfo;\n *\n * \t// In `on<MyEvent>`, `once<MyEvent>` and `listenTo<MyEvent>` calls, the `options` parameter will be of type\n * \t// `{ myOption?: boolean; priority?: PriorityString }\n * \t// Optional.\n * \tcallbackOptions: { myOption?: boolean };\n * };\n * ```\n */\nexport type BaseEvent = {\n\tname: string;\n\targs: Array<any>;\n};\n\n/**\n * Utility type that gets the `EventInfo` subclass for the given event.\n */\nexport type GetEventInfo<TEvent extends BaseEvent> = TEvent extends { eventInfo: EventInfo } ?\n\tTEvent[ 'eventInfo' ] :\n\tEventInfo<TEvent[ 'name' ], ( TEvent extends { return: infer TReturn } ? TReturn : unknown )>;\n\n/**\n * Utility type that gets the `EventInfo` subclass or event name type for the given event.\n */\nexport type GetNameOrEventInfo<TEvent extends BaseEvent> = TEvent extends { eventInfo: EventInfo } ?\n\tTEvent[ 'eventInfo' ] :\n\tTEvent[ 'name' ] | EventInfo<TEvent[ 'name' ], ( TEvent extends { return: infer TReturn } ? TReturn : unknown )>;\n\n/**\n * Utility type that gets the callback type for the given event.\n */\nexport type GetCallback<TEvent extends BaseEvent> = ( this: Emitter, ev: GetEventInfo<TEvent>, ...args: TEvent[ 'args' ] ) => void;\n\n/**\n * Utility type that gets the callback options for the given event.\n */\nexport type GetCallbackOptions<TEvent extends BaseEvent> = TEvent extends { callbackOptions: infer TOptions } ?\n\tTOptions & CallbackOptions :\n\tCallbackOptions;\n\n/**\n * Additional options for registering a callback.\n */\nexport interface CallbackOptions {\n\n\t/**\n\t * The priority of this event callback. The higher\n\t * the priority value the sooner the callback will be fired. Events having the same priority are called in the\n\t * order they were added.\n\t *\n\t * @defaultValue `'normal'`\n\t */\n\treadonly priority?: PriorityString;\n}\n\n/**\n * Checks if `listeningEmitter` listens to an emitter with given `listenedToEmitterId` and if so, returns that emitter.\n * If not, returns `null`.\n *\n * @internal\n * @param listeningEmitter An emitter that listens.\n * @param listenedToEmitterId Unique emitter id of emitter listened to.\n */\nexport function _getEmitterListenedTo( listeningEmitter: Emitter, listenedToEmitterId: string ): Emitter | null {\n\tconst listeningTo = ( listeningEmitter as EmitterInternal )[ _listeningTo ];\n\tif ( listeningTo && listeningTo[ listenedToEmitterId ] ) {\n\t\treturn listeningTo[ listenedToEmitterId ].emitter;\n\t}\n\n\treturn null;\n}\n\n/**\n * Sets emitter's unique id.\n *\n * **Note:** `_emitterId` can be set only once.\n *\n * @internal\n * @param emitter An emitter for which id will be set.\n * @param id Unique id to set. If not passed, random unique id will be set.\n */\nexport function _setEmitterId( emitter: Emitter, id?: string ): void {\n\tif ( !( emitter as EmitterInternal )[ _emitterId ] ) {\n\t\t( emitter as EmitterInternal )[ _emitterId ] = id || uid();\n\t}\n}\n\n/**\n * Returns emitter's unique id.\n *\n * @internal\n * @param emitter An emitter which id will be returned.\n */\nexport function _getEmitterId( emitter: Emitter ): string | undefined {\n\treturn ( emitter as EmitterInternal )[ _emitterId ];\n}\n\ninterface EventNode {\n\tcallbacks: Array<{ callback: Function; priority: number }>;\n\tchildEvents: Array<string>;\n}\n\n/**\n * Gets the internal `_events` property of the given object.\n * `_events` property store all lists with callbacks for registered event names.\n * If there were no events registered on the object, empty `_events` object is created.\n */\nfunction getEvents( source: EmitterInternal ): { [ eventName: string ]: EventNode } {\n\tif ( !source._events ) {\n\t\tObject.defineProperty( source, '_events', {\n\t\t\tvalue: {}\n\t\t} );\n\t}\n\n\treturn source._events!;\n}\n\n/**\n * Creates event node for generic-specific events relation architecture.\n */\nfunction makeEventNode(): EventNode {\n\treturn {\n\t\tcallbacks: [],\n\t\tchildEvents: []\n\t};\n}\n\n/**\n * Creates an architecture for generic-specific events relation.\n * If needed, creates all events for given eventName, i.e. if the first registered event\n * is foo:bar:abc, it will create foo:bar:abc, foo:bar and foo event and tie them together.\n * It also copies callbacks from more generic events to more specific events when\n * specific events are created.\n */\nfunction createEventNamespace( source: EmitterInternal, eventName: string ): void {\n\tconst events = getEvents( source );\n\n\t// First, check if the event we want to add to the structure already exists.\n\tif ( events[ eventName ] ) {\n\t\t// If it exists, we don't have to do anything.\n\t\treturn;\n\t}\n\n\t// In other case, we have to create the structure for the event.\n\t// Note, that we might need to create intermediate events too.\n\t// I.e. if foo:bar:abc is being registered and we only have foo in the structure,\n\t// we need to also register foo:bar.\n\n\t// Currently processed event name.\n\tlet name = eventName;\n\t// Name of the event that is a child event for currently processed event.\n\tlet childEventName = null;\n\n\t// Array containing all newly created specific events.\n\tconst newEventNodes = [];\n\n\t// While loop can't check for ':' index because we have to handle generic events too.\n\t// In each loop, we truncate event name, going from the most specific name to the generic one.\n\t// I.e. foo:bar:abc -> foo:bar -> foo.\n\twhile ( name !== '' ) {\n\t\tif ( events[ name ] ) {\n\t\t\t// If the currently processed event name is already registered, we can be sure\n\t\t\t// that it already has all the structure created, so we can break the loop here\n\t\t\t// as no more events need to be registered.\n\t\t\tbreak;\n\t\t}\n\n\t\t// If this event is not yet registered, create a new object for it.\n\t\tevents[ name ] = makeEventNode();\n\t\t// Add it to the array with newly created events.\n\t\tnewEventNodes.push( events[ name ] );\n\n\t\t// Add previously processed event name as a child of this event.\n\t\tif ( childEventName ) {\n\t\t\tevents[ name ].childEvents.push( childEventName );\n\t\t}\n\n\t\tchildEventName = name;\n\t\t// If `.lastIndexOf()` returns -1, `.substr()` will return '' which will break the loop.\n\t\tname = name.substr( 0, name.lastIndexOf( ':' ) );\n\t}\n\n\tif ( name !== '' ) {\n\t\t// If name is not empty, we found an already registered event that was a parent of the\n\t\t// event we wanted to register.\n\n\t\t// Copy that event's callbacks to newly registered events.\n\t\tfor ( const node of newEventNodes ) {\n\t\t\tnode.callbacks = events[ name ].callbacks.slice();\n\t\t}\n\n\t\t// Add last newly created event to the already registered event.\n\t\tevents[ name ].childEvents.push( childEventName! );\n\t}\n}\n\n/**\n * Gets an array containing callbacks list for a given event and it's more specific events.\n * I.e. if given event is foo:bar and there is also foo:bar:abc event registered, this will\n * return callback list of foo:bar and foo:bar:abc (but not foo).\n */\nfunction getCallbacksListsForNamespace( source: EmitterInternal, eventName: string ): Array<EventNode[ 'callbacks' ]> {\n\tconst eventNode = getEvents( source )[ eventName ];\n\n\tif ( !eventNode ) {\n\t\treturn [];\n\t}\n\n\tlet callbacksLists = [ eventNode.callbacks ];\n\n\tfor ( let i = 0; i < eventNode.childEvents.length; i++ ) {\n\t\tconst childCallbacksLists = getCallbacksListsForNamespace( source, eventNode.childEvents[ i ] );\n\n\t\tcallbacksLists = callbacksLists.concat( childCallbacksLists );\n\t}\n\n\treturn callbacksLists;\n}\n\n/**\n * Get the list of callbacks for a given event, but only if there any callbacks have been registered.\n * If there are no callbacks registered for given event, it checks if this is a specific event and looks\n * for callbacks for it's more generic version.\n */\nfunction getCallbacksForEvent( source: EmitterInternal, eventName: string ): EventNode[ 'callbacks' ] | null {\n\tlet event;\n\n\tif ( !source._events || !( event = source._events[ eventName ] ) || !event.callbacks.length ) {\n\t\t// There are no callbacks registered for specified eventName.\n\t\t// But this could be a specific-type event that is in a namespace.\n\t\tif ( eventName.indexOf( ':' ) > -1 ) {\n\t\t\t// If the eventName is specific, try to find callback lists for more generic event.\n\t\t\treturn getCallbacksForEvent( source, eventName.substr( 0, eventName.lastIndexOf( ':' ) ) );\n\t\t} else {\n\t\t\t// If this is a top-level generic event, return null;\n\t\t\treturn null;\n\t\t}\n\t}\n\n\treturn event.callbacks;\n}\n\n/**\n * Fires delegated events for given map of destinations.\n *\n * @param destinations A map containing `[ {@link module:utils/emittermixin~Emitter}, \"event name\" ]` pair destinations.\n * @param eventInfo The original event info object.\n * @param fireArgs Arguments the original event was fired with.\n */\nfunction fireDelegatedEvents(\n\tdestinations: Map<Emitter, string | ( ( name: string ) => string ) | undefined>,\n\teventInfo: EventInfo,\n\tfireArgs: Array<any>\n) {\n\tfor ( let [ emitter, name ] of destinations ) {\n\t\tif ( !name ) {\n\t\t\tname = eventInfo.name;\n\t\t} else if ( typeof name == 'function' ) {\n\t\t\tname = name( eventInfo.name );\n\t\t}\n\n\t\tconst delegatedInfo = new EventInfo( eventInfo.source, name );\n\n\t\tdelegatedInfo.path = [ ...eventInfo.path ];\n\n\t\temitter.fire( delegatedInfo, ...fireArgs );\n\t}\n}\n\n/**\n * Helper for registering event callback on the emitter.\n */\nfunction addEventListener<TEvent extends BaseEvent>(\n\tlistener: EmitterInternal,\n\temitter: EmitterInternal,\n\tevent: TEvent[ 'name' ],\n\tcallback: GetCallback<TEvent>,\n\toptions: CallbackOptions\n) {\n\tif ( emitter._addEventListener ) {\n\t\temitter._addEventListener( event, callback, options );\n\t} else {\n\t\t// Allow listening on objects that do not implement Emitter interface.\n\t\t// This is needed in some tests that are using mocks instead of the real objects with EmitterMixin mixed.\n\t\t( listener._addEventListener!<TEvent> ).call( emitter, event, callback, options );\n\t}\n}\n\n/**\n * Helper for removing event callback from the emitter.\n */\nfunction removeEventListener( listener: EmitterInternal, emitter: EmitterInternal, event: string, callback: Function ): void {\n\tif ( emitter._removeEventListener ) {\n\t\temitter._removeEventListener( event, callback );\n\t} else {\n\t\t// Allow listening on objects that do not implement Emitter interface.\n\t\t// This is needed in some tests that are using mocks instead of the real objects with EmitterMixin mixed.\n\t\tlistener._removeEventListener!.call( emitter, event, callback );\n\t}\n}\n\n/**\n * The return value of {@link ~Emitter#delegate}.\n */\nexport interface EmitterMixinDelegateChain {\n\n\t/**\n\t * Selects destination for {@link module:utils/emittermixin~Emitter#delegate} events.\n\t *\n\t * @param emitter An `EmitterMixin` instance which is the destination for delegated events.\n\t * @param nameOrFunction A custom event name or function which converts the original name string.\n\t */\n\tto( emitter: Emitter, nameOrFunction?: string | ( ( name: string ) => string ) ): void;\n}\n","/**\n * Checks if `value` is the\n * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types)\n * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`)\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is an object, else `false`.\n * @example\n *\n * _.isObject({});\n * // => true\n *\n * _.isObject([1, 2, 3]);\n * // => true\n *\n * _.isObject(_.noop);\n * // => true\n *\n * _.isObject(null);\n * // => false\n */\nfunction isObject(value) {\n var type = typeof value;\n return value != null && (type == 'object' || type == 'function');\n}\n\nexport default isObject;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* eslint-disable @typescript-eslint/unified-signatures */\n\n/**\n * @module utils/observablemixin\n */\n\nimport EmitterMixin, { type Emitter } from './emittermixin';\nimport CKEditorError from './ckeditorerror';\nimport type { Constructor, Mixed } from './mix';\n\nimport { isObject } from 'lodash-es';\n\nconst observablePropertiesSymbol = Symbol( 'observableProperties' );\nconst boundObservablesSymbol = Symbol( 'boundObservables' );\nconst boundPropertiesSymbol = Symbol( 'boundProperties' );\n\nconst decoratedMethods = Symbol( 'decoratedMethods' );\nconst decoratedOriginal = Symbol( 'decoratedOriginal' );\n\nconst defaultObservableClass = ObservableMixin( EmitterMixin() );\n\n/**\n * A mixin that injects the \"observable properties\" and data binding functionality described in the\n * {@link ~Observable} interface.\n *\n * This function creates a class that inherits from the provided `base` and implements `Observable` interface.\n *\n * ```ts\n * class BaseClass { ... }\n *\n * class MyClass extends ObservableMixin( BaseClass ) {\n * \t// This class derives from `BaseClass` and implements the `Observable` interface.\n * }\n * ```\n *\n * Read more about the concept of observables in the:\n * * {@glink framework/architecture/core-editor-architecture#event-system-and-observables Event system and observables}\n * section of the {@glink framework/architecture/core-editor-architecture Core editor architecture} guide,\n * * {@glink framework/deep-dive/observables Observables deep-dive} guide.\n *\n * @label EXTENDS\n */\nexport default function ObservableMixin<Base extends Constructor<Emitter>>( base: Base ): Mixed<Base, Observable>;\n\n/**\n * A mixin that injects the \"observable properties\" and data binding functionality described in the\n * {@link ~Observable} interface.\n *\n * This function creates a class that implements `Observable` interface.\n *\n * ```ts\n * class MyClass extends ObservableMixin() {\n * \t// This class implements the `Observable` interface.\n * }\n * ```\n *\n * Read more about the concept of observables in the:\n * * {@glink framework/architecture/core-editor-architecture#event-system-and-observables Event system and observables}\n * section of the {@glink framework/architecture/core-editor-architecture Core editor architecture} guide,\n * * {@glink framework/deep-dive/observables Observables deep dive} guide.\n *\n * @label NO_ARGUMENTS\n */\nexport default function ObservableMixin(): {\n\tnew (): Observable;\n\tprototype: Observable;\n};\n\nexport default function ObservableMixin( base?: Constructor<Emitter> ): unknown {\n\tif ( !base ) {\n\t\treturn defaultObservableClass;\n\t}\n\n\tabstract class Mixin extends base implements ObservableInternal {\n\t\tpublic set( name: string | { [ name: string ]: unknown }, value?: unknown ): void {\n\t\t\t// If the first parameter is an Object, iterate over its properties.\n\t\t\tif ( isObject( name ) ) {\n\t\t\t\tObject.keys( name ).forEach( property => {\n\t\t\t\t\tthis.set( property, name[ property ] );\n\t\t\t\t}, this );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tinitObservable( this );\n\n\t\t\tconst properties = this[ observablePropertiesSymbol ];\n\n\t\t\tif ( ( name in this ) && !properties!.has( name ) ) {\n\t\t\t\t/**\n\t\t\t\t * Cannot override an existing property.\n\t\t\t\t *\n\t\t\t\t * This error is thrown when trying to {@link module:utils/observablemixin~Observable#set set} a property with\n\t\t\t\t * a name of an already existing property. For example:\n\t\t\t\t *\n\t\t\t\t * ```ts\n\t\t\t\t * let observable = new Model();\n\t\t\t\t * observable.property = 1;\n\t\t\t\t * observable.set( 'property', 2 );\t\t\t// throws\n\t\t\t\t *\n\t\t\t\t * observable.set( 'property', 1 );\n\t\t\t\t * observable.set( 'property', 2 );\t\t\t// ok, because this is an existing property.\n\t\t\t\t * ```\n\t\t\t\t *\n\t\t\t\t * @error observable-set-cannot-override\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'observable-set-cannot-override', this );\n\t\t\t}\n\n\t\t\tObject.defineProperty( this, name, {\n\t\t\t\tenumerable: true,\n\t\t\t\tconfigurable: true,\n\n\t\t\t\tget() {\n\t\t\t\t\treturn properties!.get( name );\n\t\t\t\t},\n\n\t\t\t\tset( this: Observable, value ) {\n\t\t\t\t\tconst oldValue = properties!.get( name );\n\n\t\t\t\t\t// Fire `set` event before the new value will be set to make it possible\n\t\t\t\t\t// to override observable property without affecting `change` event.\n\t\t\t\t\t// See https://github.com/ckeditor/ckeditor5-utils/issues/171.\n\t\t\t\t\tlet newValue = this.fire<ObservableSetEvent>( `set:${ name }`, name, value, oldValue );\n\n\t\t\t\t\tif ( newValue === undefined ) {\n\t\t\t\t\t\tnewValue = value;\n\t\t\t\t\t}\n\n\t\t\t\t\t// Allow undefined as an initial value like A.define( 'x', undefined ) (#132).\n\t\t\t\t\t// Note: When properties map has no such own property, then its value is undefined.\n\t\t\t\t\tif ( oldValue !== newValue || !properties!.has( name ) ) {\n\t\t\t\t\t\tproperties!.set( name, newValue );\n\t\t\t\t\t\tthis.fire<ObservableChangeEvent>( `change:${ name }`, name, newValue, oldValue );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\t( this as any )[ name ] = value;\n\t\t}\n\n\t\tpublic bind( ...bindProperties: Array<string> ): any {\n\t\t\tif ( !bindProperties.length || !isStringArray( bindProperties ) ) {\n\t\t\t\t/**\n\t\t\t\t * All properties must be strings.\n\t\t\t\t *\n\t\t\t\t * @error observable-bind-wrong-properties\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'observable-bind-wrong-properties', this );\n\t\t\t}\n\n\t\t\tif ( ( new Set( bindProperties ) ).size !== bindProperties.length ) {\n\t\t\t\t/**\n\t\t\t\t * Properties must be unique.\n\t\t\t\t *\n\t\t\t\t * @error observable-bind-duplicate-properties\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'observable-bind-duplicate-properties', this );\n\t\t\t}\n\n\t\t\tinitObservable( this );\n\n\t\t\tconst boundProperties = this[ boundPropertiesSymbol ];\n\n\t\t\tbindProperties.forEach( propertyName => {\n\t\t\t\tif ( boundProperties!.has( propertyName ) ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * Cannot bind the same property more than once.\n\t\t\t\t\t *\n\t\t\t\t\t * @error observable-bind-rebind\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError( 'observable-bind-rebind', this );\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tconst bindings = new Map<string, Binding>();\n\n\t\t\tbindProperties.forEach( a => {\n\t\t\t\tconst binding = { property: a, to: [] };\n\n\t\t\t\tboundProperties!.set( a, binding );\n\t\t\t\tbindings.set( a, binding );\n\t\t\t} );\n\n\t\t\treturn {\n\t\t\t\tto: bindTo,\n\t\t\t\ttoMany: bindToMany,\n\n\t\t\t\t_observable: this,\n\t\t\t\t_bindProperties: bindProperties,\n\t\t\t\t_to: [],\n\t\t\t\t_bindings: bindings\n\t\t\t};\n\t\t}\n\n\t\tpublic unbind( ...unbindProperties: Array<keyof this & string> ): void {\n\t\t\t// Nothing to do here if not inited yet.\n\t\t\tif ( !( this[ observablePropertiesSymbol ] ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst boundProperties = this[ boundPropertiesSymbol ]!;\n\t\t\tconst boundObservables = this[ boundObservablesSymbol ]!;\n\n\t\t\tif ( unbindProperties.length ) {\n\t\t\t\tif ( !isStringArray( unbindProperties ) ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * Properties must be strings.\n\t\t\t\t\t *\n\t\t\t\t\t * @error observable-unbind-wrong-properties\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError( 'observable-unbind-wrong-properties', this );\n\t\t\t\t}\n\n\t\t\t\tunbindProperties.forEach( propertyName => {\n\t\t\t\t\tconst binding = boundProperties.get( propertyName );\n\n\t\t\t\t\t// Nothing to do if the binding is not defined\n\t\t\t\t\tif ( !binding ) {\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\n\t\t\t\t\tbinding.to.forEach( ( [ toObservable, toProperty ] ) => {\n\t\t\t\t\t\tconst toProperties = boundObservables.get( toObservable )!;\n\t\t\t\t\t\tconst toPropertyBindings = toProperties[ toProperty ];\n\n\t\t\t\t\t\ttoPropertyBindings.delete( binding );\n\n\t\t\t\t\t\tif ( !toPropertyBindings.size ) {\n\t\t\t\t\t\t\tdelete toProperties[ toProperty ];\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif ( !Object.keys( toProperties ).length ) {\n\t\t\t\t\t\t\tboundObservables.delete( toObservable );\n\t\t\t\t\t\t\tthis.stopListening( toObservable, 'change' );\n\t\t\t\t\t\t}\n\t\t\t\t\t} );\n\n\t\t\t\t\tboundProperties.delete( propertyName );\n\t\t\t\t} );\n\t\t\t} else {\n\t\t\t\tboundObservables.forEach( ( bindings, boundObservable ) => {\n\t\t\t\t\tthis.stopListening( boundObservable, 'change' );\n\t\t\t\t} );\n\n\t\t\t\tboundObservables.clear();\n\t\t\t\tboundProperties.clear();\n\t\t\t}\n\t\t}\n\n\t\tpublic decorate( this: this & { [ x: string ]: any }, methodName: keyof this & string ): void {\n\t\t\tinitObservable( this );\n\n\t\t\tconst originalMethod = this[ methodName ];\n\n\t\t\tif ( !originalMethod ) {\n\t\t\t\t/**\n\t\t\t\t * Cannot decorate an undefined method.\n\t\t\t\t *\n\t\t\t\t * @error observablemixin-cannot-decorate-undefined\n\t\t\t\t * @param {Object} object The object which method should be decorated.\n\t\t\t\t * @param {String} methodName Name of the method which does not exist.\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'observablemixin-cannot-decorate-undefined',\n\t\t\t\t\tthis,\n\t\t\t\t\t{ object: this, methodName }\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tthis.on( methodName, ( evt, args ) => {\n\t\t\t\tevt.return = originalMethod.apply( this, args );\n\t\t\t} );\n\n\t\t\tthis[ methodName ] = function( ...args: Array<unknown> ) {\n\t\t\t\treturn this.fire( methodName, args );\n\t\t\t};\n\n\t\t\tthis[ methodName ][ decoratedOriginal ] = originalMethod;\n\n\t\t\tif ( !this[ decoratedMethods ] ) {\n\t\t\t\tthis[ decoratedMethods ] = [];\n\t\t\t}\n\n\t\t\tthis[ decoratedMethods ]!.push( methodName );\n\t\t}\n\n\t\t// Override the EmitterMixin stopListening method to be able to clean (and restore) decorated methods.\n\t\t// This is needed in case of:\n\t\t// 1. Have x.foo() decorated.\n\t\t// 2. Call x.stopListening()\n\t\t// 3. Call x.foo(). Problem: nothing happens (the original foo() method is not executed)\n\t\tpublic override stopListening(\n\t\t\tthis: ObservableInternal & { [ x: string ]: any },\n\t\t\temitter?: Emitter,\n\t\t\tevent?: string,\n\t\t\tcallback?: Function\n\t\t): void {\n\t\t\t// Removing all listeners so let's clean the decorated methods to the original state.\n\t\t\tif ( !emitter && this[ decoratedMethods ] ) {\n\t\t\t\tfor ( const methodName of this[ decoratedMethods ]! ) {\n\t\t\t\t\tthis[ methodName ] = this[ methodName ][ decoratedOriginal ];\n\t\t\t\t}\n\n\t\t\t\tdelete this[ decoratedMethods ];\n\t\t\t}\n\n\t\t\tsuper.stopListening( emitter, event, callback );\n\t\t}\n\n\t\tpublic [ observablePropertiesSymbol ]?: Map<string, unknown>;\n\n\t\tpublic [ decoratedMethods ]?: Array<string>;\n\n\t\tpublic [ boundPropertiesSymbol ]?: Map<string, Binding>;\n\n\t\tpublic [ boundObservablesSymbol]?: Map<Observable, Record<string, Set<Binding>>>;\n\t}\n\n\treturn Mixin;\n}\n\n// Backward compatibility with `mix`\n( [\n\t'set', 'bind', 'unbind', 'decorate',\n\t'on', 'once', 'off', 'listenTo',\n\t'stopListening', 'fire', 'delegate', 'stopDelegating',\n\t'_addEventListener', '_removeEventListener'\n] ).forEach( key => {\n\t( ObservableMixin as any )[ key ] = ( defaultObservableClass.prototype as any )[ key ];\n} );\n\ninterface Binding {\n\n\t/**\n\t * Property which is bound.\n\t */\n\tproperty: string;\n\n\t/**\n\t * Array of observableproperty components of the binding (`{ observable: ..., property: .. }`).\n\t */\n\tto: Array<[ Observable, string ]>;\n\n\t/**\n\t * A function which processes `to` components.\n\t */\n\tcallback?: Function;\n}\n\ninterface BindChainInternal {\n\tto: Function;\n\t_observable: Observable;\n\t_bindings: Map<string, Binding>;\n\t_bindProperties: Array<string>;\n\t_to: Array<{\n\t\tobservable: Observable;\n\t\tproperties: Array<string>;\n\t}>;\n}\n\n// Init symbol properties needed for the observable mechanism to work.\nfunction initObservable( observable: ObservableInternal ): void {\n\t// Do nothing if already inited.\n\tif ( observable[ observablePropertiesSymbol ] ) {\n\t\treturn;\n\t}\n\n\t// The internal hash containing the observable's state.\n\tObject.defineProperty( observable, observablePropertiesSymbol, {\n\t\tvalue: new Map()\n\t} );\n\n\t// Map containing bindings to external observables. It shares the binding objects\n\t// (`{ observable: A, property: 'a', to: ... }`) with {@link module:utils/observablemixin~Observable#_boundProperties} and\n\t// it is used to observe external observables to update own properties accordingly.\n\t// See {@link module:utils/observablemixin~Observable#bind}.\n\t//\n\t//\t\tA.bind( 'a', 'b', 'c' ).to( B, 'x', 'y', 'x' );\n\t//\t\tconsole.log( A._boundObservables );\n\t//\n\t//\t\t\tMap( {\n\t//\t\t\t\tB: {\n\t//\t\t\t\t\tx: Set( [\n\t//\t\t\t\t\t\t{ observable: A, property: 'a', to: [ [ B, 'x' ] ] },\n\t//\t\t\t\t\t\t{ observable: A, property: 'c', to: [ [ B, 'x' ] ] }\n\t//\t\t\t\t\t] ),\n\t//\t\t\t\t\ty: Set( [\n\t//\t\t\t\t\t\t{ observable: A, property: 'b', to: [ [ B, 'y' ] ] },\n\t//\t\t\t\t\t] )\n\t//\t\t\t\t}\n\t//\t\t\t} )\n\t//\n\t//\t\tA.bind( 'd' ).to( B, 'z' ).to( C, 'w' ).as( callback );\n\t//\t\tconsole.log( A._boundObservables );\n\t//\n\t//\t\t\tMap( {\n\t//\t\t\t\tB: {\n\t//\t\t\t\t\tx: Set( [\n\t//\t\t\t\t\t\t{ observable: A, property: 'a', to: [ [ B, 'x' ] ] },\n\t//\t\t\t\t\t\t{ observable: A, property: 'c', to: [ [ B, 'x' ] ] }\n\t//\t\t\t\t\t] ),\n\t//\t\t\t\t\ty: Set( [\n\t//\t\t\t\t\t\t{ observable: A, property: 'b', to: [ [ B, 'y' ] ] },\n\t//\t\t\t\t\t] ),\n\t//\t\t\t\t\tz: Set( [\n\t//\t\t\t\t\t\t{ observable: A, property: 'd', to: [ [ B, 'z' ], [ C, 'w' ] ], callback: callback }\n\t//\t\t\t\t\t] )\n\t//\t\t\t\t},\n\t//\t\t\t\tC: {\n\t//\t\t\t\t\tw: Set( [\n\t//\t\t\t\t\t\t{ observable: A, property: 'd', to: [ [ B, 'z' ], [ C, 'w' ] ], callback: callback }\n\t//\t\t\t\t\t] )\n\t//\t\t\t\t}\n\t//\t\t\t} )\n\t//\n\tObject.defineProperty( observable, boundObservablesSymbol, {\n\t\tvalue: new Map()\n\t} );\n\n\t// Object that stores which properties of this observable are bound and how. It shares\n\t// the binding objects (`{ observable: A, property: 'a', to: ... }`) with\n\t// {@link module:utils/observablemixin~Observable#_boundObservables}. This data structure is\n\t// a reverse of {@link module:utils/observablemixin~Observable#_boundObservables} and it is helpful for\n\t// {@link module:utils/observablemixin~Observable#unbind}.\n\t//\n\t// See {@link module:utils/observablemixin~Observable#bind}.\n\t//\n\t//\t\tA.bind( 'a', 'b', 'c' ).to( B, 'x', 'y', 'x' );\n\t//\t\tconsole.log( A._boundProperties );\n\t//\n\t//\t\t\tMap( {\n\t//\t\t\t\ta: { observable: A, property: 'a', to: [ [ B, 'x' ] ] },\n\t//\t\t\t\tb: { observable: A, property: 'b', to: [ [ B, 'y' ] ] },\n\t//\t\t\t\tc: { observable: A, property: 'c', to: [ [ B, 'x' ] ] }\n\t//\t\t\t} )\n\t//\n\t//\t\tA.bind( 'd' ).to( B, 'z' ).to( C, 'w' ).as( callback );\n\t//\t\tconsole.log( A._boundProperties );\n\t//\n\t//\t\t\tMap( {\n\t//\t\t\t\ta: { observable: A, property: 'a', to: [ [ B, 'x' ] ] },\n\t//\t\t\t\tb: { observable: A, property: 'b', to: [ [ B, 'y' ] ] },\n\t//\t\t\t\tc: { observable: A, property: 'c', to: [ [ B, 'x' ] ] },\n\t//\t\t\t\td: { observable: A, property: 'd', to: [ [ B, 'z' ], [ C, 'w' ] ], callback: callback }\n\t//\t\t\t} )\n\tObject.defineProperty( observable, boundPropertiesSymbol, {\n\t\tvalue: new Map()\n\t} );\n}\n\n/**\n * A chaining for {@link module:utils/observablemixin~Observable#bind} providing `.to()` interface.\n *\n * @param args Arguments of the `.to( args )` binding.\n */\nfunction bindTo( this: BindChainInternal, ...args: Array<Observable | string | Function> ): void {\n\tconst parsedArgs = parseBindToArgs( ...args );\n\tconst bindingsKeys = Array.from( this._bindings.keys() );\n\tconst numberOfBindings = bindingsKeys.length;\n\n\t// Eliminate A.bind( 'x' ).to( B, C )\n\tif ( !parsedArgs.callback && parsedArgs.to.length > 1 ) {\n\t\t/**\n\t\t * Binding multiple observables only possible with callback.\n\t\t *\n\t\t * @error observable-bind-to-no-callback\n\t\t */\n\t\tthrow new CKEditorError( 'observable-bind-to-no-callback', this );\n\t}\n\n\t// Eliminate A.bind( 'x', 'y' ).to( B, callback )\n\tif ( numberOfBindings > 1 && parsedArgs.callback ) {\n\t\t/**\n\t\t * Cannot bind multiple properties and use a callback in one binding.\n\t\t *\n\t\t * @error observable-bind-to-extra-callback\n\t\t */\n\t\tthrow new CKEditorError(\n\t\t\t'observable-bind-to-extra-callback',\n\t\t\tthis\n\t\t);\n\t}\n\n\tparsedArgs.to.forEach( to => {\n\t\t// Eliminate A.bind( 'x', 'y' ).to( B, 'a' )\n\t\tif ( to.properties.length && to.properties.length !== numberOfBindings ) {\n\t\t\t/**\n\t\t\t * The number of properties must match.\n\t\t\t *\n\t\t\t * @error observable-bind-to-properties-length\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'observable-bind-to-properties-length', this );\n\t\t}\n\n\t\t// When no to.properties specified, observing source properties instead i.e.\n\t\t// A.bind( 'x', 'y' ).to( B ) -> Observe B.x and B.y\n\t\tif ( !to.properties.length ) {\n\t\t\tto.properties = this._bindProperties;\n\t\t}\n\t} );\n\n\tthis._to = parsedArgs.to;\n\n\t// Fill {@link BindChain#_bindings} with callback. When the callback is set there's only one binding.\n\tif ( parsedArgs.callback ) {\n\t\tthis._bindings.get( bindingsKeys[ 0 ] )!.callback = parsedArgs.callback;\n\t}\n\n\tattachBindToListeners( this._observable, this._to );\n\n\t// Update observable._boundProperties and observable._boundObservables.\n\tupdateBindToBound( this );\n\n\t// Set initial values of bound properties.\n\tthis._bindProperties.forEach( propertyName => {\n\t\tupdateBoundObservableProperty( this._observable, propertyName );\n\t} );\n}\n\n/**\n * Binds to an attribute in a set of iterable observables.\n */\nfunction bindToMany( this: BindChainInternal, observables: Array<Observable>, attribute: string, callback: Function ): void {\n\tif ( this._bindings.size > 1 ) {\n\t\t/**\n\t\t * Binding one attribute to many observables only possible with one attribute.\n\t\t *\n\t\t * @error observable-bind-to-many-not-one-binding\n\t\t */\n\t\tthrow new CKEditorError( 'observable-bind-to-many-not-one-binding', this );\n\t}\n\n\tthis.to(\n\t\t// Bind to #attribute of each observable...\n\t\t...getBindingTargets( observables, attribute ),\n\t\t// ...using given callback to parse attribute values.\n\t\tcallback\n\t);\n}\n\n/**\n * Returns an array of binding components for\n * {@link Observable#bind} from a set of iterable observables.\n */\nfunction getBindingTargets( observables: Array<Observable>, attribute: string ): Array<Observable | string> {\n\tconst observableAndAttributePairs = observables.map( observable => [ observable, attribute ] );\n\n\t// Merge pairs to one-dimension array of observables and attributes.\n\treturn Array.prototype.concat.apply( [], observableAndAttributePairs );\n}\n\n/**\n * Check if all entries of the array are of `String` type.\n */\nfunction isStringArray( arr: Array<unknown> ): arr is Array<string> {\n\treturn arr.every( a => typeof a == 'string' );\n}\n\n/**\n * Parses and validates {@link Observable#bind}`.to( args )` arguments and returns\n * an object with a parsed structure. For example\n *\n * ```ts\n * A.bind( 'x' ).to( B, 'a', C, 'b', call );\n * ```\n *\n * becomes\n *\n * ```ts\n * {\n * \tto: [\n * \t\t{ observable: B, properties: [ 'a' ] },\n * \t\t{ observable: C, properties: [ 'b' ] },\n * \t],\n * \tcallback: call\n * }\n *\n * @param args Arguments of {@link Observable#bind}`.to( args )`.\n */\nfunction parseBindToArgs( ...args: Array<Observable | string | Function> ) {\n\t// Eliminate A.bind( 'x' ).to()\n\tif ( !args.length ) {\n\t\t/**\n\t\t * Invalid argument syntax in `to()`.\n\t\t *\n\t\t * @error observable-bind-to-parse-error\n\t\t */\n\t\tthrow new CKEditorError( 'observable-bind-to-parse-error', null );\n\t}\n\n\tconst parsed: { to: BindChainInternal[ '_to' ]; callback?: Function } = { to: [] };\n\tlet lastObservable: { observable: Observable; properties: Array<string> };\n\n\tif ( typeof args[ args.length - 1 ] == 'function' ) {\n\t\tparsed.callback = args.pop() as Function;\n\t}\n\n\targs.forEach( a => {\n\t\tif ( typeof a == 'string' ) {\n\t\t\tlastObservable.properties.push( a );\n\t\t} else if ( typeof a == 'object' ) {\n\t\t\tlastObservable = { observable: a, properties: [] };\n\t\t\tparsed.to.push( lastObservable );\n\t\t} else {\n\t\t\tthrow new CKEditorError( 'observable-bind-to-parse-error', null );\n\t\t}\n\t} );\n\n\treturn parsed;\n}\n\n/**\n * Synchronizes {@link module:utils/observable#_boundObservables} with {@link Binding}.\n *\n * @param binding A binding to store in {@link Observable#_boundObservables}.\n * @param toObservable A observable, which is a new component of `binding`.\n * @param toPropertyName A name of `toObservable`'s property, a new component of the `binding`.\n */\nfunction updateBoundObservables(\n\tobservable: ObservableInternal,\n\tbinding: Binding,\n\ttoObservable: Observable,\n\ttoPropertyName: string\n): void {\n\tconst boundObservables = observable[ boundObservablesSymbol ]!;\n\tconst bindingsToObservable = boundObservables.get( toObservable );\n\tconst bindings = bindingsToObservable || {};\n\n\tif ( !bindings[ toPropertyName ] ) {\n\t\tbindings[ toPropertyName ] = new Set();\n\t}\n\n\t// Pass the binding to a corresponding Set in `observable._boundObservables`.\n\tbindings[ toPropertyName ].add( binding );\n\n\tif ( !bindingsToObservable ) {\n\t\tboundObservables.set( toObservable, bindings );\n\t}\n}\n\n/**\n * Synchronizes {@link Observable#_boundProperties} and {@link Observable#_boundObservables}\n * with {@link BindChain}.\n *\n * Assuming the following binding being created\n *\n * ```ts\n * A.bind( 'a', 'b' ).to( B, 'x', 'y' );\n * ```\n *\n * the following bindings were initialized by {@link Observable#bind} in {@link BindChain#_bindings}:\n *\n * ```ts\n * {\n * \ta: { observable: A, property: 'a', to: [] },\n * \tb: { observable: A, property: 'b', to: [] },\n * }\n * ```\n *\n * Iterate over all bindings in this chain and fill their `to` properties with\n * corresponding to( ... ) arguments (components of the binding), so\n *\n * ```ts\n * {\n * \ta: { observable: A, property: 'a', to: [ B, 'x' ] },\n * \tb: { observable: A, property: 'b', to: [ B, 'y' ] },\n * }\n * ```\n *\n * Then update the structure of {@link Observable#_boundObservables} with updated\n * binding, so it becomes:\n *\n * ```ts\n * Map( {\n * \tB: {\n * \t\tx: Set( [\n * \t\t\t{ observable: A, property: 'a', to: [ [ B, 'x' ] ] }\n * \t\t] ),\n * \t\ty: Set( [\n * \t\t\t{ observable: A, property: 'b', to: [ [ B, 'y' ] ] },\n * \t\t] )\n * \t}\n * } )\n * ```\n *\n * @param chain The binding initialized by {@link Observable#bind}.\n */\nfunction updateBindToBound( chain: BindChainInternal ): void {\n\tlet toProperty;\n\n\tchain._bindings.forEach( ( binding, propertyName ) => {\n\t\t// Note: For a binding without a callback, this will run only once\n\t\t// like in A.bind( 'x', 'y' ).to( B, 'a', 'b' )\n\t\t// TODO: ES6 destructuring.\n\t\tchain._to.forEach( to => {\n\t\t\ttoProperty = to.properties[ binding.callback ? 0 : chain._bindProperties.indexOf( propertyName ) ];\n\n\t\t\tbinding.to.push( [ to.observable, toProperty ] );\n\t\t\tupdateBoundObservables( chain._observable, binding, to.observable, toProperty );\n\t\t} );\n\t} );\n}\n\n/**\n * Updates an property of a {@link Observable} with a value\n * determined by an entry in {@link Observable#_boundProperties}.\n *\n * @param observable A observable which property is to be updated.\n * @param propertyName An property to be updated.\n */\nfunction updateBoundObservableProperty( observable: ObservableInternal, propertyName: string ): void {\n\tconst boundProperties = observable[ boundPropertiesSymbol ]!;\n\tconst binding = boundProperties.get( propertyName )!;\n\tlet propertyValue;\n\n\t// When a binding with callback is created like\n\t//\n\t// \t\tA.bind( 'a' ).to( B, 'b', C, 'c', callback );\n\t//\n\t// collect B.b and C.c, then pass them to callback to set A.a.\n\tif ( binding.callback ) {\n\t\tpropertyValue = binding.callback.apply( observable, binding.to.map( to => ( to[ 0 ] as any )[ to[ 1 ] ] ) );\n\t} else {\n\t\tpropertyValue = binding.to[ 0 ];\n\t\tpropertyValue = ( propertyValue[ 0 ] as any )[ propertyValue[ 1 ] ];\n\t}\n\n\tif ( Object.prototype.hasOwnProperty.call( observable, propertyName ) ) {\n\t\t( observable as any )[ propertyName ] = propertyValue;\n\t} else {\n\t\tobservable.set( propertyName as any, propertyValue );\n\t}\n}\n\n/**\n * Starts listening to changes in {@link BindChain._to} observables to update\n * {@link BindChain._observable} {@link BindChain._bindProperties}. Also sets the\n * initial state of {@link BindChain._observable}.\n *\n * @param chain The chain initialized by {@link Observable#bind}.\n */\nfunction attachBindToListeners( observable: ObservableInternal, toBindings: BindChainInternal[ '_to' ] ): void {\n\ttoBindings.forEach( to => {\n\t\tconst boundObservables = observable[ boundObservablesSymbol ]!;\n\t\tlet bindings;\n\n\t\t// If there's already a chain between the observables (`observable` listens to\n\t\t// `to.observable`), there's no need to create another `change` event listener.\n\t\tif ( !boundObservables.get( to.observable ) ) {\n\t\t\tobservable.listenTo<ObservableChangeEvent>( to.observable, 'change', ( evt, propertyName ) => {\n\t\t\t\tbindings = boundObservables.get( to.observable )![ propertyName ];\n\n\t\t\t\t// Note: to.observable will fire for any property change, react\n\t\t\t\t// to changes of properties which are bound only.\n\t\t\t\tif ( bindings ) {\n\t\t\t\t\tbindings.forEach( binding => {\n\t\t\t\t\t\tupdateBoundObservableProperty( observable, binding.property );\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t} );\n}\n\n/**\n * An interface which adds \"observable properties\" and data binding functionality.\n *\n * Can be easily implemented by a class by mixing the {@link module:utils/observablemixin~Observable} mixin.\n *\n * ```ts\n * class MyClass extends ObservableMixin( OtherBaseClass ) {\n * \t// This class now implements the `Observable` interface.\n * }\n * ```\n *\n * Read more about the usage of this interface in the:\n * * {@glink framework/architecture/core-editor-architecture#event-system-and-observables Event system and observables}\n * section of the {@glink framework/architecture/core-editor-architecture Core editor architecture} guide,\n * * {@glink framework/deep-dive/observables Observables deep-dive} guide.\n */\nexport interface Observable extends Emitter {\n\n\t/**\n\t * Creates and sets the value of an observable property of this object. Such a property becomes a part\n\t * of the state and is observable.\n\t *\n\t * This method throws the `observable-set-cannot-override` error if the observable instance already\n\t * has a property with the given property name. This prevents from mistakenly overriding existing\n\t * properties and methods, but means that `foo.set( 'bar', 1 )` may be slightly slower than `foo.bar = 1`.\n\t *\n\t * In TypeScript, those properties should be declared in class using `declare` keyword. In example:\n\t *\n\t * ```ts\n\t * public declare myProp: number;\n\t *\n\t * constructor() {\n\t * \tthis.set( 'myProp', 2 );\n\t * }\n\t * ```\n\t *\n\t * @label KEY_VALUE\n\t * @param name The property's name.\n\t * @param value The property's value.\n\t */\n\tset<K extends keyof this & string>( name: K, value: this[ K ] ): void;\n\n\t/**\n\t * Creates and sets the value of an observable properties of this object. Such a property becomes a part\n\t * of the state and is observable.\n\t *\n\t * It accepts a single object literal containing key/value pairs with properties to be set.\n\t *\n\t * This method throws the `observable-set-cannot-override` error if the observable instance already\n\t * has a property with the given property name. This prevents from mistakenly overriding existing\n\t * properties and methods, but means that `foo.set( 'bar', 1 )` may be slightly slower than `foo.bar = 1`.\n\t *\n\t * In TypeScript, those properties should be declared in class using `declare` keyword. In example:\n\t *\n\t * ```ts\n\t * public declare myProp1: number;\n\t * public declare myProp2: string;\n\t *\n\t * constructor() {\n\t * \tthis.set( {\n\t * \t\t'myProp1: 2,\n\t * \t\t'myProp2: 'foo'\n\t * \t} );\n\t * }\n\t * ```\n\t * @label OBJECT\n\t * @param values An object with `name=>value` pairs.\n\t */\n\tset( values: object & { readonly [ K in keyof this ]?: unknown } ): void;\n\n\t/**\n\t * Binds {@link #set observable properties} to other objects implementing the\n\t * {@link module:utils/observablemixin~Observable} interface.\n\t *\n\t * Read more in the {@glink framework/deep-dive/observables#property-bindings dedicated} guide\n\t * covering the topic of property bindings with some additional examples.\n\t *\n\t * Consider two objects: a `button` and an associated `command` (both `Observable`).\n\t *\n\t * A simple property binding could be as follows:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\t * ```\n\t *\n\t * or even shorter:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command );\n\t * ```\n\t *\n\t * which works in the following way:\n\t *\n\t * * `button.isEnabled` **instantly equals** `command.isEnabled`,\n\t * * whenever `command.isEnabled` changes, `button.isEnabled` will immediately reflect its value.\n\t *\n\t * **Note**: To release the binding, use {@link module:utils/observablemixin~Observable#unbind}.\n\t *\n\t * You can also \"rename\" the property in the binding by specifying the new name in the `to()` chain:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isWorking' );\n\t * ```\n\t *\n\t * It is possible to bind more than one property at a time to shorten the code:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled', 'value' ).to( command );\n\t * ```\n\t *\n\t * which corresponds to:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command );\n\t * button.bind( 'value' ).to( command );\n\t * ```\n\t *\n\t * The binding can include more than one observable, combining multiple data sources in a custom callback:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isEnabled', ui, 'isVisible',\n\t * \t( isCommandEnabled, isUIVisible ) => isCommandEnabled && isUIVisible );\n\t * ```\n\t *\n\t * Using a custom callback allows processing the value before passing it to the target property:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'value', value => value === 'heading1' );\n\t * ```\n\t *\n\t * It is also possible to bind to the same property in an array of observables.\n\t * To bind a `button` to multiple commands (also `Observables`) so that each and every one of them\n\t * must be enabled for the button to become enabled, use the following code:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).toMany( [ commandA, commandB, commandC ], 'isEnabled',\n\t * \t( isAEnabled, isBEnabled, isCEnabled ) => isAEnabled && isBEnabled && isCEnabled );\n\t * ```\n\t *\n\t * @label SINGLE_BIND\n\t * @param bindProperty Observable property that will be bound to other observable(s).\n\t * @returns The bind chain with the `to()` and `toMany()` methods.\n\t */\n\tbind<K extends keyof this & string>(\n\t\tbindProperty: K\n\t): SingleBindChain<K, this[ K ]>;\n\n\t/**\n\t * Binds {@link #set observable properties} to other objects implementing the\n\t * {@link module:utils/observablemixin~Observable} interface.\n\t *\n\t * Read more in the {@glink framework/deep-dive/observables#property-bindings dedicated} guide\n\t * covering the topic of property bindings with some additional examples.\n\t *\n\t * Consider two objects: a `button` and an associated `command` (both `Observable`).\n\t *\n\t * A simple property binding could be as follows:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\t * ```\n\t *\n\t * or even shorter:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command );\n\t * ```\n\t *\n\t * which works in the following way:\n\t *\n\t * * `button.isEnabled` **instantly equals** `command.isEnabled`,\n\t * * whenever `command.isEnabled` changes, `button.isEnabled` will immediately reflect its value.\n\t *\n\t * **Note**: To release the binding, use {@link module:utils/observablemixin~Observable#unbind}.\n\t *\n\t * You can also \"rename\" the property in the binding by specifying the new name in the `to()` chain:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isWorking' );\n\t * ```\n\t *\n\t * It is possible to bind more than one property at a time to shorten the code:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled', 'value' ).to( command );\n\t * ```\n\t *\n\t * which corresponds to:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command );\n\t * button.bind( 'value' ).to( command );\n\t * ```\n\t *\n\t * The binding can include more than one observable, combining multiple data sources in a custom callback:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isEnabled', ui, 'isVisible',\n\t * \t( isCommandEnabled, isUIVisible ) => isCommandEnabled && isUIVisible );\n\t * ```\n\t *\n\t * Using a custom callback allows processing the value before passing it to the target property:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'value', value => value === 'heading1' );\n\t * ```\n\t *\n\t * It is also possible to bind to the same property in an array of observables.\n\t * To bind a `button` to multiple commands (also `Observables`) so that each and every one of them\n\t * must be enabled for the button to become enabled, use the following code:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).toMany( [ commandA, commandB, commandC ], 'isEnabled',\n\t * \t( isAEnabled, isBEnabled, isCEnabled ) => isAEnabled && isBEnabled && isCEnabled );\n\t * ```\n\t *\n\t * @label DUAL_BIND\n\t * @param bindProperty1 Observable property that will be bound to other observable(s).\n\t * @param bindProperty2 Observable property that will be bound to other observable(s).\n\t * @returns The bind chain with the `to()` and `toMany()` methods.\n\t */\n\tbind<K1 extends keyof this & string, K2 extends keyof this & string>(\n\t\tbindProperty1: K1,\n\t\tbindProperty2: K2\n\t): DualBindChain<K1, this[ K1 ], K2, this[ K2 ]>;\n\n\t/**\n\t * Binds {@link #set observable properties} to other objects implementing the\n\t * {@link module:utils/observablemixin~Observable} interface.\n\t *\n\t * Read more in the {@glink framework/deep-dive/observables#property-bindings dedicated} guide\n\t * covering the topic of property bindings with some additional examples.\n\t *\n\t * Consider two objects: a `button` and an associated `command` (both `Observable`).\n\t *\n\t * A simple property binding could be as follows:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\t * ```\n\t *\n\t * or even shorter:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command );\n\t * ```\n\t *\n\t * which works in the following way:\n\t *\n\t * * `button.isEnabled` **instantly equals** `command.isEnabled`,\n\t * * whenever `command.isEnabled` changes, `button.isEnabled` will immediately reflect its value.\n\t *\n\t * **Note**: To release the binding, use {@link module:utils/observablemixin~Observable#unbind}.\n\t *\n\t * You can also \"rename\" the property in the binding by specifying the new name in the `to()` chain:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isWorking' );\n\t * ```\n\t *\n\t * It is possible to bind more than one property at a time to shorten the code:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled', 'value' ).to( command );\n\t * ```\n\t *\n\t * which corresponds to:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command );\n\t * button.bind( 'value' ).to( command );\n\t * ```\n\t *\n\t * The binding can include more than one observable, combining multiple data sources in a custom callback:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'isEnabled', ui, 'isVisible',\n\t * \t( isCommandEnabled, isUIVisible ) => isCommandEnabled && isUIVisible );\n\t * ```\n\t *\n\t * Using a custom callback allows processing the value before passing it to the target property:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).to( command, 'value', value => value === 'heading1' );\n\t * ```\n\t *\n\t * It is also possible to bind to the same property in an array of observables.\n\t * To bind a `button` to multiple commands (also `Observables`) so that each and every one of them\n\t * must be enabled for the button to become enabled, use the following code:\n\t *\n\t * ```ts\n\t * button.bind( 'isEnabled' ).toMany( [ commandA, commandB, commandC ], 'isEnabled',\n\t * \t( isAEnabled, isBEnabled, isCEnabled ) => isAEnabled && isBEnabled && isCEnabled );\n\t * ```\n\t *\n\t * @label MANY_BIND\n\t * @param bindProperties Observable properties that will be bound to other observable(s).\n\t * @returns The bind chain with the `to()` and `toMany()` methods.\n\t */\n\tbind( ...bindProperties: Array<keyof this & string> ): MultiBindChain;\n\n\t/**\n\t * Removes the binding created with {@link #bind}.\n\t *\n\t * ```ts\n\t * // Removes the binding for the 'a' property.\n\t * A.unbind( 'a' );\n\t *\n\t * // Removes bindings for all properties.\n\t * A.unbind();\n\t * ```\n\t *\n\t * @param unbindProperties Observable properties to be unbound. All the bindings will\n\t * be released if no properties are provided.\n\t */\n\tunbind( ...unbindProperties: Array<keyof this & string> ): void;\n\n\t/**\n\t * Turns the given methods of this object into event-based ones. This means that the new method will fire an event\n\t * (named after the method) and the original action will be plugged as a listener to that event.\n\t *\n\t * Read more in the {@glink framework/deep-dive/observables#decorating-object-methods dedicated} guide\n\t * covering the topic of decorating methods with some additional examples.\n\t *\n\t * Decorating the method does not change its behavior (it only adds an event),\n\t * but it allows to modify it later on by listening to the method's event.\n\t *\n\t * For example, to cancel the method execution the event can be {@link module:utils/eventinfo~EventInfo#stop stopped}:\n\t *\n\t * ```ts\n\t * class Foo extends ObservableMixin() {\n\t * \tconstructor() {\n\t * \t\tsuper();\n\t * \t\tthis.decorate( 'method' );\n\t * \t}\n\t *\n\t * \tmethod() {\n\t * \t\tconsole.log( 'called!' );\n\t * \t}\n\t * }\n\t *\n\t * const foo = new Foo();\n\t * foo.on( 'method', ( evt ) => {\n\t * \tevt.stop();\n\t * }, { priority: 'high' } );\n\t *\n\t * foo.method(); // Nothing is logged.\n\t * ```\n\t *\n\t *\n\t * **Note**: The high {@link module:utils/priorities~PriorityString priority} listener\n\t * has been used to execute this particular callback before the one which calls the original method\n\t * (which uses the \"normal\" priority).\n\t *\n\t * It is also possible to change the returned value:\n\t *\n\t * ```ts\n\t * foo.on( 'method', ( evt ) => {\n\t * \tevt.return = 'Foo!';\n\t * } );\n\t *\n\t * foo.method(); // -> 'Foo'\n\t * ```\n\t *\n\t * Finally, it is possible to access and modify the arguments the method is called with:\n\t *\n\t * ```ts\n\t * method( a, b ) {\n\t * \tconsole.log( `${ a }, ${ b }` );\n\t * }\n\t *\n\t * // ...\n\t *\n\t * foo.on( 'method', ( evt, args ) => {\n\t * \targs[ 0 ] = 3;\n\t *\n\t * \tconsole.log( args[ 1 ] ); // -> 2\n\t * }, { priority: 'high' } );\n\t *\n\t * foo.method( 1, 2 ); // -> '3, 2'\n\t * ```\n\t *\n\t * @param methodName Name of the method to decorate.\n\t */\n\tdecorate( methodName: keyof this & string ): void;\n}\n\ninterface ObservableInternal extends Observable {\n\t[ observablePropertiesSymbol ]?: Map<string, unknown>;\n\n\t[ decoratedMethods ]?: Array<string>;\n\n\t[ boundPropertiesSymbol ]?: Map<string, Binding>;\n\n\t[ boundObservablesSymbol]?: Map<Observable, Record<string, Set<Binding>>>;\n}\n\n/**\n * Fired when a property changed value.\n *\n * ```ts\n * observable.set( 'prop', 1 );\n *\n * observable.on<ObservableChangeEvent<number>>( 'change:prop', ( evt, propertyName, newValue, oldValue ) => {\n * \tconsole.log( `${ propertyName } has changed from ${ oldValue } to ${ newValue }` );\n * } );\n *\n * observable.prop = 2; // -> 'prop has changed from 1 to 2'\n * ```\n *\n * @eventName ~Observable#change:\\{property\\}\n * @param {String} name The property name.\n * @param {*} value The new property value.\n * @param {*} oldValue The previous property value.\n */\nexport type ObservableChangeEvent<TValue = any> = {\n\tname: 'change' | `change:${ string }`;\n\targs: [ name: string, value: TValue, oldValue: TValue ];\n};\n\n/**\n * Fired when a property value is going to be set but is not set yet (before the `change` event is fired).\n *\n * You can control the final value of the property by using\n * the {@link module:utils/eventinfo~EventInfo#return event's `return` property}.\n *\n * ```ts\n * observable.set( 'prop', 1 );\n *\n * observable.on<ObservableSetEvent<number>>( 'set:prop', ( evt, propertyName, newValue, oldValue ) => {\n * \tconsole.log( `Value is going to be changed from ${ oldValue } to ${ newValue }` );\n * \tconsole.log( `Current property value is ${ observable[ propertyName ] }` );\n *\n * \t// Let's override the value.\n * \tevt.return = 3;\n * } );\n *\n * observable.on<ObservableChangeEvent<number>>( 'change:prop', ( evt, propertyName, newValue, oldValue ) => {\n * \tconsole.log( `Value has changed from ${ oldValue } to ${ newValue }` );\n * } );\n *\n * observable.prop = 2; // -> 'Value is going to be changed from 1 to 2'\n * // -> 'Current property value is 1'\n * // -> 'Value has changed from 1 to 3'\n * ```\n *\n * **Note:** The event is fired even when the new value is the same as the old value.\n *\n * @eventName ~Observable#set:\\{property\\}\n * @param {String} name The property name.\n * @param {*} value The new property value.\n * @param {*} oldValue The previous property value.\n */\nexport type ObservableSetEvent<TValue = any> = {\n\tname: 'set' | `set:${ string }`;\n\targs: [ name: string, value: TValue, oldValue: TValue ];\n\treturn: TValue;\n};\n\n/**\n * Utility type that creates an event describing type from decorated method.\n *\n * ```ts\n * class Foo extends ObservableMixin() {\n * \tconstructor() {\n * \t\tsuper();\n * \t\tthis.decorate( 'method' );\n * \t}\n *\n * \tmethod( a: number, b: number ): number {\n * \t\treturn a + b;\n * \t}\n * }\n *\n * type FooMethodEvent = DecoratedMethodEvent<Foo, 'method'>;\n *\n * const foo = new Foo();\n *\n * foo.on<FooMethodEvent>( 'method', ( evt, [ a, b ] ) => {\n * \t// `a` and `b` are inferred as numbers.\n * } )\n * ```\n */\nexport type DecoratedMethodEvent<\n\tTObservable extends Observable & { [ N in TName ]: ( ...args: Array<any> ) => any },\n\tTName extends keyof TObservable & string\n> = {\n\tname: TName;\n\targs: [ Parameters<TObservable[ TName ]> ];\n\treturn: ReturnType<TObservable[ TName ]>;\n};\n\ninterface SingleBindChain<TKey extends string, TVal> {\n\ttoMany<O extends Observable, K extends keyof O>(\n\t\tobservables: ReadonlyArray<O>,\n\t\tkey: K,\n\t\tcallback: ( ...values: Array<O[ K ]> ) => TVal\n\t): void;\n\n\tto<O extends ObservableWithProperty<TKey, TVal>>(\n\t\tobservable: O\n\t): void;\n\tto<O extends ObservableWithProperty<TKey>>(\n\t\tobservable: O,\n\t\tcallback: ( value: O[ TKey ] ) => TVal\n\t): void;\n\tto<O extends ObservableWithProperty<K, TVal>, K extends keyof O>(\n\t\tobservable: O,\n\t\tkey: K\n\t): void;\n\tto<O extends Observable, K extends keyof O>(\n\t\tobservable: O,\n\t\tkey: K,\n\t\tcallback: ( value: O[ K ] ) => TVal,\n\t): void;\n\tto<\n\t\tO1 extends ObservableWithProperty<TKey>,\n\t\tO2 extends ObservableWithProperty<TKey>\n\t>(\n\t\tobservable1: O1,\n\t\tobservable2: O2,\n\t\tcallback: ( value1: O1[ TKey ], value2: O2[ TKey ] ) => TVal\n\t): void;\n\tto<\n\t\tO1 extends Observable,\n\t\tK1 extends keyof O1,\n\t\tO2 extends Observable,\n\t\tK2 extends keyof O2\n\t>(\n\t\tobservable1: O1,\n\t\tkey1: K1,\n\t\tobservable2: O2,\n\t\tkey2: K2,\n\t\tcallback: ( value1: O1[ K1 ], value2: O2[ K2 ] ) => TVal\n\t): void;\n\tto<\n\t\tO1 extends ObservableWithProperty<TKey>,\n\t\tO2 extends ObservableWithProperty<TKey>,\n\t\tO3 extends ObservableWithProperty<TKey>\n\t>(\n\t\tobservable1: O1,\n\t\tobservable2: O2,\n\t\tobservable3: O3,\n\t\tcallback: ( value1: O1[ TKey ], value2: O2[ TKey ], value3: O3[ TKey ] ) => TVal\n\t): void;\n\tto<\n\t\tO1 extends Observable,\n\t\tK1 extends keyof O1,\n\t\tO2 extends Observable,\n\t\tK2 extends keyof O2,\n\t\tO3 extends Observable,\n\t\tK3 extends keyof O3\n\t>(\n\t\tobservable1: O1,\n\t\tkey1: K1,\n\t\tobservable2: O2,\n\t\tkey2: K2,\n\t\tobservable3: O3,\n\t\tkey3: K3,\n\t\tcallback: ( value1: O1[ K1 ], value2: O2[ K2 ], value3: O3[ K3 ] ) => TVal\n\t): void;\n\tto<\n\t\tO1 extends ObservableWithProperty<TKey>,\n\t\tO2 extends ObservableWithProperty<TKey>,\n\t\tO3 extends ObservableWithProperty<TKey>,\n\t\tO4 extends ObservableWithProperty<TKey>\n\t>(\n\t\tobservable1: O1,\n\t\tobservable2: O2,\n\t\tobservable3: O3,\n\t\tobservable4: O4,\n\t\tcallback: ( value1: O1[ TKey ], value2: O2[ TKey ], value3: O3[ TKey ], value4: O4[ TKey ] ) => TVal\n\t): void;\n\tto<\n\t\tO1 extends Observable,\n\t\tK1 extends keyof O1,\n\t\tO2 extends Observable,\n\t\tK2 extends keyof O2,\n\t\tO3 extends Observable,\n\t\tK3 extends keyof O3,\n\t\tO4 extends Observable,\n\t\tK4 extends keyof O4\n\t>(\n\t\tobservable1: O1,\n\t\tkey1: K1,\n\t\tobservable2: O2,\n\t\tkey2: K2,\n\t\tobservable3: O3,\n\t\tkey3: K3,\n\t\tobservable4: O4,\n\t\tkey4: K4,\n\t\tcallback: ( value1: O1[ K1 ], value2: O2[ K2 ], value3: O3[ K3 ], value4: O4[ K4 ] ) => TVal\n\t): void;\n\tto<\n\t\tO1 extends ObservableWithProperty<TKey>,\n\t\tO2 extends ObservableWithProperty<TKey>,\n\t\tO3 extends ObservableWithProperty<TKey>,\n\t\tO4 extends ObservableWithProperty<TKey>,\n\t\tO5 extends ObservableWithProperty<TKey>\n\t>(\n\t\tobservable1: O1,\n\t\tobservable2: O2,\n\t\tobservable3: O3,\n\t\tobservable4: O4,\n\t\tobservable5: O5,\n\t\tcallback: ( value1: O1[ TKey ], value2: O2[ TKey ], value3: O3[ TKey ], value4: O4[ TKey ], value5: O5[ TKey ] ) => TVal\n\t): void;\n\tto<\n\t\tO1 extends Observable,\n\t\tK1 extends keyof O1,\n\t\tO2 extends Observable,\n\t\tK2 extends keyof O2,\n\t\tO3 extends Observable,\n\t\tK3 extends keyof O3,\n\t\tO4 extends Observable,\n\t\tK4 extends keyof O4,\n\t\tO5 extends Observable,\n\t\tK5 extends keyof O5\n\t>(\n\t\tobservable1: O1,\n\t\tkey1: K1,\n\t\tobservable2: O2,\n\t\tkey2: K2,\n\t\tobservable3: O3,\n\t\tkey3: K3,\n\t\tobservable4: O4,\n\t\tkey4: K4,\n\t\tobservable5: O5,\n\t\tkey5: K5,\n\t\tcallback: ( value1: O1[ K1 ], value2: O2[ K2 ], value3: O3[ K3 ], value4: O4[ K4 ], value5: O5[ K5 ] ) => TVal\n\t): void;\n}\n\n/**\n * A helper type that can be used as a constraint, ensuring the type is both observable and have the given property.\n *\n * ```ts\n * // Ensures that `obj` is `Observable` and have property named 'abc'.\n * function f<O extends ObservableWithProperty<'abc'>>( obj: O ) {}\n *\n * // Ensures that `obj` is `Observable` and have property named 'abc' with value `number`.\n * function f<O extends ObservableWithProperty<'abc', number>>( obj: O ) {}\n * ```\n */\nexport type ObservableWithProperty<TKey extends PropertyKey, TVal = any> = undefined extends TVal ?\n\tObservable & { [ P in TKey ]?: TVal } :\n\tObservable & { [ P in TKey ]: TVal };\n\ninterface DualBindChain<TKey1 extends string, TVal1, TKey2 extends string, TVal2> {\n\tto<\n\t\tO extends ObservableWithProperty<K1, TVal1> & ObservableWithProperty<K2, TVal2>,\n\t\tK1 extends keyof O,\n\t\tK2 extends keyof O\n\t>(\n\t\tobservable: O,\n\t\tkey1: K1,\n\t\tkey2: K2\n\t): void;\n\n\tto<\n\t\tO extends ObservableWithProperty<TKey1, TVal1> & ObservableWithProperty<TKey2, TVal2>\n\t>(\n\t\tobservable: O\n\t): void;\n}\n\ninterface MultiBindChain {\n\tto<O extends Observable>( observable: O, ...properties: Array<keyof O> ): void;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/count\n */\n\n/**\n * Returns the number of items return by the iterator.\n *\n * ```ts\n * count( [ 1, 2, 3, 4, 5 ] ); // 5;\n * ```\n *\n * @param iterable Any iterable.\n * @returns Number of items returned by that iterable.\n */\nexport default function count( iterable: Iterable<unknown> ): number {\n\tlet count = 0;\n\n\tfor ( const _ of iterable ) { // eslint-disable-line no-unused-vars\n\t\tcount++;\n\t}\n\n\treturn count;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/comparearrays\n */\n\n/**\n * Compares how given arrays relate to each other. One array can be: same as another array, prefix of another array\n * or completely different. If arrays are different, first index at which they differ is returned. Otherwise,\n * a flag specifying the relation is returned. Flags are negative numbers, so whenever a number >= 0 is returned\n * it means that arrays differ.\n *\n * ```ts\n * compareArrays( [ 0, 2 ], [ 0, 2 ] );\t\t// 'same'\n * compareArrays( [ 0, 2 ], [ 0, 2, 1 ] );\t\t// 'prefix'\n * compareArrays( [ 0, 2 ], [ 0 ] );\t\t\t// 'extension'\n * compareArrays( [ 0, 2 ], [ 1, 2 ] );\t\t// 0\n * compareArrays( [ 0, 2 ], [ 0, 1 ] );\t\t// 1\n * ```\n *\n * @param a Array that is compared.\n * @param b Array to compare with.\n * @returns How array `a` is related to `b`.\n */\nexport default function compareArrays( a: ReadonlyArray<unknown>, b: ReadonlyArray<unknown> ): ArrayRelation | number {\n\tconst minLen = Math.min( a.length, b.length );\n\n\tfor ( let i = 0; i < minLen; i++ ) {\n\t\tif ( a[ i ] != b[ i ] ) {\n\t\t\t// The arrays are different.\n\t\t\treturn i;\n\t\t}\n\t}\n\n\t// Both arrays were same at all points.\n\tif ( a.length == b.length ) {\n\t\t// If their length is also same, they are the same.\n\t\treturn 'same';\n\t} else if ( a.length < b.length ) {\n\t\t// Compared array is shorter so it is a prefix of the other array.\n\t\treturn 'prefix';\n\t} else {\n\t\t// Compared array is longer so it is an extension of the other array.\n\t\treturn 'extension';\n\t}\n}\n\n/**\n * Array relation.\n */\nexport type ArrayRelation = 'extension' | 'same' | 'prefix';\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/isiterable\n */\n\n/**\n * Checks if value implements iterator interface.\n *\n * @param value The value to check.\n * @returns True if value implements iterator interface.\n */\nexport default function isIterable( value: any ): value is Iterable<any> {\n\treturn !!( value && value[ Symbol.iterator ] );\n}\n","/** Detect free variable `global` from Node.js. */\nvar freeGlobal = typeof global == 'object' && global && global.Object === Object && global;\n\nexport default freeGlobal;\n","import freeGlobal from './_freeGlobal.js';\n\n/** Detect free variable `self`. */\nvar freeSelf = typeof self == 'object' && self && self.Object === Object && self;\n\n/** Used as a reference to the global object. */\nvar root = freeGlobal || freeSelf || Function('return this')();\n\nexport default root;\n","import root from './_root.js';\n\n/** Built-in value references. */\nvar Symbol = root.Symbol;\n\nexport default Symbol;\n","import Symbol from './_Symbol.js';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * Used to resolve the\n * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)\n * of values.\n */\nvar nativeObjectToString = objectProto.toString;\n\n/** Built-in value references. */\nvar symToStringTag = Symbol ? Symbol.toStringTag : undefined;\n\n/**\n * A specialized version of `baseGetTag` which ignores `Symbol.toStringTag` values.\n *\n * @private\n * @param {*} value The value to query.\n * @returns {string} Returns the raw `toStringTag`.\n */\nfunction getRawTag(value) {\n var isOwn = hasOwnProperty.call(value, symToStringTag),\n tag = value[symToStringTag];\n\n try {\n value[symToStringTag] = undefined;\n var unmasked = true;\n } catch (e) {}\n\n var result = nativeObjectToString.call(value);\n if (unmasked) {\n if (isOwn) {\n value[symToStringTag] = tag;\n } else {\n delete value[symToStringTag];\n }\n }\n return result;\n}\n\nexport default getRawTag;\n","/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/**\n * Used to resolve the\n * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring)\n * of values.\n */\nvar nativeObjectToString = objectProto.toString;\n\n/**\n * Converts `value` to a string using `Object.prototype.toString`.\n *\n * @private\n * @param {*} value The value to convert.\n * @returns {string} Returns the converted string.\n */\nfunction objectToString(value) {\n return nativeObjectToString.call(value);\n}\n\nexport default objectToString;\n","import Symbol from './_Symbol.js';\nimport getRawTag from './_getRawTag.js';\nimport objectToString from './_objectToString.js';\n\n/** `Object#toString` result references. */\nvar nullTag = '[object Null]',\n undefinedTag = '[object Undefined]';\n\n/** Built-in value references. */\nvar symToStringTag = Symbol ? Symbol.toStringTag : undefined;\n\n/**\n * The base implementation of `getTag` without fallbacks for buggy environments.\n *\n * @private\n * @param {*} value The value to query.\n * @returns {string} Returns the `toStringTag`.\n */\nfunction baseGetTag(value) {\n if (value == null) {\n return value === undefined ? undefinedTag : nullTag;\n }\n return (symToStringTag && symToStringTag in Object(value))\n ? getRawTag(value)\n : objectToString(value);\n}\n\nexport default baseGetTag;\n","/**\n * Checks if `value` is classified as an `Array` object.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is an array, else `false`.\n * @example\n *\n * _.isArray([1, 2, 3]);\n * // => true\n *\n * _.isArray(document.body.children);\n * // => false\n *\n * _.isArray('abc');\n * // => false\n *\n * _.isArray(_.noop);\n * // => false\n */\nvar isArray = Array.isArray;\n\nexport default isArray;\n","/**\n * Checks if `value` is object-like. A value is object-like if it's not `null`\n * and has a `typeof` result of \"object\".\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is object-like, else `false`.\n * @example\n *\n * _.isObjectLike({});\n * // => true\n *\n * _.isObjectLike([1, 2, 3]);\n * // => true\n *\n * _.isObjectLike(_.noop);\n * // => false\n *\n * _.isObjectLike(null);\n * // => false\n */\nfunction isObjectLike(value) {\n return value != null && typeof value == 'object';\n}\n\nexport default isObjectLike;\n","import baseGetTag from './_baseGetTag.js';\nimport isArray from './isArray.js';\nimport isObjectLike from './isObjectLike.js';\n\n/** `Object#toString` result references. */\nvar stringTag = '[object String]';\n\n/**\n * Checks if `value` is classified as a `String` primitive or object.\n *\n * @static\n * @since 0.1.0\n * @memberOf _\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a string, else `false`.\n * @example\n *\n * _.isString('abc');\n * // => true\n *\n * _.isString(1);\n * // => false\n */\nfunction isString(value) {\n return typeof value == 'string' ||\n (!isArray(value) && isObjectLike(value) && baseGetTag(value) == stringTag);\n}\n\nexport default isString;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/createelement\n */\n\nimport isIterable from '../isiterable';\nimport { isString } from 'lodash-es';\n\n/**\n * Attributes to be applied to the HTML element.\n */\ntype HTMLElementAttributes = { readonly [ key: string ]: string };\n\n/**\n * Attributes to be applied to the SVG element.\n */\ntype SVGElementAttributes = HTMLElementAttributes & { xmlns: string };\n\n/**\n * Element or elements that will be added to the created element as children. Strings will be automatically turned into Text nodes.\n */\ntype ChildrenElements = Node | string | Iterable<Node | string>;\n\n/**\n * Creates an HTML element with attributes and children elements.\n *\n * ```ts\n * createElement( document, 'p' ); // <p>\n * createElement( document, 'p', { class: 'foo' } ); // <p class=\"foo\">\n * createElement( document, 'p', null, 'foo' ); // <p>foo</p>\n * createElement( document, 'p', null, [ createElement(...) ] ); // <p><...></p>\n * ```\n *\n * @label HTML_ELEMENT\n * @param doc Document used to create the element.\n * @param name Name of the HTML element.\n * @param attributes Object where keys represent attribute keys and values represent attribute values.\n * @param children Child or any iterable of children. Strings will be automatically turned into Text nodes.\n * @returns HTML element.\n */\nexport default function createElement<T extends keyof HTMLElementTagNameMap>(\n\tdoc: Document,\n\tname: T,\n\tattributes?: HTMLElementAttributes,\n\tchildren?: ChildrenElements\n): HTMLElementTagNameMap[T];\n\n/**\n * Creates an SVG element with attributes and children elements.\n *\n * ```ts\n * createElement( document, 'mask', { xmlns: 'http://www.w3.org/2000/svg' } ); // <mask>\n * createElement( document, 'mask', { xmlns: 'http://www.w3.org/2000/svg', id: 'foo' } ); // <mask id=\"foo\">\n * createElement( document, 'mask', { xmlns: 'http://www.w3.org/2000/svg' }, 'foo' ); // <mask>foo</mask>\n * createElement( document, 'mask', { xmlns: 'http://www.w3.org/2000/svg' }, [ createElement(...) ] ); // <mask><...></mask>\n * ```\n *\n * @label SVG_ELEMENT\n * @param doc Document used to create the element.\n * @param name Name of the SVG element.\n * @param attributes Object where keys represent attribute keys and values represent attribute values.\n * @param children Child or any iterable of children. Strings will be automatically turned into Text nodes.\n * @returns SVG element.\n */\nexport default function createElement<T extends keyof SVGElementTagNameMap>(\n\tdoc: Document,\n\tname: T,\n\tattributes: SVGElementAttributes,\n\tchildren?: ChildrenElements\n): SVGElementTagNameMap[T];\n\n/**\n * Creates an HTML or SVG element with attributes and children elements.\n *\n * ```ts\n * createElement( document, 'p' ); // <p>\n * createElement( document, 'mask', { xmlns: 'http://www.w3.org/2000/svg' } ); // <mask>\n * ```\n *\n * @param doc Document used to create the element.\n * @param name Name of the element.\n * @param attributes Object where keys represent attribute keys and values represent attribute values.\n * @param children Child or any iterable of children. Strings will be automatically turned into Text nodes.\n * @returns HTML or SVG element.\n */\nexport default function createElement(\n\tdoc: Document,\n\tname: string,\n\tattributes: SVGElementAttributes | HTMLElementAttributes = {},\n\tchildren: ChildrenElements = []\n): Element {\n\tconst namespace = attributes && attributes.xmlns;\n\tconst element = namespace ? doc.createElementNS( namespace, name ) : doc.createElement( name );\n\n\tfor ( const key in attributes ) {\n\t\telement.setAttribute( key, attributes[ key ] );\n\t}\n\n\tif ( isString( children ) || !isIterable( children ) ) {\n\t\tchildren = [ children ];\n\t}\n\n\tfor ( let child of children ) {\n\t\tif ( isString( child ) ) {\n\t\t\tchild = doc.createTextNode( child );\n\t\t}\n\n\t\telement.appendChild( child );\n\t}\n\n\treturn element;\n}\n","/**\n * Creates a unary function that invokes `func` with its argument transformed.\n *\n * @private\n * @param {Function} func The function to wrap.\n * @param {Function} transform The argument transform.\n * @returns {Function} Returns the new function.\n */\nfunction overArg(func, transform) {\n return function(arg) {\n return func(transform(arg));\n };\n}\n\nexport default overArg;\n","import overArg from './_overArg.js';\n\n/** Built-in value references. */\nvar getPrototype = overArg(Object.getPrototypeOf, Object);\n\nexport default getPrototype;\n","import baseGetTag from './_baseGetTag.js';\nimport getPrototype from './_getPrototype.js';\nimport isObjectLike from './isObjectLike.js';\n\n/** `Object#toString` result references. */\nvar objectTag = '[object Object]';\n\n/** Used for built-in method references. */\nvar funcProto = Function.prototype,\n objectProto = Object.prototype;\n\n/** Used to resolve the decompiled source of functions. */\nvar funcToString = funcProto.toString;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/** Used to infer the `Object` constructor. */\nvar objectCtorString = funcToString.call(Object);\n\n/**\n * Checks if `value` is a plain object, that is, an object created by the\n * `Object` constructor or one with a `[[Prototype]]` of `null`.\n *\n * @static\n * @memberOf _\n * @since 0.8.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a plain object, else `false`.\n * @example\n *\n * function Foo() {\n * this.a = 1;\n * }\n *\n * _.isPlainObject(new Foo);\n * // => false\n *\n * _.isPlainObject([1, 2, 3]);\n * // => false\n *\n * _.isPlainObject({ 'x': 0, 'y': 0 });\n * // => true\n *\n * _.isPlainObject(Object.create(null));\n * // => true\n */\nfunction isPlainObject(value) {\n if (!isObjectLike(value) || baseGetTag(value) != objectTag) {\n return false;\n }\n var proto = getPrototype(value);\n if (proto === null) {\n return true;\n }\n var Ctor = hasOwnProperty.call(proto, 'constructor') && proto.constructor;\n return typeof Ctor == 'function' && Ctor instanceof Ctor &&\n funcToString.call(Ctor) == objectCtorString;\n}\n\nexport default isPlainObject;\n","/**\n * Removes all key-value entries from the list cache.\n *\n * @private\n * @name clear\n * @memberOf ListCache\n */\nfunction listCacheClear() {\n this.__data__ = [];\n this.size = 0;\n}\n\nexport default listCacheClear;\n","/**\n * Performs a\n * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)\n * comparison between two values to determine if they are equivalent.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to compare.\n * @param {*} other The other value to compare.\n * @returns {boolean} Returns `true` if the values are equivalent, else `false`.\n * @example\n *\n * var object = { 'a': 1 };\n * var other = { 'a': 1 };\n *\n * _.eq(object, object);\n * // => true\n *\n * _.eq(object, other);\n * // => false\n *\n * _.eq('a', 'a');\n * // => true\n *\n * _.eq('a', Object('a'));\n * // => false\n *\n * _.eq(NaN, NaN);\n * // => true\n */\nfunction eq(value, other) {\n return value === other || (value !== value && other !== other);\n}\n\nexport default eq;\n","import eq from './eq.js';\n\n/**\n * Gets the index at which the `key` is found in `array` of key-value pairs.\n *\n * @private\n * @param {Array} array The array to inspect.\n * @param {*} key The key to search for.\n * @returns {number} Returns the index of the matched value, else `-1`.\n */\nfunction assocIndexOf(array, key) {\n var length = array.length;\n while (length--) {\n if (eq(array[length][0], key)) {\n return length;\n }\n }\n return -1;\n}\n\nexport default assocIndexOf;\n","import assocIndexOf from './_assocIndexOf.js';\n\n/** Used for built-in method references. */\nvar arrayProto = Array.prototype;\n\n/** Built-in value references. */\nvar splice = arrayProto.splice;\n\n/**\n * Removes `key` and its value from the list cache.\n *\n * @private\n * @name delete\n * @memberOf ListCache\n * @param {string} key The key of the value to remove.\n * @returns {boolean} Returns `true` if the entry was removed, else `false`.\n */\nfunction listCacheDelete(key) {\n var data = this.__data__,\n index = assocIndexOf(data, key);\n\n if (index < 0) {\n return false;\n }\n var lastIndex = data.length - 1;\n if (index == lastIndex) {\n data.pop();\n } else {\n splice.call(data, index, 1);\n }\n --this.size;\n return true;\n}\n\nexport default listCacheDelete;\n","import assocIndexOf from './_assocIndexOf.js';\n\n/**\n * Gets the list cache value for `key`.\n *\n * @private\n * @name get\n * @memberOf ListCache\n * @param {string} key The key of the value to get.\n * @returns {*} Returns the entry value.\n */\nfunction listCacheGet(key) {\n var data = this.__data__,\n index = assocIndexOf(data, key);\n\n return index < 0 ? undefined : data[index][1];\n}\n\nexport default listCacheGet;\n","import assocIndexOf from './_assocIndexOf.js';\n\n/**\n * Checks if a list cache value for `key` exists.\n *\n * @private\n * @name has\n * @memberOf ListCache\n * @param {string} key The key of the entry to check.\n * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.\n */\nfunction listCacheHas(key) {\n return assocIndexOf(this.__data__, key) > -1;\n}\n\nexport default listCacheHas;\n","import assocIndexOf from './_assocIndexOf.js';\n\n/**\n * Sets the list cache `key` to `value`.\n *\n * @private\n * @name set\n * @memberOf ListCache\n * @param {string} key The key of the value to set.\n * @param {*} value The value to set.\n * @returns {Object} Returns the list cache instance.\n */\nfunction listCacheSet(key, value) {\n var data = this.__data__,\n index = assocIndexOf(data, key);\n\n if (index < 0) {\n ++this.size;\n data.push([key, value]);\n } else {\n data[index][1] = value;\n }\n return this;\n}\n\nexport default listCacheSet;\n","import listCacheClear from './_listCacheClear.js';\nimport listCacheDelete from './_listCacheDelete.js';\nimport listCacheGet from './_listCacheGet.js';\nimport listCacheHas from './_listCacheHas.js';\nimport listCacheSet from './_listCacheSet.js';\n\n/**\n * Creates an list cache object.\n *\n * @private\n * @constructor\n * @param {Array} [entries] The key-value pairs to cache.\n */\nfunction ListCache(entries) {\n var index = -1,\n length = entries == null ? 0 : entries.length;\n\n this.clear();\n while (++index < length) {\n var entry = entries[index];\n this.set(entry[0], entry[1]);\n }\n}\n\n// Add methods to `ListCache`.\nListCache.prototype.clear = listCacheClear;\nListCache.prototype['delete'] = listCacheDelete;\nListCache.prototype.get = listCacheGet;\nListCache.prototype.has = listCacheHas;\nListCache.prototype.set = listCacheSet;\n\nexport default ListCache;\n","import ListCache from './_ListCache.js';\n\n/**\n * Removes all key-value entries from the stack.\n *\n * @private\n * @name clear\n * @memberOf Stack\n */\nfunction stackClear() {\n this.__data__ = new ListCache;\n this.size = 0;\n}\n\nexport default stackClear;\n","/**\n * Removes `key` and its value from the stack.\n *\n * @private\n * @name delete\n * @memberOf Stack\n * @param {string} key The key of the value to remove.\n * @returns {boolean} Returns `true` if the entry was removed, else `false`.\n */\nfunction stackDelete(key) {\n var data = this.__data__,\n result = data['delete'](key);\n\n this.size = data.size;\n return result;\n}\n\nexport default stackDelete;\n","/**\n * Gets the stack value for `key`.\n *\n * @private\n * @name get\n * @memberOf Stack\n * @param {string} key The key of the value to get.\n * @returns {*} Returns the entry value.\n */\nfunction stackGet(key) {\n return this.__data__.get(key);\n}\n\nexport default stackGet;\n","/**\n * Checks if a stack value for `key` exists.\n *\n * @private\n * @name has\n * @memberOf Stack\n * @param {string} key The key of the entry to check.\n * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.\n */\nfunction stackHas(key) {\n return this.__data__.has(key);\n}\n\nexport default stackHas;\n","import baseGetTag from './_baseGetTag.js';\nimport isObject from './isObject.js';\n\n/** `Object#toString` result references. */\nvar asyncTag = '[object AsyncFunction]',\n funcTag = '[object Function]',\n genTag = '[object GeneratorFunction]',\n proxyTag = '[object Proxy]';\n\n/**\n * Checks if `value` is classified as a `Function` object.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a function, else `false`.\n * @example\n *\n * _.isFunction(_);\n * // => true\n *\n * _.isFunction(/abc/);\n * // => false\n */\nfunction isFunction(value) {\n if (!isObject(value)) {\n return false;\n }\n // The use of `Object#toString` avoids issues with the `typeof` operator\n // in Safari 9 which returns 'object' for typed arrays and other constructors.\n var tag = baseGetTag(value);\n return tag == funcTag || tag == genTag || tag == asyncTag || tag == proxyTag;\n}\n\nexport default isFunction;\n","import root from './_root.js';\n\n/** Used to detect overreaching core-js shims. */\nvar coreJsData = root['__core-js_shared__'];\n\nexport default coreJsData;\n","import coreJsData from './_coreJsData.js';\n\n/** Used to detect methods masquerading as native. */\nvar maskSrcKey = (function() {\n var uid = /[^.]+$/.exec(coreJsData && coreJsData.keys && coreJsData.keys.IE_PROTO || '');\n return uid ? ('Symbol(src)_1.' + uid) : '';\n}());\n\n/**\n * Checks if `func` has its source masked.\n *\n * @private\n * @param {Function} func The function to check.\n * @returns {boolean} Returns `true` if `func` is masked, else `false`.\n */\nfunction isMasked(func) {\n return !!maskSrcKey && (maskSrcKey in func);\n}\n\nexport default isMasked;\n","/** Used for built-in method references. */\nvar funcProto = Function.prototype;\n\n/** Used to resolve the decompiled source of functions. */\nvar funcToString = funcProto.toString;\n\n/**\n * Converts `func` to its source code.\n *\n * @private\n * @param {Function} func The function to convert.\n * @returns {string} Returns the source code.\n */\nfunction toSource(func) {\n if (func != null) {\n try {\n return funcToString.call(func);\n } catch (e) {}\n try {\n return (func + '');\n } catch (e) {}\n }\n return '';\n}\n\nexport default toSource;\n","import isFunction from './isFunction.js';\nimport isMasked from './_isMasked.js';\nimport isObject from './isObject.js';\nimport toSource from './_toSource.js';\n\n/**\n * Used to match `RegExp`\n * [syntax characters](http://ecma-international.org/ecma-262/7.0/#sec-patterns).\n */\nvar reRegExpChar = /[\\\\^$.*+?()[\\]{}|]/g;\n\n/** Used to detect host constructors (Safari). */\nvar reIsHostCtor = /^\\[object .+?Constructor\\]$/;\n\n/** Used for built-in method references. */\nvar funcProto = Function.prototype,\n objectProto = Object.prototype;\n\n/** Used to resolve the decompiled source of functions. */\nvar funcToString = funcProto.toString;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/** Used to detect if a method is native. */\nvar reIsNative = RegExp('^' +\n funcToString.call(hasOwnProperty).replace(reRegExpChar, '\\\\$&')\n .replace(/hasOwnProperty|(function).*?(?=\\\\\\()| for .+?(?=\\\\\\])/g, '$1.*?') + '$'\n);\n\n/**\n * The base implementation of `_.isNative` without bad shim checks.\n *\n * @private\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a native function,\n * else `false`.\n */\nfunction baseIsNative(value) {\n if (!isObject(value) || isMasked(value)) {\n return false;\n }\n var pattern = isFunction(value) ? reIsNative : reIsHostCtor;\n return pattern.test(toSource(value));\n}\n\nexport default baseIsNative;\n","/**\n * Gets the value at `key` of `object`.\n *\n * @private\n * @param {Object} [object] The object to query.\n * @param {string} key The key of the property to get.\n * @returns {*} Returns the property value.\n */\nfunction getValue(object, key) {\n return object == null ? undefined : object[key];\n}\n\nexport default getValue;\n","import baseIsNative from './_baseIsNative.js';\nimport getValue from './_getValue.js';\n\n/**\n * Gets the native function at `key` of `object`.\n *\n * @private\n * @param {Object} object The object to query.\n * @param {string} key The key of the method to get.\n * @returns {*} Returns the function if it's native, else `undefined`.\n */\nfunction getNative(object, key) {\n var value = getValue(object, key);\n return baseIsNative(value) ? value : undefined;\n}\n\nexport default getNative;\n","import getNative from './_getNative.js';\nimport root from './_root.js';\n\n/* Built-in method references that are verified to be native. */\nvar Map = getNative(root, 'Map');\n\nexport default Map;\n","import getNative from './_getNative.js';\n\n/* Built-in method references that are verified to be native. */\nvar nativeCreate = getNative(Object, 'create');\n\nexport default nativeCreate;\n","import nativeCreate from './_nativeCreate.js';\n\n/**\n * Removes all key-value entries from the hash.\n *\n * @private\n * @name clear\n * @memberOf Hash\n */\nfunction hashClear() {\n this.__data__ = nativeCreate ? nativeCreate(null) : {};\n this.size = 0;\n}\n\nexport default hashClear;\n","/**\n * Removes `key` and its value from the hash.\n *\n * @private\n * @name delete\n * @memberOf Hash\n * @param {Object} hash The hash to modify.\n * @param {string} key The key of the value to remove.\n * @returns {boolean} Returns `true` if the entry was removed, else `false`.\n */\nfunction hashDelete(key) {\n var result = this.has(key) && delete this.__data__[key];\n this.size -= result ? 1 : 0;\n return result;\n}\n\nexport default hashDelete;\n","import nativeCreate from './_nativeCreate.js';\n\n/** Used to stand-in for `undefined` hash values. */\nvar HASH_UNDEFINED = '__lodash_hash_undefined__';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * Gets the hash value for `key`.\n *\n * @private\n * @name get\n * @memberOf Hash\n * @param {string} key The key of the value to get.\n * @returns {*} Returns the entry value.\n */\nfunction hashGet(key) {\n var data = this.__data__;\n if (nativeCreate) {\n var result = data[key];\n return result === HASH_UNDEFINED ? undefined : result;\n }\n return hasOwnProperty.call(data, key) ? data[key] : undefined;\n}\n\nexport default hashGet;\n","import nativeCreate from './_nativeCreate.js';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * Checks if a hash value for `key` exists.\n *\n * @private\n * @name has\n * @memberOf Hash\n * @param {string} key The key of the entry to check.\n * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.\n */\nfunction hashHas(key) {\n var data = this.__data__;\n return nativeCreate ? (data[key] !== undefined) : hasOwnProperty.call(data, key);\n}\n\nexport default hashHas;\n","import nativeCreate from './_nativeCreate.js';\n\n/** Used to stand-in for `undefined` hash values. */\nvar HASH_UNDEFINED = '__lodash_hash_undefined__';\n\n/**\n * Sets the hash `key` to `value`.\n *\n * @private\n * @name set\n * @memberOf Hash\n * @param {string} key The key of the value to set.\n * @param {*} value The value to set.\n * @returns {Object} Returns the hash instance.\n */\nfunction hashSet(key, value) {\n var data = this.__data__;\n this.size += this.has(key) ? 0 : 1;\n data[key] = (nativeCreate && value === undefined) ? HASH_UNDEFINED : value;\n return this;\n}\n\nexport default hashSet;\n","import hashClear from './_hashClear.js';\nimport hashDelete from './_hashDelete.js';\nimport hashGet from './_hashGet.js';\nimport hashHas from './_hashHas.js';\nimport hashSet from './_hashSet.js';\n\n/**\n * Creates a hash object.\n *\n * @private\n * @constructor\n * @param {Array} [entries] The key-value pairs to cache.\n */\nfunction Hash(entries) {\n var index = -1,\n length = entries == null ? 0 : entries.length;\n\n this.clear();\n while (++index < length) {\n var entry = entries[index];\n this.set(entry[0], entry[1]);\n }\n}\n\n// Add methods to `Hash`.\nHash.prototype.clear = hashClear;\nHash.prototype['delete'] = hashDelete;\nHash.prototype.get = hashGet;\nHash.prototype.has = hashHas;\nHash.prototype.set = hashSet;\n\nexport default Hash;\n","import Hash from './_Hash.js';\nimport ListCache from './_ListCache.js';\nimport Map from './_Map.js';\n\n/**\n * Removes all key-value entries from the map.\n *\n * @private\n * @name clear\n * @memberOf MapCache\n */\nfunction mapCacheClear() {\n this.size = 0;\n this.__data__ = {\n 'hash': new Hash,\n 'map': new (Map || ListCache),\n 'string': new Hash\n };\n}\n\nexport default mapCacheClear;\n","/**\n * Checks if `value` is suitable for use as unique object key.\n *\n * @private\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is suitable, else `false`.\n */\nfunction isKeyable(value) {\n var type = typeof value;\n return (type == 'string' || type == 'number' || type == 'symbol' || type == 'boolean')\n ? (value !== '__proto__')\n : (value === null);\n}\n\nexport default isKeyable;\n","import isKeyable from './_isKeyable.js';\n\n/**\n * Gets the data for `map`.\n *\n * @private\n * @param {Object} map The map to query.\n * @param {string} key The reference key.\n * @returns {*} Returns the map data.\n */\nfunction getMapData(map, key) {\n var data = map.__data__;\n return isKeyable(key)\n ? data[typeof key == 'string' ? 'string' : 'hash']\n : data.map;\n}\n\nexport default getMapData;\n","import getMapData from './_getMapData.js';\n\n/**\n * Removes `key` and its value from the map.\n *\n * @private\n * @name delete\n * @memberOf MapCache\n * @param {string} key The key of the value to remove.\n * @returns {boolean} Returns `true` if the entry was removed, else `false`.\n */\nfunction mapCacheDelete(key) {\n var result = getMapData(this, key)['delete'](key);\n this.size -= result ? 1 : 0;\n return result;\n}\n\nexport default mapCacheDelete;\n","import getMapData from './_getMapData.js';\n\n/**\n * Gets the map value for `key`.\n *\n * @private\n * @name get\n * @memberOf MapCache\n * @param {string} key The key of the value to get.\n * @returns {*} Returns the entry value.\n */\nfunction mapCacheGet(key) {\n return getMapData(this, key).get(key);\n}\n\nexport default mapCacheGet;\n","import getMapData from './_getMapData.js';\n\n/**\n * Checks if a map value for `key` exists.\n *\n * @private\n * @name has\n * @memberOf MapCache\n * @param {string} key The key of the entry to check.\n * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.\n */\nfunction mapCacheHas(key) {\n return getMapData(this, key).has(key);\n}\n\nexport default mapCacheHas;\n","import getMapData from './_getMapData.js';\n\n/**\n * Sets the map `key` to `value`.\n *\n * @private\n * @name set\n * @memberOf MapCache\n * @param {string} key The key of the value to set.\n * @param {*} value The value to set.\n * @returns {Object} Returns the map cache instance.\n */\nfunction mapCacheSet(key, value) {\n var data = getMapData(this, key),\n size = data.size;\n\n data.set(key, value);\n this.size += data.size == size ? 0 : 1;\n return this;\n}\n\nexport default mapCacheSet;\n","import mapCacheClear from './_mapCacheClear.js';\nimport mapCacheDelete from './_mapCacheDelete.js';\nimport mapCacheGet from './_mapCacheGet.js';\nimport mapCacheHas from './_mapCacheHas.js';\nimport mapCacheSet from './_mapCacheSet.js';\n\n/**\n * Creates a map cache object to store key-value pairs.\n *\n * @private\n * @constructor\n * @param {Array} [entries] The key-value pairs to cache.\n */\nfunction MapCache(entries) {\n var index = -1,\n length = entries == null ? 0 : entries.length;\n\n this.clear();\n while (++index < length) {\n var entry = entries[index];\n this.set(entry[0], entry[1]);\n }\n}\n\n// Add methods to `MapCache`.\nMapCache.prototype.clear = mapCacheClear;\nMapCache.prototype['delete'] = mapCacheDelete;\nMapCache.prototype.get = mapCacheGet;\nMapCache.prototype.has = mapCacheHas;\nMapCache.prototype.set = mapCacheSet;\n\nexport default MapCache;\n","import ListCache from './_ListCache.js';\nimport Map from './_Map.js';\nimport MapCache from './_MapCache.js';\n\n/** Used as the size to enable large array optimizations. */\nvar LARGE_ARRAY_SIZE = 200;\n\n/**\n * Sets the stack `key` to `value`.\n *\n * @private\n * @name set\n * @memberOf Stack\n * @param {string} key The key of the value to set.\n * @param {*} value The value to set.\n * @returns {Object} Returns the stack cache instance.\n */\nfunction stackSet(key, value) {\n var data = this.__data__;\n if (data instanceof ListCache) {\n var pairs = data.__data__;\n if (!Map || (pairs.length < LARGE_ARRAY_SIZE - 1)) {\n pairs.push([key, value]);\n this.size = ++data.size;\n return this;\n }\n data = this.__data__ = new MapCache(pairs);\n }\n data.set(key, value);\n this.size = data.size;\n return this;\n}\n\nexport default stackSet;\n","import ListCache from './_ListCache.js';\nimport stackClear from './_stackClear.js';\nimport stackDelete from './_stackDelete.js';\nimport stackGet from './_stackGet.js';\nimport stackHas from './_stackHas.js';\nimport stackSet from './_stackSet.js';\n\n/**\n * Creates a stack cache object to store key-value pairs.\n *\n * @private\n * @constructor\n * @param {Array} [entries] The key-value pairs to cache.\n */\nfunction Stack(entries) {\n var data = this.__data__ = new ListCache(entries);\n this.size = data.size;\n}\n\n// Add methods to `Stack`.\nStack.prototype.clear = stackClear;\nStack.prototype['delete'] = stackDelete;\nStack.prototype.get = stackGet;\nStack.prototype.has = stackHas;\nStack.prototype.set = stackSet;\n\nexport default Stack;\n","/**\n * A specialized version of `_.forEach` for arrays without support for\n * iteratee shorthands.\n *\n * @private\n * @param {Array} [array] The array to iterate over.\n * @param {Function} iteratee The function invoked per iteration.\n * @returns {Array} Returns `array`.\n */\nfunction arrayEach(array, iteratee) {\n var index = -1,\n length = array == null ? 0 : array.length;\n\n while (++index < length) {\n if (iteratee(array[index], index, array) === false) {\n break;\n }\n }\n return array;\n}\n\nexport default arrayEach;\n","import getNative from './_getNative.js';\n\nvar defineProperty = (function() {\n try {\n var func = getNative(Object, 'defineProperty');\n func({}, '', {});\n return func;\n } catch (e) {}\n}());\n\nexport default defineProperty;\n","import defineProperty from './_defineProperty.js';\n\n/**\n * The base implementation of `assignValue` and `assignMergeValue` without\n * value checks.\n *\n * @private\n * @param {Object} object The object to modify.\n * @param {string} key The key of the property to assign.\n * @param {*} value The value to assign.\n */\nfunction baseAssignValue(object, key, value) {\n if (key == '__proto__' && defineProperty) {\n defineProperty(object, key, {\n 'configurable': true,\n 'enumerable': true,\n 'value': value,\n 'writable': true\n });\n } else {\n object[key] = value;\n }\n}\n\nexport default baseAssignValue;\n","import baseAssignValue from './_baseAssignValue.js';\nimport eq from './eq.js';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * Assigns `value` to `key` of `object` if the existing value is not equivalent\n * using [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero)\n * for equality comparisons.\n *\n * @private\n * @param {Object} object The object to modify.\n * @param {string} key The key of the property to assign.\n * @param {*} value The value to assign.\n */\nfunction assignValue(object, key, value) {\n var objValue = object[key];\n if (!(hasOwnProperty.call(object, key) && eq(objValue, value)) ||\n (value === undefined && !(key in object))) {\n baseAssignValue(object, key, value);\n }\n}\n\nexport default assignValue;\n","import assignValue from './_assignValue.js';\nimport baseAssignValue from './_baseAssignValue.js';\n\n/**\n * Copies properties of `source` to `object`.\n *\n * @private\n * @param {Object} source The object to copy properties from.\n * @param {Array} props The property identifiers to copy.\n * @param {Object} [object={}] The object to copy properties to.\n * @param {Function} [customizer] The function to customize copied values.\n * @returns {Object} Returns `object`.\n */\nfunction copyObject(source, props, object, customizer) {\n var isNew = !object;\n object || (object = {});\n\n var index = -1,\n length = props.length;\n\n while (++index < length) {\n var key = props[index];\n\n var newValue = customizer\n ? customizer(object[key], source[key], key, object, source)\n : undefined;\n\n if (newValue === undefined) {\n newValue = source[key];\n }\n if (isNew) {\n baseAssignValue(object, key, newValue);\n } else {\n assignValue(object, key, newValue);\n }\n }\n return object;\n}\n\nexport default copyObject;\n","/**\n * The base implementation of `_.times` without support for iteratee shorthands\n * or max array length checks.\n *\n * @private\n * @param {number} n The number of times to invoke `iteratee`.\n * @param {Function} iteratee The function invoked per iteration.\n * @returns {Array} Returns the array of results.\n */\nfunction baseTimes(n, iteratee) {\n var index = -1,\n result = Array(n);\n\n while (++index < n) {\n result[index] = iteratee(index);\n }\n return result;\n}\n\nexport default baseTimes;\n","import baseGetTag from './_baseGetTag.js';\nimport isObjectLike from './isObjectLike.js';\n\n/** `Object#toString` result references. */\nvar argsTag = '[object Arguments]';\n\n/**\n * The base implementation of `_.isArguments`.\n *\n * @private\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is an `arguments` object,\n */\nfunction baseIsArguments(value) {\n return isObjectLike(value) && baseGetTag(value) == argsTag;\n}\n\nexport default baseIsArguments;\n","import baseIsArguments from './_baseIsArguments.js';\nimport isObjectLike from './isObjectLike.js';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/** Built-in value references. */\nvar propertyIsEnumerable = objectProto.propertyIsEnumerable;\n\n/**\n * Checks if `value` is likely an `arguments` object.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is an `arguments` object,\n * else `false`.\n * @example\n *\n * _.isArguments(function() { return arguments; }());\n * // => true\n *\n * _.isArguments([1, 2, 3]);\n * // => false\n */\nvar isArguments = baseIsArguments(function() { return arguments; }()) ? baseIsArguments : function(value) {\n return isObjectLike(value) && hasOwnProperty.call(value, 'callee') &&\n !propertyIsEnumerable.call(value, 'callee');\n};\n\nexport default isArguments;\n","/**\n * This method returns `false`.\n *\n * @static\n * @memberOf _\n * @since 4.13.0\n * @category Util\n * @returns {boolean} Returns `false`.\n * @example\n *\n * _.times(2, _.stubFalse);\n * // => [false, false]\n */\nfunction stubFalse() {\n return false;\n}\n\nexport default stubFalse;\n","import root from './_root.js';\nimport stubFalse from './stubFalse.js';\n\n/** Detect free variable `exports`. */\nvar freeExports = typeof exports == 'object' && exports && !exports.nodeType && exports;\n\n/** Detect free variable `module`. */\nvar freeModule = freeExports && typeof module == 'object' && module && !module.nodeType && module;\n\n/** Detect the popular CommonJS extension `module.exports`. */\nvar moduleExports = freeModule && freeModule.exports === freeExports;\n\n/** Built-in value references. */\nvar Buffer = moduleExports ? root.Buffer : undefined;\n\n/* Built-in method references for those with the same name as other `lodash` methods. */\nvar nativeIsBuffer = Buffer ? Buffer.isBuffer : undefined;\n\n/**\n * Checks if `value` is a buffer.\n *\n * @static\n * @memberOf _\n * @since 4.3.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a buffer, else `false`.\n * @example\n *\n * _.isBuffer(new Buffer(2));\n * // => true\n *\n * _.isBuffer(new Uint8Array(2));\n * // => false\n */\nvar isBuffer = nativeIsBuffer || stubFalse;\n\nexport default isBuffer;\n","/** Used as references for various `Number` constants. */\nvar MAX_SAFE_INTEGER = 9007199254740991;\n\n/** Used to detect unsigned integer values. */\nvar reIsUint = /^(?:0|[1-9]\\d*)$/;\n\n/**\n * Checks if `value` is a valid array-like index.\n *\n * @private\n * @param {*} value The value to check.\n * @param {number} [length=MAX_SAFE_INTEGER] The upper bounds of a valid index.\n * @returns {boolean} Returns `true` if `value` is a valid index, else `false`.\n */\nfunction isIndex(value, length) {\n var type = typeof value;\n length = length == null ? MAX_SAFE_INTEGER : length;\n\n return !!length &&\n (type == 'number' ||\n (type != 'symbol' && reIsUint.test(value))) &&\n (value > -1 && value % 1 == 0 && value < length);\n}\n\nexport default isIndex;\n","/** Used as references for various `Number` constants. */\nvar MAX_SAFE_INTEGER = 9007199254740991;\n\n/**\n * Checks if `value` is a valid array-like length.\n *\n * **Note:** This method is loosely based on\n * [`ToLength`](http://ecma-international.org/ecma-262/7.0/#sec-tolength).\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a valid length, else `false`.\n * @example\n *\n * _.isLength(3);\n * // => true\n *\n * _.isLength(Number.MIN_VALUE);\n * // => false\n *\n * _.isLength(Infinity);\n * // => false\n *\n * _.isLength('3');\n * // => false\n */\nfunction isLength(value) {\n return typeof value == 'number' &&\n value > -1 && value % 1 == 0 && value <= MAX_SAFE_INTEGER;\n}\n\nexport default isLength;\n","import baseGetTag from './_baseGetTag.js';\nimport isLength from './isLength.js';\nimport isObjectLike from './isObjectLike.js';\n\n/** `Object#toString` result references. */\nvar argsTag = '[object Arguments]',\n arrayTag = '[object Array]',\n boolTag = '[object Boolean]',\n dateTag = '[object Date]',\n errorTag = '[object Error]',\n funcTag = '[object Function]',\n mapTag = '[object Map]',\n numberTag = '[object Number]',\n objectTag = '[object Object]',\n regexpTag = '[object RegExp]',\n setTag = '[object Set]',\n stringTag = '[object String]',\n weakMapTag = '[object WeakMap]';\n\nvar arrayBufferTag = '[object ArrayBuffer]',\n dataViewTag = '[object DataView]',\n float32Tag = '[object Float32Array]',\n float64Tag = '[object Float64Array]',\n int8Tag = '[object Int8Array]',\n int16Tag = '[object Int16Array]',\n int32Tag = '[object Int32Array]',\n uint8Tag = '[object Uint8Array]',\n uint8ClampedTag = '[object Uint8ClampedArray]',\n uint16Tag = '[object Uint16Array]',\n uint32Tag = '[object Uint32Array]';\n\n/** Used to identify `toStringTag` values of typed arrays. */\nvar typedArrayTags = {};\ntypedArrayTags[float32Tag] = typedArrayTags[float64Tag] =\ntypedArrayTags[int8Tag] = typedArrayTags[int16Tag] =\ntypedArrayTags[int32Tag] = typedArrayTags[uint8Tag] =\ntypedArrayTags[uint8ClampedTag] = typedArrayTags[uint16Tag] =\ntypedArrayTags[uint32Tag] = true;\ntypedArrayTags[argsTag] = typedArrayTags[arrayTag] =\ntypedArrayTags[arrayBufferTag] = typedArrayTags[boolTag] =\ntypedArrayTags[dataViewTag] = typedArrayTags[dateTag] =\ntypedArrayTags[errorTag] = typedArrayTags[funcTag] =\ntypedArrayTags[mapTag] = typedArrayTags[numberTag] =\ntypedArrayTags[objectTag] = typedArrayTags[regexpTag] =\ntypedArrayTags[setTag] = typedArrayTags[stringTag] =\ntypedArrayTags[weakMapTag] = false;\n\n/**\n * The base implementation of `_.isTypedArray` without Node.js optimizations.\n *\n * @private\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a typed array, else `false`.\n */\nfunction baseIsTypedArray(value) {\n return isObjectLike(value) &&\n isLength(value.length) && !!typedArrayTags[baseGetTag(value)];\n}\n\nexport default baseIsTypedArray;\n","/**\n * The base implementation of `_.unary` without support for storing metadata.\n *\n * @private\n * @param {Function} func The function to cap arguments for.\n * @returns {Function} Returns the new capped function.\n */\nfunction baseUnary(func) {\n return function(value) {\n return func(value);\n };\n}\n\nexport default baseUnary;\n","import freeGlobal from './_freeGlobal.js';\n\n/** Detect free variable `exports`. */\nvar freeExports = typeof exports == 'object' && exports && !exports.nodeType && exports;\n\n/** Detect free variable `module`. */\nvar freeModule = freeExports && typeof module == 'object' && module && !module.nodeType && module;\n\n/** Detect the popular CommonJS extension `module.exports`. */\nvar moduleExports = freeModule && freeModule.exports === freeExports;\n\n/** Detect free variable `process` from Node.js. */\nvar freeProcess = moduleExports && freeGlobal.process;\n\n/** Used to access faster Node.js helpers. */\nvar nodeUtil = (function() {\n try {\n // Use `util.types` for Node.js 10+.\n var types = freeModule && freeModule.require && freeModule.require('util').types;\n\n if (types) {\n return types;\n }\n\n // Legacy `process.binding('util')` for Node.js < 10.\n return freeProcess && freeProcess.binding && freeProcess.binding('util');\n } catch (e) {}\n}());\n\nexport default nodeUtil;\n","import baseIsTypedArray from './_baseIsTypedArray.js';\nimport baseUnary from './_baseUnary.js';\nimport nodeUtil from './_nodeUtil.js';\n\n/* Node.js helper references. */\nvar nodeIsTypedArray = nodeUtil && nodeUtil.isTypedArray;\n\n/**\n * Checks if `value` is classified as a typed array.\n *\n * @static\n * @memberOf _\n * @since 3.0.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a typed array, else `false`.\n * @example\n *\n * _.isTypedArray(new Uint8Array);\n * // => true\n *\n * _.isTypedArray([]);\n * // => false\n */\nvar isTypedArray = nodeIsTypedArray ? baseUnary(nodeIsTypedArray) : baseIsTypedArray;\n\nexport default isTypedArray;\n","import baseTimes from './_baseTimes.js';\nimport isArguments from './isArguments.js';\nimport isArray from './isArray.js';\nimport isBuffer from './isBuffer.js';\nimport isIndex from './_isIndex.js';\nimport isTypedArray from './isTypedArray.js';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * Creates an array of the enumerable property names of the array-like `value`.\n *\n * @private\n * @param {*} value The value to query.\n * @param {boolean} inherited Specify returning inherited property names.\n * @returns {Array} Returns the array of property names.\n */\nfunction arrayLikeKeys(value, inherited) {\n var isArr = isArray(value),\n isArg = !isArr && isArguments(value),\n isBuff = !isArr && !isArg && isBuffer(value),\n isType = !isArr && !isArg && !isBuff && isTypedArray(value),\n skipIndexes = isArr || isArg || isBuff || isType,\n result = skipIndexes ? baseTimes(value.length, String) : [],\n length = result.length;\n\n for (var key in value) {\n if ((inherited || hasOwnProperty.call(value, key)) &&\n !(skipIndexes && (\n // Safari 9 has enumerable `arguments.length` in strict mode.\n key == 'length' ||\n // Node.js 0.10 has enumerable non-index properties on buffers.\n (isBuff && (key == 'offset' || key == 'parent')) ||\n // PhantomJS 2 has enumerable non-index properties on typed arrays.\n (isType && (key == 'buffer' || key == 'byteLength' || key == 'byteOffset')) ||\n // Skip index properties.\n isIndex(key, length)\n ))) {\n result.push(key);\n }\n }\n return result;\n}\n\nexport default arrayLikeKeys;\n","/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/**\n * Checks if `value` is likely a prototype object.\n *\n * @private\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a prototype, else `false`.\n */\nfunction isPrototype(value) {\n var Ctor = value && value.constructor,\n proto = (typeof Ctor == 'function' && Ctor.prototype) || objectProto;\n\n return value === proto;\n}\n\nexport default isPrototype;\n","import overArg from './_overArg.js';\n\n/* Built-in method references for those with the same name as other `lodash` methods. */\nvar nativeKeys = overArg(Object.keys, Object);\n\nexport default nativeKeys;\n","import isPrototype from './_isPrototype.js';\nimport nativeKeys from './_nativeKeys.js';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * The base implementation of `_.keys` which doesn't treat sparse arrays as dense.\n *\n * @private\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of property names.\n */\nfunction baseKeys(object) {\n if (!isPrototype(object)) {\n return nativeKeys(object);\n }\n var result = [];\n for (var key in Object(object)) {\n if (hasOwnProperty.call(object, key) && key != 'constructor') {\n result.push(key);\n }\n }\n return result;\n}\n\nexport default baseKeys;\n","import isFunction from './isFunction.js';\nimport isLength from './isLength.js';\n\n/**\n * Checks if `value` is array-like. A value is considered array-like if it's\n * not a function and has a `value.length` that's an integer greater than or\n * equal to `0` and less than or equal to `Number.MAX_SAFE_INTEGER`.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is array-like, else `false`.\n * @example\n *\n * _.isArrayLike([1, 2, 3]);\n * // => true\n *\n * _.isArrayLike(document.body.children);\n * // => true\n *\n * _.isArrayLike('abc');\n * // => true\n *\n * _.isArrayLike(_.noop);\n * // => false\n */\nfunction isArrayLike(value) {\n return value != null && isLength(value.length) && !isFunction(value);\n}\n\nexport default isArrayLike;\n","import arrayLikeKeys from './_arrayLikeKeys.js';\nimport baseKeys from './_baseKeys.js';\nimport isArrayLike from './isArrayLike.js';\n\n/**\n * Creates an array of the own enumerable property names of `object`.\n *\n * **Note:** Non-object values are coerced to objects. See the\n * [ES spec](http://ecma-international.org/ecma-262/7.0/#sec-object.keys)\n * for more details.\n *\n * @static\n * @since 0.1.0\n * @memberOf _\n * @category Object\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of property names.\n * @example\n *\n * function Foo() {\n * this.a = 1;\n * this.b = 2;\n * }\n *\n * Foo.prototype.c = 3;\n *\n * _.keys(new Foo);\n * // => ['a', 'b'] (iteration order is not guaranteed)\n *\n * _.keys('hi');\n * // => ['0', '1']\n */\nfunction keys(object) {\n return isArrayLike(object) ? arrayLikeKeys(object) : baseKeys(object);\n}\n\nexport default keys;\n","import copyObject from './_copyObject.js';\nimport keys from './keys.js';\n\n/**\n * The base implementation of `_.assign` without support for multiple sources\n * or `customizer` functions.\n *\n * @private\n * @param {Object} object The destination object.\n * @param {Object} source The source object.\n * @returns {Object} Returns `object`.\n */\nfunction baseAssign(object, source) {\n return object && copyObject(source, keys(source), object);\n}\n\nexport default baseAssign;\n","/**\n * This function is like\n * [`Object.keys`](http://ecma-international.org/ecma-262/7.0/#sec-object.keys)\n * except that it includes inherited enumerable properties.\n *\n * @private\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of property names.\n */\nfunction nativeKeysIn(object) {\n var result = [];\n if (object != null) {\n for (var key in Object(object)) {\n result.push(key);\n }\n }\n return result;\n}\n\nexport default nativeKeysIn;\n","import isObject from './isObject.js';\nimport isPrototype from './_isPrototype.js';\nimport nativeKeysIn from './_nativeKeysIn.js';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * The base implementation of `_.keysIn` which doesn't treat sparse arrays as dense.\n *\n * @private\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of property names.\n */\nfunction baseKeysIn(object) {\n if (!isObject(object)) {\n return nativeKeysIn(object);\n }\n var isProto = isPrototype(object),\n result = [];\n\n for (var key in object) {\n if (!(key == 'constructor' && (isProto || !hasOwnProperty.call(object, key)))) {\n result.push(key);\n }\n }\n return result;\n}\n\nexport default baseKeysIn;\n","import arrayLikeKeys from './_arrayLikeKeys.js';\nimport baseKeysIn from './_baseKeysIn.js';\nimport isArrayLike from './isArrayLike.js';\n\n/**\n * Creates an array of the own and inherited enumerable property names of `object`.\n *\n * **Note:** Non-object values are coerced to objects.\n *\n * @static\n * @memberOf _\n * @since 3.0.0\n * @category Object\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of property names.\n * @example\n *\n * function Foo() {\n * this.a = 1;\n * this.b = 2;\n * }\n *\n * Foo.prototype.c = 3;\n *\n * _.keysIn(new Foo);\n * // => ['a', 'b', 'c'] (iteration order is not guaranteed)\n */\nfunction keysIn(object) {\n return isArrayLike(object) ? arrayLikeKeys(object, true) : baseKeysIn(object);\n}\n\nexport default keysIn;\n","import copyObject from './_copyObject.js';\nimport keysIn from './keysIn.js';\n\n/**\n * The base implementation of `_.assignIn` without support for multiple sources\n * or `customizer` functions.\n *\n * @private\n * @param {Object} object The destination object.\n * @param {Object} source The source object.\n * @returns {Object} Returns `object`.\n */\nfunction baseAssignIn(object, source) {\n return object && copyObject(source, keysIn(source), object);\n}\n\nexport default baseAssignIn;\n","import root from './_root.js';\n\n/** Detect free variable `exports`. */\nvar freeExports = typeof exports == 'object' && exports && !exports.nodeType && exports;\n\n/** Detect free variable `module`. */\nvar freeModule = freeExports && typeof module == 'object' && module && !module.nodeType && module;\n\n/** Detect the popular CommonJS extension `module.exports`. */\nvar moduleExports = freeModule && freeModule.exports === freeExports;\n\n/** Built-in value references. */\nvar Buffer = moduleExports ? root.Buffer : undefined,\n allocUnsafe = Buffer ? Buffer.allocUnsafe : undefined;\n\n/**\n * Creates a clone of `buffer`.\n *\n * @private\n * @param {Buffer} buffer The buffer to clone.\n * @param {boolean} [isDeep] Specify a deep clone.\n * @returns {Buffer} Returns the cloned buffer.\n */\nfunction cloneBuffer(buffer, isDeep) {\n if (isDeep) {\n return buffer.slice();\n }\n var length = buffer.length,\n result = allocUnsafe ? allocUnsafe(length) : new buffer.constructor(length);\n\n buffer.copy(result);\n return result;\n}\n\nexport default cloneBuffer;\n","/**\n * Copies the values of `source` to `array`.\n *\n * @private\n * @param {Array} source The array to copy values from.\n * @param {Array} [array=[]] The array to copy values to.\n * @returns {Array} Returns `array`.\n */\nfunction copyArray(source, array) {\n var index = -1,\n length = source.length;\n\n array || (array = Array(length));\n while (++index < length) {\n array[index] = source[index];\n }\n return array;\n}\n\nexport default copyArray;\n","/**\n * A specialized version of `_.filter` for arrays without support for\n * iteratee shorthands.\n *\n * @private\n * @param {Array} [array] The array to iterate over.\n * @param {Function} predicate The function invoked per iteration.\n * @returns {Array} Returns the new filtered array.\n */\nfunction arrayFilter(array, predicate) {\n var index = -1,\n length = array == null ? 0 : array.length,\n resIndex = 0,\n result = [];\n\n while (++index < length) {\n var value = array[index];\n if (predicate(value, index, array)) {\n result[resIndex++] = value;\n }\n }\n return result;\n}\n\nexport default arrayFilter;\n","/**\n * This method returns a new empty array.\n *\n * @static\n * @memberOf _\n * @since 4.13.0\n * @category Util\n * @returns {Array} Returns the new empty array.\n * @example\n *\n * var arrays = _.times(2, _.stubArray);\n *\n * console.log(arrays);\n * // => [[], []]\n *\n * console.log(arrays[0] === arrays[1]);\n * // => false\n */\nfunction stubArray() {\n return [];\n}\n\nexport default stubArray;\n","import arrayFilter from './_arrayFilter.js';\nimport stubArray from './stubArray.js';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Built-in value references. */\nvar propertyIsEnumerable = objectProto.propertyIsEnumerable;\n\n/* Built-in method references for those with the same name as other `lodash` methods. */\nvar nativeGetSymbols = Object.getOwnPropertySymbols;\n\n/**\n * Creates an array of the own enumerable symbols of `object`.\n *\n * @private\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of symbols.\n */\nvar getSymbols = !nativeGetSymbols ? stubArray : function(object) {\n if (object == null) {\n return [];\n }\n object = Object(object);\n return arrayFilter(nativeGetSymbols(object), function(symbol) {\n return propertyIsEnumerable.call(object, symbol);\n });\n};\n\nexport default getSymbols;\n","import copyObject from './_copyObject.js';\nimport getSymbols from './_getSymbols.js';\n\n/**\n * Copies own symbols of `source` to `object`.\n *\n * @private\n * @param {Object} source The object to copy symbols from.\n * @param {Object} [object={}] The object to copy symbols to.\n * @returns {Object} Returns `object`.\n */\nfunction copySymbols(source, object) {\n return copyObject(source, getSymbols(source), object);\n}\n\nexport default copySymbols;\n","/**\n * Appends the elements of `values` to `array`.\n *\n * @private\n * @param {Array} array The array to modify.\n * @param {Array} values The values to append.\n * @returns {Array} Returns `array`.\n */\nfunction arrayPush(array, values) {\n var index = -1,\n length = values.length,\n offset = array.length;\n\n while (++index < length) {\n array[offset + index] = values[index];\n }\n return array;\n}\n\nexport default arrayPush;\n","import arrayPush from './_arrayPush.js';\nimport getPrototype from './_getPrototype.js';\nimport getSymbols from './_getSymbols.js';\nimport stubArray from './stubArray.js';\n\n/* Built-in method references for those with the same name as other `lodash` methods. */\nvar nativeGetSymbols = Object.getOwnPropertySymbols;\n\n/**\n * Creates an array of the own and inherited enumerable symbols of `object`.\n *\n * @private\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of symbols.\n */\nvar getSymbolsIn = !nativeGetSymbols ? stubArray : function(object) {\n var result = [];\n while (object) {\n arrayPush(result, getSymbols(object));\n object = getPrototype(object);\n }\n return result;\n};\n\nexport default getSymbolsIn;\n","import copyObject from './_copyObject.js';\nimport getSymbolsIn from './_getSymbolsIn.js';\n\n/**\n * Copies own and inherited symbols of `source` to `object`.\n *\n * @private\n * @param {Object} source The object to copy symbols from.\n * @param {Object} [object={}] The object to copy symbols to.\n * @returns {Object} Returns `object`.\n */\nfunction copySymbolsIn(source, object) {\n return copyObject(source, getSymbolsIn(source), object);\n}\n\nexport default copySymbolsIn;\n","import arrayPush from './_arrayPush.js';\nimport isArray from './isArray.js';\n\n/**\n * The base implementation of `getAllKeys` and `getAllKeysIn` which uses\n * `keysFunc` and `symbolsFunc` to get the enumerable property names and\n * symbols of `object`.\n *\n * @private\n * @param {Object} object The object to query.\n * @param {Function} keysFunc The function to get the keys of `object`.\n * @param {Function} symbolsFunc The function to get the symbols of `object`.\n * @returns {Array} Returns the array of property names and symbols.\n */\nfunction baseGetAllKeys(object, keysFunc, symbolsFunc) {\n var result = keysFunc(object);\n return isArray(object) ? result : arrayPush(result, symbolsFunc(object));\n}\n\nexport default baseGetAllKeys;\n","import baseGetAllKeys from './_baseGetAllKeys.js';\nimport getSymbols from './_getSymbols.js';\nimport keys from './keys.js';\n\n/**\n * Creates an array of own enumerable property names and symbols of `object`.\n *\n * @private\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of property names and symbols.\n */\nfunction getAllKeys(object) {\n return baseGetAllKeys(object, keys, getSymbols);\n}\n\nexport default getAllKeys;\n","import baseGetAllKeys from './_baseGetAllKeys.js';\nimport getSymbolsIn from './_getSymbolsIn.js';\nimport keysIn from './keysIn.js';\n\n/**\n * Creates an array of own and inherited enumerable property names and\n * symbols of `object`.\n *\n * @private\n * @param {Object} object The object to query.\n * @returns {Array} Returns the array of property names and symbols.\n */\nfunction getAllKeysIn(object) {\n return baseGetAllKeys(object, keysIn, getSymbolsIn);\n}\n\nexport default getAllKeysIn;\n","import getNative from './_getNative.js';\nimport root from './_root.js';\n\n/* Built-in method references that are verified to be native. */\nvar DataView = getNative(root, 'DataView');\n\nexport default DataView;\n","import getNative from './_getNative.js';\nimport root from './_root.js';\n\n/* Built-in method references that are verified to be native. */\nvar Promise = getNative(root, 'Promise');\n\nexport default Promise;\n","import getNative from './_getNative.js';\nimport root from './_root.js';\n\n/* Built-in method references that are verified to be native. */\nvar Set = getNative(root, 'Set');\n\nexport default Set;\n","import getNative from './_getNative.js';\nimport root from './_root.js';\n\n/* Built-in method references that are verified to be native. */\nvar WeakMap = getNative(root, 'WeakMap');\n\nexport default WeakMap;\n","import DataView from './_DataView.js';\nimport Map from './_Map.js';\nimport Promise from './_Promise.js';\nimport Set from './_Set.js';\nimport WeakMap from './_WeakMap.js';\nimport baseGetTag from './_baseGetTag.js';\nimport toSource from './_toSource.js';\n\n/** `Object#toString` result references. */\nvar mapTag = '[object Map]',\n objectTag = '[object Object]',\n promiseTag = '[object Promise]',\n setTag = '[object Set]',\n weakMapTag = '[object WeakMap]';\n\nvar dataViewTag = '[object DataView]';\n\n/** Used to detect maps, sets, and weakmaps. */\nvar dataViewCtorString = toSource(DataView),\n mapCtorString = toSource(Map),\n promiseCtorString = toSource(Promise),\n setCtorString = toSource(Set),\n weakMapCtorString = toSource(WeakMap);\n\n/**\n * Gets the `toStringTag` of `value`.\n *\n * @private\n * @param {*} value The value to query.\n * @returns {string} Returns the `toStringTag`.\n */\nvar getTag = baseGetTag;\n\n// Fallback for data views, maps, sets, and weak maps in IE 11 and promises in Node.js < 6.\nif ((DataView && getTag(new DataView(new ArrayBuffer(1))) != dataViewTag) ||\n (Map && getTag(new Map) != mapTag) ||\n (Promise && getTag(Promise.resolve()) != promiseTag) ||\n (Set && getTag(new Set) != setTag) ||\n (WeakMap && getTag(new WeakMap) != weakMapTag)) {\n getTag = function(value) {\n var result = baseGetTag(value),\n Ctor = result == objectTag ? value.constructor : undefined,\n ctorString = Ctor ? toSource(Ctor) : '';\n\n if (ctorString) {\n switch (ctorString) {\n case dataViewCtorString: return dataViewTag;\n case mapCtorString: return mapTag;\n case promiseCtorString: return promiseTag;\n case setCtorString: return setTag;\n case weakMapCtorString: return weakMapTag;\n }\n }\n return result;\n };\n}\n\nexport default getTag;\n","/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * Initializes an array clone.\n *\n * @private\n * @param {Array} array The array to clone.\n * @returns {Array} Returns the initialized clone.\n */\nfunction initCloneArray(array) {\n var length = array.length,\n result = new array.constructor(length);\n\n // Add properties assigned by `RegExp#exec`.\n if (length && typeof array[0] == 'string' && hasOwnProperty.call(array, 'index')) {\n result.index = array.index;\n result.input = array.input;\n }\n return result;\n}\n\nexport default initCloneArray;\n","import root from './_root.js';\n\n/** Built-in value references. */\nvar Uint8Array = root.Uint8Array;\n\nexport default Uint8Array;\n","import Uint8Array from './_Uint8Array.js';\n\n/**\n * Creates a clone of `arrayBuffer`.\n *\n * @private\n * @param {ArrayBuffer} arrayBuffer The array buffer to clone.\n * @returns {ArrayBuffer} Returns the cloned array buffer.\n */\nfunction cloneArrayBuffer(arrayBuffer) {\n var result = new arrayBuffer.constructor(arrayBuffer.byteLength);\n new Uint8Array(result).set(new Uint8Array(arrayBuffer));\n return result;\n}\n\nexport default cloneArrayBuffer;\n","import cloneArrayBuffer from './_cloneArrayBuffer.js';\n\n/**\n * Creates a clone of `dataView`.\n *\n * @private\n * @param {Object} dataView The data view to clone.\n * @param {boolean} [isDeep] Specify a deep clone.\n * @returns {Object} Returns the cloned data view.\n */\nfunction cloneDataView(dataView, isDeep) {\n var buffer = isDeep ? cloneArrayBuffer(dataView.buffer) : dataView.buffer;\n return new dataView.constructor(buffer, dataView.byteOffset, dataView.byteLength);\n}\n\nexport default cloneDataView;\n","/** Used to match `RegExp` flags from their coerced string values. */\nvar reFlags = /\\w*$/;\n\n/**\n * Creates a clone of `regexp`.\n *\n * @private\n * @param {Object} regexp The regexp to clone.\n * @returns {Object} Returns the cloned regexp.\n */\nfunction cloneRegExp(regexp) {\n var result = new regexp.constructor(regexp.source, reFlags.exec(regexp));\n result.lastIndex = regexp.lastIndex;\n return result;\n}\n\nexport default cloneRegExp;\n","import Symbol from './_Symbol.js';\n\n/** Used to convert symbols to primitives and strings. */\nvar symbolProto = Symbol ? Symbol.prototype : undefined,\n symbolValueOf = symbolProto ? symbolProto.valueOf : undefined;\n\n/**\n * Creates a clone of the `symbol` object.\n *\n * @private\n * @param {Object} symbol The symbol object to clone.\n * @returns {Object} Returns the cloned symbol object.\n */\nfunction cloneSymbol(symbol) {\n return symbolValueOf ? Object(symbolValueOf.call(symbol)) : {};\n}\n\nexport default cloneSymbol;\n","import cloneArrayBuffer from './_cloneArrayBuffer.js';\n\n/**\n * Creates a clone of `typedArray`.\n *\n * @private\n * @param {Object} typedArray The typed array to clone.\n * @param {boolean} [isDeep] Specify a deep clone.\n * @returns {Object} Returns the cloned typed array.\n */\nfunction cloneTypedArray(typedArray, isDeep) {\n var buffer = isDeep ? cloneArrayBuffer(typedArray.buffer) : typedArray.buffer;\n return new typedArray.constructor(buffer, typedArray.byteOffset, typedArray.length);\n}\n\nexport default cloneTypedArray;\n","import cloneArrayBuffer from './_cloneArrayBuffer.js';\nimport cloneDataView from './_cloneDataView.js';\nimport cloneRegExp from './_cloneRegExp.js';\nimport cloneSymbol from './_cloneSymbol.js';\nimport cloneTypedArray from './_cloneTypedArray.js';\n\n/** `Object#toString` result references. */\nvar boolTag = '[object Boolean]',\n dateTag = '[object Date]',\n mapTag = '[object Map]',\n numberTag = '[object Number]',\n regexpTag = '[object RegExp]',\n setTag = '[object Set]',\n stringTag = '[object String]',\n symbolTag = '[object Symbol]';\n\nvar arrayBufferTag = '[object ArrayBuffer]',\n dataViewTag = '[object DataView]',\n float32Tag = '[object Float32Array]',\n float64Tag = '[object Float64Array]',\n int8Tag = '[object Int8Array]',\n int16Tag = '[object Int16Array]',\n int32Tag = '[object Int32Array]',\n uint8Tag = '[object Uint8Array]',\n uint8ClampedTag = '[object Uint8ClampedArray]',\n uint16Tag = '[object Uint16Array]',\n uint32Tag = '[object Uint32Array]';\n\n/**\n * Initializes an object clone based on its `toStringTag`.\n *\n * **Note:** This function only supports cloning values with tags of\n * `Boolean`, `Date`, `Error`, `Map`, `Number`, `RegExp`, `Set`, or `String`.\n *\n * @private\n * @param {Object} object The object to clone.\n * @param {string} tag The `toStringTag` of the object to clone.\n * @param {boolean} [isDeep] Specify a deep clone.\n * @returns {Object} Returns the initialized clone.\n */\nfunction initCloneByTag(object, tag, isDeep) {\n var Ctor = object.constructor;\n switch (tag) {\n case arrayBufferTag:\n return cloneArrayBuffer(object);\n\n case boolTag:\n case dateTag:\n return new Ctor(+object);\n\n case dataViewTag:\n return cloneDataView(object, isDeep);\n\n case float32Tag: case float64Tag:\n case int8Tag: case int16Tag: case int32Tag:\n case uint8Tag: case uint8ClampedTag: case uint16Tag: case uint32Tag:\n return cloneTypedArray(object, isDeep);\n\n case mapTag:\n return new Ctor;\n\n case numberTag:\n case stringTag:\n return new Ctor(object);\n\n case regexpTag:\n return cloneRegExp(object);\n\n case setTag:\n return new Ctor;\n\n case symbolTag:\n return cloneSymbol(object);\n }\n}\n\nexport default initCloneByTag;\n","import isObject from './isObject.js';\n\n/** Built-in value references. */\nvar objectCreate = Object.create;\n\n/**\n * The base implementation of `_.create` without support for assigning\n * properties to the created object.\n *\n * @private\n * @param {Object} proto The object to inherit from.\n * @returns {Object} Returns the new object.\n */\nvar baseCreate = (function() {\n function object() {}\n return function(proto) {\n if (!isObject(proto)) {\n return {};\n }\n if (objectCreate) {\n return objectCreate(proto);\n }\n object.prototype = proto;\n var result = new object;\n object.prototype = undefined;\n return result;\n };\n}());\n\nexport default baseCreate;\n","import baseCreate from './_baseCreate.js';\nimport getPrototype from './_getPrototype.js';\nimport isPrototype from './_isPrototype.js';\n\n/**\n * Initializes an object clone.\n *\n * @private\n * @param {Object} object The object to clone.\n * @returns {Object} Returns the initialized clone.\n */\nfunction initCloneObject(object) {\n return (typeof object.constructor == 'function' && !isPrototype(object))\n ? baseCreate(getPrototype(object))\n : {};\n}\n\nexport default initCloneObject;\n","import getTag from './_getTag.js';\nimport isObjectLike from './isObjectLike.js';\n\n/** `Object#toString` result references. */\nvar mapTag = '[object Map]';\n\n/**\n * The base implementation of `_.isMap` without Node.js optimizations.\n *\n * @private\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a map, else `false`.\n */\nfunction baseIsMap(value) {\n return isObjectLike(value) && getTag(value) == mapTag;\n}\n\nexport default baseIsMap;\n","import baseIsMap from './_baseIsMap.js';\nimport baseUnary from './_baseUnary.js';\nimport nodeUtil from './_nodeUtil.js';\n\n/* Node.js helper references. */\nvar nodeIsMap = nodeUtil && nodeUtil.isMap;\n\n/**\n * Checks if `value` is classified as a `Map` object.\n *\n * @static\n * @memberOf _\n * @since 4.3.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a map, else `false`.\n * @example\n *\n * _.isMap(new Map);\n * // => true\n *\n * _.isMap(new WeakMap);\n * // => false\n */\nvar isMap = nodeIsMap ? baseUnary(nodeIsMap) : baseIsMap;\n\nexport default isMap;\n","import getTag from './_getTag.js';\nimport isObjectLike from './isObjectLike.js';\n\n/** `Object#toString` result references. */\nvar setTag = '[object Set]';\n\n/**\n * The base implementation of `_.isSet` without Node.js optimizations.\n *\n * @private\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a set, else `false`.\n */\nfunction baseIsSet(value) {\n return isObjectLike(value) && getTag(value) == setTag;\n}\n\nexport default baseIsSet;\n","import baseIsSet from './_baseIsSet.js';\nimport baseUnary from './_baseUnary.js';\nimport nodeUtil from './_nodeUtil.js';\n\n/* Node.js helper references. */\nvar nodeIsSet = nodeUtil && nodeUtil.isSet;\n\n/**\n * Checks if `value` is classified as a `Set` object.\n *\n * @static\n * @memberOf _\n * @since 4.3.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a set, else `false`.\n * @example\n *\n * _.isSet(new Set);\n * // => true\n *\n * _.isSet(new WeakSet);\n * // => false\n */\nvar isSet = nodeIsSet ? baseUnary(nodeIsSet) : baseIsSet;\n\nexport default isSet;\n","import Stack from './_Stack.js';\nimport arrayEach from './_arrayEach.js';\nimport assignValue from './_assignValue.js';\nimport baseAssign from './_baseAssign.js';\nimport baseAssignIn from './_baseAssignIn.js';\nimport cloneBuffer from './_cloneBuffer.js';\nimport copyArray from './_copyArray.js';\nimport copySymbols from './_copySymbols.js';\nimport copySymbolsIn from './_copySymbolsIn.js';\nimport getAllKeys from './_getAllKeys.js';\nimport getAllKeysIn from './_getAllKeysIn.js';\nimport getTag from './_getTag.js';\nimport initCloneArray from './_initCloneArray.js';\nimport initCloneByTag from './_initCloneByTag.js';\nimport initCloneObject from './_initCloneObject.js';\nimport isArray from './isArray.js';\nimport isBuffer from './isBuffer.js';\nimport isMap from './isMap.js';\nimport isObject from './isObject.js';\nimport isSet from './isSet.js';\nimport keys from './keys.js';\nimport keysIn from './keysIn.js';\n\n/** Used to compose bitmasks for cloning. */\nvar CLONE_DEEP_FLAG = 1,\n CLONE_FLAT_FLAG = 2,\n CLONE_SYMBOLS_FLAG = 4;\n\n/** `Object#toString` result references. */\nvar argsTag = '[object Arguments]',\n arrayTag = '[object Array]',\n boolTag = '[object Boolean]',\n dateTag = '[object Date]',\n errorTag = '[object Error]',\n funcTag = '[object Function]',\n genTag = '[object GeneratorFunction]',\n mapTag = '[object Map]',\n numberTag = '[object Number]',\n objectTag = '[object Object]',\n regexpTag = '[object RegExp]',\n setTag = '[object Set]',\n stringTag = '[object String]',\n symbolTag = '[object Symbol]',\n weakMapTag = '[object WeakMap]';\n\nvar arrayBufferTag = '[object ArrayBuffer]',\n dataViewTag = '[object DataView]',\n float32Tag = '[object Float32Array]',\n float64Tag = '[object Float64Array]',\n int8Tag = '[object Int8Array]',\n int16Tag = '[object Int16Array]',\n int32Tag = '[object Int32Array]',\n uint8Tag = '[object Uint8Array]',\n uint8ClampedTag = '[object Uint8ClampedArray]',\n uint16Tag = '[object Uint16Array]',\n uint32Tag = '[object Uint32Array]';\n\n/** Used to identify `toStringTag` values supported by `_.clone`. */\nvar cloneableTags = {};\ncloneableTags[argsTag] = cloneableTags[arrayTag] =\ncloneableTags[arrayBufferTag] = cloneableTags[dataViewTag] =\ncloneableTags[boolTag] = cloneableTags[dateTag] =\ncloneableTags[float32Tag] = cloneableTags[float64Tag] =\ncloneableTags[int8Tag] = cloneableTags[int16Tag] =\ncloneableTags[int32Tag] = cloneableTags[mapTag] =\ncloneableTags[numberTag] = cloneableTags[objectTag] =\ncloneableTags[regexpTag] = cloneableTags[setTag] =\ncloneableTags[stringTag] = cloneableTags[symbolTag] =\ncloneableTags[uint8Tag] = cloneableTags[uint8ClampedTag] =\ncloneableTags[uint16Tag] = cloneableTags[uint32Tag] = true;\ncloneableTags[errorTag] = cloneableTags[funcTag] =\ncloneableTags[weakMapTag] = false;\n\n/**\n * The base implementation of `_.clone` and `_.cloneDeep` which tracks\n * traversed objects.\n *\n * @private\n * @param {*} value The value to clone.\n * @param {boolean} bitmask The bitmask flags.\n * 1 - Deep clone\n * 2 - Flatten inherited properties\n * 4 - Clone symbols\n * @param {Function} [customizer] The function to customize cloning.\n * @param {string} [key] The key of `value`.\n * @param {Object} [object] The parent object of `value`.\n * @param {Object} [stack] Tracks traversed objects and their clone counterparts.\n * @returns {*} Returns the cloned value.\n */\nfunction baseClone(value, bitmask, customizer, key, object, stack) {\n var result,\n isDeep = bitmask & CLONE_DEEP_FLAG,\n isFlat = bitmask & CLONE_FLAT_FLAG,\n isFull = bitmask & CLONE_SYMBOLS_FLAG;\n\n if (customizer) {\n result = object ? customizer(value, key, object, stack) : customizer(value);\n }\n if (result !== undefined) {\n return result;\n }\n if (!isObject(value)) {\n return value;\n }\n var isArr = isArray(value);\n if (isArr) {\n result = initCloneArray(value);\n if (!isDeep) {\n return copyArray(value, result);\n }\n } else {\n var tag = getTag(value),\n isFunc = tag == funcTag || tag == genTag;\n\n if (isBuffer(value)) {\n return cloneBuffer(value, isDeep);\n }\n if (tag == objectTag || tag == argsTag || (isFunc && !object)) {\n result = (isFlat || isFunc) ? {} : initCloneObject(value);\n if (!isDeep) {\n return isFlat\n ? copySymbolsIn(value, baseAssignIn(result, value))\n : copySymbols(value, baseAssign(result, value));\n }\n } else {\n if (!cloneableTags[tag]) {\n return object ? value : {};\n }\n result = initCloneByTag(value, tag, isDeep);\n }\n }\n // Check for circular references and return its corresponding clone.\n stack || (stack = new Stack);\n var stacked = stack.get(value);\n if (stacked) {\n return stacked;\n }\n stack.set(value, result);\n\n if (isSet(value)) {\n value.forEach(function(subValue) {\n result.add(baseClone(subValue, bitmask, customizer, subValue, value, stack));\n });\n } else if (isMap(value)) {\n value.forEach(function(subValue, key) {\n result.set(key, baseClone(subValue, bitmask, customizer, key, value, stack));\n });\n }\n\n var keysFunc = isFull\n ? (isFlat ? getAllKeysIn : getAllKeys)\n : (isFlat ? keysIn : keys);\n\n var props = isArr ? undefined : keysFunc(value);\n arrayEach(props || value, function(subValue, key) {\n if (props) {\n key = subValue;\n subValue = value[key];\n }\n // Recursively populate clone (susceptible to call stack limits).\n assignValue(result, key, baseClone(subValue, bitmask, customizer, key, value, stack));\n });\n return result;\n}\n\nexport default baseClone;\n","import baseClone from './_baseClone.js';\n\n/** Used to compose bitmasks for cloning. */\nvar CLONE_DEEP_FLAG = 1,\n CLONE_SYMBOLS_FLAG = 4;\n\n/**\n * This method is like `_.cloneWith` except that it recursively clones `value`.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to recursively clone.\n * @param {Function} [customizer] The function to customize cloning.\n * @returns {*} Returns the deep cloned value.\n * @see _.cloneWith\n * @example\n *\n * function customizer(value) {\n * if (_.isElement(value)) {\n * return value.cloneNode(true);\n * }\n * }\n *\n * var el = _.cloneDeepWith(document.body, customizer);\n *\n * console.log(el === document.body);\n * // => false\n * console.log(el.nodeName);\n * // => 'BODY'\n * console.log(el.childNodes.length);\n * // => 20\n */\nfunction cloneDeepWith(value, customizer) {\n customizer = typeof customizer == 'function' ? customizer : undefined;\n return baseClone(value, CLONE_DEEP_FLAG | CLONE_SYMBOLS_FLAG, customizer);\n}\n\nexport default cloneDeepWith;\n","import isObjectLike from './isObjectLike.js';\nimport isPlainObject from './isPlainObject.js';\n\n/**\n * Checks if `value` is likely a DOM element.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a DOM element, else `false`.\n * @example\n *\n * _.isElement(document.body);\n * // => true\n *\n * _.isElement('<body>');\n * // => false\n */\nfunction isElement(value) {\n return isObjectLike(value) && value.nodeType === 1 && !isPlainObject(value);\n}\n\nexport default isElement;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/config\n */\n\nimport { isPlainObject, isElement, cloneDeepWith } from 'lodash-es';\n\n/**\n * Handles a configuration dictionary.\n *\n * @typeParam Cfg A type of the configuration dictionary.\n */\nexport default class Config<Cfg> {\n\t/**\n\t * Store for the whole configuration.\n\t */\n\tprivate readonly _config: Record<string, any>;\n\n\t/**\n\t * Creates an instance of the {@link ~Config} class.\n\t *\n\t * @param configurations The initial configurations to be set. Usually, provided by the user.\n\t * @param defaultConfigurations The default configurations. Usually, provided by the system.\n\t */\n\tconstructor( configurations?: Partial<Cfg>, defaultConfigurations?: Partial<Cfg> ) {\n\t\tthis._config = {};\n\n\t\t// Set default configuration.\n\t\tif ( defaultConfigurations ) {\n\t\t\t// Clone the configuration to make sure that the properties will not be shared\n\t\t\t// between editors and make the watchdog feature work correctly.\n\t\t\tthis.define( cloneConfig( defaultConfigurations ) );\n\t\t}\n\n\t\t// Set initial configuration.\n\t\tif ( configurations ) {\n\t\t\tthis._setObjectToTarget( this._config, configurations );\n\t\t}\n\t}\n\n\t/**\n\t * Set configuration values.\n\t *\n\t * It also accepts setting a \"deep configuration\" by using dots in the name. For example, `'resize.width'` sets\n\t * the value for the `width` configuration in the `resize` subset.\n\t *\n\t * ```ts\n\t * config.set( 'resize.width', 500 );\n\t * ```\n\t *\n\t * It accepts both a name/value pair or an object, which properties and values will be used to set\n\t * configurations. See {@link #set:CONFIG_OBJECT}.\n\t *\n\t * @label KEY_VALUE\n\t * @param name The configuration name. Configuration names are case-sensitive.\n\t * @param value The configuration value.\n\t */\n\tpublic set<K extends string>( name: K, value: GetSubConfig<Cfg, K> ): void;\n\n\t/**\n\t * Set configuration values.\n\t *\n\t * It accepts an object, which properties and values will be used to set configurations.\n\t *\n\t * ```ts\n\t * config.set( {\n\t * \twidth: 500\n\t * \ttoolbar: {\n\t * \t\tcollapsed: true\n\t * \t}\n\t * } );\n\t *\n\t * // Equivalent to:\n\t * config.set( 'width', 500 );\n\t * config.set( 'toolbar.collapsed', true );\n\t * ```\n\t *\n\t * Passing an object as the value will amend the configuration, not replace it.\n\t *\n\t * ```ts\n\t * config.set( 'toolbar', {\n\t * \tcollapsed: true,\n\t * } );\n\t *\n\t * config.set( 'toolbar', {\n\t * \tcolor: 'red',\n\t * } );\n\t *\n\t * config.get( 'toolbar.collapsed' ); // true\n\t * config.get( 'toolbar.color' ); // 'red'\n\t * ```\n\t *\n\t * It accepts both a name/value pair or an object, which properties and values will be used to set\n\t * configurations. See {@link #set:KEY_VALUE}.\n\t *\n\t * @label CONFIG_OBJECT\n\t * @param config The configuration object from which take properties as\n\t * configuration entries. Configuration names are case-sensitive.\n\t */\n\tpublic set( config: Partial<Cfg> ): void;\n\n\tpublic set( name: string | Record<string, any>, value?: any ): void {\n\t\tthis._setToTarget( this._config, name, value );\n\t}\n\n\t/**\n\t * Does exactly the same as {@link #set:KEY_VALUE} with one exception passed configuration extends\n\t * existing one, but does not overwrite already defined values.\n\t *\n\t * This method is supposed to be called by plugin developers to setup plugin's configurations. It would be\n\t * rarely used for other needs.\n\t *\n\t * @label KEY_VALUE\n\t * @param name The configuration name. Configuration names are case-sensitive.\n\t * @param value The configuration value.\n\t */\n\tpublic define<K extends string>( name: K, value: GetSubConfig<Cfg, K> ): void;\n\n\t/**\n\t * Does exactly the same as {@link #set:CONFIG_OBJECT} with one exception passed configuration extends\n\t * existing one, but does not overwrite already defined values.\n\t *\n\t * This method is supposed to be called by plugin developers to setup plugin's configurations. It would be\n\t * rarely used for other needs.\n\t *\n\t * @label CONFIG_OBJECT\n\t * @param config The configuration object from which take properties as\n\t * configuration entries. Configuration names are case-sensitive.\n\t */\n\tpublic define( config: Partial<Cfg> ): void;\n\n\tpublic define( name: string | Record<string, any>, value?: any ): void {\n\t\tconst isDefine = true;\n\n\t\tthis._setToTarget( this._config, name, value, isDefine );\n\t}\n\n\t/**\n\t * Gets the value for a configuration entry.\n\t *\n\t * ```ts\n\t * config.get( 'name' );\n\t * ```\n\t *\n\t * Deep configurations can be retrieved by separating each part with a dot.\n\t *\n\t * ```ts\n\t * config.get( 'toolbar.collapsed' );\n\t * ```\n\t *\n\t * @param name The configuration name. Configuration names are case-sensitive.\n\t * @returns The configuration value or `undefined` if the configuration entry was not found.\n\t */\n\tpublic get<K extends string>( name: K ): GetSubConfig<Cfg, K> | undefined {\n\t\treturn this._getFromSource( this._config, name );\n\t}\n\n\t/**\n\t * Iterates over all top level configuration names.\n\t */\n\tpublic* names(): Iterable<string> {\n\t\tfor ( const name of Object.keys( this._config ) ) {\n\t\t\tyield name;\n\t\t}\n\t}\n\n\t/**\n\t * Saves passed configuration to the specified target (nested object).\n\t *\n\t * @param target Nested config object.\n\t * @param name The configuration name or an object from which take properties as\n\t * configuration entries. Configuration names are case-sensitive.\n\t * @param value The configuration value. Used if a name is passed.\n\t * @param isDefine Define if passed configuration should overwrite existing one.\n\t */\n\tprivate _setToTarget( target: any, name: any, value: any, isDefine: boolean = false ): void {\n\t\t// In case of an object, iterate through it and call `_setToTarget` again for each property.\n\t\tif ( isPlainObject( name ) ) {\n\t\t\tthis._setObjectToTarget( target, name, isDefine );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// The configuration name should be split into parts if it has dots. E.g. `resize.width` -> [`resize`, `width`].\n\t\tconst parts = name.split( '.' );\n\n\t\t// Take the name of the configuration out of the parts. E.g. `resize.width` -> `width`.\n\t\tname = parts.pop();\n\n\t\t// Iterate over parts to check if currently stored configuration has proper structure.\n\t\tfor ( const part of parts ) {\n\t\t\t// If there is no object for specified part then create one.\n\t\t\tif ( !isPlainObject( target[ part ] ) ) {\n\t\t\t\ttarget[ part ] = {};\n\t\t\t}\n\n\t\t\t// Nested object becomes a target.\n\t\t\ttarget = target[ part ];\n\t\t}\n\n\t\t// In case of value is an object.\n\t\tif ( isPlainObject( value ) ) {\n\t\t\t// We take care of proper config structure.\n\t\t\tif ( !isPlainObject( target[ name ] ) ) {\n\t\t\t\ttarget[ name ] = {};\n\t\t\t}\n\n\t\t\ttarget = target[ name ];\n\n\t\t\t// And iterate through this object calling `_setToTarget` again for each property.\n\t\t\tthis._setObjectToTarget( target, value, isDefine );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Do nothing if we are defining configuration for non empty name.\n\t\tif ( isDefine && typeof target[ name ] != 'undefined' ) {\n\t\t\treturn;\n\t\t}\n\n\t\ttarget[ name ] = value;\n\t}\n\n\t/**\n\t * Get specified configuration from specified source (nested object).\n\t *\n\t * @param source level of nested object.\n\t * @param name The configuration name. Configuration names are case-sensitive.\n\t * @returns The configuration value or `undefined` if the configuration entry was not found.\n\t */\n\tprivate _getFromSource( source: any, name: string ): any {\n\t\t// The configuration name should be split into parts if it has dots. E.g. `resize.width` -> [`resize`, `width`].\n\t\tconst parts = name.split( '.' );\n\n\t\t// Take the name of the configuration out of the parts. E.g. `resize.width` -> `width`.\n\t\tname = parts.pop()!;\n\n\t\t// Iterate over parts to check if currently stored configuration has proper structure.\n\t\tfor ( const part of parts ) {\n\t\t\tif ( !isPlainObject( source[ part ] ) ) {\n\t\t\t\tsource = null;\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\t// Nested object becomes a source.\n\t\t\tsource = source[ part ];\n\t\t}\n\n\t\t// Always returns undefined for non existing configuration.\n\t\treturn source ? cloneConfig( source[ name ] ) : undefined;\n\t}\n\n\t/**\n\t * Iterates through passed object and calls {@link #_setToTarget} method with object key and value for each property.\n\t *\n\t * @param target Nested config object.\n\t * @param configuration Configuration data set\n\t * @param isDefine Defines if passed configuration is default configuration or not.\n\t */\n\tprivate _setObjectToTarget( target: any, configuration: any, isDefine?: boolean ): void {\n\t\tObject.keys( configuration ).forEach( key => {\n\t\t\tthis._setToTarget( target, key, configuration[ key ], isDefine );\n\t\t} );\n\t}\n}\n\n/**\n * Clones configuration object or value.\n */\nfunction cloneConfig<T>( source: T ): T {\n\treturn cloneDeepWith( source, leaveDOMReferences );\n}\n\n/**\n * A customized function for cloneDeepWith.\n * It will leave references to DOM Elements instead of cloning them.\n */\nfunction leaveDOMReferences( value: unknown ): unknown {\n\treturn isElement( value ) ? value : undefined;\n}\n\n/**\n * An utility type excluding primitive values and arrays from the union.\n */\nexport type OnlyObject<T> = Exclude<T, undefined | null | string | number | boolean | Array<any>>;\n\n/**\n * An utility type extracting configuration value from the given name.\n *\n * @typeParam T The type of a configuration dictionary.\n * @typeParam K The literal type of configuration name (dot-separated path).\n */\nexport type GetSubConfig<T, K> = K extends keyof T ?\n\tT[ K ] :\n\tK extends `${ infer K1 }.${ infer K2 }` ?\n\t\tK1 extends keyof T ?\n\t\t\tGetSubConfig<OnlyObject<T[ K1 ]>, K2> :\n\t\t\tunknown :\n\t\tunknown;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/isnode\n */\n\n/**\n * Checks if the object is a native DOM Node.\n */\nexport default function isNode( obj: any ): obj is Node {\n\tif ( obj ) {\n\t\tif ( obj.defaultView ) {\n\t\t\treturn obj instanceof obj.defaultView.Document;\n\t\t} else if ( obj.ownerDocument && obj.ownerDocument.defaultView ) {\n\t\t\treturn obj instanceof obj.ownerDocument.defaultView.Node;\n\t\t}\n\t}\n\n\treturn false;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/iswindow\n */\n\n/**\n * Checks if the object is a native DOM Window.\n */\nexport default function isWindow( obj: unknown ): obj is Window {\n\tconst stringifiedObject = Object.prototype.toString.apply( obj );\n\n\t// Returns `true` for the `window` object in browser environments.\n\tif ( stringifiedObject == '[object Window]' ) {\n\t\treturn true;\n\t}\n\n\t// Returns `true` for the `window` object in the Electron environment.\n\tif ( stringifiedObject == '[object global]' ) {\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/emittermixin\n */\n\nimport EmitterMixin, {\n\t_getEmitterListenedTo,\n\t_setEmitterId,\n\ttype Emitter,\n\ttype CallbackOptions,\n\ttype BaseEvent,\n\ttype GetCallback\n} from '../emittermixin';\nimport uid from '../uid';\nimport isNode from './isnode';\nimport isWindow from './iswindow';\nimport type EventInfo from '../eventinfo';\nimport type { Constructor, Mixed } from '../mix';\n\nconst defaultEmitterClass = DomEmitterMixin( EmitterMixin() );\n\n/**\n * Mixin that injects the DOM events API into its host. It provides the API\n * compatible with {@link module:utils/emittermixin~Emitter}.\n *\n * This function creates a class that inherits from the provided `base` and implements `Emitter` interface.\n *\n * DOM emitter mixin is by default available in the {@link module:ui/view~View} class,\n * but it can also be mixed into any other class:\n *\n * ```ts\n * import DomEmitterMixin from '../utils/dom/emittermixin.js';\n *\n * class BaseClass { ... }\n *\n * class SomeView extends DomEmitterMixin( BaseClass ) {}\n *\n * const view = new SomeView();\n * view.listenTo( domElement, ( evt, domEvt ) => {\n * \tconsole.log( evt, domEvt );\n * } );\n * ```\n *\n * @label EXTENDS\n */\nexport default function DomEmitterMixin<Base extends Constructor<Emitter>>( base: Base ): Mixed<Base, DomEmitter>;\n\n/**\n * Mixin that injects the DOM events API into its host. It provides the API\n * compatible with {@link module:utils/emittermixin~Emitter}.\n *\n * This function creates a class that implements `Emitter` interface.\n *\n * DOM emitter mixin is by default available in the {@link module:ui/view~View} class,\n * but it can also be mixed into any other class:\n *\n * ```ts\n * import DomEmitterMixin from '../utils/dom/emittermixin.js';\n *\n * class SomeView extends DomEmitterMixin() {}\n *\n * const view = new SomeView();\n * view.listenTo( domElement, ( evt, domEvt ) => {\n * \tconsole.log( evt, domEvt );\n * } );\n * ```\n *\n * @label NO_ARGUMENTS\n */\nexport default function DomEmitterMixin(): {\n\tnew (): DomEmitter;\n\tprototype: DomEmitter;\n};\n\nexport default function DomEmitterMixin( base?: Constructor<Emitter> ): unknown {\n\tif ( !base ) {\n\t\treturn defaultEmitterClass;\n\t}\n\n\tabstract class Mixin extends base implements DomEmitter {\n\t\tpublic override listenTo<K extends keyof HTMLElementEventMap>(\n\t\t\temitter: Node | Window,\n\t\t\tevent: K,\n\t\t\tcallback: ( this: this, ev: EventInfo, event: HTMLElementEventMap[ K ] ) => void,\n\t\t\toptions?: CallbackOptions & { readonly useCapture?: boolean; readonly usePassive?: boolean }\n\t\t): void;\n\t\tpublic override listenTo<TEvent extends BaseEvent>(\n\t\t\temitter: Emitter,\n\t\t\tevent: TEvent[ 'name' ],\n\t\t\tcallback: GetCallback<TEvent>,\n\t\t\toptions?: CallbackOptions\n\t\t): void;\n\t\tpublic override listenTo(\n\t\t\temitter: Emitter | Node | Window,\n\t\t\tevent: string,\n\t\t\tcallback: ( ev: EventInfo, ...args: Array<any> ) => void,\n\t\t\toptions: CallbackOptions & { readonly useCapture?: boolean; readonly usePassive?: boolean } = {}\n\t\t): void {\n\t\t\t// Check if emitter is an instance of DOM Node. If so, use corresponding ProxyEmitter (or create one if not existing).\n\t\t\tif ( isNode( emitter ) || isWindow( emitter ) ) {\n\t\t\t\tconst proxyOptions = {\n\t\t\t\t\tcapture: !!options.useCapture,\n\t\t\t\t\tpassive: !!options.usePassive\n\t\t\t\t};\n\n\t\t\t\tconst proxyEmitter = this._getProxyEmitter( emitter, proxyOptions ) || new ProxyEmitter( emitter, proxyOptions );\n\n\t\t\t\tthis.listenTo( proxyEmitter, event, callback, options );\n\t\t\t} else {\n\t\t\t\t// Execute parent class method with Emitter (or ProxyEmitter) instance.\n\t\t\t\tsuper.listenTo( emitter, event, callback, options );\n\t\t\t}\n\t\t}\n\n\t\tpublic override stopListening(\n\t\t\temitter?: Emitter | Node | Window,\n\t\t\tevent?: string,\n\t\t\tcallback?: Function\n\t\t): void {\n\t\t\t// Check if the emitter is an instance of DOM Node. If so, forward the call to the corresponding ProxyEmitters.\n\t\t\tif ( isNode( emitter ) || isWindow( emitter ) ) {\n\t\t\t\tconst proxyEmitters = this._getAllProxyEmitters( emitter );\n\n\t\t\t\tfor ( const proxy of proxyEmitters ) {\n\t\t\t\t\tthis.stopListening( proxy, event, callback );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t// Execute parent class method with Emitter (or ProxyEmitter) instance.\n\t\t\t\tsuper.stopListening( emitter, event, callback );\n\t\t\t}\n\t\t}\n\n\t\t/**\n\t\t * Retrieves ProxyEmitter instance for given DOM Node residing in this Host and given options.\n\t\t *\n\t\t * @param node DOM Node of the ProxyEmitter.\n\t\t * @param options Additional options.\n\t\t * @param options.useCapture Indicates that events of this type will be dispatched to the registered\n\t\t * listener before being dispatched to any EventTarget beneath it in the DOM tree.\n\t\t * @param options.usePassive Indicates that the function specified by listener will never call preventDefault()\n\t\t * and prevents blocking browser's main thread by this event handler.\n\t\t * @returns ProxyEmitter instance bound to the DOM Node.\n\t\t */\n\t\tprivate _getProxyEmitter(\n\t\t\tnode: Node | Window,\n\t\t\toptions: { capture: boolean; passive: boolean }\n\t\t): Emitter | null {\n\t\t\treturn _getEmitterListenedTo( this, getProxyEmitterId( node, options ) );\n\t\t}\n\n\t\t/**\n\t\t * Retrieves all the ProxyEmitter instances for given DOM Node residing in this Host.\n\t\t *\n\t\t * @param node DOM Node of the ProxyEmitter.\n\t\t */\n\t\tprivate _getAllProxyEmitters( node: Node | Window ): Array<ProxyEmitter> {\n\t\t\treturn [\n\t\t\t\t{ capture: false, passive: false },\n\t\t\t\t{ capture: false, passive: true },\n\t\t\t\t{ capture: true, passive: false },\n\t\t\t\t{ capture: true, passive: true }\n\t\t\t].map( options => this._getProxyEmitter( node, options ) ).filter( proxy => !!proxy ) as any;\n\t\t}\n\t}\n\n\treturn Mixin;\n}\n\n// Backward compatibility with `mix`\n( [\n\t'_getProxyEmitter', '_getAllProxyEmitters',\n\t'on', 'once', 'off', 'listenTo',\n\t'stopListening', 'fire', 'delegate', 'stopDelegating',\n\t'_addEventListener', '_removeEventListener'\n] ).forEach( key => {\n\t( DomEmitterMixin as any )[ key ] = ( defaultEmitterClass.prototype as any )[ key ];\n} );\n\n/**\n * Creates a ProxyEmitter instance. Such an instance is a bridge between a DOM Node firing events\n * and any Host listening to them. It is backwards compatible with {@link module:utils/emittermixin~Emitter#on}.\n * There is a separate instance for each combination of modes (useCapture & usePassive). The mode is concatenated with\n * UID stored in HTMLElement to give each instance unique identifier.\n *\n * listenTo( click, ... )\n * +-----------------------------------------+\n * | stopListening( ... ) |\n * +----------------------------+ | addEventListener( click, ... )\n * | Host | | +---------------------------------------------+\n * +----------------------------+ | | removeEventListener( click, ... ) |\n * | _listeningTo: { | +----------v-------------+ |\n * | UID+mode: { | | ProxyEmitter | |\n * | emitter: ProxyEmitter, | +------------------------+ +------------v----------+\n * | callbacks: { | | events: { | | Node (HTMLElement) |\n * | click: [ callbacks ] | | click: [ callbacks ] | +-----------------------+\n * | } | | }, | | data-ck-expando: UID |\n * | } | | _domNode: Node, | +-----------------------+\n * | } | | _domListeners: {}, | |\n * | +------------------------+ | | _emitterId: UID+mode | |\n * | | DomEmitterMixin | | +--------------^---------+ |\n * | +------------------------+ | | | |\n * +--------------^-------------+ | +---------------------------------------------+\n * | | click (DOM Event)\n * +-----------------------------------------+\n * fire( click, DOM Event )\n */\nclass ProxyEmitter extends EmitterMixin() {\n\tprivate readonly _domNode: Node | Window;\n\tprivate readonly _options: { capture: boolean; passive: boolean };\n\n\t/**\n\t * @param node DOM Node that fires events.\n\t * @param options Additional options.\n\t * @param options.useCapture Indicates that events of this type will be dispatched to the registered\n\t * listener before being dispatched to any EventTarget beneath it in the DOM tree.\n\t * @param options.usePassive Indicates that the function specified by listener will never call preventDefault()\n\t * and prevents blocking browser's main thread by this event handler.\n\t */\n\tconstructor(\n\t\tnode: Node | Window,\n\t\toptions: { capture: boolean; passive: boolean }\n\t) {\n\t\tsuper();\n\n\t\t// Set emitter ID to match DOM Node \"expando\" property.\n\t\t_setEmitterId( this, getProxyEmitterId( node, options ) );\n\n\t\t// Remember the DOM Node this ProxyEmitter is bound to.\n\t\tthis._domNode = node;\n\n\t\t// And given options.\n\t\tthis._options = options;\n\t}\n\n\t/**\n\t * Collection of native DOM listeners.\n\t */\n\tprivate _domListeners?: {\n\t\t[ event: string ]: {\n\t\t\t( domEvent: unknown ): void;\n\t\t\tremoveListener(): void;\n\t\t};\n\t};\n\n\t/**\n\t * Registers a callback function to be executed when an event is fired.\n\t *\n\t * It attaches a native DOM listener to the DOM Node. When fired,\n\t * a corresponding Emitter event will also fire with DOM Event object as an argument.\n\t *\n\t * **Note**: This is automatically called by the\n\t * {@link module:utils/emittermixin~Emitter#listenTo `Emitter#listenTo()`}.\n\t *\n\t * @param event The name of the event.\n\t */\n\tpublic attach( event: string ): void {\n\t\t// If the DOM Listener for given event already exist it is pointless\n\t\t// to attach another one.\n\t\tif ( this._domListeners && this._domListeners[ event ] ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst domListener = this._createDomListener( event );\n\n\t\t// Attach the native DOM listener to DOM Node.\n\t\tthis._domNode.addEventListener( event, domListener, this._options );\n\n\t\tif ( !this._domListeners ) {\n\t\t\tthis._domListeners = {};\n\t\t}\n\n\t\t// Store the native DOM listener in this ProxyEmitter. It will be helpful\n\t\t// when stopping listening to the event.\n\t\tthis._domListeners[ event ] = domListener;\n\t}\n\n\t/**\n\t * Stops executing the callback on the given event.\n\t *\n\t * **Note**: This is automatically called by the\n\t * {@link module:utils/emittermixin~Emitter#stopListening `Emitter#stopListening()`}.\n\t *\n\t * @param event The name of the event.\n\t */\n\tpublic detach( event: string ): void {\n\t\tlet events;\n\n\t\t// Remove native DOM listeners which are orphans. If no callbacks\n\t\t// are awaiting given event, detach native DOM listener from DOM Node.\n\t\t// See: {@link attach}.\n\n\t\tif ( this._domListeners![ event ] && ( !( events = ( this as any )._events![ event ] ) || !events.callbacks.length ) ) {\n\t\t\tthis._domListeners![ event ].removeListener();\n\t\t}\n\t}\n\n\t/**\n\t * Adds callback to emitter for given event.\n\t *\n\t * @internal\n\t * @param event The name of the event.\n\t * @param callback The function to be called on event.\n\t * @param options Additional options.\n\t */\n\tpublic _addEventListener<TEvent extends BaseEvent>(\n\t\tevent: TEvent[ 'name' ],\n\t\tcallback: GetCallback<TEvent>,\n\t\toptions: CallbackOptions\n\t): void {\n\t\tthis.attach( event );\n\t\t( EmitterMixin().prototype as any )._addEventListener.call( this, event, callback, options );\n\t}\n\n\t/**\n\t * Removes callback from emitter for given event.\n\t *\n\t * @internal\n\t * @param event The name of the event.\n\t * @param callback The function to stop being called.\n\t */\n\tpublic _removeEventListener( event: string, callback: Function ) {\n\t\t( EmitterMixin().prototype as any )._removeEventListener.call( this, event, callback );\n\t\tthis.detach( event );\n\t}\n\n\t/**\n\t * Creates a native DOM listener callback. When the native DOM event\n\t * is fired it will fire corresponding event on this ProxyEmitter.\n\t * Note: A native DOM Event is passed as an argument.\n\t *\n\t * @param event The name of the event.\n\t * @returns The DOM listener callback.\n\t */\n\tprivate _createDomListener( event: string ) {\n\t\tconst domListener = ( domEvt: unknown ) => {\n\t\t\tthis.fire( event, domEvt );\n\t\t};\n\n\t\t// Supply the DOM listener callback with a function that will help\n\t\t// detach it from the DOM Node, when it is no longer necessary.\n\t\t// See: {@link detach}.\n\t\tdomListener.removeListener = () => {\n\t\t\tthis._domNode.removeEventListener( event, domListener, this._options );\n\t\t\tdelete this._domListeners![ event ];\n\t\t};\n\n\t\treturn domListener;\n\t}\n}\n\n/**\n * Gets an unique DOM Node identifier. The identifier will be set if not defined.\n *\n * @returns UID for given DOM Node.\n */\nfunction getNodeUID( node: any ): string {\n\treturn node[ 'data-ck-expando' ] || ( node[ 'data-ck-expando' ] = uid() );\n}\n\n/**\n * Gets id of the ProxyEmitter for the given node.\n */\nfunction getProxyEmitterId( node: Node | Window, options: { [ option: string ]: any } ): string {\n\tlet id = getNodeUID( node );\n\n\tfor ( const option of Object.keys( options ).sort() ) {\n\t\tif ( options[ option ] ) {\n\t\t\tid += '-' + option;\n\t\t}\n\t}\n\n\treturn id;\n}\n\nexport interface DomEventMap extends HTMLElementEventMap, WindowEventMap {\n}\n\n/**\n * Interface representing classes which mix in {@link module:utils/dom/emittermixin~DomEmitterMixin}.\n *\n * Can be easily implemented by a class by mixing the {@link module:utils/dom/emittermixin~DomEmitterMixin} mixin.\n *\n * ```ts\n * class MyClass extends DomEmitterMixin( OtherBaseClass ) {\n * \t// This class now implements the `Emitter` interface.\n * }\n * ```\n */\nexport interface DomEmitter extends Emitter {\n\n\t/**\n\t * Registers a callback function to be executed when an event is fired in a specific Emitter or DOM Node.\n\t * It is backwards compatible with {@link module:utils/emittermixin~Emitter#listenTo}.\n\t *\n\t * @label HTML_EMITTER\n\t * @param emitter The object that fires the event.\n\t * @param event The name of the event.\n\t * @param callback The function to be called on event.\n\t * @param options Additional options.\n\t * @param options.useCapture Indicates that events of this type will be dispatched to the registered\n\t * listener before being dispatched to any EventTarget beneath it in the DOM tree.\n\t * @param options.usePassive Indicates that the function specified by listener will never call preventDefault()\n\t * and prevents blocking browser's main thread by this event handler.\n\t */\n\tlistenTo<K extends keyof DomEventMap>(\n\t\temitter: Node | Window,\n\t\tevent: K,\n\t\tcallback: ( this: this, ev: EventInfo, event: DomEventMap[ K ] ) => void,\n\t\toptions?: CallbackOptions & { readonly useCapture?: boolean; readonly usePassive?: boolean }\n\t): void;\n\n\t/**\n\t * Registers a callback function to be executed when an event is fired in a specific (emitter) object.\n\t *\n\t * Events can be grouped in namespaces using `:`.\n\t * When namespaced event is fired, it additionally fires all callbacks for that namespace.\n\t *\n\t * ```ts\n\t * // myEmitter.on( ... ) is a shorthand for myEmitter.listenTo( myEmitter, ... ).\n\t * myEmitter.on( 'myGroup', genericCallback );\n\t * myEmitter.on( 'myGroup:myEvent', specificCallback );\n\t *\n\t * // genericCallback is fired.\n\t * myEmitter.fire( 'myGroup' );\n\t * // both genericCallback and specificCallback are fired.\n\t * myEmitter.fire( 'myGroup:myEvent' );\n\t * // genericCallback is fired even though there are no callbacks for \"foo\".\n\t * myEmitter.fire( 'myGroup:foo' );\n\t * ```\n\t *\n\t * An event callback can {@link module:utils/eventinfo~EventInfo#stop stop the event} and\n\t * set the {@link module:utils/eventinfo~EventInfo#return return value} of the {@link #fire} method.\n\t *\n\t * @label DOM_EMITTER\n\t * @typeParam TEvent The type describing the event. See {@link module:utils/emittermixin~BaseEvent}.\n\t * @param emitter The object that fires the event.\n\t * @param event The name of the event.\n\t * @param callback The function to be called on event.\n\t * @param options Additional options.\n\t */\n\tlistenTo<TEvent extends BaseEvent>(\n\t\temitter: Emitter,\n\t\tevent: TEvent[ 'name' ],\n\t\tcallback: GetCallback<TEvent>,\n\t\toptions?: CallbackOptions\n\t): void;\n\n\t/**\n\t * Stops listening for events. It can be used at different levels:\n\t * It is backwards compatible with {@link module:utils/emittermixin~Emitter#listenTo}.\n\t *\n\t * * To stop listening to a specific callback.\n\t * * To stop listening to a specific event.\n\t * * To stop listening to all events fired by a specific object.\n\t * * To stop listening to all events fired by all objects.\n\t *\n\t * @label DOM_STOP\n\t * @param emitter The object to stop listening to.\n\t * If omitted, stops it for all objects.\n\t * @param event (Requires the `emitter`) The name of the event to stop listening to. If omitted, stops it\n\t * for all events from `emitter`.\n\t * @param callback (Requires the `event`) The function to be removed from the call list for the given\n\t * `event`.\n\t */\n\tstopListening( emitter?: Emitter | Node | Window, event?: string, callback?: Function ): void;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals window, document */\n\n/**\n * @module utils/dom/global\n */\n\n// This interface exists to make our API pages more readable.\n/**\n * A helper (module) giving an access to the global DOM objects such as `window` and `document`.\n */\nexport interface GlobalType {\n\treadonly window: Window & typeof globalThis;\n\treadonly document: Document;\n}\n\n/**\n * A helper (module) giving an access to the global DOM objects such as `window` and\n * `document`. Accessing these objects using this helper allows easy and bulletproof\n * testing, i.e. stubbing native properties:\n *\n * ```ts\n * import { global } from 'ckeditor5/utils';\n *\n * // This stub will work for any code using global module.\n * testUtils.sinon.stub( global, 'window', {\n * \tinnerWidth: 10000\n * } );\n *\n * console.log( global.window.innerWidth );\n * ```\n */\nlet globalVar: GlobalType; // named globalVar instead of global: https://github.com/ckeditor/ckeditor5/issues/12971\n\n// In some environments window and document API might not be available.\ntry {\n\tglobalVar = { window, document };\n} catch ( e ) {\n\t// It's not possible to mock a window object to simulate lack of a window object without writing extremely convoluted code.\n\t/* istanbul ignore next -- @preserve */\n\n\t// Let's cast it to not change module's API.\n\t// We only handle this so loading editor in environments without window and document doesn't fail.\n\t// For better DX we shouldn't introduce mixed types and require developers to check the type manually.\n\t// This module should not be used on purpose in any environment outside browser.\n\tglobalVar = { window: {} as any, document: {} as any };\n}\n\nexport default globalVar;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/istext\n */\n\n/**\n * Checks if the object is a native DOM Text node.\n */\nexport default function isText( obj: unknown ): obj is Text {\n\treturn Object.prototype.toString.call( obj ) == '[object Text]';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/isrange\n */\n\n/**\n * Checks if the object is a native DOM Range.\n */\nexport default function isRange( obj: unknown ): obj is Range {\n\treturn Object.prototype.toString.apply( obj ) == '[object Range]';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/getborderwidths\n */\n\n/**\n * Returns an object containing CSS border widths of a specified HTML element.\n *\n * @param element An element which has CSS borders.\n * @returns An object containing `top`, `left`, `right` and `bottom` properties\n * with numerical values of the `border-[top,left,right,bottom]-width` CSS styles.\n */\nexport default function getBorderWidths( element: HTMLElement ): BorderWidths {\n\t// Call getComputedStyle on the window the element document belongs to.\n\tconst style = element.ownerDocument.defaultView!.getComputedStyle( element );\n\n\treturn {\n\t\ttop: parseInt( style.borderTopWidth, 10 ),\n\t\tright: parseInt( style.borderRightWidth, 10 ),\n\t\tbottom: parseInt( style.borderBottomWidth, 10 ),\n\t\tleft: parseInt( style.borderLeftWidth, 10 )\n\t};\n}\n\n/**\n * An object describing widths of `HTMLElement` borders.\n*/\nexport interface BorderWidths {\n\ttop: number;\n\tright: number;\n\tbottom: number;\n\tleft: number;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/rect\n */\n\nimport isRange from './isrange';\nimport isWindow from './iswindow';\nimport getBorderWidths from './getborderwidths';\nimport isText from './istext';\n\nconst rectProperties: Array<keyof RectLike> = [ 'top', 'right', 'bottom', 'left', 'width', 'height' ];\n\n/**\n * A helper class representing a `ClientRect` object, e.g. value returned by\n * the native `object.getBoundingClientRect()` method. Provides a set of methods\n * to manipulate the rect and compare it against other rect instances.\n */\nexport default class Rect {\n\t/**\n\t * The \"top\" value of the rect.\n\t *\n\t * @readonly\n\t */\n\tpublic top!: number;\n\n\t/**\n\t * The \"right\" value of the rect.\n\t *\n\t * @readonly\n\t */\n\tpublic right!: number;\n\n\t/**\n\t * The \"bottom\" value of the rect.\n\t *\n\t * @readonly\n\t */\n\tpublic bottom!: number;\n\n\t/**\n\t * The \"left\" value of the rect.\n\t *\n\t * @readonly\n\t */\n\tpublic left!: number;\n\n\t/**\n\t * The \"width\" value of the rect.\n\t *\n\t * @readonly\n\t */\n\tpublic width!: number;\n\n\t/**\n\t * The \"height\" value of the rect.\n\t *\n\t * @readonly\n\t */\n\tpublic height!: number;\n\n\t/**\n\t * The object this rect is for.\n\t *\n\t * @readonly\n\t */\n\tprivate _source!: RectSource;\n\n\t/**\n\t * Creates an instance of rect.\n\t *\n\t * ```ts\n\t * // Rect of an HTMLElement.\n\t * const rectA = new Rect( document.body );\n\t *\n\t * // Rect of a DOM Range.\n\t * const rectB = new Rect( document.getSelection().getRangeAt( 0 ) );\n\t *\n\t * // Rect of a window (web browser viewport).\n\t * const rectC = new Rect( window );\n\t *\n\t * // Rect out of an object.\n\t * const rectD = new Rect( { top: 0, right: 10, bottom: 10, left: 0, width: 10, height: 10 } );\n\t *\n\t * // Rect out of another Rect instance.\n\t * const rectE = new Rect( rectD );\n\t *\n\t * // Rect out of a ClientRect.\n\t * const rectF = new Rect( document.body.getClientRects().item( 0 ) );\n\t * ```\n\t *\n\t * **Note**: By default a rect of an HTML element includes its CSS borders and scrollbars (if any)\n\t * ant the rect of a `window` includes scrollbars too. Use {@link #excludeScrollbarsAndBorders}\n\t * to get the inner part of the rect.\n\t *\n\t * @param source A source object to create the rect.\n\t */\n\tconstructor( source: RectSource ) {\n\t\tconst isSourceRange = isRange( source );\n\n\t\tObject.defineProperty( this, '_source', {\n\t\t\t// If the source is a Rect instance, copy it's #_source.\n\t\t\tvalue: ( source as any )._source || source,\n\t\t\twritable: true,\n\t\t\tenumerable: false\n\t\t} );\n\n\t\tif ( isDomElement( source ) || isSourceRange ) {\n\t\t\t// The `Rect` class depends on `getBoundingClientRect` and `getClientRects` DOM methods. If the source\n\t\t\t// of a rect in an HTML element or a DOM range but it does not belong to any rendered DOM tree, these methods\n\t\t\t// will fail to obtain the geometry and the rect instance makes little sense to the features using it.\n\t\t\t// To get rid of this warning make sure the source passed to the constructor is a descendant of `window.document.body`.\n\t\t\t// @if CK_DEBUG // const sourceNode = isSourceRange ? source.startContainer : source;\n\t\t\t// @if CK_DEBUG // if ( !sourceNode.ownerDocument || !sourceNode.ownerDocument.body.contains( sourceNode ) ) {\n\t\t\t// @if CK_DEBUG // \tconsole.warn(\n\t\t\t// @if CK_DEBUG // \t\t'rect-source-not-in-dom: The source of this rect does not belong to any rendered DOM tree.',\n\t\t\t// @if CK_DEBUG // \t\t{ source } );\n\t\t\t// @if CK_DEBUG // }\n\n\t\t\tif ( isSourceRange ) {\n\t\t\t\tconst rangeRects = Rect.getDomRangeRects( source );\n\t\t\t\tcopyRectProperties( this, Rect.getBoundingRect( rangeRects )! );\n\t\t\t} else {\n\t\t\t\tcopyRectProperties( this, source.getBoundingClientRect() );\n\t\t\t}\n\t\t} else if ( isWindow( source ) ) {\n\t\t\tconst { innerWidth, innerHeight } = source;\n\n\t\t\tcopyRectProperties( this, {\n\t\t\t\ttop: 0,\n\t\t\t\tright: innerWidth,\n\t\t\t\tbottom: innerHeight,\n\t\t\t\tleft: 0,\n\t\t\t\twidth: innerWidth,\n\t\t\t\theight: innerHeight\n\t\t\t} );\n\t\t} else {\n\t\t\tcopyRectProperties( this, source );\n\t\t}\n\t}\n\n\t/**\n\t * Returns a clone of the rect.\n\t *\n\t * @returns A cloned rect.\n\t */\n\tpublic clone(): Rect {\n\t\treturn new Rect( this );\n\t}\n\n\t/**\n\t * Moves the rect so that its upperleft corner lands in desired `[ x, y ]` location.\n\t *\n\t * @param x Desired horizontal location.\n\t * @param y Desired vertical location.\n\t * @returns A rect which has been moved.\n\t */\n\tpublic moveTo( x: number, y: number ): this {\n\t\tthis.top = y;\n\t\tthis.right = x + this.width;\n\t\tthis.bottom = y + this.height;\n\t\tthis.left = x;\n\n\t\treturn this;\n\t}\n\n\t/**\n\t * Moves the rect inplace by a dedicated offset.\n\t *\n\t * @param x A horizontal offset.\n\t * @param y A vertical offset\n\t * @returns A rect which has been moved.\n\t */\n\tpublic moveBy( x: number, y: number ): this {\n\t\tthis.top += y;\n\t\tthis.right += x;\n\t\tthis.left += x;\n\t\tthis.bottom += y;\n\n\t\treturn this;\n\t}\n\n\t/**\n\t * Returns a new rect a a result of intersection with another rect.\n\t */\n\tpublic getIntersection( anotherRect: Rect ): Rect | null {\n\t\tconst rect = {\n\t\t\ttop: Math.max( this.top, anotherRect.top ),\n\t\t\tright: Math.min( this.right, anotherRect.right ),\n\t\t\tbottom: Math.min( this.bottom, anotherRect.bottom ),\n\t\t\tleft: Math.max( this.left, anotherRect.left ),\n\t\t\twidth: 0,\n\t\t\theight: 0\n\t\t};\n\n\t\trect.width = rect.right - rect.left;\n\t\trect.height = rect.bottom - rect.top;\n\n\t\tif ( rect.width < 0 || rect.height < 0 ) {\n\t\t\treturn null;\n\t\t} else {\n\t\t\treturn new Rect( rect );\n\t\t}\n\t}\n\n\t/**\n\t * Returns the area of intersection with another rect.\n\t *\n\t * @returns Area of intersection.\n\t */\n\tpublic getIntersectionArea( anotherRect: Rect ): number {\n\t\tconst rect = this.getIntersection( anotherRect );\n\n\t\tif ( rect ) {\n\t\t\treturn rect.getArea();\n\t\t} else {\n\t\t\treturn 0;\n\t\t}\n\t}\n\n\t/**\n\t * Returns the area of the rect.\n\t */\n\tpublic getArea(): number {\n\t\treturn this.width * this.height;\n\t}\n\n\t/**\n\t * Returns a new rect, a part of the original rect, which is actually visible to the user,\n\t * e.g. an original rect cropped by parent element rects which have `overflow` set in CSS\n\t * other than `\"visible\"`.\n\t *\n\t * If there's no such visible rect, which is when the rect is limited by one or many of\n\t * the ancestors, `null` is returned.\n\t *\n\t * **Note**: This method does not consider the boundaries of the viewport (window).\n\t * To get a rect cropped by all ancestors and the viewport, use an intersection such as:\n\t *\n\t * ```ts\n\t * const visibleInViewportRect = new Rect( window ).getIntersection( new Rect( source ).getVisible() );\n\t * ```\n\t *\n\t * @returns A visible rect instance or `null`, if there's none.\n\t */\n\tpublic getVisible(): Rect | null {\n\t\tconst source: RectSource & { parentNode?: Node | null; commonAncestorContainer?: Node | null } = this._source;\n\n\t\tlet visibleRect = this.clone();\n\n\t\t// There's no ancestor to crop <body> with the overflow.\n\t\tif ( isBody( source ) ) {\n\t\t\treturn visibleRect;\n\t\t}\n\n\t\tlet child: any = source;\n\t\tlet parent = source.parentNode || source.commonAncestorContainer;\n\t\tlet absolutelyPositionedChildElement;\n\n\t\t// Check the ancestors all the way up to the <body>.\n\t\twhile ( parent && !isBody( parent ) ) {\n\t\t\tif ( child instanceof HTMLElement && getElementPosition( child ) === 'absolute' ) {\n\t\t\t\tabsolutelyPositionedChildElement = child;\n\t\t\t}\n\n\t\t\t// The child will be cropped only if it has `position: absolute` and the parent has `position: relative` + some overflow.\n\t\t\t// Otherwise there's no chance of visual clipping and the parent can be skipped\n\t\t\t// https://github.com/ckeditor/ckeditor5/issues/14107.\n\t\t\tif (\n\t\t\t\tabsolutelyPositionedChildElement &&\n\t\t\t\t( getElementPosition( parent as HTMLElement ) !== 'relative' || getElementOverflow( parent as HTMLElement ) === 'visible' )\n\t\t\t) {\n\t\t\t\tchild = parent;\n\t\t\t\tparent = parent.parentNode;\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst parentRect = new Rect( parent as HTMLElement );\n\t\t\tconst intersectionRect = visibleRect.getIntersection( parentRect );\n\n\t\t\tif ( intersectionRect ) {\n\t\t\t\tif ( intersectionRect.getArea() < visibleRect.getArea() ) {\n\t\t\t\t\t// Reduce the visible rect to the intersection.\n\t\t\t\t\tvisibleRect = intersectionRect;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t// There's no intersection, the rect is completely invisible.\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tchild = parent;\n\t\t\tparent = parent.parentNode;\n\t\t}\n\n\t\treturn visibleRect;\n\t}\n\n\t/**\n\t * Checks if all property values ({@link #top}, {@link #left}, {@link #right},\n\t * {@link #bottom}, {@link #width} and {@link #height}) are the equal in both rect\n\t * instances.\n\t *\n\t * @param anotherRect A rect instance to compare with.\n\t * @returns `true` when Rects are equal. `false` otherwise.\n\t */\n\tpublic isEqual( anotherRect: Rect ): boolean {\n\t\tfor ( const prop of rectProperties ) {\n\t\t\tif ( this[ prop ] !== anotherRect[ prop ] ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Checks whether a rect fully contains another rect instance.\n\t *\n\t * @param anotherRect\n\t * @returns `true` if contains, `false` otherwise.\n\t */\n\tpublic contains( anotherRect: Rect ): boolean {\n\t\tconst intersectRect = this.getIntersection( anotherRect );\n\n\t\treturn !!( intersectRect && intersectRect.isEqual( anotherRect ) );\n\t}\n\n\t/**\n\t * Excludes scrollbars and CSS borders from the rect.\n\t *\n\t * * Borders are removed when {@link #_source} is an HTML element.\n\t * * Scrollbars are excluded from HTML elements and the `window`.\n\t *\n\t * @returns A rect which has been updated.\n\t */\n\tpublic excludeScrollbarsAndBorders(): this {\n\t\tconst source = this._source as ( HTMLElement | Window );\n\t\tlet scrollBarWidth, scrollBarHeight, direction;\n\n\t\tif ( isWindow( source ) ) {\n\t\t\tscrollBarWidth = source.innerWidth - source.document.documentElement.clientWidth;\n\t\t\tscrollBarHeight = source.innerHeight - source.document.documentElement.clientHeight;\n\t\t\tdirection = source.getComputedStyle( source.document.documentElement ).direction;\n\t\t} else {\n\t\t\tconst borderWidths = getBorderWidths( source );\n\n\t\t\tscrollBarWidth = source.offsetWidth - source.clientWidth - borderWidths.left - borderWidths.right;\n\t\t\tscrollBarHeight = source.offsetHeight - source.clientHeight - borderWidths.top - borderWidths.bottom;\n\t\t\tdirection = source.ownerDocument.defaultView!.getComputedStyle( source ).direction;\n\n\t\t\tthis.left += borderWidths.left;\n\t\t\tthis.top += borderWidths.top;\n\t\t\tthis.right -= borderWidths.right;\n\t\t\tthis.bottom -= borderWidths.bottom;\n\t\t\tthis.width = this.right - this.left;\n\t\t\tthis.height = this.bottom - this.top;\n\t\t}\n\n\t\tthis.width -= scrollBarWidth;\n\n\t\tif ( direction === 'ltr' ) {\n\t\t\tthis.right -= scrollBarWidth;\n\t\t} else {\n\t\t\tthis.left += scrollBarWidth;\n\t\t}\n\n\t\tthis.height -= scrollBarHeight;\n\t\tthis.bottom -= scrollBarHeight;\n\n\t\treturn this;\n\t}\n\n\t/**\n\t * Returns an array of rects of the given native DOM Range.\n\t *\n\t * @param range A native DOM range.\n\t * @returns DOM Range rects.\n\t */\n\tpublic static getDomRangeRects( range: Range ): Array<Rect> {\n\t\tconst rects: Array<Rect> = [];\n\t\t// Safari does not iterate over ClientRectList using for...of loop.\n\t\tconst clientRects = Array.from( range.getClientRects() );\n\n\t\tif ( clientRects.length ) {\n\t\t\tfor ( const rect of clientRects ) {\n\t\t\t\trects.push( new Rect( rect ) );\n\t\t\t}\n\t\t}\n\t\t// If there's no client rects for the Range, use parent container's bounding rect\n\t\t// instead and adjust rect's width to simulate the actual geometry of such range.\n\t\t// https://github.com/ckeditor/ckeditor5-utils/issues/153\n\t\t// https://github.com/ckeditor/ckeditor5-ui/issues/317\n\t\telse {\n\t\t\tlet startContainer = range.startContainer;\n\n\t\t\tif ( isText( startContainer ) ) {\n\t\t\t\tstartContainer = startContainer.parentNode!;\n\t\t\t}\n\n\t\t\tconst rect = new Rect( ( startContainer as Element ).getBoundingClientRect() );\n\t\t\trect.right = rect.left;\n\t\t\trect.width = 0;\n\n\t\t\trects.push( rect );\n\t\t}\n\n\t\treturn rects;\n\t}\n\n\t/**\n\t * Returns a bounding rectangle that contains all the given `rects`.\n\t *\n\t * @param rects A list of rectangles that should be contained in the result rectangle.\n\t * @returns Bounding rectangle or `null` if no `rects` were given.\n\t */\n\tpublic static getBoundingRect( rects: Iterable<Rect> ): Rect | null {\n\t\tconst boundingRectData = {\n\t\t\tleft: Number.POSITIVE_INFINITY,\n\t\t\ttop: Number.POSITIVE_INFINITY,\n\t\t\tright: Number.NEGATIVE_INFINITY,\n\t\t\tbottom: Number.NEGATIVE_INFINITY,\n\t\t\twidth: 0,\n\t\t\theight: 0\n\t\t};\n\t\tlet rectangleCount = 0;\n\n\t\tfor ( const rect of rects ) {\n\t\t\trectangleCount++;\n\n\t\t\tboundingRectData.left = Math.min( boundingRectData.left, rect.left );\n\t\t\tboundingRectData.top = Math.min( boundingRectData.top, rect.top );\n\t\t\tboundingRectData.right = Math.max( boundingRectData.right, rect.right );\n\t\t\tboundingRectData.bottom = Math.max( boundingRectData.bottom, rect.bottom );\n\t\t}\n\n\t\tif ( rectangleCount == 0 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tboundingRectData.width = boundingRectData.right - boundingRectData.left;\n\t\tboundingRectData.height = boundingRectData.bottom - boundingRectData.top;\n\n\t\treturn new Rect( boundingRectData );\n\t}\n}\n\n/**\n * A source of {@link module:utils/dom/rect~Rect}.\n */\nexport type RectSource = HTMLElement | Range | Window | RectLike;\n\n/**\n * An object that describes properties of `ClientRect` object.\n */\nexport interface RectLike {\n\treadonly top: number;\n\treadonly right: number;\n\treadonly bottom: number;\n\treadonly left: number;\n\treadonly width: number;\n\treadonly height: number;\n}\n\n/**\n * Acquires all the rect properties from the passed source.\n */\nfunction copyRectProperties( rect: Rect, source: RectLike ): void {\n\tfor ( const p of rectProperties ) {\n\t\trect[ p ] = source[ p ];\n\t}\n}\n\n/**\n * Checks if provided object is a <body> HTML element.\n */\nfunction isBody( value: unknown ): value is HTMLBodyElement {\n\tif ( !isDomElement( value ) ) {\n\t\treturn false;\n\t}\n\n\treturn value === value.ownerDocument.body;\n}\n\n/**\n * Checks if provided object \"looks like\" a DOM Element and has API required by `Rect` class.\n */\nfunction isDomElement( value: any ): value is Element {\n\t// Note: earlier we used `isElement()` from lodash library, however that function is less performant because\n\t// it makes complicated checks to make sure that given value is a DOM element.\n\treturn value !== null && typeof value === 'object' && value.nodeType === 1 && typeof value.getBoundingClientRect === 'function';\n}\n\n/**\n * Returns the value of the `position` style of an `HTMLElement`.\n */\nfunction getElementPosition( element: HTMLElement ): string {\n\treturn element.ownerDocument.defaultView!.getComputedStyle( element ).position;\n}\n\n/**\n * Returns the value of the `overflow` style of an `HTMLElement`.\n */\nfunction getElementOverflow( element: HTMLElement ): string {\n\treturn element.ownerDocument.defaultView!.getComputedStyle( element ).overflow;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/resizeobserver\n */\n\nimport global from './global';\n\n/**\n * A helper class which instances allow performing custom actions when native DOM elements are resized.\n *\n * ```ts\n * const editableElement = editor.editing.view.getDomRoot();\n *\n * const observer = new ResizeObserver( editableElement, entry => {\n * \tconsole.log( 'The editable element has been resized in DOM.' );\n * \tconsole.log( entry.target ); // -> editableElement\n * \tconsole.log( entry.contentRect.width ); // -> e.g. '423px'\n * } );\n * ```\n *\n * It uses the [native DOM resize observer](https://developer.mozilla.org/en-US/docs/Web/API/ResizeObserver)\n * under the hood.\n */\nexport default class ResizeObserver {\n\t/**\n\t * The element observed by this observer.\n\t */\n\tprivate readonly _element: Element;\n\n\t/**\n\t * The callback executed each time {@link #_element} is resized.\n\t */\n\tprivate readonly _callback: ( entry: ResizeObserverEntry ) => void;\n\n\t/**\n\t * The single native observer instance shared across all {@link module:utils/dom/resizeobserver~ResizeObserver} instances.\n\t */\n\tprivate static _observerInstance: InstanceType<typeof global.window.ResizeObserver> | null = null;\n\n\t/**\n\t * A mapping of native DOM elements and their callbacks shared across all\n\t * {@link module:utils/dom/resizeobserver~ResizeObserver} instances.\n\t */\n\tprivate static _elementCallbacks: Map<Element, Set<( entry: ResizeObserverEntry ) => void>> | null = null;\n\n\t/**\n\t * Creates an instance of the `ResizeObserver` class.\n\t *\n\t * @param element A DOM element that is to be observed for resizing. Note that\n\t * the element must be visible (i.e. not detached from DOM) for the observer to work.\n\t * @param callback A function called when the observed element was resized. It passes\n\t * the [`ResizeObserverEntry`](https://developer.mozilla.org/en-US/docs/Web/API/ResizeObserverEntry)\n\t * object with information about the resize event.\n\t */\n\tconstructor( element: Element, callback: ( entry: ResizeObserverEntry ) => void ) {\n\t\t// **Note**: For the maximum performance, this class ensures only a single instance of the native\n\t\t// observer is used no matter how many instances of this class were created.\n\t\tif ( !ResizeObserver._observerInstance ) {\n\t\t\tResizeObserver._createObserver();\n\t\t}\n\n\t\tthis._element = element;\n\t\tthis._callback = callback;\n\n\t\tResizeObserver._addElementCallback( element, callback );\n\t\tResizeObserver._observerInstance!.observe( element );\n\t}\n\n\t/**\n\t * The element observed by this observer.\n\t */\n\tpublic get element(): Element {\n\t\treturn this._element;\n\t}\n\n\t/**\n\t * Destroys the observer which disables the `callback` passed to the {@link #constructor}.\n\t */\n\tpublic destroy(): void {\n\t\tResizeObserver._deleteElementCallback( this._element, this._callback );\n\t}\n\n\t/**\n\t * Registers a new resize callback for the DOM element.\n\t */\n\tprivate static _addElementCallback( element: Element, callback: ( entry: ResizeObserverEntry ) => void ): void {\n\t\tif ( !ResizeObserver._elementCallbacks ) {\n\t\t\tResizeObserver._elementCallbacks = new Map();\n\t\t}\n\n\t\tlet callbacks = ResizeObserver._elementCallbacks.get( element );\n\n\t\tif ( !callbacks ) {\n\t\t\tcallbacks = new Set();\n\t\t\tResizeObserver._elementCallbacks.set( element, callbacks );\n\t\t}\n\n\t\tcallbacks.add( callback );\n\t}\n\n\t/**\n\t * Removes a resize callback from the DOM element. If no callbacks are left\n\t * for the element, it removes the element from the native observer.\n\t */\n\tprivate static _deleteElementCallback( element: Element, callback: ( entry: ResizeObserverEntry ) => void ): void {\n\t\tconst callbacks = ResizeObserver._getElementCallbacks( element );\n\n\t\t// Remove the element callback. Check if exist first in case someone\n\t\t// called destroy() twice.\n\t\tif ( callbacks ) {\n\t\t\tcallbacks.delete( callback );\n\n\t\t\t// If no callbacks left for the element, also remove the element.\n\t\t\tif ( !callbacks.size ) {\n\t\t\t\tResizeObserver._elementCallbacks!.delete( element );\n\t\t\t\tResizeObserver._observerInstance!.unobserve( element );\n\t\t\t}\n\t\t}\n\n\t\tif ( ResizeObserver._elementCallbacks && !ResizeObserver._elementCallbacks.size ) {\n\t\t\tResizeObserver._observerInstance = null;\n\t\t\tResizeObserver._elementCallbacks = null;\n\t\t}\n\t}\n\n\t/**\n\t * Returns are registered resize callbacks for the DOM element.\n\t */\n\tprivate static _getElementCallbacks( element: Element ): Set<( entry: ResizeObserverEntry ) => void> | null | undefined {\n\t\tif ( !ResizeObserver._elementCallbacks ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn ResizeObserver._elementCallbacks.get( element );\n\t}\n\n\t/**\n\t * Creates the single native observer shared across all `ResizeObserver` instances.\n\t */\n\tprivate static _createObserver(): void {\n\t\tResizeObserver._observerInstance = new global.window.ResizeObserver( entries => {\n\t\t\tfor ( const entry of entries ) {\n\t\t\t\tconst callbacks = ResizeObserver._getElementCallbacks( entry.target );\n\n\t\t\t\tif ( callbacks ) {\n\t\t\t\t\tfor ( const callback of callbacks ) {\n\t\t\t\t\t\tcallback( entry );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/setdatainelement\n */\n\n/* globals HTMLTextAreaElement */\n\n/**\n * Sets data in a given element.\n *\n * @param el The element in which the data will be set.\n * @param data The data string.\n */\nexport default function setDataInElement( el: HTMLElement, data: string ): void {\n\tif ( el instanceof HTMLTextAreaElement ) {\n\t\tel.value = data;\n\t}\n\n\tel.innerHTML = data;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/tounit\n */\n\n/**\n * Returns a helper function, which adds a desired trailing\n * `unit` to the passed value.\n *\n * @param unit An unit like \"px\" or \"em\".\n */\nexport default function toUnit( unit: string ): ToUnitHelper {\n\treturn value => value + unit;\n}\n\n/**\n * A function, which adds a predefined trailing `unit`\n * to the passed `value`.\n *\n * @param value A value to be given the unit.\n * @returns A value with the trailing unit.\n */\nexport type ToUnitHelper = ( value: string | number ) => string;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/indexof\n */\n\n/**\n * Returns index of the node in the parent element.\n *\n * @param node Node which index is tested.\n * @returns Index of the node in the parent element. Returns 0 if node has no parent.\n */\nexport default function indexOf( node: Node ): number {\n\tlet index = 0;\n\n\twhile ( node.previousSibling ) {\n\t\tnode = node.previousSibling;\n\t\tindex++;\n\t}\n\n\treturn index;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/insertat\n */\n\n/**\n * Inserts node to the parent at given index.\n *\n * @param parentElement Parent element.\n * @param index Insertions index.\n * @param nodeToInsert Node to insert.\n */\nexport default function insertAt(\n\tparentElement: Element,\n\tindex: number,\n\tnodeToInsert: Node\n): void {\n\tparentElement.insertBefore( nodeToInsert, parentElement.childNodes[ index ] || null );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals Node */\n\n/**\n * @module utils/dom/iscomment\n */\n\n/**\n * Checks whether the object is a native DOM Comment node.\n */\nexport default function isComment( obj: any ): obj is Comment {\n\treturn obj && obj.nodeType === Node.COMMENT_NODE;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/isvisible\n */\n\n/**\n * Checks whether the element is visible to the user in DOM:\n *\n * * connected to the root of the document,\n * * has no `display: none`,\n * * has no ancestors with `display: none`.\n *\n * **Note**: This helper does not check whether the element is hidden by cropping, overflow, etc..\n * To check that, use {@link module:utils/dom/rect~Rect} instead.\n */\nexport default function isVisible( element: HTMLElement | null | undefined ): boolean {\n\treturn !!( element && element.getClientRects && element.getClientRects().length );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/position\n */\n\nimport global from './global';\nimport Rect, { type RectSource } from './rect';\nimport getPositionedAncestor from './getpositionedancestor';\nimport getBorderWidths from './getborderwidths';\nimport { isFunction } from 'lodash-es';\n\n// @if CK_DEBUG_POSITION // const RectDrawer = require( '@ckeditor/ckeditor5-utils/tests/_utils/rectdrawer' ).default\n\n/**\n * Calculates the `position: absolute` coordinates of a given element so it can be positioned with respect to the\n * target in the visually most efficient way, taking various restrictions like viewport or limiter geometry\n * into consideration.\n *\n * ```ts\n * // The element which is to be positioned.\n * const element = document.body.querySelector( '#toolbar' );\n *\n * // A target to which the element is positioned relatively.\n * const target = document.body.querySelector( '#container' );\n *\n * // Finding the optimal coordinates for the positioning.\n * const { left, top, name } = getOptimalPosition( {\n * \telement: element,\n * \ttarget: target,\n *\n * \t// The algorithm will chose among these positions to meet the requirements such\n * \t// as \"limiter\" element or \"fitInViewport\", set below. The positions are considered\n * \t// in the order of the array.\n * \tpositions: [\n * \t\t//\n * \t \t//\t[ Target ]\n * \t\t//\t+-----------------+\n * \t\t//\t| Element |\n * \t\t//\t+-----------------+\n * \t\t//\n * \t\ttargetRect => ( {\n * \t\t\ttop: targetRect.bottom,\n * \t\t\tleft: targetRect.left,\n * \t\t\tname: 'mySouthEastPosition'\n * \t\t} ),\n *\n * \t\t//\n * \t\t//\t+-----------------+\n * \t\t//\t| Element |\n * \t\t//\t+-----------------+\n * \t\t//\t[ Target ]\n * \t\t//\n * \t\t( targetRect, elementRect ) => ( {\n * \t\t\ttop: targetRect.top - elementRect.height,\n * \t\t\tleft: targetRect.left,\n * \t\t\tname: 'myNorthEastPosition'\n * \t\t} )\n * \t],\n *\n * \t// Find a position such guarantees the element remains within visible boundaries of <body>.\n * \tlimiter: document.body,\n *\n * \t// Find a position such guarantees the element remains within visible boundaries of the browser viewport.\n * \tfitInViewport: true\n * } );\n *\n * // The best position which fits into document.body and the viewport. May be useful\n * // to set proper class on the `element`.\n * console.log( name ); // -> \"myNorthEastPosition\"\n *\n * // Using the absolute coordinates which has been found to position the element\n * // as in the diagram depicting the \"myNorthEastPosition\" position.\n * element.style.top = top;\n * element.style.left = left;\n * ```\n *\n * @param options The input data and configuration of the helper.\n */\nexport function getOptimalPosition( { element, target, positions, limiter, fitInViewport, viewportOffsetConfig }: Options ): Position {\n\t// If the {@link module:utils/dom/position~Options#target} is a function, use what it returns.\n\t// https://github.com/ckeditor/ckeditor5-utils/issues/157\n\tif ( isFunction( target ) ) {\n\t\ttarget = target();\n\t}\n\n\t// If the {@link module:utils/dom/position~Options#limiter} is a function, use what it returns.\n\t// https://github.com/ckeditor/ckeditor5-ui/issues/260\n\tif ( isFunction( limiter ) ) {\n\t\tlimiter = limiter();\n\t}\n\n\tconst positionedElementAncestor = getPositionedAncestor( element );\n\tconst elementRect = new Rect( element );\n\tconst targetRect = new Rect( target );\n\n\tlet bestPosition: Position;\n\n\t// @if CK_DEBUG_POSITION // RectDrawer.clear();\n\t// @if CK_DEBUG_POSITION // RectDrawer.draw( targetRect, { outlineWidth: '5px' }, 'Target' );\n\n\tconst viewportRect = fitInViewport && getConstrainedViewportRect( viewportOffsetConfig ) || null;\n\tconst positionOptions = { targetRect, elementRect, positionedElementAncestor, viewportRect };\n\n\t// If there are no limits, just grab the very first position and be done with that drama.\n\tif ( !limiter && !fitInViewport ) {\n\t\tbestPosition = new PositionObject( positions[ 0 ], positionOptions );\n\t} else {\n\t\tconst limiterRect = limiter && new Rect( limiter ).getVisible();\n\n\t\t// @if CK_DEBUG_POSITION // if ( viewportRect ) {\n\t\t// @if CK_DEBUG_POSITION //\t\tRectDrawer.draw( viewportRect, { outlineWidth: '5px' }, 'Viewport' );\n\t\t// @if CK_DEBUG_POSITION // }\n\n\t\t// @if CK_DEBUG_POSITION // if ( limiter ) {\n\t\t// @if CK_DEBUG_POSITION // \tRectDrawer.draw( limiterRect, { outlineWidth: '5px', outlineColor: 'green' }, 'Visible limiter' );\n\t\t// @if CK_DEBUG_POSITION // }\n\n\t\tObject.assign( positionOptions, { limiterRect, viewportRect } );\n\n\t\t// If there's no best position found, i.e. when all intersections have no area because\n\t\t// rects have no width or height, then just use the first available position.\n\t\tbestPosition = getBestPosition( positions, positionOptions ) || new PositionObject( positions[ 0 ], positionOptions );\n\t}\n\n\treturn bestPosition;\n}\n\n/**\n * Returns a viewport `Rect` shrunk by the viewport offset config from all sides.\n */\nfunction getConstrainedViewportRect( viewportOffsetConfig: Options[ 'viewportOffsetConfig' ] ): Rect {\n\tviewportOffsetConfig = Object.assign( { top: 0, bottom: 0, left: 0, right: 0 }, viewportOffsetConfig );\n\n\tconst viewportRect = new Rect( global.window );\n\n\tviewportRect.top += viewportOffsetConfig.top!;\n\tviewportRect.height -= viewportOffsetConfig.top!;\n\tviewportRect.bottom -= viewportOffsetConfig.bottom!;\n\tviewportRect.height -= viewportOffsetConfig.bottom!;\n\n\treturn viewportRect;\n}\n\n/**\n * For a given array of positioning functions, returns such that provides the best\n * fit of the `elementRect` into the `limiterRect` and `viewportRect`.\n */\nfunction getBestPosition(\n\tpositions: ReadonlyArray<PositioningFunction>,\n\toptions: ConstructorParameters<typeof PositionObject>[ 1 ]\n): Position | null {\n\tconst { elementRect } = options;\n\n\t// This is when element is fully visible.\n\tconst elementRectArea = elementRect.getArea();\n\n\tconst positionInstances = positions\n\t\t.map( positioningFunction => new PositionObject( positioningFunction, options ) )\n\t\t// Some positioning functions may return `null` if they don't want to participate.\n\t\t.filter( position => !!position.name );\n\n\tlet maxFitFactor = 0;\n\tlet bestPosition = null;\n\n\tfor ( const position of positionInstances ) {\n\t\tconst { limiterIntersectionArea, viewportIntersectionArea } = position;\n\n\t\t// If a such position is found that element is fully contained by the limiter then, obviously,\n\t\t// there will be no better one, so finishing.\n\t\tif ( limiterIntersectionArea === elementRectArea ) {\n\t\t\treturn position;\n\t\t}\n\n\t\t// To maximize both viewport and limiter intersection areas we use distance on _viewportIntersectionArea\n\t\t// and _limiterIntersectionArea plane (without sqrt because we are looking for max value).\n\t\tconst fitFactor = viewportIntersectionArea ** 2 + limiterIntersectionArea ** 2;\n\n\t\tif ( fitFactor > maxFitFactor ) {\n\t\t\tmaxFitFactor = fitFactor;\n\t\t\tbestPosition = position;\n\t\t}\n\t}\n\n\treturn bestPosition;\n}\n\n/**\n * For a given absolute Rect coordinates object and a positioned element ancestor, it updates its\n * coordinates that make up for the position and the scroll of the ancestor.\n *\n * This is necessary because while Rects (and DOMRects) are relative to the browser's viewport, their coordinates\n * are used in reallife to position elements with `position: absolute`, which are scoped by any positioned\n * (and scrollable) ancestors.\n */\nfunction shiftRectToCompensatePositionedAncestor( rect: Rect, positionedElementAncestor: HTMLElement ): void {\n\tconst ancestorPosition = getRectForAbsolutePositioning( new Rect( positionedElementAncestor ) );\n\tconst ancestorBorderWidths = getBorderWidths( positionedElementAncestor );\n\n\tlet moveX = 0;\n\tlet moveY = 0;\n\n\t// (https://github.com/ckeditor/ckeditor5-ui-default/issues/126)\n\t// If there's some positioned ancestor of the panel, then its `Rect` must be taken into\n\t// consideration. `Rect` is always relative to the viewport while `position: absolute` works\n\t// with respect to that positioned ancestor.\n\tmoveX -= ancestorPosition.left;\n\tmoveY -= ancestorPosition.top;\n\n\t// (https://github.com/ckeditor/ckeditor5-utils/issues/139)\n\t// If there's some positioned ancestor of the panel, not only its position must be taken into\n\t// consideration (see above) but also its internal scrolls. Scroll have an impact here because `Rect`\n\t// is relative to the viewport (it doesn't care about scrolling), while `position: absolute`\n\t// must compensate that scrolling.\n\tmoveX += positionedElementAncestor.scrollLeft;\n\tmoveY += positionedElementAncestor.scrollTop;\n\n\t// (https://github.com/ckeditor/ckeditor5-utils/issues/139)\n\t// If there's some positioned ancestor of the panel, then its `Rect` includes its CSS `borderWidth`\n\t// while `position: absolute` positioning does not consider it.\n\t// E.g. `{ position: absolute, top: 0, left: 0 }` means upper left corner of the element,\n\t// not upper-left corner of its border.\n\tmoveX -= ancestorBorderWidths.left;\n\tmoveY -= ancestorBorderWidths.top;\n\n\trect.moveBy( moveX, moveY );\n}\n\n/**\n * DOMRect (also Rect) works in a scrollindependent geometry but `position: absolute` doesn't.\n * This function converts Rect to `position: absolute` coordinates.\n */\nfunction getRectForAbsolutePositioning( rect: Rect ): Rect {\n\tconst { scrollX, scrollY } = global.window;\n\n\treturn rect.clone().moveBy( scrollX, scrollY );\n}\n\n/**\n * A position object which instances are created and used by the {@link module:utils/dom/position~getOptimalPosition} helper.\n *\n * {@link module:utils/dom/position~Position#top} and {@link module:utils/dom/position~Position#left} properties of the position instance\n * translate directly to the `top` and `left` properties in CSS \"`position: absolute` coordinate system\". If set on the positioned element\n * in DOM, they will make it display it in the right place in the viewport.\n */\nexport interface Position {\n\n\t/**\n\t * Position name.\n\t */\n\treadonly name?: string;\n\n\t/**\n\t * Additional position configuration, as passed from the {@link module:utils/dom/position~PositioningFunction positioning function}.\n\t *\n\t * This object can be use, for instance, to pass through presentation options used by the consumer of the\n\t * {@link module:utils/dom/position~getOptimalPosition} helper.\n\t */\n\treadonly config?: object;\n\n\t/**\n\t * The left value in pixels in the CSS `position: absolute` coordinate system.\n\t * Set it on the positioned element in DOM to move it to the position.\n\t */\n\treadonly left: number;\n\n\t/**\n\t * The top value in pixels in the CSS `position: absolute` coordinate system.\n\t * Set it on the positioned element in DOM to move it to the position.\n\t */\n\treadonly top: number;\n}\n\n/**\n * A position class which instances are created and used by the {@link module:utils/dom/position~getOptimalPosition} helper.\n *\n * {@link module:utils/dom/position~Position#top} and {@link module:utils/dom/position~Position#left} properties of the position instance\n * translate directly to the `top` and `left` properties in CSS \"`position: absolute` coordinate system\". If set on the positioned element\n * in DOM, they will make it display it in the right place in the viewport.\n */\nclass PositionObject implements Position {\n\tpublic name?: string;\n\tpublic config?: object;\n\n\tprivate _positioningFunctionCorrdinates!: { left: number; top: number };\n\tprivate _options!: ConstructorParameters<typeof PositionObject>[ 1 ];\n\tprivate _cachedRect?: Rect;\n\tprivate _cachedAbsoluteRect?: Rect;\n\n\t/**\n\t * Creates an instance of the {@link module:utils/dom/position~PositionObject} class.\n\t *\n\t * @param positioningFunction function The function that defines the expected\n\t * coordinates the positioned element should move to.\n\t * @param options options object.\n\t * @param options.elementRect The positioned element rect.\n\t * @param options.targetRect The target element rect.\n\t * @param options.viewportRect The viewport rect.\n\t * @param options.limiterRect The limiter rect.\n\t * @param options.positionedElementAncestor Nearest element ancestor element which CSS position is not \"static\".\n\t */\n\tconstructor(\n\t\tpositioningFunction: PositioningFunction,\n\t\toptions: {\n\t\t\treadonly elementRect: Rect;\n\t\t\treadonly targetRect: Rect;\n\t\t\treadonly viewportRect: Rect | null;\n\t\t\treadonly positionedElementAncestor?: HTMLElement | null;\n\t\t\treadonly limiterRect?: Rect;\n\t\t}\n\t) {\n\t\tconst positioningFunctionOutput = positioningFunction( options.targetRect, options.elementRect, options.viewportRect );\n\n\t\t// Nameless position for a function that didn't participate.\n\t\tif ( !positioningFunctionOutput ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst { left, top, name, config } = positioningFunctionOutput;\n\n\t\tthis.name = name;\n\t\tthis.config = config;\n\n\t\tthis._positioningFunctionCorrdinates = { left, top };\n\t\tthis._options = options;\n\t}\n\n\t/**\n\t * The left value in pixels in the CSS `position: absolute` coordinate system.\n\t * Set it on the positioned element in DOM to move it to the position.\n\t */\n\tpublic get left(): number {\n\t\treturn this._absoluteRect.left;\n\t}\n\n\t/**\n\t * The top value in pixels in the CSS `position: absolute` coordinate system.\n\t * Set it on the positioned element in DOM to move it to the position.\n\t */\n\tpublic get top(): number {\n\t\treturn this._absoluteRect.top;\n\t}\n\n\t/**\n\t * An intersection area between positioned element and limiter within viewport constraints.\n\t */\n\tpublic get limiterIntersectionArea(): number {\n\t\tconst limiterRect = this._options.limiterRect;\n\n\t\tif ( limiterRect ) {\n\t\t\tconst viewportRect = this._options.viewportRect;\n\n\t\t\tif ( viewportRect ) {\n\t\t\t\t// Consider only the part of the limiter which is visible in the viewport. So the limiter is getting limited.\n\t\t\t\tconst limiterViewportIntersectRect = limiterRect.getIntersection( viewportRect );\n\n\t\t\t\tif ( limiterViewportIntersectRect ) {\n\t\t\t\t\t// If the limiter is within the viewport, then check the intersection between that part of the\n\t\t\t\t\t// limiter and actual position.\n\t\t\t\t\treturn limiterViewportIntersectRect.getIntersectionArea( this._rect );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\treturn limiterRect.getIntersectionArea( this._rect );\n\t\t\t}\n\t\t}\n\n\t\treturn 0;\n\t}\n\n\t/**\n\t * An intersection area between positioned element and viewport.\n\t */\n\tpublic get viewportIntersectionArea(): number {\n\t\tconst viewportRect = this._options.viewportRect;\n\n\t\tif ( viewportRect ) {\n\t\t\treturn viewportRect.getIntersectionArea( this._rect );\n\t\t}\n\n\t\treturn 0;\n\t}\n\n\t/**\n\t * An already positioned element rect. A clone of the element rect passed to the constructor\n\t * but placed in the viewport according to the positioning function.\n\t */\n\tprivate get _rect(): Rect {\n\t\tif ( this._cachedRect ) {\n\t\t\treturn this._cachedRect;\n\t\t}\n\n\t\tthis._cachedRect = this._options.elementRect.clone().moveTo(\n\t\t\tthis._positioningFunctionCorrdinates.left,\n\t\t\tthis._positioningFunctionCorrdinates.top\n\t\t);\n\n\t\treturn this._cachedRect;\n\t}\n\n\t/**\n\t * An already absolutely positioned element rect. See ({@link #_rect}).\n\t */\n\tprivate get _absoluteRect(): Rect {\n\t\tif ( this._cachedAbsoluteRect ) {\n\t\t\treturn this._cachedAbsoluteRect;\n\t\t}\n\n\t\tthis._cachedAbsoluteRect = getRectForAbsolutePositioning( this._rect );\n\n\t\tif ( this._options.positionedElementAncestor ) {\n\t\t\tshiftRectToCompensatePositionedAncestor( this._cachedAbsoluteRect, this._options.positionedElementAncestor );\n\t\t}\n\n\t\treturn this._cachedAbsoluteRect!;\n\t}\n}\n\n/**\n * The `getOptimalPosition()` helper options.\n */\nexport interface Options {\n\n\t/**\n\t * Element that is to be positioned.\n\t */\n\treadonly element: HTMLElement;\n\n\t/**\n\t * Target with respect to which the `element` is to be positioned.\n\t */\n\treadonly target: RectSource | ( () => RectSource );\n\n\t/**\n\t * An array of positioning functions.\n\t *\n\t * **Note**: Positioning functions are processed in the order of preference. The first function that works\n\t * in the current environment (e.g. offers the complete fit in the viewport geometry) will be picked by\n\t * `getOptimalPosition()`.\n\t *\n\t * **Note**: Any positioning function returning `null` is ignored.\n\t */\n\treadonly positions: ReadonlyArray<PositioningFunction>;\n\n\t/**\n\t * When set, the algorithm will chose position which fits the most in the\n\t * limiter's bounding rect.\n\t */\n\treadonly limiter?: RectSource | ( () => ( RectSource | null ) ) | null;\n\n\t/**\n\t * When set, the algorithm will chose such a position which fits `element`\n\t * the most inside visible viewport.\n\t */\n\treadonly fitInViewport?: boolean;\n\n\t/**\n\t * Viewport offset config object. It restricts the visible viewport available to the `getOptimalPosition()` from each side.\n\t *\n\t * ```ts\n\t * {\n\t * \ttop: 50,\n\t * \tright: 50,\n\t * \tbottom: 50,\n\t * \tleft: 50\n\t * }\n\t * ```\n\t */\n\treadonly viewportOffsetConfig?: {\n\t\treadonly top?: number;\n\t\treadonly right?: number;\n\t\treadonly bottom?: number;\n\t\treadonly left?: number;\n\t};\n}\n\n/**\n * A positioning function which, based on positioned element and target {@link module:utils/dom/rect~Rect Rects}, returns rect coordinates\n * representing the geometrical relation between them. Used by the {@link module:utils/dom/position~getOptimalPosition} helper.\n *\n * ```ts\n * // This simple position will place the element directly under the target, in the middle:\n * //\n * //\t [ Target ]\n * //\t+-----------------+\n * //\t| Element |\n * //\t+-----------------+\n * //\n * const position = ( targetRect, elementRect, [ viewportRect ] ) => ( {\n * \ttop: targetRect.bottom,\n * \tleft: targetRect.left + targetRect.width / 2 - elementRect.width / 2,\n * \tname: 'bottomMiddle',\n *\n * \t// Note: The config is optional.\n * \tconfig: {\n * \t\tzIndex: '999'\n * \t}\n * } );\n * ```\n *\n * @param elementRect The rect of the element to be positioned.\n * @param targetRect The rect of the target the element (its rect) is relatively positioned to.\n * @param viewportRect The rect of the visual browser viewport.\n * @returns When the function returns `null`, it will not be considered by {@link module:utils/dom/position~getOptimalPosition}.\n */\nexport type PositioningFunction = ( elementRect: Rect, targetRect: Rect, viewportRect: Rect | null ) => PositioningFunctionResult | null;\n\n/**\n * The result of {@link module:utils/dom/position~PositioningFunction}.\n */\nexport interface PositioningFunctionResult {\n\n\t/**\n\t * The `top` value of the element rect that would represent the position.\n\t */\n\ttop: number;\n\n\t/**\n\t * The `left` value of the element rect that would represent the position.\n\t */\n\tleft: number;\n\n\t/**\n\t * The name of the position. It helps the user of the {@link module:utils/dom/position~getOptimalPosition}\n\t * helper to recognize different positioning function results. It will pass through to the {@link module:utils/dom/position~Position}\n\t * returned by the helper.\n\t */\n\tname?: string;\n\n\t/**\n\t * An optional configuration that will pass-through the {@link module:utils/dom/position~getOptimalPosition} helper\n\t * to the {@link module:utils/dom/position~Position} returned by this helper.\n\t * This configuration may, for instance, let the user of {@link module:utils/dom/position~getOptimalPosition} know that this particular\n\t * position comes with a certain presentation.\n\t */\n\tconfig?: object;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/getpositionedancestor\n */\n\nimport global from './global';\n\n/**\n * For a given element, returns the nearest ancestor element which CSS position is not \"static\".\n *\n * @param element The native DOM element to be checked.\n */\nexport default function getPositionedAncestor( element?: HTMLElement ): HTMLElement | null {\n\tif ( !element || !element.parentNode ) {\n\t\treturn null;\n\t}\n\n\tif ( element.offsetParent === global.document.body ) {\n\t\treturn null;\n\t}\n\n\treturn element.offsetParent as HTMLElement;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/remove\n */\n\n/**\n * Removes given node from parent.\n *\n * @param node Node to remove.\n */\nexport default function remove( node: Node ): void {\n\tconst parent = node.parentNode;\n\n\tif ( parent ) {\n\t\tparent.removeChild( node );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/scroll\n */\n\nimport isRange from './isrange';\nimport Rect from './rect';\nimport isText from './istext';\n\ntype IfTrue<T> = T extends true ? true : never;\n\n/**\n * Makes any page `HTMLElement` or `Range` (`target`) visible inside the browser viewport.\n * This helper will scroll all `target` ancestors and the web browser viewport to reveal the target to\n * the user. If the `target` is already visible, nothing will happen.\n *\n * @param options Additional configuration of the scrolling behavior.\n * @param options.target A target, which supposed to become visible to the user.\n * @param options.viewportOffset An offset from the edge of the viewport (in pixels)\n * the `target` will be moved by if the viewport is scrolled. It enhances the user experience\n * by keeping the `target` some distance from the edge of the viewport and thus making it easier to\n * read or edit by the user.\n * @param options.ancestorOffset An offset from the boundary of scrollable ancestors (if any)\n * the `target` will be moved by if the viewport is scrolled. It enhances the user experience\n * by keeping the `target` some distance from the edge of the ancestors and thus making it easier to\n * read or edit by the user.\n * @param options.alignToTop When set `true`, the helper will make sure the `target` is scrolled up\n * to the top boundary of the viewport and/or scrollable ancestors if scrolled up. When not set\n * (default), the `target` will be revealed by scrolling as little as possible. This option will\n * not affect `targets` that must be scrolled down because they will appear at the top of the boundary\n * anyway.\n *\n * ```\n * scrollViewportToShowTarget() with scrollViewportToShowTarget() with\n * Initial state alignToTop unset (default) alignToTop = true\n *\n * ┌────────────────────────────────┬─┐ ┌────────────────────────────────┬─┐ ┌────────────────────────────────┬─┐\n * │ │▲│ │ │▲│ │ [ Target to be revealed ] │▲│\n * │ │ │ │ │ │ │ │ │\n * │ │█│ │ │ │ │ │ │\n * │ │█│ │ │ │ │ │ │\n * │ │ │ │ │█│ │ │ │\n * │ │ │ │ │█│ │ │█│\n * │ │ │ │ │ │ │ │█│\n * │ │▼│ │ [ Target to be revealed ] │▼│ │ │▼│\n * └────────────────────────────────┴─┘ └────────────────────────────────┴─┘ └────────────────────────────────┴─┘\n *\n *\n * [ Target to be revealed ]\n *```\n *\n * @param options.forceScroll When set `true`, the `target` will be aligned to the top of the viewport\n * and scrollable ancestors whether it is already visible or not. This option will only work when `alignToTop`\n * is `true`\n */\nexport function scrollViewportToShowTarget<T extends boolean, U extends IfTrue<T>>(\n\t{\n\t\ttarget,\n\t\tviewportOffset = 0,\n\t\tancestorOffset = 0,\n\t\talignToTop,\n\t\tforceScroll\n\t}:\n\t{\n\t\treadonly target: HTMLElement | Range;\n\t\treadonly viewportOffset?: number | { top: number; bottom: number; left: number; right: number };\n\t\treadonly ancestorOffset?: number;\n\t\treadonly alignToTop?: T;\n\t\treadonly forceScroll?: U;\n\t}\n): void {\n\tconst targetWindow = getWindow( target );\n\tlet currentWindow: Window | null = targetWindow;\n\tlet currentFrame: HTMLElement | null = null;\n\n\tviewportOffset = normalizeViewportOffset( viewportOffset );\n\n\t// Iterate over all windows, starting from target's parent window up to window#top.\n\twhile ( currentWindow ) {\n\t\tlet firstAncestorToScroll: HTMLElement;\n\n\t\t// Let's scroll target's ancestors first to reveal it. Then, once the ancestor scrolls\n\t\t// settled down, the algorithm can eventually scroll the viewport of the current window.\n\t\t//\n\t\t// Note: If the current window is target's **original** window (e.g. the first one),\n\t\t// start scrolling the closest parent of the target. If not, scroll the closest parent\n\t\t// of an iframe that resides in the current window.\n\t\tif ( currentWindow == targetWindow ) {\n\t\t\tfirstAncestorToScroll = getParentElement( target );\n\t\t} else {\n\t\t\tfirstAncestorToScroll = getParentElement( currentFrame! );\n\t\t}\n\n\t\t// Scroll the target's ancestors first. Once done, scrolling the viewport is easy.\n\t\tscrollAncestorsToShowRect( {\n\t\t\tparent: firstAncestorToScroll,\n\t\t\tgetRect: () => {\n\t\t\t\t// Note: If the target does not belong to the current window **directly**,\n\t\t\t\t// i.e. it resides in an iframe belonging to the window, obtain the target's rect\n\t\t\t\t// in the coordinates of the current window. By default, a Rect returns geometry\n\t\t\t\t// relative to the current window's viewport. To make it work in a parent window,\n\t\t\t\t// it must be shifted.\n\t\t\t\treturn getRectRelativeToWindow( target, currentWindow! );\n\t\t\t},\n\t\t\talignToTop,\n\t\t\tancestorOffset,\n\t\t\tforceScroll\n\t\t} );\n\n\t\t// Obtain the rect of the target after it has been scrolled within its ancestors.\n\t\t// It's time to scroll the viewport.\n\t\tconst targetRect = getRectRelativeToWindow( target, currentWindow );\n\n\t\tscrollWindowToShowRect( {\n\t\t\twindow: currentWindow,\n\t\t\trect: targetRect,\n\t\t\tviewportOffset,\n\t\t\talignToTop,\n\t\t\tforceScroll\n\t\t} );\n\n\t\tif ( currentWindow.parent != currentWindow ) {\n\t\t\t// Keep the reference to the <iframe> element the \"previous current window\" was\n\t\t\t// rendered within. It will be useful to recalculate the rect of the target\n\t\t\t// in the parent window's relative geometry. The target's rect must be shifted\n\t\t\t// by it's iframe's position.\n\t\t\tcurrentFrame = currentWindow.frameElement as HTMLElement | null;\n\t\t\tcurrentWindow = currentWindow.parent;\n\n\t\t\t// If the current window has some parent but frameElement is inaccessible, then they have\n\t\t\t// different domains/ports and, due to security reasons, accessing and scrolling\n\t\t\t// the parent window won't be possible.\n\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/930.\n\t\t\tif ( !currentFrame ) {\n\t\t\t\treturn;\n\t\t\t}\n\t\t} else {\n\t\t\tcurrentWindow = null;\n\t\t}\n\t}\n}\n\n/**\n * Makes any page `HTMLElement` or `Range` (target) visible within its scrollable ancestors,\n * e.g. if they have `overflow: scroll` CSS style.\n *\n * @param target A target, which supposed to become visible to the user.\n * @param ancestorOffset An offset between the target and the boundary of scrollable ancestors\n * to be maintained while scrolling.\n * @param limiterElement The outermost ancestor that should be scrolled. If specified, it can prevent\n * scrolling the whole page.\n */\nexport function scrollAncestorsToShowTarget( target: HTMLElement | Range, ancestorOffset?: number, limiterElement?: HTMLElement ): void {\n\tconst targetParent = getParentElement( target );\n\n\tscrollAncestorsToShowRect( {\n\t\tparent: targetParent,\n\t\tgetRect: () => new Rect( target ),\n\t\tancestorOffset,\n\t\tlimiterElement\n\t} );\n}\n\n/**\n * Makes a given rect visible within its parent window.\n *\n * Note: Avoid the situation where the caret is still in the viewport, but totally\n * at the edge of it. In such situation, if it moved beyond the viewport in the next\n * action e.g. after paste, the scrolling would move it to the viewportOffset level\n * and it all would look like the caret visually moved up/down:\n *\n * 1.\n * ```\n * | foo[]\n * | <--- N px of space below the caret\n * +---------------------------------...\n * ```\n *\n * 2. *paste*\n * 3.\n * ```\n * |\n * |\n * +-foo-----------------------------...\n * bar[] <--- caret below viewport, scrolling...\n * ```\n *\n * 4. *scrolling*\n * 5.\n * ```\n * |\n * | foo\n * | bar[] <--- caret precisely at the edge\n * +---------------------------------...\n * ```\n *\n * To prevent this, this method checks the rects moved by the viewportOffset to cover\n * the upper/lower edge of the viewport. It makes sure if the action repeats, there's\n * no twitching it's a purely visual improvement:\n *\n * 5. (after fix)\n * ```\n * |\n * | foo\n * | bar[]\n * | <--- N px of space below the caret\n * +---------------------------------...\n * ```\n *\n * @param options Additional configuration of the scrolling behavior.\n * @param options.window A window which is scrolled to reveal the rect.\n * @param options.rect A rect which is to be revealed.\n * @param options.viewportOffset An offset from the edge of the viewport (in pixels) the `rect` will be\n * moved by if the viewport is scrolled.\n * @param options.alignToTop When set `true`, the helper will make sure the `rect` is scrolled up\n * to the top boundary of the viewport if scrolled up. When not set (default), the `rect` will be\n * revealed by scrolling as little as possible. This option will not affect rects that must be scrolled\n * down because they will appear at the top of the boundary anyway.\n * @param options.forceScroll When set `true`, the `rect` will be aligned to the top of the viewport\n * whether it is already visible or not. This option will only work when `alignToTop` is `true`\n */\nfunction scrollWindowToShowRect<T extends boolean, U extends IfTrue<T>>(\n\t{\n\t\twindow,\n\t\trect,\n\t\talignToTop,\n\t\tforceScroll,\n\t\tviewportOffset\n\t}: {\n\t\treadonly window: Window;\n\t\treadonly rect: Rect;\n\t\treadonly viewportOffset: { top: number; bottom: number; left: number; right: number };\n\t\treadonly alignToTop?: T;\n\t\treadonly forceScroll?: U;\n\t}\n): void {\n\tconst targetShiftedDownRect = rect.clone().moveBy( 0, viewportOffset.bottom );\n\tconst targetShiftedUpRect = rect.clone().moveBy( 0, -viewportOffset.top );\n\tconst viewportRect = new Rect( window ).excludeScrollbarsAndBorders();\n\n\tconst rects = [ targetShiftedUpRect, targetShiftedDownRect ];\n\tconst forceScrollToTop = alignToTop && forceScroll;\n\tconst allRectsFitInViewport = rects.every( rect => viewportRect.contains( rect ) );\n\n\tlet { scrollX, scrollY } = window;\n\tconst initialScrollX = scrollX;\n\tconst initialScrollY = scrollY;\n\n\tif ( forceScrollToTop ) {\n\t\tscrollY -= ( viewportRect.top - rect.top ) + viewportOffset.top;\n\t} else if ( !allRectsFitInViewport ) {\n\t\tif ( isAbove( targetShiftedUpRect, viewportRect ) ) {\n\t\t\tscrollY -= viewportRect.top - rect.top + viewportOffset.top;\n\t\t} else if ( isBelow( targetShiftedDownRect, viewportRect ) ) {\n\t\t\tif ( alignToTop ) {\n\t\t\t\tscrollY += rect.top - viewportRect.top - viewportOffset.top;\n\t\t\t} else {\n\t\t\t\tscrollY += rect.bottom - viewportRect.bottom + viewportOffset.bottom;\n\t\t\t}\n\t\t}\n\t}\n\n\tif ( !allRectsFitInViewport ) {\n\t\t// TODO: Web browsers scroll natively to place the target in the middle\n\t\t// of the viewport. It's not a very popular case, though.\n\t\tif ( isLeftOf( rect, viewportRect ) ) {\n\t\t\tscrollX -= viewportRect.left - rect.left + viewportOffset.left;\n\t\t} else if ( isRightOf( rect, viewportRect ) ) {\n\t\t\tscrollX += rect.right - viewportRect.right + viewportOffset.right;\n\t\t}\n\t}\n\n\tif ( scrollX != initialScrollX || scrollY !== initialScrollY ) {\n\t\twindow.scrollTo( scrollX, scrollY );\n\t}\n}\n\n/**\n * Recursively scrolls element ancestors to visually reveal a rect.\n *\n * @param options Additional configuration of the scrolling behavior.\n * @param options.parent The first parent ancestor to start scrolling.\n * @param options.getRect A function which returns the Rect, which is to be revealed.\n * @param options.ancestorOffset An offset from the boundary of scrollable ancestors (if any)\n * the `Rect` instance will be moved by if the viewport is scrolled.\n * @param options.alignToTop When set `true`, the helper will make sure the `Rect` instance is scrolled up\n * to the top boundary of the scrollable ancestors if scrolled up. When not set (default), the `rect`\n * will be revealed by scrolling as little as possible. This option will not affect rects that must be\n * scrolled down because they will appear at the top of the boundary\n * anyway.\n * @param options.forceScroll When set `true`, the `rect` will be aligned to the top of scrollable ancestors\n * whether it is already visible or not. This option will only work when `alignToTop` is `true`\n * @param options.limiterElement The outermost ancestor that should be scrolled. Defaults to the `<body>` element.\n */\nfunction scrollAncestorsToShowRect<T extends boolean, U extends IfTrue<T>>(\n\t{\n\t\tparent,\n\t\tgetRect,\n\t\talignToTop,\n\t\tforceScroll,\n\t\tancestorOffset = 0,\n\t\tlimiterElement\n\t}: {\n\t\treadonly parent: HTMLElement;\n\t\treadonly getRect: () => Rect;\n\t\treadonly alignToTop?: T;\n\t\treadonly forceScroll?: U;\n\t\treadonly ancestorOffset?: number;\n\t\treadonly limiterElement?: HTMLElement;\n\t}\n): void {\n\tconst parentWindow = getWindow( parent );\n\tconst forceScrollToTop = alignToTop && forceScroll;\n\tlet parentRect: Rect, targetRect: Rect, targetFitsInTarget: boolean;\n\n\tconst limiter = limiterElement || parentWindow.document.body;\n\n\twhile ( parent != limiter ) {\n\t\ttargetRect = getRect();\n\t\tparentRect = new Rect( parent ).excludeScrollbarsAndBorders();\n\t\ttargetFitsInTarget = parentRect.contains( targetRect );\n\n\t\tif ( forceScrollToTop ) {\n\t\t\tparent.scrollTop -= ( parentRect.top - targetRect.top ) + ancestorOffset;\n\t\t} else if ( !targetFitsInTarget ) {\n\t\t\tif ( isAbove( targetRect, parentRect ) ) {\n\t\t\t\tparent.scrollTop -= parentRect.top - targetRect.top + ancestorOffset;\n\t\t\t} else if ( isBelow( targetRect, parentRect ) ) {\n\t\t\t\tif ( alignToTop ) {\n\t\t\t\t\tparent.scrollTop += targetRect.top - parentRect.top - ancestorOffset;\n\t\t\t\t} else {\n\t\t\t\t\tparent.scrollTop += targetRect.bottom - parentRect.bottom + ancestorOffset;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif ( !targetFitsInTarget ) {\n\t\t\tif ( isLeftOf( targetRect, parentRect ) ) {\n\t\t\t\tparent.scrollLeft -= parentRect.left - targetRect.left + ancestorOffset;\n\t\t\t} else if ( isRightOf( targetRect, parentRect ) ) {\n\t\t\t\tparent.scrollLeft += targetRect.right - parentRect.right + ancestorOffset;\n\t\t\t}\n\t\t}\n\n\t\tparent = parent.parentNode as HTMLElement;\n\t}\n}\n\n/**\n * Determines if a given `Rect` extends beyond the bottom edge of the second `Rect`.\n */\nfunction isBelow( firstRect: Rect, secondRect: Rect ): boolean {\n\treturn firstRect.bottom > secondRect.bottom;\n}\n\n/**\n * Determines if a given `Rect` extends beyond the top edge of the second `Rect`.\n */\nfunction isAbove( firstRect: Rect, secondRect: Rect ): boolean {\n\treturn firstRect.top < secondRect.top;\n}\n\n/**\n * Determines if a given `Rect` extends beyond the left edge of the second `Rect`.\n */\nfunction isLeftOf( firstRect: Rect, secondRect: Rect ): boolean {\n\treturn firstRect.left < secondRect.left;\n}\n\n/**\n * Determines if a given `Rect` extends beyond the right edge of the second `Rect`.\n */\nfunction isRightOf( firstRect: Rect, secondRect: Rect ): boolean {\n\treturn firstRect.right > secondRect.right;\n}\n\n/**\n * Returns the closest window of an element or range.\n */\nfunction getWindow( elementOrRange: HTMLElement | Range ): Window {\n\tif ( isRange( elementOrRange ) ) {\n\t\treturn elementOrRange.startContainer.ownerDocument!.defaultView!;\n\t} else {\n\t\treturn elementOrRange.ownerDocument.defaultView!;\n\t}\n}\n\n/**\n * Returns the closest parent of an element or DOM range.\n */\nfunction getParentElement( elementOrRange: HTMLElement | Range ): HTMLElement {\n\tif ( isRange( elementOrRange ) ) {\n\t\tlet parent = elementOrRange.commonAncestorContainer as HTMLElement;\n\n\t\t// If a Range is attached to the Text, use the closest element ancestor.\n\t\tif ( isText( parent ) ) {\n\t\t\tparent = parent.parentNode as HTMLElement;\n\t\t}\n\n\t\treturn parent;\n\t} else {\n\t\treturn elementOrRange.parentNode as HTMLElement;\n\t}\n}\n\n/**\n * Returns the rect of an element or range residing in an iframe.\n * The result rect is relative to the geometry of the passed window instance.\n *\n * @param target Element or range which rect should be returned.\n * @param relativeWindow A window the rect should be relative to.\n */\nfunction getRectRelativeToWindow( target: HTMLElement | Range, relativeWindow: Window ): Rect {\n\tconst targetWindow = getWindow( target );\n\tconst rect = new Rect( target );\n\n\tif ( targetWindow === relativeWindow ) {\n\t\treturn rect;\n\t} else {\n\t\tlet currentWindow = targetWindow;\n\n\t\twhile ( currentWindow != relativeWindow ) {\n\t\t\tconst frame = currentWindow.frameElement as HTMLElement;\n\t\t\tconst frameRect = new Rect( frame ).excludeScrollbarsAndBorders();\n\n\t\t\trect.moveBy( frameRect.left, frameRect.top );\n\n\t\t\tcurrentWindow = currentWindow.parent;\n\t\t}\n\t}\n\n\treturn rect;\n}\n\n/**\n * A helper that explodes the `viewportOffset` configuration if defined as a plain number into an object\n * with `top`, `bottom`, `left`, and `right` properties.\n *\n * If an object value is passed, this helper will pass it through.\n *\n * @param viewportOffset Viewport offset to be normalized.\n */\nfunction normalizeViewportOffset( viewportOffset: number | { top: number; bottom: number; left: number; right: number } ) {\n\tif ( typeof viewportOffset === 'number' ) {\n\t\treturn {\n\t\t\ttop: viewportOffset,\n\t\t\tbottom: viewportOffset,\n\t\t\tleft: viewportOffset,\n\t\t\tright: viewportOffset\n\t\t};\n\t}\n\n\treturn viewportOffset;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * A set of utilities related to keyboard support.\n *\n * @module utils/keyboard\n */\n\nimport type { LanguageDirection } from './language';\nimport CKEditorError from './ckeditorerror';\nimport env from './env';\n\nconst modifiersToGlyphsMac = {\n\tctrl: '⌃',\n\tcmd: '⌘',\n\talt: '⌥',\n\tshift: '⇧'\n} as const;\n\nconst modifiersToGlyphsNonMac = {\n\tctrl: 'Ctrl+',\n\talt: 'Alt+',\n\tshift: 'Shift+'\n} as const;\n\n/**\n * An object with `keyName => keyCode` pairs for a set of known keys.\n *\n * Contains:\n *\n * * `a-z`,\n * * `0-9`,\n * * `f1-f12`,\n * * `` ` ``, `-`, `=`, `[`, `]`, `;`, `'`, `,`, `.`, `/`, `\\`,\n * * `arrow(left|up|right|bottom)`,\n * * `backspace`, `delete`, `enter`, `esc`, `tab`,\n * * `ctrl`, `cmd`, `shift`, `alt`.\n */\nexport const keyCodes = generateKnownKeyCodes();\n\nconst keyCodeNames = Object.fromEntries(\n\tObject.entries( keyCodes ).map( ( [ name, code ] ) => [ code, name.charAt( 0 ).toUpperCase() + name.slice( 1 ) ] )\n);\n\n/**\n * Converts a key name or {@link module:utils/keyboard~KeystrokeInfo keystroke info} into a key code.\n *\n * Note: Key names are matched with {@link module:utils/keyboard#keyCodes} in a case-insensitive way.\n *\n * @param key A key name (see {@link module:utils/keyboard#keyCodes}) or a keystroke data object.\n * @returns Key or keystroke code.\n */\nexport function getCode( key: string | Readonly<KeystrokeInfo> ): number {\n\tlet keyCode: number | undefined;\n\n\tif ( typeof key == 'string' ) {\n\t\tkeyCode = keyCodes[ key.toLowerCase() ];\n\n\t\tif ( !keyCode ) {\n\t\t\t/**\n\t\t\t * Unknown key name. Only key names included in the {@link module:utils/keyboard#keyCodes} can be used.\n\t\t\t *\n\t\t\t * @error keyboard-unknown-key\n\t\t\t * @param {String} key\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'keyboard-unknown-key', null, { key } );\n\t\t}\n\t} else {\n\t\tkeyCode = key.keyCode +\n\t\t\t( key.altKey ? keyCodes.alt : 0 ) +\n\t\t\t( key.ctrlKey ? keyCodes.ctrl : 0 ) +\n\t\t\t( key.shiftKey ? keyCodes.shift : 0 ) +\n\t\t\t( key.metaKey ? keyCodes.cmd : 0 );\n\t}\n\n\treturn keyCode;\n}\n\n/**\n * Parses the keystroke and returns a keystroke code that will match the code returned by\n * {@link module:utils/keyboard~getCode} for the corresponding {@link module:utils/keyboard~KeystrokeInfo keystroke info}.\n *\n * The keystroke can be passed in two formats:\n *\n * * as a single string e.g. `ctrl + A`,\n * * as an array of {@link module:utils/keyboard~keyCodes known key names} and key codes e.g.:\n * * `[ 'ctrl', 32 ]` (ctrl + space),\n * * `[ 'ctrl', 'a' ]` (ctrl + A).\n *\n * Note: Key names are matched with {@link module:utils/keyboard#keyCodes} in a case-insensitive way.\n *\n * Note: Only keystrokes with a single non-modifier key are supported (e.g. `ctrl+A` is OK, but `ctrl+A+B` is not).\n *\n * Note: On macOS, keystroke handling is translating the `Ctrl` key to the `Cmd` key and handling only that keystroke.\n * For example, a registered keystroke `Ctrl+A` will be translated to `Cmd+A` on macOS. To disable the translation of some keystroke,\n * use the forced modifier: `Ctrl!+A` (note the exclamation mark).\n *\n * @param keystroke The keystroke definition.\n * @returns Keystroke code.\n */\nexport function parseKeystroke( keystroke: string | ReadonlyArray<number | string> ): number {\n\tif ( typeof keystroke == 'string' ) {\n\t\tkeystroke = splitKeystrokeText( keystroke );\n\t}\n\n\treturn keystroke\n\t\t.map( key => ( typeof key == 'string' ) ? getEnvKeyCode( key ) : key )\n\t\t.reduce( ( key, sum ) => sum + key, 0 );\n}\n\n/**\n * Translates any keystroke string text like `\"Ctrl+A\"` to an\n * environmentspecific keystroke, i.e. `\"⌘A\"` on macOS.\n *\n * @param keystroke The keystroke text.\n * @returns The keystroke text specific for the environment.\n */\nexport function getEnvKeystrokeText( keystroke: string ): string {\n\tlet keystrokeCode = parseKeystroke( keystroke );\n\n\tconst modifiersToGlyphs = Object.entries( env.isMac ? modifiersToGlyphsMac : modifiersToGlyphsNonMac );\n\n\tconst modifiers = modifiersToGlyphs.reduce( ( modifiers, [ name, glyph ] ) => {\n\t\t// Modifier keys are stored as a bit mask so extract those from the keystroke code.\n\t\tif ( ( keystrokeCode & keyCodes[ name ] ) != 0 ) {\n\t\t\tkeystrokeCode &= ~keyCodes[ name ];\n\t\t\tmodifiers += glyph;\n\t\t}\n\n\t\treturn modifiers;\n\t}, '' );\n\n\treturn modifiers + ( keystrokeCode ? keyCodeNames[ keystrokeCode ] : '' );\n}\n\n/**\n * Returns `true` if the provided key code represents one of the arrow keys.\n *\n * @param keyCode A key code as in {@link module:utils/keyboard~KeystrokeInfo#keyCode}.\n */\nexport function isArrowKeyCode( keyCode: number ): boolean {\n\treturn keyCode == keyCodes.arrowright ||\n\t\tkeyCode == keyCodes.arrowleft ||\n\t\tkeyCode == keyCodes.arrowup ||\n\t\tkeyCode == keyCodes.arrowdown;\n}\n\n/**\n * String representing a direction of an arrow key kode.\n */\nexport type ArrowKeyCodeDirection = 'left' | 'up' | 'right' | 'down';\n\n/**\n * Returns the direction in which the {@link module:engine/model/documentselection~DocumentSelection selection}\n * will move when the provided arrow key code is pressed considering the language direction of the editor content.\n *\n * For instance, in righttoleft (RTL) content languages, pressing the left arrow means moving the selection right (forward)\n * in the model structure. Similarly, pressing the right arrow moves the selection left (backward).\n *\n * @param keyCode A key code as in {@link module:utils/keyboard~KeystrokeInfo#keyCode}.\n * @param contentLanguageDirection The content language direction, corresponding to\n * {@link module:utils/locale~Locale#contentLanguageDirection}.\n * @returns Localized arrow direction or `undefined` for non-arrow key codes.\n */\nexport function getLocalizedArrowKeyCodeDirection(\n\tkeyCode: number,\n\tcontentLanguageDirection: LanguageDirection\n): ArrowKeyCodeDirection | undefined {\n\tconst isLtrContent = contentLanguageDirection === 'ltr';\n\n\tswitch ( keyCode ) {\n\t\tcase keyCodes.arrowleft:\n\t\t\treturn isLtrContent ? 'left' : 'right';\n\n\t\tcase keyCodes.arrowright:\n\t\t\treturn isLtrContent ? 'right' : 'left';\n\n\t\tcase keyCodes.arrowup:\n\t\t\treturn 'up';\n\n\t\tcase keyCodes.arrowdown:\n\t\t\treturn 'down';\n\t}\n}\n\n/**\n * Converts a key name to the key code with mapping based on the env.\n *\n * See: {@link module:utils/keyboard~getCode}.\n *\n * @param key The key name (see {@link module:utils/keyboard#keyCodes}).\n * @returns Key code.\n */\nfunction getEnvKeyCode( key: string ): number {\n\t// Don't remap modifier key for forced modifiers.\n\tif ( key.endsWith( '!' ) ) {\n\t\treturn getCode( key.slice( 0, -1 ) );\n\t}\n\n\tconst code = getCode( key );\n\n\treturn env.isMac && code == keyCodes.ctrl ? keyCodes.cmd : code;\n}\n\n/**\n * Determines if the provided key code moves the {@link module:engine/model/documentselection~DocumentSelection selection}\n * forward or backward considering the language direction of the editor content.\n *\n * For instance, in righttoleft (RTL) languages, pressing the left arrow means moving forward\n * in the model structure. Similarly, pressing the right arrow moves the selection backward.\n *\n * @param keyCode A key code as in {@link module:utils/keyboard~KeystrokeInfo#keyCode}.\n * @param contentLanguageDirection The content language direction, corresponding to\n * {@link module:utils/locale~Locale#contentLanguageDirection}.\n */\nexport function isForwardArrowKeyCode(\n\tkeyCode: number,\n\tcontentLanguageDirection: LanguageDirection\n): boolean {\n\tconst localizedKeyCodeDirection = getLocalizedArrowKeyCodeDirection( keyCode, contentLanguageDirection );\n\n\treturn localizedKeyCodeDirection === 'down' || localizedKeyCodeDirection === 'right';\n}\n\nfunction generateKnownKeyCodes(): { readonly [ keyCode: string ]: number } {\n\tconst keyCodes: { [keyCode: string]: number } = {\n\t\tarrowleft: 37,\n\t\tarrowup: 38,\n\t\tarrowright: 39,\n\t\tarrowdown: 40,\n\t\tbackspace: 8,\n\t\tdelete: 46,\n\t\tenter: 13,\n\t\tspace: 32,\n\t\tesc: 27,\n\t\ttab: 9,\n\n\t\t// The idea about these numbers is that they do not collide with any real key codes, so we can use them\n\t\t// like bit masks.\n\t\tctrl: 0x110000,\n\t\tshift: 0x220000,\n\t\talt: 0x440000,\n\t\tcmd: 0x880000\n\t};\n\n\t// a-z\n\tfor ( let code = 65; code <= 90; code++ ) {\n\t\tconst letter = String.fromCharCode( code );\n\n\t\tkeyCodes[ letter.toLowerCase() ] = code;\n\t}\n\n\t// 0-9\n\tfor ( let code = 48; code <= 57; code++ ) {\n\t\tkeyCodes[ code - 48 ] = code;\n\t}\n\n\t// F1-F12\n\tfor ( let code = 112; code <= 123; code++ ) {\n\t\tkeyCodes[ 'f' + ( code - 111 ) ] = code;\n\t}\n\n\t// other characters\n\tfor ( const char of '`-=[];\\',./\\\\' ) {\n\t\tkeyCodes[ char ] = char.charCodeAt( 0 );\n\t}\n\n\treturn keyCodes;\n}\n\nfunction splitKeystrokeText( keystroke: string ): Array<string> {\n\treturn keystroke.split( '+' ).map( key => key.trim() );\n}\n\n/**\n * Information about the keystroke.\n */\nexport interface KeystrokeInfo {\n\n\t/**\n\t * The [key code](https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/keyCode).\n\t */\n\tkeyCode: number;\n\n\t/**\n\t * Whether the <kbd>Alt</kbd> modifier was pressed.\n\t */\n\taltKey: boolean;\n\n\t/**\n\t * Whether the <kbd>Cmd</kbd> modifier was pressed.\n\t */\n\tmetaKey: boolean;\n\n\t/**\n\t * Whether the <kbd>Ctrl</kbd> modifier was pressed.\n\t */\n\tctrlKey: boolean;\n\n\t/**\n\t * Whether the <kbd>Shift</kbd> modifier was pressed.\n\t */\n\tshiftKey: boolean;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/toarray\n */\n\n/**\n * Transforms any value to an array. If the provided value is already an array, it is returned unchanged.\n *\n * @label MUTABLE\n * @param data The value to transform to an array.\n * @returns An array created from data.\n */\nexport default function toArray<T>( data: ArrayOrItem<T> ): Array<T>;\n\n/**\n * Transforms any value to an array. If the provided value is already an array, it is returned unchanged.\n *\n * @label IMMUTABLE\n * @param data The value to transform to an array.\n * @returns An array created from data.\n */\nexport default function toArray<T>( data: ReadonlyArrayOrItem<T> ): ReadonlyArray<T>;\n\nexport default function toArray<T>( data: ArrayOrItem<T> ): Array<T> {\n\treturn Array.isArray( data ) ? data : [ data ];\n}\n\nexport type ArrayOrItem<T> = T | Array<T>;\nexport type ReadonlyArrayOrItem<T> = T | ReadonlyArray<T>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* eslint-disable no-var */\n\n/**\n * @module utils/translation-service\n */\n\nimport CKEditorError from './ckeditorerror';\nimport global from './dom/global';\n\ndeclare global {\n\tvar CKEDITOR_TRANSLATIONS: {\n\t\t[ language: string ]: {\n\t\t\tdictionary: { [ messageId: string ]: string | ReadonlyArray<string> };\n\t\t\tgetPluralForm?: ( n: number ) => number;\n\t\t};\n\t};\n}\n\n/* istanbul ignore else -- @preserve */\nif ( !global.window.CKEDITOR_TRANSLATIONS ) {\n\tglobal.window.CKEDITOR_TRANSLATIONS = {};\n}\n\n/**\n * Adds translations to existing ones or overrides the existing translations. These translations will later\n * be available for the {@link module:utils/locale~Locale#t `t()`} function.\n *\n * The `translations` is an object which consists of `messageId: translation` pairs. Note that the message ID can be\n * either constructed from the message string or from the message ID if it was passed\n * (this happens rarely and mostly for short messages or messages with placeholders).\n * Since the editor displays only the message string, the message ID can be found either in the source code or in the\n * built translations for another language.\n *\n * ```ts\n * add( 'pl', {\n * \t'Cancel': 'Anuluj',\n * \t'IMAGE': 'obraz', // Note that the `IMAGE` comes from the message ID, while the string can be `image`.\n * } );\n * ```\n *\n * If the message is supposed to support various plural forms, make sure to provide an array with the singular form and all plural forms:\n *\n * ```ts\n * add( 'pl', {\n * \t'Add space': [ 'Dodaj spację', 'Dodaj %0 spacje', 'Dodaj %0 spacji' ]\n * } );\n * ```\n *\n * You should also specify the third argument (the `getPluralForm()` function) that will be used to determine the plural form if no\n * language file was loaded for that language. All language files coming from CKEditor 5 sources will have this option set, so\n * these plural form rules will be reused by other translations added to the registered languages. The `getPluralForm()` function\n * can return either a Boolean or a number.\n *\n * ```ts\n * add( 'en', {\n * \t// ... Translations.\n * }, n => n !== 1 );\n * add( 'pl', {\n * \t// ... Translations.\n * }, n => n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && ( n % 100 < 10 || n % 100 >= 20 ) ? 1 : 2 );\n * ```\n *\n * All translations extend the global `window.CKEDITOR_TRANSLATIONS` object. An example of this object can be found below:\n *\n * ```ts\n * {\n * \tpl: {\n * \t\tdictionary: {\n * \t\t\t'Cancel': 'Anuluj',\n * \t\t\t'Add space': [ 'Dodaj spację', 'Dodaj %0 spacje', 'Dodaj %0 spacji' ]\n * \t\t},\n * \t\t// A function that returns the plural form index.\n * \t\tgetPluralForm: n => n == 1 ? 0 : n % 10 >= 2 && n % 10 <= 4 && ( n % 100 < 10 || n % 100 >= 20 ) ? 1 : 2 );\n * \t}\n * \t// Other languages.\n * \t}\n * ```\n *\n * If you cannot import this function from this module (e.g. because you use a CKEditor 5 build), you can\n * still add translations by extending the global `window.CKEDITOR_TRANSLATIONS` object by using a function like\n * the one below:\n *\n * ```ts\n * function addTranslations( language, translations, getPluralForm ) {\n * \tif ( !global.window.CKEDITOR_TRANSLATIONS ) {\n * \t\tglobal.window.CKEDITOR_TRANSLATIONS = {};\n * \t}\n\n * \tif ( !global.window.CKEDITOR_TRANSLATIONS[ language ] ) {\n * \t\tglobal.window.CKEDITOR_TRANSLATIONS[ language ] = {};\n * \t}\n *\n * \tconst languageTranslations = global.window.CKEDITOR_TRANSLATIONS[ language ];\n *\n * \tlanguageTranslations.dictionary = languageTranslations.dictionary || {};\n * \tlanguageTranslations.getPluralForm = getPluralForm || languageTranslations.getPluralForm;\n *\n * \t// Extend the dictionary for the given language.\n * \tObject.assign( languageTranslations.dictionary, translations );\n * }\n * ```\n *\n * @param language Target language.\n * @param translations An object with translations which will be added to the dictionary.\n * For each message ID the value should be either a translation or an array of translations if the message\n * should support plural forms.\n * @param getPluralForm A function that returns the plural form index (a number).\n */\nexport function add(\n\tlanguage: string,\n\ttranslations: { readonly [ messageId: string ]: string | ReadonlyArray<string> },\n\tgetPluralForm?: ( n: number ) => number\n): void {\n\tif ( !global.window.CKEDITOR_TRANSLATIONS[ language ] ) {\n\t\tglobal.window.CKEDITOR_TRANSLATIONS[ language ] = {} as any;\n\t}\n\n\tconst languageTranslations = global.window.CKEDITOR_TRANSLATIONS[ language ];\n\n\tlanguageTranslations.dictionary = languageTranslations.dictionary || {};\n\tlanguageTranslations.getPluralForm = getPluralForm || languageTranslations.getPluralForm;\n\n\tObject.assign( languageTranslations.dictionary, translations );\n}\n\n/**\n * **Note:** This method is internal, use {@link module:utils/locale~Locale#t the `t()` function} instead to translate\n * the editor UI parts.\n *\n * This function is responsible for translating messages to the specified language. It uses translations added perviously\n * by {@link module:utils/translation-service~add} (a translations dictionary and the `getPluralForm()` function\n * to provide accurate translations of plural forms).\n *\n * When no translation is defined in the dictionary or the dictionary does not exist, this function returns\n * the original message string or the message plural depending on the number of elements.\n *\n * ```ts\n * translate( 'pl', { string: 'Cancel' } ); // 'Cancel'\n * ```\n *\n * The third optional argument is the number of elements, based on which the single form or one of the plural forms\n * should be picked when the message is supposed to support various plural forms.\n *\n * ```ts\n * translate( 'en', { string: 'Add a space', plural: 'Add %0 spaces' }, 1 ); // 'Add a space'\n * translate( 'en', { string: 'Add a space', plural: 'Add %0 spaces' }, 3 ); // 'Add %0 spaces'\n * ```\n *\n * The message should provide an ID using the `id` property when the message strings are not unique and their\n * translations should be different.\n *\n * ```ts\n * translate( 'en', { string: 'image', id: 'ADD_IMAGE' } );\n * translate( 'en', { string: 'image', id: 'AN_IMAGE' } );\n * ```\n *\n * @internal\n * @param language Target language.\n * @param message A message that will be translated.\n * @param quantity The number of elements for which a plural form should be picked from the target language dictionary.\n * @returns Translated sentence.\n */\nexport function _translate( language: string, message: Message, quantity: number = 1 ): string {\n\tif ( typeof quantity !== 'number' ) {\n\t\t/**\n\t\t * An incorrect value was passed to the translation function. This was probably caused\n\t\t * by an incorrect message interpolation of a plural form. Note that for messages supporting plural forms\n\t\t * the second argument of the `t()` function should always be a number or an array with a number as the first element.\n\t\t *\n\t\t * @error translation-service-quantity-not-a-number\n\t\t */\n\t\tthrow new CKEditorError( 'translation-service-quantity-not-a-number', null, { quantity } );\n\t}\n\n\tconst numberOfLanguages = getNumberOfLanguages();\n\n\tif ( numberOfLanguages === 1 ) {\n\t\t// Override the language to the only supported one.\n\t\t// This can't be done in the `Locale` class, because the translations comes after the `Locale` class initialization.\n\t\tlanguage = Object.keys( global.window.CKEDITOR_TRANSLATIONS )[ 0 ];\n\t}\n\n\tconst messageId = message.id || message.string;\n\n\tif ( numberOfLanguages === 0 || !hasTranslation( language, messageId ) ) {\n\t\tif ( quantity !== 1 ) {\n\t\t\t// Return the default plural form that was passed in the `message.plural` parameter.\n\t\t\treturn message.plural!;\n\t\t}\n\n\t\treturn message.string;\n\t}\n\n\tconst dictionary = global.window.CKEDITOR_TRANSLATIONS[ language ].dictionary;\n\tconst getPluralForm = global.window.CKEDITOR_TRANSLATIONS[ language ].getPluralForm || ( n => n === 1 ? 0 : 1 );\n\tconst translation = dictionary[ messageId ];\n\n\tif ( typeof translation === 'string' ) {\n\t\treturn translation;\n\t}\n\n\tconst pluralFormIndex = Number( getPluralForm( quantity ) );\n\n\t// Note: The `translate` function is not responsible for replacing `%0, %1, ...` with values.\n\treturn translation[ pluralFormIndex ];\n}\n\n/**\n * Clears dictionaries for test purposes.\n *\n * @internal\n */\nexport function _clear(): void {\n\tglobal.window.CKEDITOR_TRANSLATIONS = {};\n}\n\n/**\n * Checks whether the dictionary exists and translation in that dictionary exists.\n */\nfunction hasTranslation( language: string, messageId: string ): boolean {\n\treturn (\n\t\t!!global.window.CKEDITOR_TRANSLATIONS[ language ] &&\n\t\t!!global.window.CKEDITOR_TRANSLATIONS[ language ].dictionary[ messageId ]\n\t);\n}\n\nfunction getNumberOfLanguages(): number {\n\treturn Object.keys( global.window.CKEDITOR_TRANSLATIONS ).length;\n}\n\n/**\n * The internationalization message interface. A message that implements this interface can be passed to the `t()` function\n * to be translated to the target UI language.\n */\nexport interface Message {\n\n\t/**\n\t * The message string to translate. Acts as a default translation if the translation for a given language\n\t * is not defined. When the message is supposed to support plural forms, the string should be the English singular form of the message.\n\t */\n\treadonly string: string;\n\n\t/**\n\t * The message ID. If passed, the message ID is taken from this property instead of the `message.string`.\n\t * This property is useful when various messages share the same message string, for example, the `editor` string in `in the editor`\n\t * and `my editor` sentences.\n\t */\n\treadonly id?: string;\n\n\t/**\n\t * The plural form of the message. This property should be skipped when a message is not supposed\n\t * to support plural forms. Otherwise it should always be set to a string with the English plural form of the message.\n\t */\n\treadonly plural?: string;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/language\n */\n\n/**\n * String representing a language direction.\n */\nexport type LanguageDirection = 'ltr' | 'rtl';\n\nconst RTL_LANGUAGE_CODES = [\n\t'ar', 'ara', // Arabic\n\t'fa', 'per', 'fas', // Persian\n\t'he', 'heb', // Hebrew\n\t'ku', 'kur', // Kurdish\n\t'ug', 'uig' // Uighur, Uyghur\n];\n\n/**\n * Helps determine whether a language text direction is LTR or RTL.\n *\n * @param languageCode The ISO 639-1 or ISO 639-2 language code.\n */\nexport function getLanguageDirection( languageCode: string ): LanguageDirection {\n\treturn RTL_LANGUAGE_CODES.includes( languageCode ) ? 'rtl' : 'ltr';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/locale\n */\n\n/* globals console */\n\nimport toArray from './toarray';\nimport { _translate, type Message } from './translation-service';\nimport { getLanguageDirection, type LanguageDirection } from './language';\n\n/**\n * Represents the localization services.\n */\nexport default class Locale {\n\t/**\n\t * The editor UI language code in the [ISO 639-1](https://en.wikipedia.org/wiki/ISO_639-1) format.\n\t *\n\t * If the {@link #contentLanguage content language} was not specified in the `Locale` constructor,\n\t * it also defines the language of the content.\n\t */\n\tpublic readonly uiLanguage: string;\n\n\t/**\n\t * Text direction of the {@link #uiLanguage editor UI language}. Either `'ltr'` or `'rtl'`.\n\t */\n\tpublic readonly uiLanguageDirection: LanguageDirection;\n\n\t/**\n\t * The editor content language code in the [ISO 639-1](https://en.wikipedia.org/wiki/ISO_639-1) format.\n\t *\n\t * Usually the same as the {@link #uiLanguage editor language}, it can be customized by passing an optional\n\t * argument to the `Locale` constructor.\n\t */\n\tpublic readonly contentLanguage: string;\n\n\t/**\n\t * Text direction of the {@link #contentLanguage editor content language}.\n\t *\n\t * If the content language was passed directly to the `Locale` constructor, this property represents the\n\t * direction of that language.\n\t *\n\t * If the {@link #contentLanguage editor content language} was derived from the {@link #uiLanguage editor language},\n\t * the content language direction is the same as the {@link #uiLanguageDirection UI language direction}.\n\t *\n\t * The value is either `'ltr'` or `'rtl'`.\n\t */\n\tpublic readonly contentLanguageDirection: LanguageDirection;\n\n\t/**\n\t * Translates the given message to the {@link #uiLanguage}. This method is also available in\n\t * {@link module:core/editor/editor~Editor#t `Editor`} and {@link module:ui/view~View#t `View`}.\n\t *\n\t * This method's context is statically bound to the `Locale` instance and **should always be called as a function**:\n\t *\n\t * ```ts\n\t * const t = locale.t;\n\t * t( 'Label' );\n\t * ```\n\t *\n\t * The message can be either a string or an object implementing the {@link module:utils/translation-service~Message} interface.\n\t *\n\t * The message may contain placeholders (`%<index>`) for value(s) that are passed as a `values` parameter.\n\t * For an array of values, the `%<index>` will be changed to an element of that array at the given index.\n\t * For a single value passed as the second argument, only the `%0` placeholders will be changed to the provided value.\n\t *\n\t * ```ts\n\t * t( 'Created file \"%0\" in %1ms.', [ fileName, timeTaken ] );\n\t * t( 'Created file \"%0\", fileName );\n\t * ```\n\t *\n\t * The message supports plural forms. To specify the plural form, use the `plural` property. Singular or plural form\n\t * will be chosen depending on the first value from the passed `values`. The value of the `plural` property is used\n\t * as a default plural translation when the translation for the target language is missing.\n\t *\n\t * ```ts\n\t * t( { string: 'Add a space', plural: 'Add %0 spaces' }, 1 ); // 'Add a space' for the English language.\n\t * t( { string: 'Add a space', plural: 'Add %0 spaces' }, 5 ); // 'Add 5 spaces' for the English language.\n\t * t( { string: '%1 a space', plural: '%1 %0 spaces' }, [ 2, 'Add' ] ); // 'Add 2 spaces' for the English language.\n\t *\n\t * t( { string: 'Add a space', plural: 'Add %0 spaces' }, 1 ); // 'Dodaj spację' for the Polish language.\n\t * t( { string: 'Add a space', plural: 'Add %0 spaces' }, 5 ); // 'Dodaj 5 spacji' for the Polish language.\n\t * t( { string: '%1 a space', plural: '%1 %0 spaces' }, [ 2, 'Add' ] ); // 'Dodaj 2 spacje' for the Polish language.\n\t * ```\n\t *\n\t * * The message should provide an ID using the `id` property when the message strings are not unique and their\n\t * translations should be different.\n\t *\n\t * ```ts\n\t * translate( 'en', { string: 'image', id: 'ADD_IMAGE' } );\n\t * translate( 'en', { string: 'image', id: 'AN_IMAGE' } );\n\t * ```\n\t */\n\tpublic readonly t: LocaleTranslate;\n\n\t/**\n\t * Creates a new instance of the locale class. Learn more about\n\t * {@glink features/ui-language configuring the language of the editor}.\n\t *\n\t * @param options Locale configuration.\n\t * @param options.uiLanguage The editor UI language code in the\n\t * [ISO 639-1](https://en.wikipedia.org/wiki/ISO_639-1) format. See {@link #uiLanguage}.\n\t * @param options.contentLanguage The editor content language code in the\n\t * [ISO 639-1](https://en.wikipedia.org/wiki/ISO_639-1) format. If not specified, the same as `options.language`.\n\t * See {@link #contentLanguage}.\n\t */\n\tconstructor( { uiLanguage = 'en', contentLanguage }: { readonly uiLanguage?: string; readonly contentLanguage?: string } = {} ) {\n\t\tthis.uiLanguage = uiLanguage;\n\t\tthis.contentLanguage = contentLanguage || this.uiLanguage;\n\t\tthis.uiLanguageDirection = getLanguageDirection( this.uiLanguage );\n\t\tthis.contentLanguageDirection = getLanguageDirection( this.contentLanguage );\n\n\t\tthis.t = ( message, values ) => this._t( message, values );\n\t}\n\n\t/**\n\t * The editor UI language code in the [ISO 639-1](https://en.wikipedia.org/wiki/ISO_639-1) format.\n\t *\n\t * **Note**: This property was deprecated. Please use {@link #uiLanguage} and {@link #contentLanguage}\n\t * properties instead.\n\t *\n\t * @deprecated\n\t */\n\tpublic get language(): string {\n\t\t/**\n\t\t * The {@link module:utils/locale~Locale#language `Locale#language`} property was deprecated and will\n\t\t * be removed in the near future. Please use the {@link module:utils/locale~Locale#uiLanguage `Locale#uiLanguage`} and\n\t\t * {@link module:utils/locale~Locale#contentLanguage `Locale#contentLanguage`} properties instead.\n\t\t *\n\t\t * @error locale-deprecated-language-property\n\t\t */\n\t\tconsole.warn(\n\t\t\t'locale-deprecated-language-property: ' +\n\t\t\t'The Locale#language property has been deprecated and will be removed in the near future. ' +\n\t\t\t'Please use #uiLanguage and #contentLanguage properties instead.' );\n\n\t\treturn this.uiLanguage;\n\t}\n\n\t/**\n\t * An unbound version of the {@link #t} method.\n\t */\n\tprivate _t( message: string | Message, values: number | string | ReadonlyArray<number | string> = [] ): string {\n\t\tvalues = toArray( values );\n\n\t\tif ( typeof message === 'string' ) {\n\t\t\tmessage = { string: message };\n\t\t}\n\n\t\tconst hasPluralForm = !!message.plural;\n\t\tconst quantity = hasPluralForm ? values[ 0 ] as number : 1;\n\n\t\tconst translatedString = _translate( this.uiLanguage, message, quantity );\n\n\t\treturn interpolateString( translatedString, values );\n\t}\n}\n\n/**\n * @param message A message that will be localized (translated).\n * @param values A value or an array of values that will fill message placeholders.\n * For messages supporting plural forms the first value will determine the plural form.\n */\nexport type LocaleTranslate = ( message: string | Message, values?: number | string | ReadonlyArray<number | string> ) => string;\n\n/**\n * Fills the `%0, %1, ...` string placeholders with values.\n */\nfunction interpolateString( string: string, values: ReadonlyArray<any> ): string {\n\treturn string.replace( /%(\\d+)/g, ( match, index ) => {\n\t\treturn ( index < values.length ) ? values[ index ] : match;\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/collection\n */\n\nimport EmitterMixin from './emittermixin';\nimport CKEditorError from './ckeditorerror';\nimport uid from './uid';\nimport isIterable from './isiterable';\n\n/**\n * Collections are ordered sets of objects. Items in the collection can be retrieved by their indexes\n * in the collection (like in an array) or by their ids.\n *\n * If an object without an `id` property is being added to the collection, the `id` property will be generated\n * automatically. Note that the automatically generated id is unique only within this single collection instance.\n *\n * By default an item in the collection is identified by its `id` property. The name of the identifier can be\n * configured through the constructor of the collection.\n *\n * @typeParam T The type of the collection element.\n */\nexport default class Collection<T extends Record<string, any>> extends EmitterMixin() implements Iterable<T> {\n\t/**\n\t * The internal list of items in the collection.\n\t */\n\tprivate readonly _items: Array<T>;\n\n\t/**\n\t * The internal map of items in the collection.\n\t */\n\tprivate readonly _itemMap: Map<string, T>;\n\n\t/**\n\t * The name of the property which is considered to identify an item.\n\t */\n\tprivate readonly _idProperty: string;\n\n\t/**\n\t * A collection instance this collection is bound to as a result\n\t * of calling {@link #bindTo} method.\n\t */\n\tprivate _bindToCollection?: Collection<any> | null;\n\n\t/**\n\t * A helper mapping external items of a bound collection ({@link #bindTo})\n\t * and actual items of this collection. It provides information\n\t * necessary to properly remove items bound to another collection.\n\t *\n\t * See {@link #_bindToInternalToExternalMap}.\n\t */\n\tprivate readonly _bindToExternalToInternalMap: WeakMap<any, T>;\n\n\t/**\n\t * A helper mapping items of this collection to external items of a bound collection\n\t * ({@link #bindTo}). It provides information necessary to manage the bindings, e.g.\n\t * to avoid loops in twoway bindings.\n\t *\n\t * See {@link #_bindToExternalToInternalMap}.\n\t */\n\tprivate readonly _bindToInternalToExternalMap: WeakMap<T, any>;\n\n\t/**\n\t * Stores indexes of skipped items from bound external collection.\n\t */\n\tprivate _skippedIndexesFromExternal: Array<number>;\n\n\t/**\n\t * Creates a new Collection instance.\n\t *\n\t * You can pass a configuration object as the argument of the constructor:\n\t *\n\t * ```ts\n\t * const emptyCollection = new Collection<{ name: string }>( { idProperty: 'name' } );\n\t * emptyCollection.add( { name: 'John' } );\n\t * console.log( collection.get( 'John' ) ); // -> { name: 'John' }\n\t * ```\n\t *\n\t * The collection is empty by default. You can add new items using the {@link #add} method:\n\t *\n\t * ```ts\n\t * const collection = new Collection<{ id: string }>();\n\t *\n\t * collection.add( { id: 'John' } );\n\t * console.log( collection.get( 0 ) ); // -> { id: 'John' }\n\t * ```\n\t *\n\t * @label NO_ITEMS\n\t * @param options The options object.\n\t * @param options.idProperty The name of the property which is used to identify an item.\n\t * Items that do not have such a property will be assigned one when added to the collection.\n\t */\n\tconstructor( options?: { readonly idProperty?: string } );\n\n\t/**\n\t * Creates a new Collection instance with specified initial items.\n\t *\n\t * ```ts\n\t * const collection = new Collection<{ id: string }>( [ { id: 'John' }, { id: 'Mike' } ] );\n\t *\n\t * console.log( collection.get( 0 ) ); // -> { id: 'John' }\n\t * console.log( collection.get( 1 ) ); // -> { id: 'Mike' }\n\t * console.log( collection.get( 'Mike' ) ); // -> { id: 'Mike' }\n\t * ```\n\t *\n\t * You can always pass a configuration object as the last argument of the constructor:\n\t *\n\t * ```ts\n\t * const nonEmptyCollection = new Collection<{ name: string }>( [ { name: 'John' } ], { idProperty: 'name' } );\n\t * nonEmptyCollection.add( { name: 'George' } );\n\t * console.log( collection.get( 'George' ) ); // -> { name: 'George' }\n\t * console.log( collection.get( 'John' ) ); // -> { name: 'John' }\n\t * ```\n\t *\n\t * @label INITIAL_ITEMS\n\t * @param initialItems The initial items of the collection.\n\t * @param options The options object.\n\t * @param options.idProperty The name of the property which is used to identify an item.\n\t * Items that do not have such a property will be assigned one when added to the collection.\n\t */\n\tconstructor( initialItems: Iterable<T>, options?: { readonly idProperty?: string } );\n\n\tconstructor(\n\t\tinitialItemsOrOptions: Iterable<T> | { readonly idProperty?: string } = {},\n\t\toptions: { readonly idProperty?: string } = {}\n\t) {\n\t\tsuper();\n\n\t\tconst hasInitialItems = isIterable( initialItemsOrOptions );\n\n\t\tif ( !hasInitialItems ) {\n\t\t\toptions = initialItemsOrOptions;\n\t\t}\n\n\t\tthis._items = [];\n\t\tthis._itemMap = new Map();\n\t\tthis._idProperty = options.idProperty || 'id';\n\t\tthis._bindToExternalToInternalMap = new WeakMap();\n\t\tthis._bindToInternalToExternalMap = new WeakMap();\n\t\tthis._skippedIndexesFromExternal = [];\n\n\t\t// Set the initial content of the collection (if provided in the constructor).\n\t\tif ( hasInitialItems ) {\n\t\t\tfor ( const item of initialItemsOrOptions ) {\n\t\t\t\tthis._items.push( item );\n\t\t\t\tthis._itemMap.set( this._getItemIdBeforeAdding( item ), item );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * The number of items available in the collection.\n\t */\n\tpublic get length(): number {\n\t\treturn this._items.length;\n\t}\n\n\t/**\n\t * Returns the first item from the collection or null when collection is empty.\n\t */\n\tpublic get first(): T | null {\n\t\treturn this._items[ 0 ] || null;\n\t}\n\n\t/**\n\t * Returns the last item from the collection or null when collection is empty.\n\t */\n\tpublic get last(): T | null {\n\t\treturn this._items[ this.length - 1 ] || null;\n\t}\n\n\t/**\n\t * Adds an item into the collection.\n\t *\n\t * If the item does not have an id, then it will be automatically generated and set on the item.\n\t *\n\t * @param item\n\t * @param index The position of the item in the collection. The item\n\t * is pushed to the collection when `index` not specified.\n\t * @fires add\n\t * @fires change\n\t */\n\tpublic add( item: T, index?: number ): this {\n\t\treturn this.addMany( [ item ], index );\n\t}\n\n\t/**\n\t * Adds multiple items into the collection.\n\t *\n\t * Any item not containing an id will get an automatically generated one.\n\t *\n\t * @param items\n\t * @param index The position of the insertion. Items will be appended if no `index` is specified.\n\t * @fires add\n\t * @fires change\n\t */\n\tpublic addMany( items: Iterable<T>, index?: number ): this {\n\t\tif ( index === undefined ) {\n\t\t\tindex = this._items.length;\n\t\t} else if ( index > this._items.length || index < 0 ) {\n\t\t\t/**\n\t\t\t * The `index` passed to {@link module:utils/collection~Collection#addMany `Collection#addMany()`}\n\t\t\t * is invalid. It must be a number between 0 and the collection's length.\n\t\t\t *\n\t\t\t * @error collection-add-item-invalid-index\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'collection-add-item-invalid-index', this );\n\t\t}\n\n\t\tlet offset = 0;\n\n\t\tfor ( const item of items ) {\n\t\t\tconst itemId = this._getItemIdBeforeAdding( item );\n\t\t\tconst currentItemIndex = index + offset;\n\n\t\t\tthis._items.splice( currentItemIndex, 0, item );\n\t\t\tthis._itemMap.set( itemId, item );\n\n\t\t\tthis.fire<CollectionAddEvent<T>>( 'add', item, currentItemIndex );\n\n\t\t\toffset++;\n\t\t}\n\n\t\tthis.fire<CollectionChangeEvent<T>>( 'change', {\n\t\t\tadded: items,\n\t\t\tremoved: [],\n\t\t\tindex\n\t\t} );\n\n\t\treturn this;\n\t}\n\n\t/**\n\t * Gets an item by its ID or index.\n\t *\n\t * @param idOrIndex The item ID or index in the collection.\n\t * @returns The requested item or `null` if such item does not exist.\n\t */\n\tpublic get( idOrIndex: string | number ): T | null {\n\t\tlet item: T | undefined;\n\n\t\tif ( typeof idOrIndex == 'string' ) {\n\t\t\titem = this._itemMap.get( idOrIndex );\n\t\t} else if ( typeof idOrIndex == 'number' ) {\n\t\t\titem = this._items[ idOrIndex ];\n\t\t} else {\n\t\t\t/**\n\t\t\t * An index or ID must be given.\n\t\t\t *\n\t\t\t * @error collection-get-invalid-arg\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'collection-get-invalid-arg', this );\n\t\t}\n\n\t\treturn item || null;\n\t}\n\n\t/**\n\t * Returns a Boolean indicating whether the collection contains an item.\n\t *\n\t * @param itemOrId The item or its ID in the collection.\n\t * @returns `true` if the collection contains the item, `false` otherwise.\n\t */\n\tpublic has( itemOrId: T | string ): boolean {\n\t\tif ( typeof itemOrId == 'string' ) {\n\t\t\treturn this._itemMap.has( itemOrId );\n\t\t} else { // Object\n\t\t\tconst idProperty = this._idProperty;\n\t\t\tconst id = itemOrId[ idProperty ];\n\n\t\t\treturn id && this._itemMap.has( id );\n\t\t}\n\t}\n\n\t/**\n\t * Gets an index of an item in the collection.\n\t * When an item is not defined in the collection, the index will equal -1.\n\t *\n\t * @param itemOrId The item or its ID in the collection.\n\t * @returns The index of a given item.\n\t */\n\tpublic getIndex( itemOrId: T | string ): number {\n\t\tlet item: T | undefined;\n\n\t\tif ( typeof itemOrId == 'string' ) {\n\t\t\titem = this._itemMap.get( itemOrId );\n\t\t} else {\n\t\t\titem = itemOrId;\n\t\t}\n\n\t\treturn item ? this._items.indexOf( item ) : -1;\n\t}\n\n\t/**\n\t * Removes an item from the collection.\n\t *\n\t * @param subject The item to remove, its ID or index in the collection.\n\t * @returns The removed item.\n\t * @fires remove\n\t * @fires change\n\t */\n\tpublic remove( subject: T | number | string ): T {\n\t\tconst [ item, index ] = this._remove( subject );\n\n\t\tthis.fire<CollectionChangeEvent<T>>( 'change', {\n\t\t\tadded: [],\n\t\t\tremoved: [ item ],\n\t\t\tindex\n\t\t} );\n\n\t\treturn item;\n\t}\n\n\t/**\n\t * Executes the callback for each item in the collection and composes an array or values returned by this callback.\n\t *\n\t * @typeParam U The result type of the callback.\n\t * @param callback\n\t * @param ctx Context in which the `callback` will be called.\n\t * @returns The result of mapping.\n\t */\n\tpublic map<U>(\n\t\tcallback: ( item: T, index: number ) => U,\n\t\tctx?: any\n\t): Array<U> {\n\t\treturn this._items.map( callback, ctx );\n\t}\n\n\t/**\n\t * Finds the first item in the collection for which the `callback` returns a true value.\n\t *\n\t * @param callback\n\t * @param ctx Context in which the `callback` will be called.\n\t * @returns The item for which `callback` returned a true value.\n\t */\n\tpublic find(\n\t\tcallback: ( item: T, index: number ) => boolean,\n\t\tctx?: any\n\t): T | undefined {\n\t\treturn this._items.find( callback, ctx );\n\t}\n\n\t/**\n\t * Returns an array with items for which the `callback` returned a true value.\n\t *\n\t * @param callback\n\t * @param ctx Context in which the `callback` will be called.\n\t * @returns The array with matching items.\n\t */\n\tpublic filter(\n\t\tcallback: ( item: T, index: number ) => boolean,\n\t\tctx?: any\n\t): Array<T> {\n\t\treturn this._items.filter( callback, ctx );\n\t}\n\n\t/**\n\t * Removes all items from the collection and destroys the binding created using\n\t * {@link #bindTo}.\n\t *\n\t * @fires remove\n\t * @fires change\n\t */\n\tpublic clear(): void {\n\t\tif ( this._bindToCollection ) {\n\t\t\tthis.stopListening( this._bindToCollection );\n\t\t\tthis._bindToCollection = null;\n\t\t}\n\n\t\tconst removedItems = Array.from( this._items );\n\n\t\twhile ( this.length ) {\n\t\t\tthis._remove( 0 );\n\t\t}\n\n\t\tthis.fire<CollectionChangeEvent<T>>( 'change', {\n\t\t\tadded: [],\n\t\t\tremoved: removedItems,\n\t\t\tindex: 0\n\t\t} );\n\t}\n\n\t/**\n\t * Binds and synchronizes the collection with another one.\n\t *\n\t * The binding can be a simple factory:\n\t *\n\t * ```ts\n\t * class FactoryClass {\n\t * \tpublic label: string;\n\t *\n\t * \tconstructor( data: { label: string } ) {\n\t * \t\tthis.label = data.label;\n\t * \t}\n\t * }\n\t *\n\t * const source = new Collection<{ label: string }>( { idProperty: 'label' } );\n\t * const target = new Collection<FactoryClass>();\n\t *\n\t * target.bindTo( source ).as( FactoryClass );\n\t *\n\t * source.add( { label: 'foo' } );\n\t * source.add( { label: 'bar' } );\n\t *\n\t * console.log( target.length ); // 2\n\t * console.log( target.get( 1 ).label ); // 'bar'\n\t *\n\t * source.remove( 0 );\n\t * console.log( target.length ); // 1\n\t * console.log( target.get( 0 ).label ); // 'bar'\n\t * ```\n\t *\n\t * or the factory driven by a custom callback:\n\t *\n\t * ```ts\n\t * class FooClass {\n\t * \tpublic label: string;\n\t *\n\t * \tconstructor( data: { label: string } ) {\n\t * \t\tthis.label = data.label;\n\t * \t}\n\t * }\n\t *\n\t * class BarClass {\n\t * \tpublic label: string;\n\t *\n\t * \tconstructor( data: { label: string } ) {\n\t * \t\tthis.label = data.label;\n\t * \t}\n\t * }\n\t *\n\t * const source = new Collection<{ label: string }>( { idProperty: 'label' } );\n\t * const target = new Collection<FooClass | BarClass>();\n\t *\n\t * target.bindTo( source ).using( ( item ) => {\n\t * \tif ( item.label == 'foo' ) {\n\t * \t\treturn new FooClass( item );\n\t * \t} else {\n\t * \t\treturn new BarClass( item );\n\t * \t}\n\t * } );\n\t *\n\t * source.add( { label: 'foo' } );\n\t * source.add( { label: 'bar' } );\n\t *\n\t * console.log( target.length ); // 2\n\t * console.log( target.get( 0 ) instanceof FooClass ); // true\n\t * console.log( target.get( 1 ) instanceof BarClass ); // true\n\t * ```\n\t *\n\t * or the factory out of property name:\n\t *\n\t * ```ts\n\t * const source = new Collection<{ nested: { value: string } }>();\n\t * const target = new Collection<{ value: string }>();\n\t *\n\t * target.bindTo( source ).using( 'nested' );\n\t *\n\t * source.add( { nested: { value: 'foo' } } );\n\t * source.add( { nested: { value: 'bar' } } );\n\t *\n\t * console.log( target.length ); // 2\n\t * console.log( target.get( 0 ).value ); // 'foo'\n\t * console.log( target.get( 1 ).value ); // 'bar'\n\t * ```\n\t *\n\t * It's possible to skip specified items by returning null value:\n\t *\n\t * ```ts\n\t * const source = new Collection<{ hidden: boolean }>();\n\t * const target = new Collection<{ hidden: boolean }>();\n\t *\n\t * target.bindTo( source ).using( item => {\n\t * \tif ( item.hidden ) {\n\t * \t\treturn null;\n\t * \t}\n\t *\n\t * \treturn item;\n\t * } );\n\t *\n\t * source.add( { hidden: true } );\n\t * source.add( { hidden: false } );\n\t *\n\t * console.log( source.length ); // 2\n\t * console.log( target.length ); // 1\n\t * ```\n\t *\n\t * **Note**: {@link #clear} can be used to break the binding.\n\t *\n\t * @typeParam S The type of `externalCollection` element.\n\t * @param externalCollection A collection to be bound.\n\t * @returns The binding chain object.\n\t */\n\tpublic bindTo<S extends Record<string, any>>(\n\t\texternalCollection: Collection<S>\n\t): CollectionBindToChain<S, T> {\n\t\tif ( this._bindToCollection ) {\n\t\t\t/**\n\t\t\t * The collection cannot be bound more than once.\n\t\t\t *\n\t\t\t * @error collection-bind-to-rebind\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'collection-bind-to-rebind', this );\n\t\t}\n\n\t\tthis._bindToCollection = externalCollection;\n\n\t\treturn {\n\t\t\tas: Class => {\n\t\t\t\tthis._setUpBindToBinding<S>( item => new Class( item ) );\n\t\t\t},\n\n\t\t\tusing: callbackOrProperty => {\n\t\t\t\tif ( typeof callbackOrProperty == 'function' ) {\n\t\t\t\t\tthis._setUpBindToBinding<S>( callbackOrProperty );\n\t\t\t\t} else {\n\t\t\t\t\tthis._setUpBindToBinding<S>( item => item[ callbackOrProperty ] );\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\t}\n\n\t/**\n\t * Finalizes and activates a binding initiated by {@link #bindTo}.\n\t *\n\t * @param factory A function which produces collection items.\n\t */\n\tprivate _setUpBindToBinding<S>( factory: ( item: S ) => T | null ): void {\n\t\tconst externalCollection = this._bindToCollection!;\n\n\t\t// Adds the item to the collection once a change has been done to the external collection.\n\t\tconst addItem = ( evt: unknown, externalItem: S, index: number ) => {\n\t\t\tconst isExternalBoundToThis = externalCollection._bindToCollection == this;\n\t\t\tconst externalItemBound = externalCollection._bindToInternalToExternalMap.get( externalItem );\n\n\t\t\t// If an external collection is bound to this collection, which makes it a 2way binding,\n\t\t\t// and the particular external collection item is already bound, don't add it here.\n\t\t\t// The external item has been created **out of this collection's item** and (re)adding it will\n\t\t\t// cause a loop.\n\t\t\tif ( isExternalBoundToThis && externalItemBound ) {\n\t\t\t\tthis._bindToExternalToInternalMap.set( externalItem, externalItemBound );\n\t\t\t\tthis._bindToInternalToExternalMap.set( externalItemBound, externalItem );\n\t\t\t} else {\n\t\t\t\tconst item = factory( externalItem );\n\n\t\t\t\t// When there is no item we need to remember skipped index first and then we can skip this item.\n\t\t\t\tif ( !item ) {\n\t\t\t\t\tthis._skippedIndexesFromExternal.push( index );\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// Lets try to put item at the same index as index in external collection\n\t\t\t\t// but when there are a skipped items in one or both collections we need to recalculate this index.\n\t\t\t\tlet finalIndex = index;\n\n\t\t\t\t// When we try to insert item after some skipped items from external collection we need\n\t\t\t\t// to include this skipped items and decrease index.\n\t\t\t\t//\n\t\t\t\t// For the following example:\n\t\t\t\t// external -> [ 'A', 'B - skipped for internal', 'C - skipped for internal' ]\n\t\t\t\t// internal -> [ A ]\n\t\t\t\t//\n\t\t\t\t// Another item is been added at the end of external collection:\n\t\t\t\t// external.add( 'D' )\n\t\t\t\t// external -> [ 'A', 'B - skipped for internal', 'C - skipped for internal', 'D' ]\n\t\t\t\t//\n\t\t\t\t// We can't just add 'D' to internal at the same index as index in external because\n\t\t\t\t// this will produce empty indexes what is invalid:\n\t\t\t\t// internal -> [ 'A', empty, empty, 'D' ]\n\t\t\t\t//\n\t\t\t\t// So we need to include skipped items and decrease index\n\t\t\t\t// internal -> [ 'A', 'D' ]\n\t\t\t\tfor ( const skipped of this._skippedIndexesFromExternal ) {\n\t\t\t\t\tif ( index > skipped ) {\n\t\t\t\t\t\tfinalIndex--;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// We need to take into consideration that external collection could skip some items from\n\t\t\t\t// internal collection.\n\t\t\t\t//\n\t\t\t\t// For the following example:\n\t\t\t\t// internal -> [ 'A', 'B - skipped for external', 'C - skipped for external' ]\n\t\t\t\t// external -> [ A ]\n\t\t\t\t//\n\t\t\t\t// Another item is been added at the end of external collection:\n\t\t\t\t// external.add( 'D' )\n\t\t\t\t// external -> [ 'A', 'D' ]\n\t\t\t\t//\n\t\t\t\t// We need to include skipped items and place new item after them:\n\t\t\t\t// internal -> [ 'A', 'B - skipped for external', 'C - skipped for external', 'D' ]\n\t\t\t\tfor ( const skipped of externalCollection._skippedIndexesFromExternal ) {\n\t\t\t\t\tif ( finalIndex >= skipped ) {\n\t\t\t\t\t\tfinalIndex++;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tthis._bindToExternalToInternalMap.set( externalItem, item );\n\t\t\t\tthis._bindToInternalToExternalMap.set( item, externalItem );\n\t\t\t\tthis.add( item, finalIndex );\n\n\t\t\t\t// After adding new element to internal collection we need update indexes\n\t\t\t\t// of skipped items in external collection.\n\t\t\t\tfor ( let i = 0; i < externalCollection._skippedIndexesFromExternal.length; i++ ) {\n\t\t\t\t\tif ( finalIndex <= externalCollection._skippedIndexesFromExternal[ i ] ) {\n\t\t\t\t\t\texternalCollection._skippedIndexesFromExternal[ i ]++;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\n\t\t// Load the initial content of the collection.\n\t\tfor ( const externalItem of externalCollection ) {\n\t\t\taddItem( null, externalItem, externalCollection.getIndex( externalItem ) );\n\t\t}\n\n\t\t// Synchronize the with collection as new items are added.\n\t\tthis.listenTo<CollectionAddEvent<S>>( externalCollection, 'add', addItem );\n\n\t\t// Synchronize the with collection as new items are removed.\n\t\tthis.listenTo<CollectionRemoveEvent<S>>( externalCollection, 'remove', ( evt, externalItem, index ) => {\n\t\t\tconst item = this._bindToExternalToInternalMap.get( externalItem );\n\n\t\t\tif ( item ) {\n\t\t\t\tthis.remove( item );\n\t\t\t}\n\n\t\t\t// After removing element from external collection we need update/remove indexes\n\t\t\t// of skipped items in internal collection.\n\t\t\tthis._skippedIndexesFromExternal = this._skippedIndexesFromExternal.reduce( ( result, skipped ) => {\n\t\t\t\tif ( index < skipped ) {\n\t\t\t\t\tresult.push( skipped - 1 );\n\t\t\t\t}\n\n\t\t\t\tif ( index > skipped ) {\n\t\t\t\t\tresult.push( skipped );\n\t\t\t\t}\n\n\t\t\t\treturn result;\n\t\t\t}, [] as Array<number> );\n\t\t} );\n\t}\n\n\t/**\n\t * Returns an unique id property for a given `item`.\n\t *\n\t * The method will generate new id and assign it to the `item` if it doesn't have any.\n\t *\n\t * @param item Item to be added.\n\t */\n\tprivate _getItemIdBeforeAdding( item: any ): string {\n\t\tconst idProperty = this._idProperty;\n\t\tlet itemId: string | undefined;\n\n\t\tif ( ( idProperty in item ) ) {\n\t\t\titemId = item[ idProperty ];\n\n\t\t\tif ( typeof itemId != 'string' ) {\n\t\t\t\t/**\n\t\t\t\t * This item's ID should be a string.\n\t\t\t\t *\n\t\t\t\t * @error collection-add-invalid-id\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'collection-add-invalid-id', this );\n\t\t\t}\n\n\t\t\tif ( this.get( itemId ) ) {\n\t\t\t\t/**\n\t\t\t\t * This item already exists in the collection.\n\t\t\t\t *\n\t\t\t\t * @error collection-add-item-already-exists\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'collection-add-item-already-exists', this );\n\t\t\t}\n\t\t} else {\n\t\t\titem[ idProperty ] = itemId = uid();\n\t\t}\n\n\t\treturn itemId;\n\t}\n\n\t/**\n\t * Core {@link #remove} method implementation shared in other functions.\n\t *\n\t * In contrast this method **does not** fire the {@link #event:change} event.\n\t *\n\t * @param subject The item to remove, its id or index in the collection.\n\t * @returns Returns an array with the removed item and its index.\n\t * @fires remove\n\t */\n\tprivate _remove( subject: T | number | string ): [ item: T, index: number ] {\n\t\tlet index: number, id: string, item: T;\n\t\tlet itemDoesNotExist = false;\n\t\tconst idProperty = this._idProperty;\n\n\t\tif ( typeof subject == 'string' ) {\n\t\t\tid = subject;\n\t\t\titem = this._itemMap.get( id )!;\n\t\t\titemDoesNotExist = !item;\n\n\t\t\tif ( item ) {\n\t\t\t\tindex = this._items.indexOf( item );\n\t\t\t}\n\t\t} else if ( typeof subject == 'number' ) {\n\t\t\tindex = subject;\n\t\t\titem = this._items[ index ];\n\t\t\titemDoesNotExist = !item;\n\n\t\t\tif ( item ) {\n\t\t\t\tid = item[ idProperty ]!;\n\t\t\t}\n\t\t} else {\n\t\t\titem = subject;\n\t\t\tid = item[ idProperty ]!;\n\t\t\tindex = this._items.indexOf( item );\n\t\t\titemDoesNotExist = ( index == -1 || !this._itemMap.get( id ) );\n\t\t}\n\n\t\tif ( itemDoesNotExist ) {\n\t\t\t/**\n\t\t\t * Item not found.\n\t\t\t *\n\t\t\t * @error collection-remove-404\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'collection-remove-404', this );\n\t\t}\n\n\t\tthis._items.splice( index!, 1 );\n\t\tthis._itemMap.delete( id! );\n\n\t\tconst externalItem = this._bindToInternalToExternalMap.get( item );\n\t\tthis._bindToInternalToExternalMap.delete( item );\n\t\tthis._bindToExternalToInternalMap.delete( externalItem );\n\n\t\tthis.fire<CollectionRemoveEvent<T>>( 'remove', item, index! );\n\n\t\treturn [ item, index! ];\n\t}\n\n\t/**\n\t * Iterable interface.\n\t */\n\tpublic [ Symbol.iterator ](): Iterator<T> {\n\t\treturn this._items[ Symbol.iterator ]();\n\t}\n}\n\n/**\n * Fired when an item is added to the collection.\n *\n * @eventName ~Collection#add\n * @param item The added item.\n * @param index An index where the addition occurred.\n */\nexport type CollectionAddEvent<T = any> = {\n\tname: 'add';\n\targs: [ item: T, index: number ];\n};\n\n/**\n * Fired when the collection was changed due to adding or removing items.\n *\n * @eventName ~Collection#change\n * @param data Changed items.\n */\nexport type CollectionChangeEvent<T = any> = {\n\tname: 'change';\n\targs: [ data: CollectionChangeEventData<T> ];\n};\n\n/**\n * A structure describing the {@link ~Collection#event:change `Collection#change`} event.\n */\nexport type CollectionChangeEventData<T = any> = {\n\n\t/**\n\t * A list of added items.\n\t */\n\tadded: Iterable<T>;\n\n\t/**\n\t * A list of removed items.\n\t */\n\tremoved: Iterable<T>;\n\n\t/**\n\t * An index where the addition or removal occurred.\n\t */\n\tindex: number;\n};\n\n/**\n * Fired when an item is removed from the collection.\n *\n * @eventName ~Collection#remove\n * @param item The removed item.\n * @param index Index from which item was removed.\n */\nexport type CollectionRemoveEvent<T = any> = {\n\tname: 'remove';\n\targs: [ item: T, index: number ];\n};\n\n/**\n * An object returned by the {@link module:utils/collection~Collection#bindTo `bindTo()`} method\n * providing functions that specify the type of the binding.\n *\n * See the {@link module:utils/collection~Collection#bindTo `bindTo()`} documentation for examples.\n */\nexport interface CollectionBindToChain<S, T> {\n\n\t/**\n\t * Creates the class factory binding in which items of the source collection are passed to\n\t * the constructor of the specified class.\n\t *\n\t * @param Class The class constructor used to create instances in the factory.\n\t */\n\tas( Class: new ( item: S ) => T ): void;\n\n\t/**\n\t * Creates a callback or a property binding.\n\t *\n\t * @param callbackOrProperty When the function is passed, it should return\n\t * the collection items. When the string is provided, the property value is used to create the bound collection items.\n\t */\n\tusing( callbackOrProperty: keyof S | ( ( item: S ) => T | null ) ): void;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/first\n */\n\n/**\n * Returns first item of the given `iterator`.\n */\nexport default function first<T>( iterator: Iterator<T> ): T | null {\n\tconst iteratorItem = iterator.next();\n\n\tif ( iteratorItem.done ) {\n\t\treturn null;\n\t}\n\n\treturn iteratorItem.value;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* global setTimeout, clearTimeout */\n\n/**\n * @module utils/focustracker\n */\n\nimport DomEmitterMixin from './dom/emittermixin';\nimport ObservableMixin from './observablemixin';\nimport CKEditorError from './ckeditorerror';\n\n/**\n * Allows observing a group of `Element`s whether at least one of them is focused.\n *\n * Used by the {@link module:core/editor/editor~Editor} in order to track whether the focus is still within the application,\n * or were used outside of its UI.\n *\n * **Note** `focus` and `blur` listeners use event capturing, so it is only needed to register wrapper `Element`\n * which contain other `focusable` elements. But note that this wrapper element has to be focusable too\n * (have e.g. `tabindex=\"-1\"`).\n *\n * Check out the {@glink framework/deep-dive/ui/focus-tracking \"Deep dive into focus tracking\"} guide to learn more.\n */\nexport default class FocusTracker extends DomEmitterMixin( ObservableMixin() ) {\n\t/**\n\t * True when one of the registered elements is focused.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public isFocused: boolean;\n\n\t/**\n\t * The currently focused element.\n\t *\n\t * While {@link #isFocused `isFocused`} remains `true`, the focus can\n\t * move between different UI elements. This property tracks those\n\t * elements and tells which one is currently focused.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public focusedElement: Element | null;\n\n\t/**\n\t * List of registered elements.\n\t *\n\t * @internal\n\t */\n\tpublic _elements: Set<Element> = new Set();\n\n\t/**\n\t * Event loop timeout.\n\t */\n\tprivate _nextEventLoopTimeout: ReturnType<typeof setTimeout> | null = null;\n\n\tconstructor() {\n\t\tsuper();\n\n\t\tthis.set( 'isFocused', false );\n\t\tthis.set( 'focusedElement', null );\n\t}\n\n\t/**\n\t * Starts tracking the specified element.\n\t */\n\tpublic add( element: Element ): void {\n\t\tif ( this._elements.has( element ) ) {\n\t\t\t/**\n\t\t\t * This element is already tracked by {@link module:utils/focustracker~FocusTracker}.\n\t\t\t *\n\t\t\t * @error focustracker-add-element-already-exist\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'focustracker-add-element-already-exist', this );\n\t\t}\n\n\t\tthis.listenTo( element, 'focus', () => this._focus( element ), { useCapture: true } );\n\t\tthis.listenTo( element, 'blur', () => this._blur(), { useCapture: true } );\n\t\tthis._elements.add( element );\n\t}\n\n\t/**\n\t * Stops tracking the specified element and stops listening on this element.\n\t */\n\tpublic remove( element: Element ): void {\n\t\tif ( element === this.focusedElement ) {\n\t\t\tthis._blur();\n\t\t}\n\n\t\tif ( this._elements.has( element ) ) {\n\t\t\tthis.stopListening( element );\n\t\t\tthis._elements.delete( element );\n\t\t}\n\t}\n\n\t/**\n\t * Destroys the focus tracker by:\n\t * - Disabling all event listeners attached to tracked elements.\n\t * - Removing all tracked elements that were previously added.\n\t */\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Stores currently focused element and set {@link #isFocused} as `true`.\n\t */\n\tprivate _focus( element: Element ): void {\n\t\tclearTimeout( this._nextEventLoopTimeout! );\n\n\t\tthis.focusedElement = element;\n\t\tthis.isFocused = true;\n\t}\n\n\t/**\n\t * Clears currently focused element and set {@link #isFocused} as `false`.\n\t * This method uses `setTimeout` to change order of fires `blur` and `focus` events.\n\t */\n\tprivate _blur(): void {\n\t\tclearTimeout( this._nextEventLoopTimeout! );\n\n\t\tthis._nextEventLoopTimeout = setTimeout( () => {\n\t\t\tthis.focusedElement = null;\n\t\t\tthis.isFocused = false;\n\t\t}, 0 );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/keystrokehandler\n */\n\nimport DomEmitterMixin, { type DomEmitter } from './dom/emittermixin';\nimport type { Emitter } from './emittermixin';\nimport { getCode, parseKeystroke, type KeystrokeInfo } from './keyboard';\nimport type { PriorityString } from './priorities';\n\n/**\n * Keystroke handler allows registering callbacks for given keystrokes.\n *\n * The most frequent use of this class is through the {@link module:core/editor/editor~Editor#keystrokes `editor.keystrokes`}\n * property. It allows listening to keystrokes executed in the editing view:\n *\n * ```ts\n * editor.keystrokes.set( 'Ctrl+A', ( keyEvtData, cancel ) => {\n * \tconsole.log( 'Ctrl+A has been pressed' );\n * \tcancel();\n * } );\n * ```\n *\n * However, this utility class can be used in various part of the UI. For instance, a certain {@link module:ui/view~View}\n * can use it like this:\n *\n * ```ts\n * class MyView extends View {\n * \tconstructor() {\n * \t\tthis.keystrokes = new KeystrokeHandler();\n *\n * \t\tthis.keystrokes.set( 'tab', handleTabKey );\n * \t}\n *\n * \trender() {\n * \t\tsuper.render();\n *\n * \t\tthis.keystrokes.listenTo( this.element );\n * \t}\n * }\n * ```\n *\n * That keystroke handler will listen to `keydown` events fired in this view's main element.\n *\n */\nexport default class KeystrokeHandler {\n\t/**\n\t * Listener used to listen to events for easier keystroke handler destruction.\n\t */\n\tprivate readonly _listener: DomEmitter;\n\n\t/**\n\t * Creates an instance of the keystroke handler.\n\t */\n\tconstructor() {\n\t\tthis._listener = new ( DomEmitterMixin() )();\n\t}\n\n\t/**\n\t * Starts listening for `keydown` events from a given emitter.\n\t */\n\tpublic listenTo( emitter: Emitter | HTMLElement | Window ): void {\n\t\t// The #_listener works here as a kind of dispatcher. It groups the events coming from the same\n\t\t// keystroke so the listeners can be attached to them with different priorities.\n\t\t//\n\t\t// E.g. all the keystrokes with the `keyCode` of 42 coming from the `emitter` are propagated\n\t\t// as a `_keydown:42` event by the `_listener`. If there's a callback created by the `set`\n\t\t// method for this 42 keystroke, it listens to the `_listener#_keydown:42` event only and interacts\n\t\t// only with other listeners of this particular event, thus making it possible to prioritize\n\t\t// the listeners and safely cancel execution, when needed. Instead of duplicating the Emitter logic,\n\t\t// the KeystrokeHandler reuses it to do its job.\n\t\tthis._listener.listenTo( emitter as HTMLElement | Window, 'keydown', ( evt, keyEvtData ) => {\n\t\t\tthis._listener.fire( '_keydown:' + getCode( keyEvtData ), keyEvtData );\n\t\t} );\n\t}\n\n\t/**\n\t * Registers a handler for the specified keystroke.\n\t *\n\t * @param keystroke Keystroke defined in a format accepted by\n\t * the {@link module:utils/keyboard~parseKeystroke} function.\n\t * @param callback A function called with the\n\t * {@link module:engine/view/observer/keyobserver~KeyEventData key event data} object and\n\t * a helper function to call both `preventDefault()` and `stopPropagation()` on the underlying event.\n\t * @param options Additional options.\n\t * @param options.priority The priority of the keystroke\n\t * callback. The higher the priority value the sooner the callback will be executed. Keystrokes having the same priority\n\t * are called in the order they were added.\n\t */\n\tpublic set(\n\t\tkeystroke: string | ReadonlyArray<string | number>,\n\t\tcallback: ( ev: KeyboardEvent, cancel: () => void ) => void,\n\t\toptions: { readonly priority?: PriorityString } = {}\n\t): void {\n\t\tconst keyCode = parseKeystroke( keystroke );\n\t\tconst priority = options.priority;\n\n\t\t// Execute the passed callback on KeystrokeHandler#_keydown.\n\t\t// TODO: https://github.com/ckeditor/ckeditor5-utils/issues/144\n\t\tthis._listener.listenTo( this._listener, '_keydown:' + keyCode, ( evt, keyEvtData: KeyboardEvent ) => {\n\t\t\tcallback( keyEvtData, () => {\n\t\t\t\t// Stop the event in the DOM: no listener in the web page\n\t\t\t\t// will be triggered by this event.\n\t\t\t\tkeyEvtData.preventDefault();\n\t\t\t\tkeyEvtData.stopPropagation();\n\n\t\t\t\t// Stop the event in the KeystrokeHandler: no more callbacks\n\t\t\t\t// will be executed for this keystroke.\n\t\t\t\tevt.stop();\n\t\t\t} );\n\n\t\t\t// Mark this keystroke as handled by the callback. See: #press.\n\t\t\tevt.return = true;\n\t\t}, { priority } );\n\t}\n\n\t/**\n\t * Triggers a keystroke handler for a specified key combination, if such a keystroke was {@link #set defined}.\n\t *\n\t * @param keyEvtData Key event data.\n\t * @returns Whether the keystroke was handled.\n\t */\n\tpublic press( keyEvtData: Readonly<KeystrokeInfo> ): boolean {\n\t\treturn !!this._listener.fire( '_keydown:' + getCode( keyEvtData ), keyEvtData );\n\t}\n\n\t/**\n\t * Stops listening to `keydown` events from the given emitter.\n\t */\n\tpublic stopListening( emitter?: Emitter | HTMLElement | Window ): void {\n\t\tthis._listener.stopListening( emitter );\n\t}\n\n\t/**\n\t * Destroys the keystroke handler.\n\t */\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/tomap\n */\n\nimport objectToMap from './objecttomap';\nimport isIterable from './isiterable';\n\n/**\n * Transforms object or iterable to map. Iterable needs to be in the format acceptable by the `Map` constructor.\n *\n * ```ts\n * map = toMap( { 'foo': 1, 'bar': 2 } );\n * map = toMap( [ [ 'foo', 1 ], [ 'bar', 2 ] ] );\n * map = toMap( anotherMap );\n * ```\n *\n * @param data Object or iterable to transform.\n * @returns Map created from data.\n */\nexport default function toMap<T>( data: { readonly [ key: string ]: T } | Iterable<readonly [ string, T ]> | null | undefined ):\n\tMap<string, T>\n{\n\tif ( isIterable( data ) ) {\n\t\treturn new Map( data );\n\t} else {\n\t\treturn objectToMap( data );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/objecttomap\n */\n\n/**\n * Transforms object to map.\n *\n * ```ts\n * const map = objectToMap( { 'foo': 1, 'bar': 2 } );\n * map.get( 'foo' ); // 1\n * ```\n *\n * **Note**: For mixed data (`Object` or `Iterable`) there's a dedicated {@link module:utils/tomap~toMap} function.\n *\n * @param obj Object to transform.\n * @returns Map created from object.\n */\nexport default function objectToMap<T>( obj: { readonly [ key: string ]: T } | null | undefined ): Map<string, T> {\n\tconst map = new Map<string, T>();\n\n\tfor ( const key in obj ) {\n\t\tmap.set( key, obj[ key ] );\n\t}\n\n\treturn map;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/delay\n */\n\n/* globals setTimeout, clearTimeout */\n\n/**\n * Returns a function wrapper that will trigger a function after a specified wait time.\n * The timeout can be canceled by calling the cancel function on the returned wrapped function.\n *\n * @param func The function to wrap.\n * @param wait The timeout in ms.\n */\nexport default function delay<T extends ( ...args: Array<any> ) => any>( func: T, wait: number ): DelayedFunc<T> {\n\tlet timer: ReturnType<typeof setTimeout>;\n\n\tfunction delayed( ...args: Parameters<T> ) {\n\t\tdelayed.cancel();\n\t\ttimer = setTimeout( () => func( ...args ), wait );\n\t}\n\n\tdelayed.cancel = () => {\n\t\tclearTimeout( timer );\n\t};\n\n\treturn delayed;\n}\n\nexport interface DelayedFunc<T extends ( ...args: Array<any> ) => any> {\n\t( ...args: Parameters<T> ): void;\n\tcancel(): void;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * Set of utils to handle unicode characters.\n *\n * @module utils/unicode\n */\n\n/**\n * Checks whether given `character` is a combining mark.\n *\n * @param character Character to check.\n */\nexport function isCombiningMark( character: string ): boolean {\n\t// eslint-disable-next-line no-misleading-character-class\n\treturn !!character && character.length == 1 && /[\\u0300-\\u036f\\u1ab0-\\u1aff\\u1dc0-\\u1dff\\u20d0-\\u20ff\\ufe20-\\ufe2f]/.test( character );\n}\n\n/**\n * Checks whether given `character` is a high half of surrogate pair.\n *\n * Using UTF-16 terminology, a surrogate pair denotes UTF-16 character using two UTF-8 characters. The surrogate pair\n * consist of high surrogate pair character followed by low surrogate pair character.\n *\n * @param character Character to check.\n */\nexport function isHighSurrogateHalf( character: string ): boolean {\n\treturn !!character && character.length == 1 && /[\\ud800-\\udbff]/.test( character );\n}\n\n/**\n * Checks whether given `character` is a low half of surrogate pair.\n *\n * Using UTF-16 terminology, a surrogate pair denotes UTF-16 character using two UTF-8 characters. The surrogate pair\n * consist of high surrogate pair character followed by low surrogate pair character.\n *\n * @param character Character to check.\n */\nexport function isLowSurrogateHalf( character: string ): boolean {\n\treturn !!character && character.length == 1 && /[\\udc00-\\udfff]/.test( character );\n}\n\n/**\n * Checks whether given offset in a string is inside a surrogate pair (between two surrogate halves).\n *\n * @param string String to check.\n * @param offset Offset to check.\n */\nexport function isInsideSurrogatePair( string: string, offset: number ): boolean {\n\treturn isHighSurrogateHalf( string.charAt( offset - 1 ) ) && isLowSurrogateHalf( string.charAt( offset ) );\n}\n\n/**\n * Checks whether given offset in a string is between base character and combining mark or between two combining marks.\n *\n * @param string String to check.\n * @param offset Offset to check.\n */\nexport function isInsideCombinedSymbol( string: string, offset: number ): boolean {\n\treturn isCombiningMark( string.charAt( offset ) );\n}\n\nconst EMOJI_PATTERN = buildEmojiRegexp();\n\n/**\n * Checks whether given offset in a string is inside multi-character emoji sequence.\n *\n * @param string String to check.\n * @param offset Offset to check.\n */\nexport function isInsideEmojiSequence( string: string, offset: number ): boolean {\n\tconst matches = String( string ).matchAll( EMOJI_PATTERN );\n\n\treturn Array.from( matches ).some( match => match.index! < offset && offset < match.index! + match[ 0 ].length );\n}\n\nfunction buildEmojiRegexp(): RegExp {\n\tconst parts = [\n\t\t// Emoji Tag Sequence (ETS)\n\t\t/\\p{Emoji}[\\u{E0020}-\\u{E007E}]+\\u{E007F}/u,\n\n\t\t// Emoji Keycap Sequence\n\t\t/\\p{Emoji}\\u{FE0F}?\\u{20E3}/u,\n\n\t\t// Emoji Presentation Sequence\n\t\t/\\p{Emoji}\\u{FE0F}/u,\n\n\t\t// Single-Character Emoji / Emoji Modifier Sequence\n\t\t/(?=\\p{General_Category=Other_Symbol})\\p{Emoji}\\p{Emoji_Modifier}*/u\n\t];\n\n\tconst flagSequence = /\\p{Regional_Indicator}{2}/u.source;\n\tconst emoji = '(?:' + parts.map( part => part.source ).join( '|' ) + ')';\n\tconst sequence = `${ flagSequence }|${ emoji }(?:\\u{200D}${ emoji })*`;\n\n\treturn new RegExp( sequence, 'ug' );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/plugin\n */\n\n/* eslint-disable @typescript-eslint/no-invalid-void-type */\n\nimport { ObservableMixin, type ObservableSetEvent, type EventInfo } from '@ckeditor/ckeditor5-utils';\n\nimport type Editor from './editor/editor';\n\n/**\n * The base class for CKEditor plugin classes.\n */\nexport default class Plugin extends ObservableMixin() implements PluginInterface {\n\t/**\n\t * The editor instance.\n\t *\n\t * Note that most editors implement the {@link module:core/editor/editor~Editor#ui} property.\n\t * However, editors with an external UI (i.e. Bootstrap-based) or a headless editor may not have this property or\n\t * throw an error when accessing it.\n\t *\n\t * Because of above, to make plugins more universal, it is recommended to split features into:\n\t * - The \"editing\" part that uses the {@link module:core/editor/editor~Editor} class without `ui` property.\n\t * - The \"UI\" part that uses the {@link module:core/editor/editor~Editor} class and accesses `ui` property.\n\t */\n\tpublic readonly editor: Editor;\n\n\t/**\n\t * Flag indicating whether a plugin is enabled or disabled.\n\t * A disabled plugin will not transform text.\n\t *\n\t * Plugin can be simply disabled like that:\n\t *\n\t * ```ts\n\t * // Disable the plugin so that no toolbars are visible.\n\t * editor.plugins.get( 'TextTransformation' ).isEnabled = false;\n\t * ```\n\t *\n\t * You can also use {@link #forceDisabled} method.\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tpublic declare isEnabled: boolean;\n\n\t/**\n\t * Holds identifiers for {@link #forceDisabled} mechanism.\n\t */\n\tprivate _disableStack = new Set<string>();\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper();\n\n\t\tthis.editor = editor;\n\n\t\tthis.set( 'isEnabled', true );\n\t}\n\n\t/**\n\t * Disables the plugin.\n\t *\n\t * Plugin may be disabled by multiple features or algorithms (at once). When disabling a plugin, unique id should be passed\n\t * (e.g. feature name). The same identifier should be used when {@link #clearForceDisabled enabling back} the plugin.\n\t * The plugin becomes enabled only after all features {@link #clearForceDisabled enabled it back}.\n\t *\n\t * Disabling and enabling a plugin:\n\t *\n\t * ```ts\n\t * plugin.isEnabled; // -> true\n\t * plugin.forceDisabled( 'MyFeature' );\n\t * plugin.isEnabled; // -> false\n\t * plugin.clearForceDisabled( 'MyFeature' );\n\t * plugin.isEnabled; // -> true\n\t * ```\n\t *\n\t * Plugin disabled by multiple features:\n\t *\n\t * ```ts\n\t * plugin.forceDisabled( 'MyFeature' );\n\t * plugin.forceDisabled( 'OtherFeature' );\n\t * plugin.clearForceDisabled( 'MyFeature' );\n\t * plugin.isEnabled; // -> false\n\t * plugin.clearForceDisabled( 'OtherFeature' );\n\t * plugin.isEnabled; // -> true\n\t * ```\n\t *\n\t * Multiple disabling with the same identifier is redundant:\n\t *\n\t * ```ts\n\t * plugin.forceDisabled( 'MyFeature' );\n\t * plugin.forceDisabled( 'MyFeature' );\n\t * plugin.clearForceDisabled( 'MyFeature' );\n\t * plugin.isEnabled; // -> true\n\t * ```\n\t *\n\t * **Note:** some plugins or algorithms may have more complex logic when it comes to enabling or disabling certain plugins,\n\t * so the plugin might be still disabled after {@link #clearForceDisabled} was used.\n\t *\n\t * @param id Unique identifier for disabling. Use the same id when {@link #clearForceDisabled enabling back} the plugin.\n\t */\n\tpublic forceDisabled( id: string ): void {\n\t\tthis._disableStack.add( id );\n\n\t\tif ( this._disableStack.size == 1 ) {\n\t\t\tthis.on<ObservableSetEvent<boolean>>( 'set:isEnabled', forceDisable, { priority: 'highest' } );\n\t\t\tthis.isEnabled = false;\n\t\t}\n\t}\n\n\t/**\n\t * Clears forced disable previously set through {@link #forceDisabled}. See {@link #forceDisabled}.\n\t *\n\t * @param id Unique identifier, equal to the one passed in {@link #forceDisabled} call.\n\t */\n\tpublic clearForceDisabled( id: string ): void {\n\t\tthis._disableStack.delete( id );\n\n\t\tif ( this._disableStack.size == 0 ) {\n\t\t\tthis.off( 'set:isEnabled', forceDisable );\n\t\t\tthis.isEnabled = true;\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get isContextPlugin(): false {\n\t\treturn false;\n\t}\n}\n\n/**\n * The base interface for CKEditor plugins.\n *\n * In its minimal form a plugin can be a simple function that accepts {@link module:core/editor/editor~Editor the editor}\n * as a parameter:\n *\n * ```ts\n * // A simple plugin that enables a data processor.\n * function MyPlugin( editor ) {\n * \teditor.data.processor = new MyDataProcessor();\n * }\n * ```\n *\n * In most cases however, you will want to inherit from the {@link ~Plugin} class which implements the\n * {@link module:utils/observablemixin~Observable} and is, therefore, more convenient:\n *\n * ```ts\n * class MyPlugin extends Plugin {\n * \tinit() {\n * \t\t// `listenTo()` and `editor` are available thanks to `Plugin`.\n * \t\t// By using `listenTo()` you will ensure that the listener is removed when\n * \t\t// the plugin is destroyed.\n * \t\tthis.listenTo( this.editor.data, 'ready', () => {\n * \t\t\t// Do something when the data is ready.\n * \t\t} );\n * \t}\n * }\n * ```\n *\n * The plugin class can have `pluginName` and `requires` static members. See {@link ~PluginStaticMembers} for more details.\n *\n * The plugin can also implement methods (e.g. {@link ~PluginInterface#init `init()`} or\n * {@link ~PluginInterface#destroy `destroy()`}) which, when present, will be used to properly\n * initialize and destroy the plugin.\n *\n * **Note:** When defined as a plain function, the plugin acts as a constructor and will be\n * called in parallel with other plugins' {@link ~PluginConstructor constructors}.\n * This means the code of that plugin will be executed **before** {@link ~PluginInterface#init `init()`} and\n * {@link ~PluginInterface#afterInit `afterInit()`} methods of other plugins and, for instance,\n * you cannot use it to extend other plugins' {@glink framework/architecture/editing-engine#schema schema}\n * rules as they are defined later on during the `init()` stage.\n */\nexport interface PluginInterface {\n\n\t/**\n\t * The second stage (after plugin constructor) of the plugin initialization.\n\t * Unlike the plugin constructor this method can be asynchronous.\n\t *\n\t * A plugin's `init()` method is called after its {@link ~PluginStaticMembers#requires dependencies} are initialized,\n\t * so in the same order as the constructors of these plugins.\n\t *\n\t * **Note:** This method is optional. A plugin instance does not need to have it defined.\n\t */\n\tinit?(): Promise<unknown> | null | undefined | void;\n\n\t/**\n\t * The third (and last) stage of the plugin initialization. See also {@link ~PluginConstructor} and {@link ~PluginInterface#init}.\n\t *\n\t * **Note:** This method is optional. A plugin instance does not need to have it defined.\n\t */\n\tafterInit?(): Promise<unknown> | null | undefined | void;\n\n\t/**\n\t * Destroys the plugin.\n\t *\n\t * **Note:** This method is optional. A plugin instance does not need to have it defined.\n\t */\n\tdestroy(): Promise<unknown> | null | undefined | void;\n}\n\n/**\n * Creates a new plugin instance. This is the first step of the plugin initialization.\n * See also {@link ~PluginInterface#init} and {@link ~PluginInterface#afterInit}.\n *\n * The plugin static properties should conform to {@link ~PluginStaticMembers `PluginStaticMembers` interface}.\n *\n * A plugin is always instantiated after its {@link ~PluginStaticMembers#requires dependencies} and the\n * {@link ~PluginInterface#init} and {@link ~PluginInterface#afterInit} methods are called in the same order.\n *\n * Usually, you will want to put your plugin's initialization code in the {@link ~PluginInterface#init} method.\n * The constructor can be understood as \"before init\" and used in special cases, just like\n * {@link ~PluginInterface#afterInit} serves the special \"after init\" scenarios (e.g.the code which depends on other\n * plugins, but which does not {@link ~PluginStaticMembers#requires explicitly require} them).\n */\nexport type PluginConstructor<TContext = Editor> =\n\t( PluginClassConstructor<TContext> | PluginFunctionConstructor<TContext> ) & PluginStaticMembers<TContext>;\n\n/**\n * In most cases, you will want to inherit from the {@link ~Plugin} class which implements the\n * {@link module:utils/observablemixin~Observable} and is, therefore, more convenient:\n *\n * ```ts\n * class MyPlugin extends Plugin {\n * \tinit() {\n * \t\t// `listenTo()` and `editor` are available thanks to `Plugin`.\n * \t\t// By using `listenTo()` you will ensure that the listener is removed when\n * \t\t// the plugin is destroyed.\n * \t\tthis.listenTo( this.editor.data, 'ready', () => {\n * \t\t\t// Do something when the data is ready.\n * \t\t} );\n * \t}\n * }\n * ```\n */\nexport type PluginClassConstructor<TContext = Editor> = new ( editor: TContext ) => PluginInterface;\n\n/**\n * In its minimal form a plugin can be a simple function that accepts {@link module:core/editor/editor~Editor the editor}\n * as a parameter:\n *\n * ```ts\n * // A simple plugin that enables a data processor.\n * function MyPlugin( editor ) {\n * \teditor.data.processor = new MyDataProcessor();\n * }\n * ```\n */\nexport type PluginFunctionConstructor<TContext = Editor> = ( editor: TContext ) => void;\n\n/**\n * Static properties of a plugin.\n */\nexport interface PluginStaticMembers<TContext = Editor> {\n\n\t/**\n\t * An array of plugins required by this plugin.\n\t *\n\t * To keep the plugin class definition tight it is recommended to define this property as a static getter:\n\t *\n\t * ```ts\n\t * import Image from './image.js';\n\t *\n\t * export default class ImageCaption {\n\t * \tstatic get requires() {\n\t * \t\treturn [ Image ];\n\t * \t}\n\t * }\n\t * ```\n\t */\n\treadonly requires?: PluginDependencies<TContext>;\n\n\t/**\n\t * An optional name of the plugin. If set, the plugin will be available in\n\t * {@link module:core/plugincollection~PluginCollection#get} by its\n\t * name and its constructor. If not, then only by its constructor.\n\t *\n\t * The name should reflect the constructor name.\n\t *\n\t * To keep the plugin class definition tight, it is recommended to define this property as a static getter:\n\t *\n\t * ```ts\n\t * export default class ImageCaption {\n\t * \tstatic get pluginName() {\n\t * \t\treturn 'ImageCaption';\n\t * \t}\n\t * }\n\t * ```\n\t *\n\t * Note: The native `Function.name` property could not be used to keep the plugin name because\n\t * it will be mangled during code minification.\n\t *\n\t * Naming a plugin is necessary to enable removing it through the\n\t * {@link module:core/editor/editorconfig~EditorConfig#removePlugins `config.removePlugins`} option.\n\t */\n\treadonly pluginName?: string;\n\n\t/**\n\t * A flag which defines if a plugin is allowed or not allowed to be used directly by a {@link module:core/context~Context}.\n\t */\n\treadonly isContextPlugin?: boolean;\n}\n\nexport type PluginDependencies<TContext = Editor> = ReadonlyArray<PluginConstructor<TContext> | string>;\n\n/**\n * An array of loaded plugins.\n */\nexport type LoadedPlugins = Array<PluginInterface>;\n\n/**\n * Helper function that forces plugin to be disabled.\n */\nfunction forceDisable( evt: EventInfo<string, boolean> ) {\n\tevt.return = false;\n\tevt.stop();\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/command\n */\n\nimport {\n\tObservableMixin,\n\ttype EventInfo,\n\ttype ObservableChangeEvent,\n\ttype DecoratedMethodEvent,\n\ttype ObservableSetEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport type Editor from './editor/editor';\n\n/**\n * Base class for the CKEditor commands.\n *\n * Commands are the main way to manipulate the editor contents and state. They are mostly used by UI elements (or by other\n * commands) to make changes in the model. Commands are available in every part of the code that has access to\n * the {@link module:core/editor/editor~Editor editor} instance.\n *\n * Instances of registered commands can be retrieved from {@link module:core/editor/editor~Editor#commands `editor.commands`}.\n * The easiest way to execute a command is through {@link module:core/editor/editor~Editor#execute `editor.execute()`}.\n *\n * By default, commands are disabled when the editor is in the {@link module:core/editor/editor~Editor#isReadOnly read-only} mode\n * but commands with the {@link module:core/command~Command#affectsData `affectsData`} flag set to `false` will not be disabled.\n */\nexport default class Command extends ObservableMixin() {\n\t/**\n\t * The editor on which this command will be used.\n\t */\n\tpublic readonly editor: Editor;\n\n\t/**\n\t * The value of the command. A given command class should define what it represents for it.\n\t *\n\t * For example, the `'bold'` command's value indicates whether the selection starts in a bolded text.\n\t * And the value of the `'link'` command may be an object with link details.\n\t *\n\t * It is possible for a command to have no value (e.g. for stateless actions such as `'uploadImage'`).\n\t *\n\t * A given command class should control this value by overriding the {@link #refresh `refresh()`} method.\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public value: unknown;\n\n\t/**\n\t * Flag indicating whether a command is enabled or disabled.\n\t * A disabled command will do nothing when executed.\n\t *\n\t * A given command class should control this value by overriding the {@link #refresh `refresh()`} method.\n\t *\n\t * It is possible to disable a command \"from outside\" using {@link #forceDisabled} method.\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public isEnabled: boolean;\n\n\t/**\n\t * A flag indicating whether a command's `isEnabled` state should be changed depending on where the document\n\t * selection is placed.\n\t *\n\t * By default, it is set to `true`. If the document selection is placed in a\n\t * {@link module:engine/model/model~Model#canEditAt non-editable} place (such as non-editable root), the command becomes disabled.\n\t *\n\t * The flag should be changed to `false` in a concrete command's constructor if the command should not change its `isEnabled`\n\t * accordingly to the document selection.\n\t */\n\tprotected _isEnabledBasedOnSelection: boolean;\n\n\t/**\n\t * A flag indicating whether a command execution changes the editor data or not.\n\t *\n\t * @see #affectsData\n\t */\n\tprivate _affectsData: boolean;\n\n\t/**\n\t * Holds identifiers for {@link #forceDisabled} mechanism.\n\t */\n\tprivate readonly _disableStack: Set<string>;\n\n\t/**\n\t * Creates a new `Command` instance.\n\t *\n\t * @param editor The editor on which this command will be used.\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper();\n\n\t\tthis.editor = editor;\n\t\tthis.set( 'value', undefined );\n\t\tthis.set( 'isEnabled', false );\n\n\t\tthis._affectsData = true;\n\t\tthis._isEnabledBasedOnSelection = true;\n\t\tthis._disableStack = new Set();\n\n\t\tthis.decorate( 'execute' );\n\n\t\t// By default, every command is refreshed when changes are applied to the model.\n\t\tthis.listenTo( this.editor.model.document, 'change', () => {\n\t\t\tthis.refresh();\n\t\t} );\n\n\t\tthis.listenTo<ObservableChangeEvent<boolean>>( editor, 'change:isReadOnly', () => {\n\t\t\tthis.refresh();\n\t\t} );\n\n\t\t// By default, commands are disabled if the selection is in non-editable place or editor is in read-only mode.\n\t\tthis.on<ObservableSetEvent<boolean>>( 'set:isEnabled', evt => {\n\t\t\tif ( !this.affectsData ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst selection = editor.model.document.selection;\n\t\t\tconst selectionInGraveyard = selection.getFirstPosition()!.root.rootName == '$graveyard';\n\t\t\tconst canEditAtSelection = !selectionInGraveyard && editor.model.canEditAt( selection );\n\n\t\t\t// Disable if editor is read only, or when selection is in a place which cannot be edited.\n\t\t\t//\n\t\t\t// Checking `editor.isReadOnly` is needed for all commands that have `_isEnabledBasedOnSelection == false`.\n\t\t\t// E.g. undo does not base on selection, but affects data and should be disabled when the editor is in read-only mode.\n\t\t\tif ( editor.isReadOnly || this._isEnabledBasedOnSelection && !canEditAtSelection ) {\n\t\t\t\tevt.return = false;\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { priority: 'highest' } );\n\n\t\tthis.on<CommandExecuteEvent>( 'execute', evt => {\n\t\t\tif ( !this.isEnabled ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * A flag indicating whether a command execution changes the editor data or not.\n\t *\n\t * Commands with `affectsData` set to `false` will not be automatically disabled in\n\t * the {@link module:core/editor/editor~Editor#isReadOnly read-only mode} and\n\t * {@glink features/read-only#related-features other editor modes} with restricted user write permissions.\n\t *\n\t * **Note:** You do not have to set it for your every command. It is `true` by default.\n\t *\n\t * @default true\n\t */\n\tpublic get affectsData(): boolean {\n\t\treturn this._affectsData;\n\t}\n\n\tprotected set affectsData( affectsData: boolean ) {\n\t\tthis._affectsData = affectsData;\n\t}\n\n\t/**\n\t * Refreshes the command. The command should update its {@link #isEnabled} and {@link #value} properties\n\t * in this method.\n\t *\n\t * This method is automatically called when\n\t * {@link module:engine/model/document~Document#event:change any changes are applied to the document}.\n\t */\n\tpublic refresh(): void {\n\t\tthis.isEnabled = true;\n\t}\n\n\t/**\n\t * Disables the command.\n\t *\n\t * Command may be disabled by multiple features or algorithms (at once). When disabling a command, unique id should be passed\n\t * (e.g. the feature name). The same identifier should be used when {@link #clearForceDisabled enabling back} the command.\n\t * The command becomes enabled only after all features {@link #clearForceDisabled enabled it back}.\n\t *\n\t * Disabling and enabling a command:\n\t *\n\t * ```ts\n\t * command.isEnabled; // -> true\n\t * command.forceDisabled( 'MyFeature' );\n\t * command.isEnabled; // -> false\n\t * command.clearForceDisabled( 'MyFeature' );\n\t * command.isEnabled; // -> true\n\t * ```\n\t *\n\t * Command disabled by multiple features:\n\t *\n\t * ```ts\n\t * command.forceDisabled( 'MyFeature' );\n\t * command.forceDisabled( 'OtherFeature' );\n\t * command.clearForceDisabled( 'MyFeature' );\n\t * command.isEnabled; // -> false\n\t * command.clearForceDisabled( 'OtherFeature' );\n\t * command.isEnabled; // -> true\n\t * ```\n\t *\n\t * Multiple disabling with the same identifier is redundant:\n\t *\n\t * ```ts\n\t * command.forceDisabled( 'MyFeature' );\n\t * command.forceDisabled( 'MyFeature' );\n\t * command.clearForceDisabled( 'MyFeature' );\n\t * command.isEnabled; // -> true\n\t * ```\n\t *\n\t * **Note:** some commands or algorithms may have more complex logic when it comes to enabling or disabling certain commands,\n\t * so the command might be still disabled after {@link #clearForceDisabled} was used.\n\t *\n\t * @param id Unique identifier for disabling. Use the same id when {@link #clearForceDisabled enabling back} the command.\n\t */\n\tpublic forceDisabled( id: string ): void {\n\t\tthis._disableStack.add( id );\n\n\t\tif ( this._disableStack.size == 1 ) {\n\t\t\tthis.on<ObservableSetEvent<boolean>>( 'set:isEnabled', forceDisable, { priority: 'highest' } );\n\t\t\tthis.isEnabled = false;\n\t\t}\n\t}\n\n\t/**\n\t * Clears forced disable previously set through {@link #forceDisabled}. See {@link #forceDisabled}.\n\t *\n\t * @param id Unique identifier, equal to the one passed in {@link #forceDisabled} call.\n\t */\n\tpublic clearForceDisabled( id: string ): void {\n\t\tthis._disableStack.delete( id );\n\n\t\tif ( this._disableStack.size == 0 ) {\n\t\t\tthis.off( 'set:isEnabled', forceDisable );\n\t\t\tthis.refresh();\n\t\t}\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * A command may accept parameters. They will be passed from {@link module:core/editor/editor~Editor#execute `editor.execute()`}\n\t * to the command.\n\t *\n\t * The `execute()` method will automatically abort when the command is disabled ({@link #isEnabled} is `false`).\n\t * This behavior is implemented by a high priority listener to the {@link #event:execute} event.\n\t *\n\t * In order to see how to disable a command from \"outside\" see the {@link #isEnabled} documentation.\n\t *\n\t * This method may return a value, which would be forwarded all the way down to the\n\t * {@link module:core/editor/editor~Editor#execute `editor.execute()`}.\n\t *\n\t * @fires execute\n\t */\n\tpublic execute( ...args: Array<unknown> ): unknown { return undefined; }\n\n\t/**\n\t * Destroys the command.\n\t */\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\t}\n}\n\n/**\n * Helper function that forces command to be disabled.\n */\nfunction forceDisable( evt: EventInfo<string, boolean> ) {\n\tevt.return = false;\n\tevt.stop();\n}\n\n/**\n * Event fired by the {@link module:core/command~Command#execute} method. The command action is a listener to this event so it's\n * possible to change/cancel the behavior of the command by listening to this event.\n *\n * See {@link module:utils/observablemixin~Observable#decorate} for more information and samples.\n *\n * **Note:** This event is fired even if command is disabled. However, it is automatically blocked\n * by a high priority listener in order to prevent command execution.\n *\n * @eventName ~Command#execute\n */\nexport type CommandExecuteEvent = DecoratedMethodEvent<Command, 'execute'>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/multicommand\n */\n\nimport Command from './command';\nimport type Editor from './editor/editor';\n\nimport {\n\tinsertToPriorityArray,\n\ttype PriorityString,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\n/**\n * A CKEditor command that aggregates other commands.\n *\n * This command is used to proxy multiple commands. The multi-command is enabled when\n * at least one of its registered child commands is enabled.\n * When executing a multi-command, the first enabled command with highest priority will be executed.\n *\n * ```ts\n * const multiCommand = new MultiCommand( editor );\n *\n * const commandFoo = new Command( editor );\n * const commandBar = new Command( editor );\n *\n * // Register a child command.\n * multiCommand.registerChildCommand( commandFoo );\n * // Register a child command with a low priority.\n * multiCommand.registerChildCommand( commandBar, { priority: 'low' } );\n *\n * // Enable one of the commands.\n * commandBar.isEnabled = true;\n *\n * multiCommand.execute(); // Will execute commandBar.\n * ```\n */\nexport default class MultiCommand extends Command {\n\t/**\n\t * Registered child commands definitions.\n\t */\n\tprivate _childCommandsDefinitions: Array<{ command: Command; priority: PriorityString }> = [];\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\t// Override base command refresh(): the command's state is changed when one of child commands changes states.\n\t}\n\n\t/**\n\t * Executes the first enabled command which has the highest priority of all registered child commands.\n\t *\n\t * @returns The value returned by the {@link module:core/command~Command#execute `command.execute()`}.\n\t */\n\tpublic override execute( ...args: Array<unknown> ): unknown {\n\t\tconst command = this._getFirstEnabledCommand();\n\n\t\treturn !!command && command.execute( args );\n\t}\n\n\t/**\n\t * Registers a child command.\n\t *\n\t * @param options An object with configuration options.\n\t * @param options.priority Priority of a command to register.\n\t */\n\tpublic registerChildCommand(\n\t\tcommand: Command,\n\t\toptions: { priority?: PriorityString } = {}\n\t): void {\n\t\tinsertToPriorityArray( this._childCommandsDefinitions, { command, priority: options.priority || 'normal' } );\n\n\t\t// Change multi-command enabled state when one of registered commands changes state.\n\t\tcommand.on<ObservableChangeEvent<boolean>>( 'change:isEnabled', () => this._checkEnabled() );\n\n\t\tthis._checkEnabled();\n\t}\n\n\t/**\n\t * Checks if any of child commands is enabled.\n\t */\n\tprivate _checkEnabled(): void {\n\t\tthis.isEnabled = !!this._getFirstEnabledCommand();\n\t}\n\n\t/**\n\t * Returns a first enabled command with the highest priority or `undefined` if none of them is enabled.\n\t */\n\tprivate _getFirstEnabledCommand(): Command | undefined {\n\t\tconst commandDefinition = this._childCommandsDefinitions.find( ( { command } ) => command.isEnabled );\n\n\t\treturn commandDefinition && commandDefinition.command;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/plugincollection\n */\n\nimport { CKEditorError, EmitterMixin } from '@ckeditor/ckeditor5-utils';\nimport type { LoadedPlugins, PluginClassConstructor, PluginConstructor, PluginInterface } from './plugin';\n\n/**\n * Manages a list of CKEditor plugins, including loading, resolving dependencies and initialization.\n */\nexport default class PluginCollection<TContext extends object> extends EmitterMixin() implements Iterable<PluginEntry<TContext>> {\n\tprivate _context: TContext;\n\n\tprivate _plugins = new Map<PluginConstructor<TContext> | string, PluginInterface>();\n\n\t/**\n\t * A map of plugin constructors that can be retrieved by their names.\n\t */\n\tprivate _availablePlugins: Map<string, PluginConstructor<TContext>>;\n\n\t/**\n\t * Map of {@link module:core/contextplugin~ContextPlugin context plugins} which can be retrieved by their constructors or instances.\n\t */\n\tprivate _contextPlugins: Map<PluginConstructor<TContext> | PluginInterface, PluginConstructor<TContext> | PluginInterface>;\n\n\t/**\n\t * Creates an instance of the plugin collection class.\n\t * Allows loading and initializing plugins and their dependencies.\n\t * Allows providing a list of already loaded plugins. These plugins will not be destroyed along with this collection.\n\t *\n\t * @param availablePlugins Plugins (constructors) which the collection will be able to use\n\t * when {@link module:core/plugincollection~PluginCollection#init} is used with the plugin names (strings, instead of constructors).\n\t * Usually, the editor will pass its built-in plugins to the collection so they can later be\n\t * used in `config.plugins` or `config.removePlugins` by names.\n\t * @param contextPlugins A list of already initialized plugins represented by a `[ PluginConstructor, pluginInstance ]` pair.\n\t */\n\tconstructor(\n\t\tcontext: TContext,\n\t\tavailablePlugins: Iterable<PluginConstructor<TContext>> = [],\n\t\tcontextPlugins: Iterable<PluginEntry<TContext>> = []\n\t) {\n\t\tsuper();\n\n\t\tthis._context = context;\n\t\tthis._availablePlugins = new Map();\n\n\t\tfor ( const PluginConstructor of availablePlugins ) {\n\t\t\tif ( PluginConstructor.pluginName ) {\n\t\t\t\tthis._availablePlugins.set( PluginConstructor.pluginName, PluginConstructor );\n\t\t\t}\n\t\t}\n\n\t\tthis._contextPlugins = new Map();\n\n\t\tfor ( const [ PluginConstructor, pluginInstance ] of contextPlugins ) {\n\t\t\tthis._contextPlugins.set( PluginConstructor, pluginInstance );\n\t\t\tthis._contextPlugins.set( pluginInstance, PluginConstructor );\n\n\t\t\t// To make it possible to require a plugin by its name.\n\t\t\tif ( PluginConstructor.pluginName ) {\n\t\t\t\tthis._availablePlugins.set( PluginConstructor.pluginName, PluginConstructor );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * Returns `[ PluginConstructor, pluginInstance ]` pairs.\n\t */\n\tpublic* [ Symbol.iterator ](): IterableIterator<PluginEntry<TContext>> {\n\t\tfor ( const entry of this._plugins ) {\n\t\t\tif ( typeof entry[ 0 ] == 'function' ) {\n\t\t\t\tyield entry as any;\n\t\t\t}\n\t\t}\n\t}\n\n\tpublic get<TConstructor extends PluginClassConstructor<TContext>>( key: TConstructor ): InstanceType<TConstructor>;\n\tpublic get<TName extends string>( key: TName ): PluginsMap[ TName ];\n\n\t/**\n\t * Gets the plugin instance by its constructor or name.\n\t *\n\t * ```ts\n\t * // Check if 'Clipboard' plugin was loaded.\n\t * if ( editor.plugins.has( 'ClipboardPipeline' ) ) {\n\t * \t// Get clipboard plugin instance\n\t * \tconst clipboard = editor.plugins.get( 'ClipboardPipeline' );\n\t *\n\t * \tthis.listenTo( clipboard, 'inputTransformation', ( evt, data ) => {\n\t * \t\t// Do something on clipboard input.\n\t * \t} );\n\t * }\n\t * ```\n\t *\n\t * **Note**: This method will throw an error if a plugin is not loaded. Use `{@link #has editor.plugins.has()}`\n\t * to check if a plugin is available.\n\t *\n\t * @param key The plugin constructor or {@link module:core/plugin~PluginStaticMembers#pluginName name}.\n\t */\n\tpublic get( key: PluginConstructor<TContext> | string ): PluginInterface {\n\t\tconst plugin = this._plugins.get( key );\n\n\t\tif ( !plugin ) {\n\t\t\tlet pluginName = key;\n\n\t\t\tif ( typeof key == 'function' ) {\n\t\t\t\tpluginName = key.pluginName || key.name;\n\t\t\t}\n\n\t\t\t/**\n\t\t\t * The plugin is not loaded and could not be obtained.\n\t\t\t *\n\t\t\t * Plugin classes (constructors) need to be provided to the editor and must be loaded before they can be obtained from\n\t\t\t * the plugin collection.\n\t\t\t * This is usually done in CKEditor 5 builds by setting the {@link module:core/editor/editor~Editor.builtinPlugins}\n\t\t\t * property.\n\t\t\t *\n\t\t\t * **Note**: You can use `{@link module:core/plugincollection~PluginCollection#has editor.plugins.has()}`\n\t\t\t * to check if a plugin was loaded.\n\t\t\t *\n\t\t\t * @error plugincollection-plugin-not-loaded\n\t\t\t * @param plugin The name of the plugin which is not loaded.\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'plugincollection-plugin-not-loaded', this._context, { plugin: pluginName } );\n\t\t}\n\n\t\treturn plugin;\n\t}\n\n\t/**\n\t * Checks if a plugin is loaded.\n\t *\n\t * ```ts\n\t * // Check if the 'Clipboard' plugin was loaded.\n\t * if ( editor.plugins.has( 'ClipboardPipeline' ) ) {\n\t * \t// Now use the clipboard plugin instance:\n\t * \tconst clipboard = editor.plugins.get( 'ClipboardPipeline' );\n\t *\n\t * \t// ...\n\t * }\n\t * ```\n\t *\n\t * @param key The plugin constructor or {@link module:core/plugin~PluginStaticMembers#pluginName name}.\n\t */\n\tpublic has( key: PluginConstructor<TContext> | string ): boolean {\n\t\treturn this._plugins.has( key );\n\t}\n\n\t/**\n\t * Initializes a set of plugins and adds them to the collection.\n\t *\n\t * @param plugins An array of {@link module:core/plugin~PluginInterface plugin constructors}\n\t * or {@link module:core/plugin~PluginStaticMembers#pluginName plugin names}.\n\t * @param pluginsToRemove Names of the plugins or plugin constructors\n\t * that should not be loaded (despite being specified in the `plugins` array).\n\t * @param pluginsSubstitutions An array of {@link module:core/plugin~PluginInterface plugin constructors}\n\t * that will be used to replace plugins of the same names that were passed in `plugins` or that are in their dependency tree.\n\t * A useful option for replacing built-in plugins while creating tests (for mocking their APIs). Plugins that will be replaced\n\t * must follow these rules:\n\t * * The new plugin must be a class.\n\t * * The new plugin must be named.\n\t * * Both plugins must not depend on other plugins.\n\t * @returns A promise which gets resolved once all plugins are loaded and available in the collection.\n\t */\n\tpublic init(\n\t\tplugins: ReadonlyArray<PluginConstructor<TContext> | string>,\n\t\tpluginsToRemove: ReadonlyArray<PluginConstructor<TContext> | string> = [],\n\t\tpluginsSubstitutions: ReadonlyArray<PluginConstructor<TContext>> = []\n\t): Promise<LoadedPlugins> {\n\t\t// Plugin initialization procedure consists of 2 main steps:\n\t\t// 1) collecting all available plugin constructors,\n\t\t// 2) verification whether all required plugins can be instantiated.\n\t\t//\n\t\t// In the first step, all plugin constructors, available in the provided `plugins` array and inside\n\t\t// plugin's dependencies (from the `Plugin.requires` array), are recursively collected and added to the existing\n\t\t// `this._availablePlugins` map, but without any verification at the given moment. Performing the verification\n\t\t// at this point (during the plugin constructor searching) would cause false errors to occur, that some plugin\n\t\t// is missing but in fact it may be defined further in the array as the dependency of other plugin. After\n\t\t// traversing the entire dependency tree, it will be checked if all required \"top level\" plugins are available.\n\t\t//\n\t\t// In the second step, the list of plugins that have not been explicitly removed is traversed to get all the\n\t\t// plugin constructors to be instantiated in the correct order and to validate against some rules. Finally, if\n\t\t// no plugin is missing and no other error has been found, they all will be instantiated.\n\t\t// eslint-disable-next-line @typescript-eslint/no-this-alias\n\t\tconst that = this;\n\t\tconst context = this._context;\n\n\t\tfindAvailablePluginConstructors( plugins );\n\n\t\tvalidatePlugins( plugins );\n\n\t\tconst pluginsToLoad = plugins.filter( plugin => !isPluginRemoved( plugin, pluginsToRemove ) );\n\n\t\tconst pluginConstructors = [ ...getPluginConstructors( pluginsToLoad ) ];\n\n\t\tsubstitutePlugins( pluginConstructors, pluginsSubstitutions );\n\n\t\tconst pluginInstances = loadPlugins( pluginConstructors );\n\n\t\treturn initPlugins( pluginInstances, 'init' )\n\t\t\t.then( () => initPlugins( pluginInstances, 'afterInit' ) )\n\t\t\t.then( () => pluginInstances );\n\n\t\tfunction isPluginConstructor( plugin: PluginConstructor<TContext> | string | null ): plugin is PluginConstructor<TContext> {\n\t\t\treturn typeof plugin === 'function';\n\t\t}\n\n\t\tfunction isContextPlugin(\n\t\t\tplugin: PluginConstructor<TContext> | string | null\n\t\t): plugin is PluginConstructor<TContext> & { isContextPlugin: true } {\n\t\t\treturn isPluginConstructor( plugin ) && !!plugin.isContextPlugin;\n\t\t}\n\n\t\tfunction isPluginRemoved(\n\t\t\tplugin: PluginConstructor<TContext> | string,\n\t\t\tpluginsToRemove: ReadonlyArray<PluginConstructor<TContext> | string>\n\t\t) {\n\t\t\treturn pluginsToRemove.some( removedPlugin => {\n\t\t\t\tif ( removedPlugin === plugin ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\tif ( getPluginName( plugin ) === removedPlugin ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\tif ( getPluginName( removedPlugin ) === plugin ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\treturn false;\n\t\t\t} );\n\t\t}\n\n\t\tfunction getPluginName( plugin: PluginConstructor<TContext> | string ) {\n\t\t\treturn isPluginConstructor( plugin ) ?\n\t\t\t\tplugin.pluginName || plugin.name :\n\t\t\t\tplugin;\n\t\t}\n\n\t\tfunction findAvailablePluginConstructors(\n\t\t\tplugins: ReadonlyArray<PluginConstructor<TContext> | string>,\n\t\t\tprocessed = new Set<PluginConstructor<TContext>>()\n\t\t) {\n\t\t\tplugins.forEach( plugin => {\n\t\t\t\tif ( !isPluginConstructor( plugin ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tif ( processed.has( plugin ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tprocessed.add( plugin );\n\n\t\t\t\tif ( plugin.pluginName && !that._availablePlugins.has( plugin.pluginName ) ) {\n\t\t\t\t\tthat._availablePlugins.set( plugin.pluginName, plugin );\n\t\t\t\t}\n\n\t\t\t\tif ( plugin.requires ) {\n\t\t\t\t\tfindAvailablePluginConstructors( plugin.requires, processed );\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\n\t\tfunction getPluginConstructors(\n\t\t\tplugins: ReadonlyArray<PluginConstructor<TContext> | string>,\n\t\t\tprocessed = new Set<PluginConstructor<TContext>>()\n\t\t) {\n\t\t\treturn plugins\n\t\t\t\t.map( plugin => {\n\t\t\t\t\treturn isPluginConstructor( plugin ) ?\n\t\t\t\t\t\tplugin :\n\t\t\t\t\t\tthat._availablePlugins.get( plugin )!;\n\t\t\t\t} )\n\t\t\t\t.reduce( ( result, plugin ) => {\n\t\t\t\t\tif ( processed.has( plugin ) ) {\n\t\t\t\t\t\treturn result;\n\t\t\t\t\t}\n\n\t\t\t\t\tprocessed.add( plugin );\n\n\t\t\t\t\tif ( plugin.requires ) {\n\t\t\t\t\t\tvalidatePlugins( plugin.requires, plugin );\n\n\t\t\t\t\t\tgetPluginConstructors( plugin.requires, processed ).forEach( plugin => result.add( plugin ) );\n\t\t\t\t\t}\n\n\t\t\t\t\treturn result.add( plugin );\n\t\t\t\t}, new Set<PluginConstructor<TContext>>() );\n\t\t}\n\n\t\tfunction validatePlugins(\n\t\t\tplugins: ReadonlyArray<PluginConstructor<TContext> | string>,\n\t\t\tparentPluginConstructor: PluginConstructor<TContext> | null = null\n\t\t) {\n\t\t\tplugins\n\t\t\t\t.map( plugin => {\n\t\t\t\t\treturn isPluginConstructor( plugin ) ?\n\t\t\t\t\t\tplugin :\n\t\t\t\t\t\tthat._availablePlugins.get( plugin ) || plugin;\n\t\t\t\t} )\n\t\t\t\t.forEach( plugin => {\n\t\t\t\t\tcheckMissingPlugin( plugin, parentPluginConstructor );\n\t\t\t\t\tcheckContextPlugin( plugin, parentPluginConstructor );\n\t\t\t\t\tcheckRemovedPlugin( plugin, parentPluginConstructor );\n\t\t\t\t} );\n\t\t}\n\n\t\tfunction checkMissingPlugin(\n\t\t\tplugin: PluginConstructor<TContext> | string,\n\t\t\tparentPluginConstructor: PluginConstructor<TContext> | null\n\t\t) {\n\t\t\tif ( isPluginConstructor( plugin ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( parentPluginConstructor ) {\n\t\t\t\t/**\n\t\t\t\t * A required \"soft\" dependency was not found on the plugin list.\n\t\t\t\t *\n\t\t\t\t * When configuring the editor, either prior to building (via\n\t\t\t\t * {@link module:core/editor/editor~Editor.builtinPlugins `Editor.builtinPlugins`}) or when\n\t\t\t\t * creating a new instance of the editor (e.g. via\n\t\t\t\t * {@link module:core/editor/editorconfig~EditorConfig#plugins `config.plugins`}), you need to provide\n\t\t\t\t * some of the dependencies for other plugins that you used.\n\t\t\t\t *\n\t\t\t\t * This error is thrown when one of these dependencies was not provided. The name of the missing plugin\n\t\t\t\t * can be found in `missingPlugin` and the plugin that required it in `requiredBy`.\n\t\t\t\t *\n\t\t\t\t * In order to resolve it, you need to import the missing plugin and add it to the\n\t\t\t\t * current list of plugins (`Editor.builtinPlugins` or `config.plugins`/`config.extraPlugins`).\n\t\t\t\t *\n\t\t\t\t * Soft requirements were introduced in version 26.0.0. If you happen to stumble upon this error\n\t\t\t\t * when upgrading to version 26.0.0, read also the\n\t\t\t\t * {@glink updating/guides/update-to-26 Migration to 26.0.0} guide.\n\t\t\t\t *\n\t\t\t\t * @error plugincollection-soft-required\n\t\t\t\t * @param missingPlugin The name of the required plugin.\n\t\t\t\t * @param requiredBy The name of the plugin that requires the other plugin.\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'plugincollection-soft-required',\n\t\t\t\t\tcontext,\n\t\t\t\t\t{ missingPlugin: plugin, requiredBy: getPluginName( parentPluginConstructor ) }\n\t\t\t\t);\n\t\t\t}\n\n\t\t\t/**\n\t\t\t * A plugin is not available and could not be loaded.\n\t\t\t *\n\t\t\t * Plugin classes (constructors) need to be provided to the editor before they can be loaded by name.\n\t\t\t * This is usually done in CKEditor 5 builds by setting the {@link module:core/editor/editor~Editor.builtinPlugins}\n\t\t\t * property.\n\t\t\t *\n\t\t\t * **If you see this warning when using one of the {@glink installation/getting-started/predefined-builds\n\t\t\t * CKEditor 5 Builds}**,\n\t\t\t * it means that you try to enable a plugin which was not included in that build. This may be due to a typo\n\t\t\t * in the plugin name or simply because that plugin is not a part of this build. In the latter scenario,\n\t\t\t * read more about {@glink installation/getting-started/quick-start custom builds}.\n\t\t\t *\n\t\t\t * **If you see this warning when using one of the editor creators directly** (not a build), then it means\n\t\t\t * that you tried loading plugins by name. However, unlike CKEditor 4, CKEditor 5 does not implement a \"plugin loader\".\n\t\t\t * This means that CKEditor 5 does not know where to load the plugin modules from. Therefore, you need to\n\t\t\t * provide each plugin through a reference (as a constructor function). Check out the examples in\n\t\t\t * {@glink installation/advanced/alternative-setups/integrating-from-source-webpack \"Building from source\"}.\n\t\t\t *\n\t\t\t * @error plugincollection-plugin-not-found\n\t\t\t * @param plugin The name of the plugin which could not be loaded.\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'plugincollection-plugin-not-found',\n\t\t\t\tcontext,\n\t\t\t\t{ plugin }\n\t\t\t);\n\t\t}\n\n\t\tfunction checkContextPlugin(\n\t\t\tplugin: PluginConstructor<TContext> | string,\n\t\t\tparentPluginConstructor: PluginConstructor<TContext> | null\n\t\t) {\n\t\t\tif ( !isContextPlugin( parentPluginConstructor ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( isContextPlugin( plugin ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t/**\n\t\t\t * If a plugin is a context plugin, all plugins it requires should also be context plugins\n\t\t\t * instead of plugins. In other words, if one plugin can be used in the context,\n\t\t\t * all its requirements should also be ready to be used in the context. Note that the context\n\t\t\t * provides only a part of the API provided by the editor. If one plugin needs a full\n\t\t\t * editor API, all plugins which require it are considered as plugins that need a full\n\t\t\t * editor API.\n\t\t\t *\n\t\t\t * @error plugincollection-context-required\n\t\t\t * @param plugin The name of the required plugin.\n\t\t\t * @param requiredBy The name of the parent plugin.\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'plugincollection-context-required',\n\t\t\t\tcontext,\n\t\t\t\t{ plugin: getPluginName( plugin ), requiredBy: getPluginName( parentPluginConstructor ) }\n\t\t\t);\n\t\t}\n\n\t\tfunction checkRemovedPlugin(\n\t\t\tplugin: PluginConstructor<TContext> | string,\n\t\t\tparentPluginConstructor: PluginConstructor<TContext> | null\n\t\t) {\n\t\t\tif ( !parentPluginConstructor ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( !isPluginRemoved( plugin, pluginsToRemove ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t/**\n\t\t\t * Cannot load a plugin because one of its dependencies is listed in the `removePlugins` option.\n\t\t\t *\n\t\t\t * @error plugincollection-required\n\t\t\t * @param plugin The name of the required plugin.\n\t\t\t * @param requiredBy The name of the parent plugin.\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'plugincollection-required',\n\t\t\t\tcontext,\n\t\t\t\t{ plugin: getPluginName( plugin ), requiredBy: getPluginName( parentPluginConstructor ) }\n\t\t\t);\n\t\t}\n\n\t\tfunction loadPlugins( pluginConstructors: ReadonlyArray<PluginConstructor<TContext>> ) {\n\t\t\treturn pluginConstructors.map( PluginConstructor => {\n\t\t\t\tlet pluginInstance = that._contextPlugins.get( PluginConstructor ) as ( PluginInterface | undefined );\n\n\t\t\t\tpluginInstance = pluginInstance || new ( PluginConstructor as PluginClassConstructor<TContext> )( context );\n\n\t\t\t\tthat._add( PluginConstructor, pluginInstance );\n\n\t\t\t\treturn pluginInstance;\n\t\t\t} );\n\t\t}\n\n\t\tfunction initPlugins( pluginInstances: ReadonlyArray<PluginInterface>, method: 'init' | 'afterInit' ) {\n\t\t\treturn pluginInstances.reduce<Promise<unknown>>( ( promise, plugin ) => {\n\t\t\t\tif ( !plugin[ method ] ) {\n\t\t\t\t\treturn promise;\n\t\t\t\t}\n\n\t\t\t\tif ( that._contextPlugins.has( plugin ) ) {\n\t\t\t\t\treturn promise;\n\t\t\t\t}\n\n\t\t\t\treturn promise.then( plugin[ method ]!.bind( plugin ) );\n\t\t\t}, Promise.resolve() );\n\t\t}\n\n\t\t/**\n\t\t * Replaces plugin constructors with the specified set of plugins.\n\t\t */\n\t\tfunction substitutePlugins(\n\t\t\tpluginConstructors: Array<PluginConstructor<TContext>>,\n\t\t\tpluginsSubstitutions: ReadonlyArray<PluginConstructor<TContext>>\n\t\t) {\n\t\t\tfor ( const pluginItem of pluginsSubstitutions ) {\n\t\t\t\tif ( typeof pluginItem != 'function' ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * The plugin replacing an existing plugin must be a function.\n\t\t\t\t\t *\n\t\t\t\t\t * @error plugincollection-replace-plugin-invalid-type\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError( 'plugincollection-replace-plugin-invalid-type', null, { pluginItem } );\n\t\t\t\t}\n\n\t\t\t\tconst pluginName = pluginItem.pluginName;\n\n\t\t\t\tif ( !pluginName ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * The plugin replacing an existing plugin must have a name.\n\t\t\t\t\t *\n\t\t\t\t\t * @error plugincollection-replace-plugin-missing-name\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError( 'plugincollection-replace-plugin-missing-name', null, { pluginItem } );\n\t\t\t\t}\n\n\t\t\t\tif ( pluginItem.requires && pluginItem.requires.length ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * The plugin replacing an existing plugin cannot depend on other plugins.\n\t\t\t\t\t *\n\t\t\t\t\t * @error plugincollection-plugin-for-replacing-cannot-have-dependencies\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError( 'plugincollection-plugin-for-replacing-cannot-have-dependencies', null, { pluginName } );\n\t\t\t\t}\n\n\t\t\t\tconst pluginToReplace = that._availablePlugins.get( pluginName );\n\n\t\t\t\tif ( !pluginToReplace ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * The replaced plugin does not exist in the\n\t\t\t\t\t * {@link module:core/plugincollection~PluginCollection available plugins} collection.\n\t\t\t\t\t *\n\t\t\t\t\t * @error plugincollection-plugin-for-replacing-not-exist\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError( 'plugincollection-plugin-for-replacing-not-exist', null, { pluginName } );\n\t\t\t\t}\n\n\t\t\t\tconst indexInPluginConstructors = pluginConstructors.indexOf( pluginToReplace );\n\n\t\t\t\tif ( indexInPluginConstructors === -1 ) {\n\t\t\t\t\t// The Context feature can substitute plugins as well.\n\t\t\t\t\t// It may happen that the editor will be created with the given context, where the plugin for substitute\n\t\t\t\t\t// was already replaced. In such a case, we don't want to do it again.\n\t\t\t\t\tif ( that._contextPlugins.has( pluginToReplace ) ) {\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\n\t\t\t\t\t/**\n\t\t\t\t\t * The replaced plugin will not be loaded so it cannot be replaced.\n\t\t\t\t\t *\n\t\t\t\t\t * @error plugincollection-plugin-for-replacing-not-loaded\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError( 'plugincollection-plugin-for-replacing-not-loaded', null, { pluginName } );\n\t\t\t\t}\n\n\t\t\t\tif ( pluginToReplace.requires && pluginToReplace.requires.length ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * The replaced plugin cannot depend on other plugins.\n\t\t\t\t\t *\n\t\t\t\t\t * @error plugincollection-replaced-plugin-cannot-have-dependencies\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError( 'plugincollection-replaced-plugin-cannot-have-dependencies', null, { pluginName } );\n\t\t\t\t}\n\n\t\t\t\tpluginConstructors.splice( indexInPluginConstructors, 1, pluginItem );\n\t\t\t\tthat._availablePlugins.set( pluginName, pluginItem );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Destroys all loaded plugins.\n\t */\n\tpublic destroy(): Promise<unknown> {\n\t\tconst promises: Array<unknown> = [];\n\n\t\tfor ( const [ , pluginInstance ] of this ) {\n\t\t\tif ( typeof pluginInstance.destroy == 'function' && !this._contextPlugins.has( pluginInstance ) ) {\n\t\t\t\tpromises.push( pluginInstance.destroy() );\n\t\t\t}\n\t\t}\n\n\t\treturn Promise.all( promises );\n\t}\n\n\t/**\n\t * Adds the plugin to the collection. Exposed mainly for testing purposes.\n\t *\n\t * @param PluginConstructor The plugin constructor.\n\t * @param plugin The instance of the plugin.\n\t */\n\tprivate _add( PluginConstructor: PluginConstructor<TContext>, plugin: PluginInterface ) {\n\t\tthis._plugins.set( PluginConstructor, plugin );\n\n\t\tconst pluginName = PluginConstructor.pluginName;\n\n\t\tif ( !pluginName ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( this._plugins.has( pluginName ) ) {\n\t\t\t/**\n\t\t\t * Two plugins with the same {@link module:core/plugin~PluginStaticMembers#pluginName} were loaded.\n\t\t\t * This will lead to runtime conflicts between these plugins.\n\t\t\t *\n\t\t\t * In practice, this warning usually means that new plugins were added to an existing CKEditor 5 build.\n\t\t\t * Plugins should always be added to a source version of the editor (`@ckeditor/ckeditor5-editor-*`),\n\t\t\t * not to an editor imported from one of the `@ckeditor/ckeditor5-build-*` packages.\n\t\t\t *\n\t\t\t * Check your import paths and the list of plugins passed to\n\t\t\t * {@link module:core/editor/editor~Editor.create `Editor.create()`}\n\t\t\t * or specified in {@link module:core/editor/editor~Editor.builtinPlugins `Editor.builtinPlugins`}.\n\t\t\t *\n\t\t\t * The second option is that your `node_modules/` directory contains duplicated versions of the same\n\t\t\t * CKEditor 5 packages. Normally, on clean installations, npm deduplicates packages in `node_modules/`, so\n\t\t\t * it may be enough to call `rm -rf node_modules && npm i`. However, if you installed conflicting versions\n\t\t\t * of some packages, their dependencies may need to be installed in more than one version which may lead to this\n\t\t\t * warning.\n\t\t\t *\n\t\t\t * Technically speaking, this error occurs because after adding a plugin to an existing editor build\n\t\t\t * the dependencies of this plugin are being duplicated.\n\t\t\t * They are already built into that editor build and now get added for the second time as dependencies\n\t\t\t * of the plugin you are installing.\n\t\t\t *\n\t\t\t * Read more about {@glink installation/plugins/installing-plugins Installing plugins}.\n\t\t\t *\n\t\t\t * @error plugincollection-plugin-name-conflict\n\t\t\t * @param pluginName The duplicated plugin name.\n\t\t\t * @param plugin1 The first plugin constructor.\n\t\t\t * @param plugin2 The second plugin constructor.\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'plugincollection-plugin-name-conflict',\n\t\t\t\tnull,\n\t\t\t\t{ pluginName, plugin1: this._plugins.get( pluginName )!.constructor, plugin2: PluginConstructor }\n\t\t\t);\n\t\t}\n\n\t\tthis._plugins.set( pluginName, plugin );\n\t}\n}\n\n/**\n * A `[ PluginConstructor, pluginInstance ]` pair.\n */\nexport type PluginEntry<TContext> = [ PluginConstructor<TContext>, PluginInterface ];\n\n/**\n * Helper type that maps plugin names to their types.\n * It is meant to be extended with module augmentation.\n *\n * ```ts\n * class MyPlugin extends Plugin {\n * \tpublic static pluginName() {\n * \t\treturn 'MyPlugin' as const;\n * \t}\n * }\n *\n * declare module '@ckeditor/ckeditor5-core' {\n * \tinterface PluginsMap {\n * \t\t[ MyPlugin.pluginName ]: MyPlugin;\n * \t}\n * }\n *\n * // Returns `MyPlugin`.\n * const myPlugin = editor.plugins.get( 'MyPlugin' );\n * ```\n */\nexport interface PluginsMap {\n\t[ name: string ]: PluginInterface;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/context\n */\n\nimport {\n\tConfig,\n\tCollection,\n\tCKEditorError,\n\tLocale,\n\ttype LocaleTranslate\n} from '@ckeditor/ckeditor5-utils';\n\nimport PluginCollection from './plugincollection';\nimport type Editor from './editor/editor';\nimport type { LoadedPlugins, PluginConstructor } from './plugin';\nimport type { EditorConfig } from './editor/editorconfig';\n\n/**\n * Provides a common, higher-level environment for solutions that use multiple {@link module:core/editor/editor~Editor editors}\n * or plugins that work outside the editor. Use it instead of {@link module:core/editor/editor~Editor.create `Editor.create()`}\n * in advanced application integrations.\n *\n * All configuration options passed to a context will be used as default options for the editor instances initialized in that context.\n *\n * {@link module:core/contextplugin~ContextPlugin Context plugins} passed to a context instance will be shared among all\n * editor instances initialized in this context. These will be the same plugin instances for all the editors.\n *\n * **Note:** The context can only be initialized with {@link module:core/contextplugin~ContextPlugin context plugins}\n * (e.g. [comments](https://ckeditor.com/collaboration/comments/)). Regular {@link module:core/plugin~Plugin plugins} require an\n * editor instance to work and cannot be added to a context.\n *\n * **Note:** You can add a context plugin to an editor instance, though.\n *\n * If you are using multiple editor instances on one page and use any context plugins, create a context to share the configuration and\n * plugins among these editors. Some plugins will use the information about all existing editors to better integrate between them.\n *\n * If you are using plugins that do not require an editor to work (e.g. [comments](https://ckeditor.com/collaboration/comments/)),\n * enable and configure them using the context.\n *\n * If you are using only a single editor on each page, use {@link module:core/editor/editor~Editor.create `Editor.create()`} instead.\n * In such a case, a context instance will be created by the editor instance in a transparent way.\n *\n * See {@link ~Context.create `Context.create()`} for usage examples.\n */\nexport default class Context {\n\t/**\n\t * Stores all the configurations specific to this context instance.\n\t */\n\tpublic readonly config: Config<ContextConfig>;\n\n\t/**\n\t * The plugins loaded and in use by this context instance.\n\t */\n\tpublic readonly plugins: PluginCollection<Context | Editor>;\n\n\tpublic readonly locale: Locale;\n\n\t/**\n\t * Shorthand for {@link module:utils/locale~Locale#t}.\n\t */\n\tpublic readonly t: LocaleTranslate;\n\n\t/**\n\t * A list of editors that this context instance is injected to.\n\t */\n\tpublic readonly editors: Collection<Editor>;\n\n\t/**\n\t * The default configuration which is built into the `Context` class.\n\t *\n\t * It is used in CKEditor 5 builds featuring `Context` to provide the default configuration options which are later used during the\n\t * context initialization.\n\t *\n\t * ```ts\n\t * Context.defaultConfig = {\n\t * \tfoo: 1,\n\t * \tbar: 2\n\t * };\n\t *\n\t * Context\n\t * \t.create()\n\t * \t.then( context => {\n\t * \t\tcontext.config.get( 'foo' ); // -> 1\n\t * \t\tcontext.config.get( 'bar' ); // -> 2\n\t * \t} );\n\t *\n\t * // The default options can be overridden by the configuration passed to create().\n\t * Context\n\t * \t.create( { bar: 3 } )\n\t * \t.then( context => {\n\t * \t\tcontext.config.get( 'foo' ); // -> 1\n\t * \t\tcontext.config.get( 'bar' ); // -> 3\n\t * \t} );\n\t * ```\n\t *\n\t * See also {@link module:core/context~Context.builtinPlugins `Context.builtinPlugins`}\n\t * and {@link module:core/editor/editor~Editor.defaultConfig `Editor.defaultConfig`}.\n\t */\n\tpublic static defaultConfig: ContextConfig;\n\n\t/**\n\t * An array of plugins built into the `Context` class.\n\t *\n\t * It is used in CKEditor 5 builds featuring `Context` to provide a list of context plugins which are later automatically initialized\n\t * during the context initialization.\n\t *\n\t * They will be automatically initialized by `Context` unless `config.plugins` is passed.\n\t *\n\t * ```ts\n\t * // Build some context plugins into the Context class first.\n\t * Context.builtinPlugins = [ FooPlugin, BarPlugin ];\n\t *\n\t * // Normally, you need to define config.plugins, but since Context.builtinPlugins was\n\t * // defined, now you can call create() without any configuration.\n\t * Context\n\t * \t.create()\n\t * \t.then( context => {\n\t * \t\tcontext.plugins.get( FooPlugin ); // -> An instance of the Foo plugin.\n\t * \t\tcontext.plugins.get( BarPlugin ); // -> An instance of the Bar plugin.\n\t * \t} );\n\t * ```\n\t *\n\t * See also {@link module:core/context~Context.defaultConfig `Context.defaultConfig`}\n\t * and {@link module:core/editor/editor~Editor.builtinPlugins `Editor.builtinPlugins`}.\n\t */\n\tpublic static builtinPlugins: Array<PluginConstructor<Context | Editor>>;\n\n\t/**\n\t * Reference to the editor which created the context.\n\t * Null when the context was created outside of the editor.\n\t *\n\t * It is used to destroy the context when removing the editor that has created the context.\n\t */\n\tprivate _contextOwner: Editor | null = null;\n\n\t/**\n\t * Creates a context instance with a given configuration.\n\t *\n\t * Usually not to be used directly. See the static {@link module:core/context~Context.create `create()`} method.\n\t *\n\t * @param config The context configuration.\n\t */\n\tconstructor( config?: ContextConfig ) {\n\t\tthis.config = new Config<ContextConfig>( config, ( this.constructor as typeof Context ).defaultConfig );\n\n\t\tconst availablePlugins = ( this.constructor as typeof Context ).builtinPlugins;\n\n\t\tthis.config.define( 'plugins', availablePlugins );\n\n\t\tthis.plugins = new PluginCollection<Context | Editor>( this, availablePlugins );\n\n\t\tconst languageConfig = this.config.get( 'language' ) || {};\n\n\t\tthis.locale = new Locale( {\n\t\t\tuiLanguage: typeof languageConfig === 'string' ? languageConfig : languageConfig.ui,\n\t\t\tcontentLanguage: this.config.get( 'language.content' )\n\t\t} );\n\n\t\tthis.t = this.locale.t;\n\n\t\tthis.editors = new Collection<Editor>();\n\t}\n\n\t/**\n\t * Loads and initializes plugins specified in the configuration.\n\t *\n\t * @returns A promise which resolves once the initialization is completed, providing an array of loaded plugins.\n\t */\n\tpublic initPlugins(): Promise<LoadedPlugins> {\n\t\tconst plugins = this.config.get( 'plugins' ) || [];\n\t\tconst substitutePlugins = this.config.get( 'substitutePlugins' ) || [];\n\n\t\t// Plugins for substitution should be checked as well.\n\t\tfor ( const Plugin of plugins.concat( substitutePlugins ) ) {\n\t\t\tif ( typeof Plugin != 'function' ) {\n\t\t\t\t/**\n\t\t\t\t * Only a constructor function is allowed as a {@link module:core/contextplugin~ContextPlugin context plugin}.\n\t\t\t\t *\n\t\t\t\t * @error context-initplugins-constructor-only\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'context-initplugins-constructor-only',\n\t\t\t\t\tnull,\n\t\t\t\t\t{ Plugin }\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tif ( Plugin.isContextPlugin !== true ) {\n\t\t\t\t/**\n\t\t\t\t * Only a plugin marked as a {@link module:core/contextplugin~ContextPlugin.isContextPlugin context plugin}\n\t\t\t\t * is allowed to be used with a context.\n\t\t\t\t *\n\t\t\t\t * @error context-initplugins-invalid-plugin\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'context-initplugins-invalid-plugin',\n\t\t\t\t\tnull,\n\t\t\t\t\t{ Plugin }\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\n\t\treturn this.plugins.init( plugins, [], substitutePlugins );\n\t}\n\n\t/**\n\t * Destroys the context instance and all editors used with the context,\n\t * releasing all resources used by the context.\n\t *\n\t * @returns A promise that resolves once the context instance is fully destroyed.\n\t */\n\tpublic destroy(): Promise<unknown> {\n\t\treturn Promise.all( Array.from( this.editors, editor => editor.destroy() ) )\n\t\t\t.then( () => this.plugins.destroy() );\n\t}\n\n\t/**\n\t * Adds a reference to the editor which is used with this context.\n\t *\n\t * When the given editor has created the context, the reference to this editor will be stored\n\t * as a {@link ~Context#_contextOwner}.\n\t *\n\t * This method should only be used by the editor.\n\t *\n\t * @internal\n\t * @param isContextOwner Stores the given editor as a context owner.\n\t */\n\tpublic _addEditor( editor: Editor, isContextOwner: boolean ): void {\n\t\tif ( this._contextOwner ) {\n\t\t\t/**\n\t\t\t * Cannot add multiple editors to the context which is created by the editor.\n\t\t\t *\n\t\t\t * @error context-addeditor-private-context\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'context-addeditor-private-context' );\n\t\t}\n\n\t\tthis.editors.add( editor );\n\n\t\tif ( isContextOwner ) {\n\t\t\tthis._contextOwner = editor;\n\t\t}\n\t}\n\n\t/**\n\t * Removes a reference to the editor which was used with this context.\n\t * When the context was created by the given editor, the context will be destroyed.\n\t *\n\t * This method should only be used by the editor.\n\t *\n\t * @internal\n\t * @return A promise that resolves once the editor is removed from the context or when the context was destroyed.\n\t */\n\tpublic _removeEditor( editor: Editor ): Promise<unknown> {\n\t\tif ( this.editors.has( editor ) ) {\n\t\t\tthis.editors.remove( editor );\n\t\t}\n\n\t\tif ( this._contextOwner === editor ) {\n\t\t\treturn this.destroy();\n\t\t}\n\n\t\treturn Promise.resolve();\n\t}\n\n\t/**\n\t * Returns the context configuration which will be copied to the editors created using this context.\n\t *\n\t * The configuration returned by this method has the plugins configuration removed &mdash; plugins are shared with all editors\n\t * through another mechanism.\n\t *\n\t * This method should only be used by the editor.\n\t *\n\t * @internal\n\t * @returns Configuration as a plain object.\n\t */\n\tpublic _getEditorConfig(): Partial<EditorConfig> {\n\t\tconst result: Record<string, unknown> = {};\n\n\t\tfor ( const name of this.config.names() ) {\n\t\t\tif ( ![ 'plugins', 'removePlugins', 'extraPlugins' ].includes( name ) ) {\n\t\t\t\tresult[ name ] = this.config.get( name );\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t}\n\n\t/**\n\t * Creates and initializes a new context instance.\n\t *\n\t * ```ts\n\t * const commonConfig = { ... }; // Configuration for all the plugins and editors.\n\t * const editorPlugins = [ ... ]; // Regular plugins here.\n\t *\n\t * Context\n\t * \t.create( {\n\t * \t\t// Only context plugins here.\n\t * \t\tplugins: [ ... ],\n\t *\n\t * \t\t// Configure the language for all the editors (it cannot be overwritten).\n\t * \t\tlanguage: { ... },\n\t *\n\t * \t\t// Configuration for context plugins.\n\t * \t\tcomments: { ... },\n\t * \t\t...\n\t *\n\t * \t\t// Default configuration for editor plugins.\n\t * \t\ttoolbar: { ... },\n\t * \t\timage: { ... },\n\t * \t\t...\n\t * \t} )\n\t * \t.then( context => {\n\t * \t\tconst promises = [];\n\t *\n\t * \t\tpromises.push( ClassicEditor.create(\n\t * \t\t\tdocument.getElementById( 'editor1' ),\n\t * \t\t\t{\n\t * \t\t\t\teditorPlugins,\n\t * \t\t\t\tcontext\n\t * \t\t\t}\n\t * \t\t) );\n\t *\n\t * \t\tpromises.push( ClassicEditor.create(\n\t * \t\t\tdocument.getElementById( 'editor2' ),\n\t * \t\t\t{\n\t * \t\t\t\teditorPlugins,\n\t * \t\t\t\tcontext,\n\t * \t\t\t\ttoolbar: { ... } // You can overwrite the configuration of the context.\n\t * \t\t\t}\n\t * \t\t) );\n\t *\n\t * \t\treturn Promise.all( promises );\n\t * \t} );\n\t * ```\n\t *\n\t * @param config The context configuration.\n\t * @returns A promise resolved once the context is ready. The promise resolves with the created context instance.\n\t */\n\tpublic static create( config?: ContextConfig ): Promise<Context> {\n\t\treturn new Promise( resolve => {\n\t\t\tconst context = new this( config );\n\n\t\t\tresolve( context.initPlugins().then( () => context ) );\n\t\t} );\n\t}\n}\n\n/**\n * The context configuration.\n */\nexport type ContextConfig = {\n\tplugins?: Array<PluginConstructor<Context | Editor>>;\n\tsubstitutePlugins?: Array<PluginConstructor<Context | Editor>>;\n} & Omit<EditorConfig, 'plugins' | 'substitutePlugins' | 'removePlugins' | 'extraPlugins'>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/contextplugin\n */\n\nimport {\n\tObservableMixin,\n\ttype Collection,\n\ttype Config,\n\ttype Locale,\n\ttype LocaleTranslate\n} from '@ckeditor/ckeditor5-utils';\n\nimport type Editor from './editor/editor';\nimport type { EditorConfig } from './editor/editorconfig';\nimport type Context from './context';\nimport type { PluginDependencies, PluginInterface } from './plugin';\nimport type PluginCollection from './plugincollection';\n\n/**\n * The base class for {@link module:core/context~Context} plugin classes.\n *\n * A context plugin can either be initialized for an {@link module:core/editor/editor~Editor editor} or for\n * a {@link module:core/context~Context context}. In other words, it can either\n * work within one editor instance or with one or more editor instances that use a single context.\n * It is the context plugin's role to implement handling for both modes.\n *\n * There are a few rules for interaction between the editor plugins and context plugins:\n *\n * * A context plugin can require another context plugin.\n * * An {@link module:core/plugin~Plugin editor plugin} can require a context plugin.\n * * A context plugin MUST NOT require an {@link module:core/plugin~Plugin editor plugin}.\n */\nexport default class ContextPlugin extends ObservableMixin() implements PluginInterface {\n\t/**\n\t * The context or editor instance.\n\t */\n\tpublic readonly context: ContextInterface;\n\n\t/**\n\t * Creates a new plugin instance.\n\t */\n\tconstructor( context: Context | Editor ) {\n\t\tsuper();\n\n\t\tthis.context = context as ContextInterface;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get isContextPlugin(): true {\n\t\treturn true;\n\t}\n}\n\n/**\n * The common interface of {@link module:core/context~Context} and {@link module:core/editor/editor~Editor}.\n */\nexport interface ContextInterface {\n\tconfig: Config<Omit<EditorConfig, 'plugins' | 'substitutePlugins' | 'removePlugins' | 'extraPlugins'>>;\n\tplugins: PluginCollection<Context | Editor>;\n\tlocale: Locale;\n\tt: LocaleTranslate;\n\teditors?: Collection<Editor>;\n}\n\nexport type ContextPluginDependencies = PluginDependencies<Context | Editor>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/editingkeystrokehandler\n */\n\nimport { KeystrokeHandler, type PriorityString } from '@ckeditor/ckeditor5-utils';\n\nimport type Editor from './editor/editor';\n\n/**\n * A keystroke handler for editor editing. Its instance is available\n * in {@link module:core/editor/editor~Editor#keystrokes} so plugins\n * can register their keystrokes.\n *\n * E.g. an undo plugin would do this:\n *\n * ```ts\n * editor.keystrokes.set( 'Ctrl+Z', 'undo' );\n * editor.keystrokes.set( 'Ctrl+Shift+Z', 'redo' );\n * editor.keystrokes.set( 'Ctrl+Y', 'redo' );\n * ```\n */\nexport default class EditingKeystrokeHandler extends KeystrokeHandler {\n\t/**\n\t * The editor instance.\n\t */\n\tpublic readonly editor: Editor;\n\n\t/**\n\t * Creates an instance of the keystroke handler.\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper();\n\t\tthis.editor = editor;\n\t}\n\n\t/**\n\t * Registers a handler for the specified keystroke.\n\t *\n\t * The handler can be specified as a command name or a callback.\n\t *\n\t * @param keystroke Keystroke defined in a format accepted by\n\t * the {@link module:utils/keyboard~parseKeystroke} function.\n\t * @param callback If a string is passed, then the keystroke will\n\t * {@link module:core/editor/editor~Editor#execute execute a command}.\n\t * If a function, then it will be called with the\n\t * {@link module:engine/view/observer/keyobserver~KeyEventData key event data} object and\n\t * a `cancel()` helper to both `preventDefault()` and `stopPropagation()` of the event.\n\t * @param options Additional options.\n\t * @param options.priority The priority of the keystroke callback. The higher the priority value\n\t * the sooner the callback will be executed. Keystrokes having the same priority\n\t * are called in the order they were added.\n\t */\n\tpublic override set(\n\t\tkeystroke: string | Array<string | number>,\n\t\tcallback: EditingKeystrokeCallback,\n\t\toptions: { readonly priority?: PriorityString } = {}\n\t): void {\n\t\tif ( typeof callback == 'string' ) {\n\t\t\tconst commandName = callback;\n\n\t\t\tcallback = ( evtData, cancel ) => {\n\t\t\t\tthis.editor.execute( commandName );\n\t\t\t\tcancel();\n\t\t\t};\n\t\t}\n\n\t\tsuper.set( keystroke, callback, options );\n\t}\n}\n\n/**\n * Command name or a callback to be executed when a given keystroke is pressed.\n */\nexport type EditingKeystrokeCallback = string | ( ( ev: KeyboardEvent, cancel: () => void ) => void );\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./placeholder.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/placeholder\n */\n\nimport '../../theme/placeholder.css';\n\nimport type Document from './document';\nimport type DowncastWriter from './downcastwriter';\nimport type EditableElement from './editableelement';\nimport type Element from './element';\nimport type View from './view';\n\nimport { logWarning, type ObservableChangeEvent } from '@ckeditor/ckeditor5-utils';\n\n// Each document stores information about its placeholder elements and check functions.\nconst documentPlaceholders = new WeakMap<Document, Map<Element, PlaceholderConfig>>();\n\nlet hasDisplayedPlaceholderDeprecationWarning = false;\n\n/**\n * A helper that enables a placeholder on the provided view element (also updates its visibility).\n * The placeholder is a CSS pseudoelement (with a text content) attached to the element.\n *\n * To change the placeholder text, change value of the `placeholder` property in the provided `element`.\n *\n * To disable the placeholder, use {@link module:engine/view/placeholder~disablePlaceholder `disablePlaceholder()`} helper.\n *\n * @param options Configuration options of the placeholder.\n * @param options.view Editing view instance.\n * @param options.element Element that will gain a placeholder. See `options.isDirectHost` to learn more.\n * @param options.isDirectHost If set `false`, the placeholder will not be enabled directly\n * in the passed `element` but in one of its children (selected automatically, i.e. a first empty child element).\n * Useful when attaching placeholders to elements that can host other elements (not just text), for instance,\n * editable root elements.\n * @param options.text Placeholder text. It's **deprecated** and will be removed soon. Use\n * {@link module:engine/view/placeholder~PlaceholderableElement#placeholder `options.element.placeholder`} instead.\n * @param options.keepOnFocus If set `true`, the placeholder stay visible when the host element is focused.\n */\nexport function enablePlaceholder( { view, element, text, isDirectHost = true, keepOnFocus = false }: {\n\tview: View;\n\telement: PlaceholderableElement | EditableElement;\n\tisDirectHost?: boolean;\n\ttext?: string;\n\tkeepOnFocus?: boolean;\n} ): void {\n\tconst doc = view.document;\n\n\t// Use a single a single post fixer per—document to update all placeholders.\n\tif ( !documentPlaceholders.has( doc ) ) {\n\t\tdocumentPlaceholders.set( doc, new Map() );\n\n\t\t// If a post-fixer callback makes a change, it should return `true` so other postfixers\n\t\t// can reevaluate the document again.\n\t\tdoc.registerPostFixer( writer => updateDocumentPlaceholders( doc, writer ) );\n\n\t\t// Update placeholders on isComposing state change since rendering is disabled while in composition mode.\n\t\tdoc.on<ObservableChangeEvent>( 'change:isComposing', () => {\n\t\t\tview.change( writer => updateDocumentPlaceholders( doc, writer ) );\n\t\t}, { priority: 'high' } );\n\t}\n\n\tif ( element.is( 'editableElement' ) ) {\n\t\telement.on( 'change:placeholder', ( evtInfo, evt, text ) => {\n\t\t\tsetPlaceholder( text );\n\t\t} );\n\t}\n\n\tif ( element.placeholder ) {\n\t\tsetPlaceholder( element.placeholder );\n\t} else if ( text ) {\n\t\tsetPlaceholder( text );\n\t}\n\n\tif ( text ) {\n\t\tshowPlaceholderTextDeprecationWarning();\n\t}\n\n\tfunction setPlaceholder( text: string ) {\n\t\t// Store information about the element placeholder under its document.\n\t\tdocumentPlaceholders.get( doc )!.set( element, {\n\t\t\ttext,\n\t\t\tisDirectHost,\n\t\t\tkeepOnFocus,\n\t\t\thostElement: isDirectHost ? element : null\n\t\t} );\n\n\t\t// Update the placeholders right away.\n\t\tview.change( writer => updateDocumentPlaceholders( doc, writer ) );\n\t}\n}\n\n/**\n * Disables the placeholder functionality from a given element.\n *\n * See {@link module:engine/view/placeholder~enablePlaceholder `enablePlaceholder()`} to learn more.\n */\nexport function disablePlaceholder( view: View, element: Element ): void {\n\tconst doc = element.document;\n\n\tif ( !documentPlaceholders.has( doc ) ) {\n\t\treturn;\n\t}\n\n\tview.change( writer => {\n\t\tconst placeholders = documentPlaceholders.get( doc )!;\n\t\tconst config = placeholders.get( element )!;\n\n\t\twriter.removeAttribute( 'data-placeholder', config.hostElement! );\n\t\thidePlaceholder( writer, config.hostElement! );\n\n\t\tplaceholders.delete( element );\n\t} );\n}\n\n/**\n * Shows a placeholder in the provided element by changing related attributes and CSS classes.\n *\n * **Note**: This helper will not update the placeholder visibility nor manage the\n * it in any way in the future. What it does is a onetime state change of an element. Use\n * {@link module:engine/view/placeholder~enablePlaceholder `enablePlaceholder()`} and\n * {@link module:engine/view/placeholder~disablePlaceholder `disablePlaceholder()`} for full\n * placeholder functionality.\n *\n * **Note**: This helper will blindly show the placeholder directly in the root editable element if\n * one is passed, which could result in a visual clash if the editable element has some children\n * (for instance, an empty paragraph). Use {@link module:engine/view/placeholder~enablePlaceholder `enablePlaceholder()`}\n * in that case or make sure the correct element is passed to the helper.\n *\n * @returns `true`, if any changes were made to the `element`.\n */\nexport function showPlaceholder( writer: DowncastWriter, element: Element ): boolean {\n\tif ( !element.hasClass( 'ck-placeholder' ) ) {\n\t\twriter.addClass( 'ck-placeholder', element );\n\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n\n/**\n * Hides a placeholder in the element by changing related attributes and CSS classes.\n *\n * **Note**: This helper will not update the placeholder visibility nor manage the\n * it in any way in the future. What it does is a onetime state change of an element. Use\n * {@link module:engine/view/placeholder~enablePlaceholder `enablePlaceholder()`} and\n * {@link module:engine/view/placeholder~disablePlaceholder `disablePlaceholder()`} for full\n * placeholder functionality.\n *\n * @returns `true`, if any changes were made to the `element`.\n */\nexport function hidePlaceholder( writer: DowncastWriter, element: Element ): boolean {\n\tif ( element.hasClass( 'ck-placeholder' ) ) {\n\t\twriter.removeClass( 'ck-placeholder', element );\n\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n\n/**\n * Checks if a placeholder should be displayed in the element.\n *\n * **Note**: This helper will blindly check the possibility of showing a placeholder directly in the\n * root editable element if one is passed, which may not be the expected result. If an element can\n * host other elements (not just text), most likely one of its children should be checked instead\n * because it will be the final host for the placeholder. Use\n * {@link module:engine/view/placeholder~enablePlaceholder `enablePlaceholder()`} in that case or make\n * sure the correct element is passed to the helper.\n *\n * @param element Element that holds the placeholder.\n * @param keepOnFocus Focusing the element will keep the placeholder visible.\n */\nexport function needsPlaceholder( element: Element, keepOnFocus: boolean ): boolean {\n\tif ( !element.isAttached() ) {\n\t\treturn false;\n\t}\n\n\t// Anything but uiElement(s) counts as content.\n\tconst hasContent = Array.from( element.getChildren() )\n\t\t.some( element => !element.is( 'uiElement' ) );\n\n\tif ( hasContent ) {\n\t\treturn false;\n\t}\n\n\tconst doc = element.document;\n\tconst viewSelection = doc.selection;\n\tconst selectionAnchor = viewSelection.anchor;\n\n\tif ( doc.isComposing && selectionAnchor && selectionAnchor.parent === element ) {\n\t\treturn false;\n\t}\n\n\t// Skip the focus check and make the placeholder visible already regardless of document focus state.\n\tif ( keepOnFocus ) {\n\t\treturn true;\n\t}\n\n\t// If the document is blurred.\n\tif ( !doc.isFocused ) {\n\t\treturn true;\n\t}\n\n\t// If document is focused and the element is empty but the selection is not anchored inside it.\n\treturn !!selectionAnchor && selectionAnchor.parent !== element;\n}\n\n/**\n * Updates all placeholders associated with a document in a postfixer callback.\n *\n * @returns True if any changes were made to the view document.\n */\nfunction updateDocumentPlaceholders( doc: Document, writer: DowncastWriter ): boolean {\n\tconst placeholders = documentPlaceholders.get( doc )!;\n\tconst directHostElements: Array<Element> = [];\n\tlet wasViewModified = false;\n\n\t// First set placeholders on the direct hosts.\n\tfor ( const [ element, config ] of placeholders ) {\n\t\tif ( config.isDirectHost ) {\n\t\t\tdirectHostElements.push( element );\n\n\t\t\tif ( updatePlaceholder( writer, element, config ) ) {\n\t\t\t\twasViewModified = true;\n\t\t\t}\n\t\t}\n\t}\n\n\t// Then set placeholders on the indirect hosts but only on those that does not already have an direct host placeholder.\n\tfor ( const [ element, config ] of placeholders ) {\n\t\tif ( config.isDirectHost ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst hostElement = getChildPlaceholderHostSubstitute( element );\n\n\t\t// When not a direct host, it could happen that there is no child element\n\t\t// capable of displaying a placeholder.\n\t\tif ( !hostElement ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\t// Don't override placeholder if the host element already has some direct placeholder.\n\t\tif ( directHostElements.includes( hostElement ) ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\t// Update the host element (used for setting and removing the placeholder).\n\t\tconfig.hostElement = hostElement;\n\n\t\tif ( updatePlaceholder( writer, element, config ) ) {\n\t\t\twasViewModified = true;\n\t\t}\n\t}\n\n\treturn wasViewModified;\n}\n\n/**\n * Updates a single placeholder in a postfixer callback.\n *\n * @returns True if any changes were made to the view document.\n */\nfunction updatePlaceholder( writer: DowncastWriter, element: Element, config: PlaceholderConfig ) {\n\tconst { text, isDirectHost, hostElement } = config;\n\n\tlet wasViewModified = false;\n\n\t// This may be necessary when updating the placeholder text to something else.\n\tif ( hostElement!.getAttribute( 'data-placeholder' ) !== text ) {\n\t\twriter.setAttribute( 'data-placeholder', text, hostElement! );\n\t\twasViewModified = true;\n\t}\n\n\t// If the host element is not a direct host then placeholder is needed only when there is only one element.\n\tconst isOnlyChild = isDirectHost || element.childCount == 1;\n\n\tif ( isOnlyChild && needsPlaceholder( hostElement!, config.keepOnFocus ) ) {\n\t\tif ( showPlaceholder( writer, hostElement! ) ) {\n\t\t\twasViewModified = true;\n\t\t}\n\t} else if ( hidePlaceholder( writer, hostElement! ) ) {\n\t\twasViewModified = true;\n\t}\n\n\treturn wasViewModified;\n}\n\n/**\n * Gets a child element capable of displaying a placeholder if a parent element can host more\n * than just text (for instance, when it is a root editable element). The child element\n * can then be used in other placeholder helpers as a substitute of its parent.\n */\nfunction getChildPlaceholderHostSubstitute( parent: Element ): Element | null {\n\tif ( parent.childCount ) {\n\t\tconst firstChild = parent.getChild( 0 )!;\n\n\t\tif ( firstChild.is( 'element' ) && !firstChild.is( 'uiElement' ) && !firstChild.is( 'attributeElement' ) ) {\n\t\t\treturn firstChild;\n\t\t}\n\t}\n\n\treturn null;\n}\n\n/**\n * Displays a deprecation warning message in the console, but only once per page load.\n */\nfunction showPlaceholderTextDeprecationWarning() {\n\tif ( !hasDisplayedPlaceholderDeprecationWarning ) {\n\t\t/**\n\t\t * The \"text\" option in the {@link module:engine/view/placeholder~enablePlaceholder `enablePlaceholder()`}\n\t\t * function is deprecated and will be removed soon.\n\t\t *\n\t\t * See the {@glink updating/guides/update-to-39#view-element-placeholder Migration to v39} guide for\n\t\t * more information on how to apply this change.\n\t\t *\n\t\t * @error enableplaceholder-deprecated-text-option\n\t\t */\n\t\tlogWarning( 'enableplaceholder-deprecated-text-option' );\n\t}\n\n\thasDisplayedPlaceholderDeprecationWarning = true;\n}\n\n/**\n * Configuration of the placeholder.\n */\ninterface PlaceholderConfig {\n\ttext: string;\n\tisDirectHost: boolean;\n\tkeepOnFocus: boolean;\n\thostElement: Element | null;\n}\n\n/**\n * Element that could have a placeholder.\n */\nexport interface PlaceholderableElement extends Element {\n\n\t/**\n\t * The text of element's placeholder.\n\t */\n\tplaceholder?: string;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport type AttributeElement from './attributeelement';\nimport type ContainerElement from './containerelement';\nimport type DocumentFragment from './documentfragment';\nimport type DocumentSelection from './documentselection';\nimport type EditableElement from './editableelement';\nimport type Element from './element';\nimport type EmptyElement from './emptyelement';\nimport type Node from './node';\nimport type Position from './position';\nimport type Range from './range';\nimport type RawElement from './rawelement';\nimport type RootEditableElement from './rooteditableelement';\nimport type Selection from './selection';\nimport type Text from './text';\nimport type TextProxy from './textproxy';\nimport type UIElement from './uielement';\n\n/**\n * @module engine/view/typecheckable\n */\n\nexport default abstract class TypeCheckable {\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/node~Node} or its subclass.\n\t *\n\t * This method is useful when processing view objects that are of unknown type. For example, a function\n\t * may return a {@link module:engine/view/documentfragment~DocumentFragment} or a {@link module:engine/view/node~Node}\n\t * that can be either a text node or an element. This method can be used to check what kind of object is returned.\n\t *\n\t * ```ts\n\t * someObject.is( 'element' ); // -> true if this is an element\n\t * someObject.is( 'node' ); // -> true if this is a node (a text node or an element)\n\t * someObject.is( 'documentFragment' ); // -> true if this is a document fragment\n\t * ```\n\t *\n\t * Since this method is also available on a range of model objects, you can prefix the type of the object with\n\t * `model:` or `view:` to check, for example, if this is the model's or view's element:\n\t *\n\t * ```ts\n\t * viewElement.is( 'view:element' ); // -> true\n\t * viewElement.is( 'model:element' ); // -> false\n\t * ```\n\t *\n\t * By using this method it is also possible to check a name of an element:\n\t *\n\t * ```ts\n\t * imgElement.is( 'element', 'img' ); // -> true\n\t * imgElement.is( 'view:element', 'img' ); // -> same as above, but more precise\n\t * ```\n\t * @label NODE\n\t */\n\tpublic is( type: 'node' | 'view:node' ): this is (\n\t\tNode |\n\t\tText |\n\t\tElement |\n\t\tAttributeElement |\n\t\tContainerElement |\n\t\tEditableElement |\n\t\tEmptyElement |\n\t\tRawElement |\n\t\tRootEditableElement |\n\t\tUIElement\n\t);\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/element~Element} or its subclass.\n\t *\n\t * ```ts\n\t * element.is( 'element' ); // -> true\n\t * element.is( 'node' ); // -> true\n\t * element.is( 'view:element' ); // -> true\n\t * element.is( 'view:node' ); // -> true\n\t *\n\t * element.is( 'model:element' ); // -> false\n\t * element.is( 'documentSelection' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is an element, you can also check its\n\t * {@link module:engine/view/element~Element#name name}:\n\t *\n\t * ```ts\n\t * element.is( 'element', 'img' ); // -> true if this is an <img> element\n\t * text.is( 'element', 'img' ); -> false\n\t * ```\n\t *\n\t * @label ELEMENT\n\t */\n\tpublic is( type: 'element' | 'view:element' ): this is (\n\t\tElement |\n\t\tAttributeElement |\n\t\tContainerElement |\n\t\tEditableElement |\n\t\tEmptyElement |\n\t\tRawElement |\n\t\tRootEditableElement |\n\t\tUIElement\n\t);\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/attributeelement~AttributeElement}.\n\t *\n\t * ```ts\n\t * attributeElement.is( 'attributeElement' ); // -> true\n\t * attributeElement.is( 'element' ); // -> true\n\t * attributeElement.is( 'node' ); // -> true\n\t * attributeElement.is( 'view:attributeElement' ); // -> true\n\t * attributeElement.is( 'view:element' ); // -> true\n\t * attributeElement.is( 'view:node' ); // -> true\n\t *\n\t * attributeElement.is( 'model:element' ); // -> false\n\t * attributeElement.is( 'documentFragment' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is an attribute element, you can also check its\n\t * {@link module:engine/view/attributeelement~AttributeElement#name name}:\n\t *\n\t * ```ts\n\t * attributeElement.is( 'element', 'b' ); // -> true if this is a bold element\n\t * attributeElement.is( 'attributeElement', 'b' ); // -> same as above\n\t * text.is( 'element', 'b' ); -> false\n\t * ```\n\t *\n\t * @label ATTRIBUTE_ELEMENT\n\t */\n\tpublic is( type: 'attributeElement' | 'view:attributeElement' ): this is AttributeElement;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/containerelement~ContainerElement} or its subclass.\n\t *\n\t * ```ts\n\t * containerElement.is( 'containerElement' ); // -> true\n\t * containerElement.is( 'element' ); // -> true\n\t * containerElement.is( 'node' ); // -> true\n\t * containerElement.is( 'view:containerElement' ); // -> true\n\t * containerElement.is( 'view:element' ); // -> true\n\t * containerElement.is( 'view:node' ); // -> true\n\t *\n\t * containerElement.is( 'model:element' ); // -> false\n\t * containerElement.is( 'documentFragment' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is a container element, you can also check its\n\t * {@link module:engine/view/containerelement~ContainerElement#name name}:\n\t *\n\t * ```ts\n\t * containerElement.is( 'element', 'div' ); // -> true if this is a div container element\n\t * containerElement.is( 'contaienrElement', 'div' ); // -> same as above\n\t * text.is( 'element', 'div' ); -> false\n\t * ```\n\t *\n\t * @label CONTAINER_ELEMENT\n\t */\n\tpublic is( type: 'containerElement' | 'view:containerElement' ): this is ContainerElement | EditableElement | RootEditableElement;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/editableelement~EditableElement} or its subclass.\n\t *\n\t * ```ts\n\t * editableElement.is( 'editableElement' ); // -> true\n\t * editableElement.is( 'element' ); // -> true\n\t * editableElement.is( 'node' ); // -> true\n\t * editableElement.is( 'view:editableElement' ); // -> true\n\t * editableElement.is( 'view:element' ); // -> true\n\t * editableElement.is( 'view:node' ); // -> true\n\t *\n\t * editableElement.is( 'model:element' ); // -> false\n\t * editableElement.is( 'documentFragment' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is an editbale element, you can also check its\n\t * {@link module:engine/view/editableelement~EditableElement#name name}:\n\t *\n\t * ```ts\n\t * editableElement.is( 'element', 'div' ); // -> true if this is a div element\n\t * editableElement.is( 'editableElement', 'div' ); // -> same as above\n\t * text.is( 'element', 'div' ); -> false\n\t * ```\n\t *\n\t * @label EDITABLE_ELEMENT\n\t */\n\tpublic is( type: 'editableElement' | 'view:editableElement' ): this is EditableElement | RootEditableElement;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/emptyelement~EmptyElement}.\n\t *\n\t * ```ts\n\t * emptyElement.is( 'emptyElement' ); // -> true\n\t * emptyElement.is( 'element' ); // -> true\n\t * emptyElement.is( 'node' ); // -> true\n\t * emptyElement.is( 'view:emptyElement' ); // -> true\n\t * emptyElement.is( 'view:element' ); // -> true\n\t * emptyElement.is( 'view:node' ); // -> true\n\t *\n\t * emptyElement.is( 'model:element' ); // -> false\n\t * emptyElement.is( 'documentFragment' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is an empty element, you can also check its\n\t * {@link module:engine/view/emptyelement~EmptyElement#name name}:\n\t *\n\t * ```ts\n\t * emptyElement.is( 'element', 'img' ); // -> true if this is a img element\n\t * emptyElement.is( 'emptyElement', 'img' ); // -> same as above\n\t * text.is( 'element', 'img' ); -> false\n\t * ```\n\t *\n\t * @label EMPTY_ELEMENT\n\t */\n\tpublic is( type: 'emptyElement' | 'view:emptyElement' ): this is EmptyElement;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/rawelement~RawElement}.\n\t *\n\t * ```ts\n\t * rawElement.is( 'rawElement' ); // -> true\n\t * rawElement.is( 'element' ); // -> true\n\t * rawElement.is( 'node' ); // -> true\n\t * rawElement.is( 'view:rawElement' ); // -> true\n\t * rawElement.is( 'view:element' ); // -> true\n\t * rawElement.is( 'view:node' ); // -> true\n\t *\n\t * rawElement.is( 'model:element' ); // -> false\n\t * rawElement.is( 'documentFragment' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is a raw element, you can also check its\n\t * {@link module:engine/view/rawelement~RawElement#name name}:\n\t *\n\t * ```ts\n\t * rawElement.is( 'img' ); // -> true if this is an img element\n\t * rawElement.is( 'rawElement', 'img' ); // -> same as above\n\t * text.is( 'img' ); -> false\n\t * ```\n\t *\n\t * @label RAW_ELEMENT\n\t */\n\tpublic is( type: 'rawElement' | 'view:rawElement' ): this is RawElement;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/rooteditableelement~RootEditableElement}.\n\t *\n\t * ```ts\n\t * rootEditableElement.is( 'rootElement' ); // -> true\n\t * rootEditableElement.is( 'editableElement' ); // -> true\n\t * rootEditableElement.is( 'element' ); // -> true\n\t * rootEditableElement.is( 'node' ); // -> true\n\t * rootEditableElement.is( 'view:editableElement' ); // -> true\n\t * rootEditableElement.is( 'view:element' ); // -> true\n\t * rootEditableElement.is( 'view:node' ); // -> true\n\t *\n\t * rootEditableElement.is( 'model:element' ); // -> false\n\t * rootEditableElement.is( 'documentFragment' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is a root editable element, you can also check its\n\t * {@link module:engine/view/rooteditableelement~RootEditableElement#name name}:\n\t *\n\t * ```ts\n\t * rootEditableElement.is( 'element', 'div' ); // -> true if this is a div root editable element\n\t * rootEditableElement.is( 'rootElement', 'div' ); // -> same as above\n\t * text.is( 'element', 'div' ); -> false\n\t * ```\n\t *\n\t * @label ROOT_ELEMENT\n\t */\n\tpublic is( type: 'rootElement' | 'view:rootElement' ): this is RootEditableElement;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/uielement~UIElement}.\n\t *\n\t * ```ts\n\t * uiElement.is( 'uiElement' ); // -> true\n\t * uiElement.is( 'element' ); // -> true\n\t * uiElement.is( 'node' ); // -> true\n\t * uiElement.is( 'view:uiElement' ); // -> true\n\t * uiElement.is( 'view:element' ); // -> true\n\t * uiElement.is( 'view:node' ); // -> true\n\t *\n\t * uiElement.is( 'model:element' ); // -> false\n\t * uiElement.is( 'documentFragment' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is an ui element, you can also check its\n\t * {@link module:engine/view/uielement~UIElement#name name}:\n\t *\n\t * ```ts\n\t * uiElement.is( 'element', 'span' ); // -> true if this is a span ui element\n\t * uiElement.is( 'uiElement', 'span' ); // -> same as above\n\t * text.is( 'element', 'span' ); -> false\n\t * ```\n\t *\n\t * @label UI_ELEMENT\n\t */\n\tpublic is( type: 'uiElement' | 'view:uiElement' ): this is UIElement;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/text~Text}.\n\t *\n\t * ```ts\n\t * text.is( '$text' ); // -> true\n\t * text.is( 'node' ); // -> true\n\t * text.is( 'view:$text' ); // -> true\n\t * text.is( 'view:node' ); // -> true\n\t *\n\t * text.is( 'model:$text' ); // -> false\n\t * text.is( 'element' ); // -> false\n\t * text.is( 'range' ); // -> false\n\t * ```\n\t *\n\t * @label TEXT\n\t */\n\tpublic is( type: '$text' | 'view:$text' ): this is Text;\n\n\t/**\n\t * hecks whether this object is of type {@link module:engine/view/documentfragment~DocumentFragment}.\n\t *\n\t * ```ts\n\t * docFrag.is( 'documentFragment' ); // -> true\n\t * docFrag.is( 'view:documentFragment' ); // -> true\n\t *\n\t * docFrag.is( 'model:documentFragment' ); // -> false\n\t * docFrag.is( 'element' ); // -> false\n\t * docFrag.is( 'node' ); // -> false\n\t * ```\n\t *\n\t * @label DOCUMENT_FRAGMENT\n\t */\n\tpublic is( type: 'documentFragment' | 'view:documentFragment' ): this is DocumentFragment;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/textproxy~TextProxy}.\n\t *\n\t * ```ts\n\t * textProxy.is( '$textProxy' ); // -> true\n\t * textProxy.is( 'view:$textProxy' ); // -> true\n\t *\n\t * textProxy.is( 'model:$textProxy' ); // -> false\n\t * textProxy.is( 'element' ); // -> false\n\t * textProxy.is( 'range' ); // -> false\n\t * ```\n\t *\n\t * **Note:** Until version 20.0.0 this method wasn't accepting `'$textProxy'` type. The legacy `'textProxy'` type is still\n\t * accepted for backward compatibility.\n\t *\n\t * @label TEXT_PROXY\n\t */\n\tpublic is( type: '$textProxy' | 'view:$textProxy' ): this is TextProxy;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/position~Position}.\n\t *\n\t * ```ts\n\t * position.is( 'position' ); // -> true\n\t * position.is( 'view:position' ); // -> true\n\t *\n\t * position.is( 'model:position' ); // -> false\n\t * position.is( 'element' ); // -> false\n\t * position.is( 'range' ); // -> false\n\t * ```\n\t *\n\t * @label POSITION\n\t */\n\tpublic is( type: 'position' | 'view:position' ): this is Position;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/range~Range}.\n\t *\n\t * ```ts\n\t * range.is( 'range' ); // -> true\n\t * range.is( 'view:range' ); // -> true\n\t *\n\t * range.is( 'model:range' ); // -> false\n\t * range.is( 'element' ); // -> false\n\t * range.is( 'selection' ); // -> false\n\t * ```\n\t *\n\t * @label RANGE\n\t */\n\tpublic is( type: 'range' | 'view:range' ): this is Range;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/selection~Selection} or\n\t * {@link module:engine/view/documentselection~DocumentSelection}.\n\t *\n\t * ```ts\n\t * selection.is( 'selection' ); // -> true\n\t * selection.is( 'view:selection' ); // -> true\n\t *\n\t * selection.is( 'model:selection' ); // -> false\n\t * selection.is( 'element' ); // -> false\n\t * selection.is( 'range' ); // -> false\n\t * ```\n\t *\n\t * @label SELECTION\n\t */\n\tpublic is( type: 'selection' | 'view:selection' ): this is Selection | DocumentSelection;\n\n\t/**\n\t * Checks whether this object is of type {@link module:engine/view/documentselection~DocumentSelection}.\n\t *\n\t * ```ts\n\t * `docSelection.is( 'selection' ); // -> true\n\t * docSelection.is( 'documentSelection' ); // -> true\n\t * docSelection.is( 'view:selection' ); // -> true\n\t * docSelection.is( 'view:documentSelection' ); // -> true\n\t *\n\t * docSelection.is( 'model:documentSelection' ); // -> false\n\t * docSelection.is( 'element' ); // -> false\n\t * docSelection.is( 'node' ); // -> false\n\t * ```\n\t *\n\t * @label DOCUMENT_SELECTION\n\t */\n\tpublic is( type: 'documentSelection' | 'view:documentSelection' ): this is DocumentSelection;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/view/element~Element} or its subclass and has the specified `name`.\n\t *\n\t * @label ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'element' | 'view:element', name: N ): this is (\n\t\tElement |\n\t\tAttributeElement |\n\t\tContainerElement |\n\t\tEditableElement |\n\t\tEmptyElement |\n\t\tRawElement |\n\t\tRootEditableElement |\n\t\tUIElement\n\t) & { name: N };\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/view/attributeelement~AttributeElement} and has the specified `name`.\n\t *\n\t * @label ATTRIBUTE_ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'attributeElement' | 'view:attributeElement', name: N ): this is AttributeElement & { name: N };\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/view/containerelement~ContainerElement}\n\t * or its subclass and has the specified `name`.\n\t *\n\t * @label CONTAINER_ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'containerElement' | 'view:containerElement', name: N ): this is (\n\t\tContainerElement |\n\t\tEditableElement |\n\t\tRootEditableElement\n\t) & { name: N };\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/view/editableelement~EditableElement}\n\t * or its subclass and has the specified `name`.\n\t *\n\t * @label EDITABLE_ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'editableElement' | 'view:editableElement', name: N ): this is (\n\t\tEditableElement |\n\t\tRootEditableElement\n\t) & { name: N };\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/view/emptyelement~EmptyElement} has the specified `name`.\n\t *\n\t * @label EMPTY_ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'emptyElement' | 'view:emptyElement', name: N ): this is EmptyElement & { name: N };\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/view/rawelement~RawElement} and has the specified `name`.\n\t *\n\t * @label RAW_ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'rawElement' | 'view:rawElement', name: N ): this is RawElement & { name: N };\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/view/rooteditableelement~RootEditableElement} and has the specified `name`.\n\t *\n\t * @label ROOT_ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'rootElement' | 'view:rootElement', name: N ): this is RootEditableElement & { name: N };\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/view/uielement~UIElement} and has the specified `name`.\n\t *\n\t * @label UI_ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'uiElement' | 'view:uiElement', name: N ): this is UIElement & { name: N };\n\n\t/* istanbul ignore next -- @preserve */\n\tpublic is(): boolean {\n\t\t// There are a lot of overloads above.\n\t\t// Overriding method in derived classes remove them and only `is( type: string ): boolean` is visible which we don't want.\n\t\t// One option would be to copy them all to all classes, but that's ugly.\n\t\t// It's best when TypeScript compiler doesn't see those overloads, except the one in the top base class.\n\t\t// To overload a method, but not let the compiler see it, do after class definition:\n\t\t// `MyClass.prototype.is = function( type: string ) {...}`\n\t\tthrow new Error( 'is() method is abstract' );\n\t}\n}\n","import baseClone from './_baseClone.js';\n\n/** Used to compose bitmasks for cloning. */\nvar CLONE_SYMBOLS_FLAG = 4;\n\n/**\n * Creates a shallow clone of `value`.\n *\n * **Note:** This method is loosely based on the\n * [structured clone algorithm](https://mdn.io/Structured_clone_algorithm)\n * and supports cloning arrays, array buffers, booleans, date objects, maps,\n * numbers, `Object` objects, regexes, sets, strings, symbols, and typed\n * arrays. The own enumerable properties of `arguments` objects are cloned\n * as plain objects. An empty object is returned for uncloneable values such\n * as error objects, functions, DOM nodes, and WeakMaps.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Lang\n * @param {*} value The value to clone.\n * @returns {*} Returns the cloned value.\n * @see _.cloneDeep\n * @example\n *\n * var objects = [{ 'a': 1 }, { 'b': 2 }];\n *\n * var shallow = _.clone(objects);\n * console.log(shallow[0] === objects[0]);\n * // => true\n */\nfunction clone(value) {\n return baseClone(value, CLONE_SYMBOLS_FLAG);\n}\n\nexport default clone;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/node\n */\n\nimport TypeCheckable from './typecheckable';\n\nimport {\n\tCKEditorError,\n\tEmitterMixin,\n\tcompareArrays\n} from '@ckeditor/ckeditor5-utils';\n\nimport { clone } from 'lodash-es';\n\n// To check if component is loaded more than once.\nimport '@ckeditor/ckeditor5-utils/src/version';\n\nimport type { default as Document, ChangeType } from './document';\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\n\n/**\n * Abstract view node class.\n *\n * This is an abstract class. Its constructor should not be used directly.\n * Use the {@link module:engine/view/downcastwriter~DowncastWriter} or {@link module:engine/view/upcastwriter~UpcastWriter}\n * to create new instances of view nodes.\n */\nexport default abstract class Node extends EmitterMixin( TypeCheckable ) {\n\t/**\n\t * The document instance to which this node belongs.\n\t */\n\tpublic readonly document: Document;\n\n\t/**\n\t * Parent element. Null by default. Set by {@link module:engine/view/element~Element#_insertChild}.\n\t */\n\tpublic readonly parent: Element | DocumentFragment | null;\n\n\t/**\n\t * Creates a tree view node.\n\t *\n\t * @param document The document instance to which this node belongs.\n\t */\n\tprotected constructor( document: Document ) {\n\t\tsuper();\n\n\t\tthis.document = document;\n\t\tthis.parent = null;\n\t}\n\n\t/**\n\t * Index of the node in the parent element or null if the node has no parent.\n\t *\n\t * Accessing this property throws an error if this node's parent element does not contain it.\n\t * This means that view tree got broken.\n\t */\n\tpublic get index(): number | null {\n\t\tlet pos;\n\n\t\tif ( !this.parent ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// No parent or child doesn't exist in parent's children.\n\t\tif ( ( pos = this.parent.getChildIndex( this ) ) == -1 ) {\n\t\t\t/**\n\t\t\t * The node's parent does not contain this node. It means that the document tree is corrupted.\n\t\t\t *\n\t\t\t * @error view-node-not-found-in-parent\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-node-not-found-in-parent', this );\n\t\t}\n\n\t\treturn pos;\n\t}\n\n\t/**\n\t * Node's next sibling, or `null` if it is the last child.\n\t */\n\tpublic get nextSibling(): Node | null {\n\t\tconst index = this.index;\n\n\t\treturn ( index !== null && this.parent!.getChild( index + 1 ) ) || null;\n\t}\n\n\t/**\n\t * Node's previous sibling, or `null` if it is the first child.\n\t */\n\tpublic get previousSibling(): Node | null {\n\t\tconst index = this.index;\n\n\t\treturn ( index !== null && this.parent!.getChild( index - 1 ) ) || null;\n\t}\n\n\t/**\n\t * Top-most ancestor of the node. If the node has no parent it is the root itself.\n\t */\n\tpublic get root(): Element | DocumentFragment {\n\t\t// eslint-disable-next-line @typescript-eslint/no-this-alias, consistent-this\n\t\tlet root: Node | DocumentFragment = this;\n\n\t\twhile ( root.parent ) {\n\t\t\troot = root.parent;\n\t\t}\n\n\t\treturn root as any;\n\t}\n\n\t/**\n\t * Returns true if the node is in a tree rooted in the document (is a descendant of one of its roots).\n\t */\n\tpublic isAttached(): boolean {\n\t\treturn this.root.is( 'rootElement' );\n\t}\n\n\t/**\n\t * Gets a path to the node. The path is an array containing indices of consecutive ancestors of this node,\n\t * beginning from {@link module:engine/view/node~Node#root root}, down to this node's index.\n\t *\n\t * ```ts\n\t * const abc = downcastWriter.createText( 'abc' );\n\t * const foo = downcastWriter.createText( 'foo' );\n\t * const h1 = downcastWriter.createElement( 'h1', null, downcastWriter.createText( 'header' ) );\n\t * const p = downcastWriter.createElement( 'p', null, [ abc, foo ] );\n\t * const div = downcastWriter.createElement( 'div', null, [ h1, p ] );\n\t * foo.getPath(); // Returns [ 1, 3 ]. `foo` is in `p` which is in `div`. `p` starts at offset 1, while `foo` at 3.\n\t * h1.getPath(); // Returns [ 0 ].\n\t * div.getPath(); // Returns [].\n\t * ```\n\t *\n\t * @returns The path.\n\t */\n\tpublic getPath(): Array<number> {\n\t\tconst path = [];\n\t\t// eslint-disable-next-line @typescript-eslint/no-this-alias, consistent-this\n\t\tlet node: Node | DocumentFragment = this;\n\n\t\twhile ( node.parent ) {\n\t\t\tpath.unshift( node.index! );\n\t\t\tnode = node.parent;\n\t\t}\n\n\t\treturn path;\n\t}\n\n\t/**\n\t * Returns ancestors array of this node.\n\t *\n\t * @param options Options object.\n\t * @param options.includeSelf When set to `true` this node will be also included in parent's array.\n\t * @param options.parentFirst When set to `true`, array will be sorted from node's parent to root element,\n\t * otherwise root element will be the first item in the array.\n\t * @returns Array with ancestors.\n\t */\n\tpublic getAncestors( options: { includeSelf?: boolean; parentFirst?: boolean } = {} ): Array<Node | DocumentFragment> {\n\t\tconst ancestors: Array<Node | DocumentFragment> = [];\n\t\tlet parent = options.includeSelf ? this : this.parent;\n\n\t\twhile ( parent ) {\n\t\t\tancestors[ options.parentFirst ? 'push' : 'unshift' ]( parent );\n\t\t\tparent = parent.parent;\n\t\t}\n\n\t\treturn ancestors;\n\t}\n\n\t/**\n\t * Returns a {@link module:engine/view/element~Element} or {@link module:engine/view/documentfragment~DocumentFragment}\n\t * which is a common ancestor of both nodes.\n\t *\n\t * @param node The second node.\n\t * @param options Options object.\n\t * @param options.includeSelf When set to `true` both nodes will be considered \"ancestors\" too.\n\t * Which means that if e.g. node A is inside B, then their common ancestor will be B.\n\t */\n\tpublic getCommonAncestor( node: Node, options: { includeSelf?: boolean } = {} ): Element | DocumentFragment | null {\n\t\tconst ancestorsA = this.getAncestors( options );\n\t\tconst ancestorsB = node.getAncestors( options );\n\n\t\tlet i = 0;\n\n\t\twhile ( ancestorsA[ i ] == ancestorsB[ i ] && ancestorsA[ i ] ) {\n\t\t\ti++;\n\t\t}\n\n\t\treturn i === 0 ? null : ancestorsA[ i - 1 ] as ( Element | DocumentFragment );\n\t}\n\n\t/**\n\t * Returns whether this node is before given node. `false` is returned if nodes are in different trees (for example,\n\t * in different {@link module:engine/view/documentfragment~DocumentFragment}s).\n\t *\n\t * @param node Node to compare with.\n\t */\n\tpublic isBefore( node: Node ): boolean {\n\t\t// Given node is not before this node if they are same.\n\t\tif ( this == node ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Return `false` if it is impossible to compare nodes.\n\t\tif ( this.root !== node.root ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst thisPath = this.getPath();\n\t\tconst nodePath = node.getPath();\n\n\t\tconst result = compareArrays( thisPath, nodePath );\n\n\t\tswitch ( result ) {\n\t\t\tcase 'prefix':\n\t\t\t\treturn true;\n\n\t\t\tcase 'extension':\n\t\t\t\treturn false;\n\n\t\t\tdefault:\n\t\t\t\treturn thisPath[ result as number ] < nodePath[ result as number ];\n\t\t}\n\t}\n\n\t/**\n\t * Returns whether this node is after given node. `false` is returned if nodes are in different trees (for example,\n\t * in different {@link module:engine/view/documentfragment~DocumentFragment}s).\n\t *\n\t * @param node Node to compare with.\n\t */\n\tpublic isAfter( node: Node ): boolean {\n\t\t// Given node is not before this node if they are same.\n\t\tif ( this == node ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Return `false` if it is impossible to compare nodes.\n\t\tif ( this.root !== node.root ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// In other cases, just check if the `node` is before, and return the opposite.\n\t\treturn !this.isBefore( node );\n\t}\n\n\t/**\n\t * Removes node from parent.\n\t *\n\t * @internal\n\t */\n\tpublic _remove(): void {\n\t\tthis.parent!._removeChildren( this.index! );\n\t}\n\n\t/**\n\t * @internal\n\t * @param type Type of the change.\n\t * @param node Changed node.\n\t * @fires change\n\t */\n\tpublic _fireChange( type: ChangeType, node: Node ): void {\n\t\tthis.fire<ViewNodeChangeEvent>( `change:${ type }`, node );\n\n\t\tif ( this.parent ) {\n\t\t\tthis.parent._fireChange( type, node );\n\t\t}\n\t}\n\n\t/**\n\t * Custom toJSON method to solve child-parent circular dependencies.\n\t *\n\t * @returns Clone of this object with the parent property removed.\n\t */\n\tpublic toJSON(): unknown {\n\t\tconst json: any = clone( this );\n\n\t\t// Due to circular references we need to remove parent reference.\n\t\tdelete json.parent;\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * Clones this node.\n\t *\n\t * @internal\n\t * @returns Clone of this node.\n\t */\n\tpublic abstract _clone( deep?: boolean ): Node;\n\n\t/**\n\t * Checks if provided node is similar to this node.\n\t *\n\t * @returns True if nodes are similar.\n\t */\n\tpublic abstract isSimilar( other: Node ): boolean;\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nNode.prototype.is = function( type: string ): boolean {\n\treturn type === 'node' || type === 'view:node';\n};\n\n/**\n * Fired when list of {@link module:engine/view/element~Element elements} children, attributes or data changes.\n *\n * Change event is bubbled it is fired on all ancestors.\n *\n * @eventName ~Node#change\n * @eventName ~Node#change:children\n * @eventName ~Node#change:attributes\n * @eventName ~Node#change:text\n */\nexport type ViewNodeChangeEvent = {\n\tname: 'change' | `change:${ ChangeType }`;\n\targs: [ changedNode: Node ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/text\n */\n\nimport Node from './node';\n\nimport type Document from './document';\n\n/**\n * Tree view text node.\n *\n * The constructor of this class should not be used directly. To create a new text node instance\n * use the {@link module:engine/view/downcastwriter~DowncastWriter#createText `DowncastWriter#createText()`}\n * method when working on data downcasted from the model or the\n * {@link module:engine/view/upcastwriter~UpcastWriter#createText `UpcastWriter#createText()`}\n * method when working on non-semantic views.\n */\nexport default class Text extends Node {\n\t/**\n\t * The text content.\n\t *\n\t * Setting the data fires the {@link module:engine/view/node~Node#event:change:text change event}.\n\t */\n\tprivate _textData: string;\n\n\t/**\n\t * Creates a tree view text node.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#createText\n\t * @internal\n\t * @param document The document instance to which this text node belongs.\n\t * @param data The text's data.\n\t */\n\tconstructor( document: Document, data: string ) {\n\t\tsuper( document );\n\n\t\tthis._textData = data;\n\t}\n\n\t/**\n\t * The text content.\n\t */\n\tpublic get data(): string {\n\t\treturn this._textData;\n\t}\n\n\t/**\n\t * The `_data` property is controlled by a getter and a setter.\n\t *\n\t * The getter is required when using the addition assignment operator on protected property:\n\t *\n\t * ```ts\n\t * const foo = downcastWriter.createText( 'foo' );\n\t * const bar = downcastWriter.createText( 'bar' );\n\t *\n\t * foo._data += bar.data; // executes: `foo._data = foo._data + bar.data`\n\t * console.log( foo.data ); // prints: 'foobar'\n\t * ```\n\t *\n\t * If the protected getter didn't exist, `foo._data` will return `undefined` and result of the merge will be invalid.\n\t *\n\t * The setter sets data and fires the {@link module:engine/view/node~Node#event:change:text change event}.\n\t *\n\t * @internal\n\t */\n\tpublic get _data(): string {\n\t\treturn this.data;\n\t}\n\n\tpublic set _data( data: string ) {\n\t\tthis._fireChange( 'text', this );\n\n\t\tthis._textData = data;\n\t}\n\n\t/**\n\t * Checks if this text node is similar to other text node.\n\t * Both nodes should have the same data to be considered as similar.\n\t *\n\t * @param otherNode Node to check if it is same as this node.\n\t */\n\tpublic isSimilar( otherNode: Node ): boolean {\n\t\tif ( !( otherNode instanceof Text ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn this === otherNode || this.data === otherNode.data;\n\t}\n\n\t/**\n\t * Clones this node.\n\t *\n\t * @internal\n\t * @returns Text node that is a clone of this node.\n\t */\n\tpublic _clone(): Text {\n\t\treturn new Text( this.document, this.data );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `#${ this.data }`;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ViewText: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logExtended(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ViewText: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nText.prototype.is = function( type: string ): boolean {\n\treturn type === '$text' || type === 'view:$text' ||\n\t\t// This are legacy values kept for backward compatibility.\n\t\ttype === 'text' || type === 'view:text' ||\n\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\ttype === 'node' || type === 'view:node';\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/textproxy\n */\n\nimport TypeCheckable from './typecheckable';\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type Document from './document';\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\nimport type Node from './node';\nimport type Text from './text';\n\n/**\n * TextProxy is a wrapper for substring of {@link module:engine/view/text~Text}. Instance of this class is created by\n * {@link module:engine/view/treewalker~TreeWalker} when only a part of {@link module:engine/view/text~Text} needs to be returned.\n *\n * `TextProxy` has an API similar to {@link module:engine/view/text~Text Text} and allows to do most of the common tasks performed\n * on view nodes.\n *\n * **Note:** Some `TextProxy` instances may represent whole text node, not just a part of it.\n * See {@link module:engine/view/textproxy~TextProxy#isPartial}.\n *\n * **Note:** `TextProxy` is a readonly interface.\n *\n * **Note:** `TextProxy` instances are created on the fly basing on the current state of parent {@link module:engine/view/text~Text}.\n * Because of this it is highly unrecommended to store references to `TextProxy instances because they might get\n * invalidated due to operations on Document. Also TextProxy is not a {@link module:engine/view/node~Node} so it can not be\n * inserted as a child of {@link module:engine/view/element~Element}.\n *\n * `TextProxy` instances are created by {@link module:engine/view/treewalker~TreeWalker view tree walker}. You should not need to create\n * an instance of this class by your own.\n */\nexport default class TextProxy extends TypeCheckable {\n\t/**\n\t * Reference to the {@link module:engine/view/text~Text} element which TextProxy is a substring.\n\t */\n\tpublic readonly textNode: Text;\n\n\t/**\n\t * Text data represented by this text proxy.\n\t */\n\tpublic readonly data: string;\n\n\t/**\n\t * Offset in the `textNode` where this `TextProxy` instance starts.\n\t */\n\tpublic readonly offsetInText: number;\n\n\t/**\n\t * Creates a text proxy.\n\t *\n\t * @internal\n\t * @param textNode Text node which part is represented by this text proxy.\n\t * @param offsetInText Offset in {@link module:engine/view/textproxy~TextProxy#textNode text node}\n\t * from which the text proxy starts.\n\t * @param length Text proxy length, that is how many text node's characters, starting from `offsetInText` it represents.\n\t * @constructor\n\t */\n\tconstructor( textNode: Text, offsetInText: number, length: number ) {\n\t\tsuper();\n\n\t\tthis.textNode = textNode;\n\n\t\tif ( offsetInText < 0 || offsetInText > textNode.data.length ) {\n\t\t\t/**\n\t\t\t * Given offsetInText value is incorrect.\n\t\t\t *\n\t\t\t * @error view-textproxy-wrong-offsetintext\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-textproxy-wrong-offsetintext', this );\n\t\t}\n\n\t\tif ( length < 0 || offsetInText + length > textNode.data.length ) {\n\t\t\t/**\n\t\t\t * Given length value is incorrect.\n\t\t\t *\n\t\t\t * @error view-textproxy-wrong-length\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-textproxy-wrong-length', this );\n\t\t}\n\n\t\tthis.data = textNode.data.substring( offsetInText, offsetInText + length );\n\t\tthis.offsetInText = offsetInText;\n\t}\n\n\t/**\n\t * Offset size of this node.\n\t */\n\tpublic get offsetSize(): number {\n\t\treturn this.data.length;\n\t}\n\n\t/**\n\t * Flag indicating whether `TextProxy` instance covers only part of the original {@link module:engine/view/text~Text text node}\n\t * (`true`) or the whole text node (`false`).\n\t *\n\t * This is `false` when text proxy starts at the very beginning of {@link module:engine/view/textproxy~TextProxy#textNode textNode}\n\t * ({@link module:engine/view/textproxy~TextProxy#offsetInText offsetInText} equals `0`) and text proxy sizes is equal to\n\t * text node size.\n\t */\n\tpublic get isPartial(): boolean {\n\t\treturn this.data.length !== this.textNode.data.length;\n\t}\n\n\t/**\n\t * Parent of this text proxy, which is same as parent of text node represented by this text proxy.\n\t */\n\tpublic get parent(): Element | DocumentFragment | null {\n\t\treturn this.textNode.parent;\n\t}\n\n\t/**\n\t * Root of this text proxy, which is same as root of text node represented by this text proxy.\n\t */\n\tpublic get root(): Node | DocumentFragment {\n\t\treturn this.textNode.root;\n\t}\n\n\t/**\n\t * {@link module:engine/view/document~Document View document} that owns this text proxy, or `null` if the text proxy is inside\n\t * {@link module:engine/view/documentfragment~DocumentFragment document fragment}.\n\t */\n\tpublic get document(): Document | null {\n\t\treturn this.textNode.document;\n\t}\n\n\t/**\n\t * Returns ancestors array of this text proxy.\n\t *\n\t * @param options Options object.\n\t * @param options.includeSelf When set to `true`, textNode will be also included in parent's array.\n\t * @param options.parentFirst When set to `true`, array will be sorted from text proxy parent to\n\t * root element, otherwise root element will be the first item in the array.\n\t * @returns Array with ancestors.\n\t */\n\tpublic getAncestors( options: {\n\t\tincludeSelf?: boolean;\n\t\tparentFirst?: boolean;\n\t} = {} ): Array<Text | Element | DocumentFragment> {\n\t\tconst ancestors: Array<Text | Element | DocumentFragment> = [];\n\t\tlet parent: Text | Element | DocumentFragment | null = options.includeSelf ? this.textNode : this.parent;\n\n\t\twhile ( parent !== null ) {\n\t\t\tancestors[ options.parentFirst ? 'push' : 'unshift' ]( parent );\n\t\t\tparent = parent.parent;\n\t\t}\n\n\t\treturn ancestors;\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `#${ this.data }`;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ViewTextProxy: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logExtended(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ViewTextProxy: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nTextProxy.prototype.is = function( type: string ): boolean {\n\treturn type === '$textProxy' || type === 'view:$textProxy' ||\n\t\t// This are legacy values kept for backward compatibility.\n\t\ttype === 'textProxy' || type === 'view:textProxy';\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/matcher\n */\n\nimport type Element from './element';\n\nimport { isPlainObject } from 'lodash-es';\n\nimport { logWarning } from '@ckeditor/ckeditor5-utils';\n\n/**\n * View matcher class.\n * Instance of this class can be used to find {@link module:engine/view/element~Element elements} that match given pattern.\n */\nexport default class Matcher {\n\tprivate readonly _patterns: Array<MatcherFunctionPattern | MatcherObjectPattern> = [];\n\n\t/**\n\t * Creates new instance of Matcher.\n\t *\n\t * @param pattern Match patterns. See {@link module:engine/view/matcher~Matcher#add add method} for more information.\n\t */\n\tconstructor( ...pattern: Array<MatcherPattern> ) {\n\t\tthis.add( ...pattern );\n\t}\n\n\t/**\n\t * Adds pattern or patterns to matcher instance.\n\t *\n\t * ```ts\n\t * // String.\n\t * matcher.add( 'div' );\n\t *\n\t * // Regular expression.\n\t * matcher.add( /^\\w/ );\n\t *\n\t * // Single class.\n\t * matcher.add( {\n\t * \tclasses: 'foobar'\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/view/matcher~MatcherPattern} for more examples.\n\t *\n\t * Multiple patterns can be added in one call:\n\t *\n\t * ```ts\n\t * matcher.add( 'div', { classes: 'foobar' } );\n\t * ```\n\t *\n\t * @param pattern Object describing pattern details. If string or regular expression\n\t * is provided it will be used to match element's name. Pattern can be also provided in a form\n\t * of a function - then this function will be called with each {@link module:engine/view/element~Element element} as a parameter.\n\t * Function's return value will be stored under `match` key of the object returned from\n\t * {@link module:engine/view/matcher~Matcher#match match} or {@link module:engine/view/matcher~Matcher#matchAll matchAll} methods.\n\t */\n\tpublic add( ...pattern: Array<MatcherPattern> ): void {\n\t\tfor ( let item of pattern ) {\n\t\t\t// String or RegExp pattern is used as element's name.\n\t\t\tif ( typeof item == 'string' || item instanceof RegExp ) {\n\t\t\t\titem = { name: item };\n\t\t\t}\n\n\t\t\tthis._patterns.push( item );\n\t\t}\n\t}\n\n\t/**\n\t * Matches elements for currently stored patterns. Returns match information about first found\n\t * {@link module:engine/view/element~Element element}, otherwise returns `null`.\n\t *\n\t * Example of returned object:\n\t *\n\t * ```ts\n\t * {\n\t * \telement: <instance of found element>,\n\t * \tpattern: <pattern used to match found element>,\n\t * \tmatch: {\n\t * \t\tname: true,\n\t * \t\tattributes: [ 'title', 'href' ],\n\t * \t\tclasses: [ 'foo' ],\n\t * \t\tstyles: [ 'color', 'position' ]\n\t * \t}\n\t * }\n\t * ```\n\t *\n\t * @see module:engine/view/matcher~Matcher#add\n\t * @see module:engine/view/matcher~Matcher#matchAll\n\t * @param element View element to match against stored patterns.\n\t */\n\tpublic match( ...element: Array<Element> ): MatchResult | null {\n\t\tfor ( const singleElement of element ) {\n\t\t\tfor ( const pattern of this._patterns ) {\n\t\t\t\tconst match = isElementMatching( singleElement, pattern );\n\n\t\t\t\tif ( match ) {\n\t\t\t\t\treturn {\n\t\t\t\t\t\telement: singleElement,\n\t\t\t\t\t\tpattern,\n\t\t\t\t\t\tmatch\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Matches elements for currently stored patterns. Returns array of match information with all found\n\t * {@link module:engine/view/element~Element elements}. If no element is found - returns `null`.\n\t *\n\t * @see module:engine/view/matcher~Matcher#add\n\t * @see module:engine/view/matcher~Matcher#match\n\t * @param element View element to match against stored patterns.\n\t * @returns Array with match information about found elements or `null`. For more information\n\t * see {@link module:engine/view/matcher~Matcher#match match method} description.\n\t */\n\tpublic matchAll( ...element: Array<Element> ): Array<MatchResult> | null {\n\t\tconst results: Array<MatchResult> = [];\n\n\t\tfor ( const singleElement of element ) {\n\t\t\tfor ( const pattern of this._patterns ) {\n\t\t\t\tconst match = isElementMatching( singleElement, pattern );\n\n\t\t\t\tif ( match ) {\n\t\t\t\t\tresults.push( {\n\t\t\t\t\t\telement: singleElement,\n\t\t\t\t\t\tpattern,\n\t\t\t\t\t\tmatch\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn results.length > 0 ? results : null;\n\t}\n\n\t/**\n\t * Returns the name of the element to match if there is exactly one pattern added to the matcher instance\n\t * and it matches element name defined by `string` (not `RegExp`). Otherwise, returns `null`.\n\t *\n\t * @returns Element name trying to match.\n\t */\n\tpublic getElementName(): string | null {\n\t\tif ( this._patterns.length !== 1 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst pattern = this._patterns[ 0 ];\n\t\tconst name = pattern.name;\n\n\t\treturn ( typeof pattern != 'function' && name && !( name instanceof RegExp ) ) ? name : null;\n\t}\n}\n\n/**\n * Returns match information if {@link module:engine/view/element~Element element} is matching provided pattern.\n * If element cannot be matched to provided pattern - returns `null`.\n *\n * @returns Returns object with match information or null if element is not matching.\n */\nfunction isElementMatching( element: Element, pattern: MatcherFunctionPattern | MatcherObjectPattern ): Match | null {\n\t// If pattern is provided as function - return result of that function;\n\tif ( typeof pattern == 'function' ) {\n\t\treturn pattern( element );\n\t}\n\n\tconst match: Match = {};\n\n\t// Check element's name.\n\tif ( pattern.name ) {\n\t\tmatch.name = matchName( pattern.name, element.name );\n\n\t\tif ( !match.name ) {\n\t\t\treturn null;\n\t\t}\n\t}\n\n\t// Check element's attributes.\n\tif ( pattern.attributes ) {\n\t\tmatch.attributes = matchAttributes( pattern.attributes, element );\n\n\t\tif ( !match.attributes ) {\n\t\t\treturn null;\n\t\t}\n\t}\n\n\t// Check element's classes.\n\tif ( pattern.classes ) {\n\t\tmatch.classes = matchClasses( pattern.classes, element );\n\n\t\tif ( !match.classes ) {\n\t\t\treturn null;\n\t\t}\n\t}\n\n\t// Check element's styles.\n\tif ( pattern.styles ) {\n\t\tmatch.styles = matchStyles( pattern.styles, element );\n\n\t\tif ( !match.styles ) {\n\t\t\treturn null;\n\t\t}\n\t}\n\n\treturn match;\n}\n\n/**\n * Checks if name can be matched by provided pattern.\n *\n * @returns Returns `true` if name can be matched, `false` otherwise.\n */\nfunction matchName( pattern: string | RegExp, name: string ): boolean {\n\t// If pattern is provided as RegExp - test against this regexp.\n\tif ( pattern instanceof RegExp ) {\n\t\treturn !!name.match( pattern );\n\t}\n\n\treturn pattern === name;\n}\n\n/**\n * Checks if an array of key/value pairs can be matched against provided patterns.\n *\n * Patterns can be provided in a following ways:\n * - a boolean value matches any attribute with any value (or no value):\n *\n * ```ts\n * pattern: true\n * ```\n *\n * - a RegExp expression or object matches any attribute name:\n *\n * ```ts\n * pattern: /h[1-6]/\n * ```\n *\n * - an object matches any attribute that has the same name as the object item's key, where object item's value is:\n * \t- equal to `true`, which matches any attribute value:\n *\n * ```ts\n * pattern: {\n * \trequired: true\n * }\n * ```\n *\n * \t- a string that is equal to attribute value:\n *\n * ```ts\n * pattern: {\n * \trel: 'nofollow'\n * }\n * ```\n *\n * \t- a regular expression that matches attribute value,\n *\n * ```ts\n * pattern: {\n * \tsrc: /^https/\n * }\n * ```\n *\n * - an array with items, where the item is:\n * \t- a string that is equal to attribute value:\n *\n * ```ts\n * pattern: [ 'data-property-1', 'data-property-2' ],\n * ```\n *\n * \t- an object with `key` and `value` property, where `key` is a regular expression matching attribute name and\n * \t\t`value` is either regular expression matching attribute value or a string equal to attribute value:\n *\n * ```ts\n * pattern: [\n * \t{ key: /^data-property-/, value: true },\n * \t// or:\n * \t{ key: /^data-property-/, value: 'foobar' },\n * \t// or:\n * \t{ key: /^data-property-/, value: /^foo/ }\n * ]\n * ```\n *\n * @param patterns Object with information about attributes to match.\n * @param keys Attribute, style or class keys.\n * @param valueGetter A function providing value for a given item key.\n * @returns Returns array with matched attribute names or `null` if no attributes were matched.\n */\nfunction matchPatterns(\n\tpatterns: PropertyPatterns,\n\tkeys: Iterable<string>,\n\tvalueGetter: ( value: string ) => unknown\n): Array<string> | undefined {\n\tconst normalizedPatterns = normalizePatterns( patterns );\n\tconst normalizedItems = Array.from( keys );\n\tconst match: Array<string> = [];\n\n\tnormalizedPatterns.forEach( ( [ patternKey, patternValue ] ) => {\n\t\tnormalizedItems.forEach( itemKey => {\n\t\t\tif (\n\t\t\t\tisKeyMatched( patternKey, itemKey ) &&\n\t\t\t\tisValueMatched( patternValue, itemKey, valueGetter )\n\t\t\t) {\n\t\t\t\tmatch.push( itemKey );\n\t\t\t}\n\t\t} );\n\t} );\n\n\t// Return matches only if there are at least as many of them as there are patterns.\n\t// The RegExp pattern can match more than one item.\n\tif ( !normalizedPatterns.length || match.length < normalizedPatterns.length ) {\n\t\treturn undefined;\n\t}\n\n\treturn match;\n}\n\n/**\n * Bring all the possible pattern forms to an array of arrays where first item is a key and second is a value.\n *\n * Examples:\n *\n * Boolean pattern value:\n *\n * ```ts\n * true\n * ```\n *\n * to\n *\n * ```ts\n * [ [ true, true ] ]\n * ```\n *\n * Textual pattern value:\n *\n * ```ts\n * 'attribute-name-or-class-or-style'\n * ```\n *\n * to\n *\n * ```ts\n * [ [ 'attribute-name-or-class-or-style', true ] ]\n * ```\n *\n * Regular expression:\n *\n * ```ts\n * /^data-.*$/\n * ```\n *\n * to\n *\n * ```ts\n * [ [ /^data-.*$/, true ] ]\n * ```\n *\n * Objects (plain or with `key` and `value` specified explicitly):\n *\n * ```ts\n * {\n * \tsrc: /^https:.*$/\n * }\n * ```\n *\n * or\n *\n * ```ts\n * [ {\n * \tkey: 'src',\n * \tvalue: /^https:.*$/\n * } ]\n * ```\n *\n * to:\n *\n * ```ts\n * [ [ 'src', /^https:.*$/ ] ]\n * ```\n *\n * @returns Returns an array of objects or null if provided patterns were not in an expected form.\n */\nfunction normalizePatterns( patterns: PropertyPatterns ): Array<[ true | string | RegExp, true | string | RegExp ]> {\n\tif ( Array.isArray( patterns ) ) {\n\t\treturn patterns.map( ( pattern: any ) => {\n\t\t\tif ( isPlainObject( pattern ) ) {\n\t\t\t\tif ( pattern.key === undefined || pattern.value === undefined ) {\n\t\t\t\t\t// Documented at the end of matcher.js.\n\t\t\t\t\tlogWarning( 'matcher-pattern-missing-key-or-value', pattern );\n\t\t\t\t}\n\n\t\t\t\treturn [ pattern.key, pattern.value ];\n\t\t\t}\n\n\t\t\t// Assume the pattern is either String or RegExp.\n\t\t\treturn [ pattern, true ];\n\t\t} );\n\t}\n\n\tif ( isPlainObject( patterns ) ) {\n\t\treturn Object.entries( patterns );\n\t}\n\n\t// Other cases (true, string or regexp).\n\treturn [ [ patterns as any, true ] ];\n}\n\n/**\n * @param patternKey A pattern representing a key we want to match.\n * @param itemKey An actual item key (e.g. `'src'`, `'background-color'`, `'ck-widget'`) we're testing against pattern.\n */\nfunction isKeyMatched( patternKey: true | string | RegExp, itemKey: string ): unknown {\n\treturn patternKey === true ||\n\t\tpatternKey === itemKey ||\n\t\tpatternKey instanceof RegExp && itemKey.match( patternKey );\n}\n\n/**\n * @param patternValue A pattern representing a value we want to match.\n * @param itemKey An item key, e.g. `background`, `href`, 'rel', etc.\n * @param valueGetter A function used to provide a value for a given `itemKey`.\n */\nfunction isValueMatched(\n\tpatternValue: true | string | RegExp,\n\titemKey: string,\n\tvalueGetter: ( itemKey: string ) => unknown\n): boolean {\n\tif ( patternValue === true ) {\n\t\treturn true;\n\t}\n\n\tconst itemValue = valueGetter( itemKey );\n\n\t// For now, the reducers are not returning the full tree of properties.\n\t// Casting to string preserves the old behavior until the root cause is fixed.\n\t// More can be found in https://github.com/ckeditor/ckeditor5/issues/10399.\n\treturn patternValue === itemValue ||\n\t\tpatternValue instanceof RegExp && !!String( itemValue ).match( patternValue );\n}\n\n/**\n * Checks if attributes of provided element can be matched against provided patterns.\n *\n * @param patterns Object with information about attributes to match. Each key of the object will be\n * used as attribute name. Value of each key can be a string or regular expression to match against attribute value.\n * @param element Element which attributes will be tested.\n * @returns Returns array with matched attribute names or `null` if no attributes were matched.\n */\nfunction matchAttributes(\n\tpatterns: AttributePatterns,\n\telement: Element\n): Array<string> | undefined {\n\tconst attributeKeys = new Set( element.getAttributeKeys() );\n\n\t// `style` and `class` attribute keys are deprecated. Only allow them in object pattern\n\t// for backward compatibility.\n\tif ( isPlainObject( patterns ) ) {\n\t\tif ( ( patterns as any ).style !== undefined ) {\n\t\t\t// Documented at the end of matcher.js.\n\t\t\tlogWarning( 'matcher-pattern-deprecated-attributes-style-key', patterns as any );\n\t\t}\n\t\tif ( ( patterns as any ).class !== undefined ) {\n\t\t\t// Documented at the end of matcher.js.\n\t\t\tlogWarning( 'matcher-pattern-deprecated-attributes-class-key', patterns as any );\n\t\t}\n\t} else {\n\t\tattributeKeys.delete( 'style' );\n\t\tattributeKeys.delete( 'class' );\n\t}\n\n\treturn matchPatterns( patterns, attributeKeys, key => element.getAttribute( key ) );\n}\n\n/**\n * Checks if classes of provided element can be matched against provided patterns.\n *\n * @param patterns Array of strings or regular expressions to match against element's classes.\n * @param element Element which classes will be tested.\n * @returns Returns array with matched class names or `null` if no classes were matched.\n */\nfunction matchClasses( patterns: ClassPatterns, element: Element ): Array<string> | undefined {\n\t// We don't need `getter` here because patterns for classes are always normalized to `[ className, true ]`.\n\treturn matchPatterns( patterns, element.getClassNames(), /* istanbul ignore next -- @preserve */ () => {} );\n}\n\n/**\n * Checks if styles of provided element can be matched against provided patterns.\n *\n * @param patterns Object with information about styles to match. Each key of the object will be\n * used as style name. Value of each key can be a string or regular expression to match against style value.\n * @param element Element which styles will be tested.\n * @returns Returns array with matched style names or `null` if no styles were matched.\n */\nfunction matchStyles( patterns: StylePatterns, element: Element ): Array<string> | undefined {\n\treturn matchPatterns( patterns, element.getStyleNames( true ), key => element.getStyle( key ) );\n}\n\n/**\n * An entity that is a valid pattern recognized by a matcher. `MatcherPattern` is used by {@link ~Matcher} to recognize\n * if a view element fits in a group of view elements described by the pattern.\n *\n * `MatcherPattern` can be given as a `String`, a `RegExp`, an `Object` or a `Function`.\n *\n * If `MatcherPattern` is given as a `String` or `RegExp`, it will match any view element that has a matching name:\n *\n * ```ts\n * // Match any element with name equal to 'div'.\n * const pattern = 'div';\n *\n * // Match any element which name starts on 'p'.\n * const pattern = /^p/;\n * ```\n *\n * If `MatcherPattern` is given as an `Object`, all the object's properties will be matched with view element properties.\n * If the view element does not meet all of the object's pattern properties, the match will not happen.\n * Available `Object` matching properties:\n *\n * Matching view element:\n *\n * ```ts\n * // Match view element's name using String:\n * const pattern = { name: 'p' };\n *\n * // or by providing RegExp:\n * const pattern = { name: /^(ul|ol)$/ };\n *\n * // The name can also be skipped to match any view element with matching attributes:\n * const pattern = {\n * \tattributes: {\n * \t\t'title': true\n * \t}\n * };\n * ```\n *\n * Matching view element attributes:\n *\n * ```ts\n * // Match view element with any attribute value.\n * const pattern = {\n * \tname: 'p',\n * \tattributes: true\n * };\n *\n * // Match view element which has matching attributes (String).\n * const pattern = {\n * \tname: 'figure',\n * \tattributes: 'title' // Match title attribute (can be empty).\n * };\n *\n * // Match view element which has matching attributes (RegExp).\n * const pattern = {\n * \tname: 'figure',\n * \tattributes: /^data-.*$/ // Match attributes starting with `data-` e.g. `data-foo` with any value (can be empty).\n * };\n *\n * // Match view element which has matching attributes (Object).\n * const pattern = {\n * \tname: 'figure',\n * \tattributes: {\n * \t\ttitle: 'foobar', // Match `title` attribute with 'foobar' value.\n * \t\talt: true, // Match `alt` attribute with any value (can be empty).\n * \t\t'data-type': /^(jpg|png)$/ // Match `data-type` attribute with `jpg` or `png` value.\n * \t}\n * };\n *\n * // Match view element which has matching attributes (Array).\n * const pattern = {\n * \tname: 'figure',\n * \tattributes: [\n * \t\t'title', // Match `title` attribute (can be empty).\n * \t\t/^data-*$/ // Match attributes starting with `data-` e.g. `data-foo` with any value (can be empty).\n * \t]\n * };\n *\n * // Match view element which has matching attributes (key-value pairs).\n * const pattern = {\n * \tname: 'input',\n * \tattributes: [\n * \t\t{\n * \t\t\tkey: 'type', // Match `type` as an attribute key.\n * \t\t\tvalue: /^(text|number|date)$/\t // Match `text`, `number` or `date` values.\n * \t\t},\n * \t\t{\n * \t\t\tkey: /^data-.*$/, // Match attributes starting with `data-` e.g. `data-foo`.\n * \t\t\tvalue: true // Match any value (can be empty).\n * \t\t}\n * \t]\n * };\n * ```\n *\n * Matching view element styles:\n *\n * ```ts\n * // Match view element with any style.\n * const pattern = {\n * \tname: 'p',\n * \tstyles: true\n * };\n *\n * // Match view element which has matching styles (String).\n * const pattern = {\n * \tname: 'p',\n * \tstyles: 'color' // Match attributes with `color` style.\n * };\n *\n * // Match view element which has matching styles (RegExp).\n * const pattern = {\n * \tname: 'p',\n * \tstyles: /^border.*$/ // Match view element with any border style.\n * };\n *\n * // Match view element which has matching styles (Object).\n * const pattern = {\n * \tname: 'p',\n * \tstyles: {\n * \t\tcolor: /rgb\\((\\d{1,3}), (\\d{1,3}), (\\d{1,3})\\)/, // Match `color` in RGB format only.\n * \t\t'font-weight': 600, // Match `font-weight` only if it's `600`.\n * \t\t'text-decoration': true // Match any text decoration.\n * \t}\n * };\n *\n * // Match view element which has matching styles (Array).\n * const pattern = {\n * \tname: 'p',\n * \tstyles: [\n * \t\t'color', // Match `color` with any value.\n * \t\t/^border.*$/ // Match all border properties.\n * \t]\n * };\n *\n * // Match view element which has matching styles (key-value pairs).\n * const pattern = {\n * \tname: 'p',\n * \tstyles: [\n * \t\t{\n * \t\t\tkey: 'color', \t\t// Match `color` as an property key.\n * \t\t\tvalue: /rgb\\((\\d{1,3}), (\\d{1,3}), (\\d{1,3})\\)/\t\t// Match RGB format only.\n * \t\t},\n * \t\t{\n * \t\t\tkey: /^border.*$/, // Match any border style.\n * \t\t\tvalue: true // Match any value.\n * \t\t}\n * \t]\n * };\n * ```\n *\n * Matching view element classes:\n *\n * ```ts\n * // Match view element with any class.\n * const pattern = {\n * \tname: 'p',\n * \tclasses: true\n * };\n *\n * // Match view element which has matching class (String).\n * const pattern = {\n * \tname: 'p',\n * \tclasses: 'highlighted' // Match `highlighted` class.\n * };\n *\n * // Match view element which has matching classes (RegExp).\n * const pattern = {\n * \tname: 'figure',\n * \tclasses: /^image-side-(left|right)$/ // Match `image-side-left` or `image-side-right` class.\n * };\n *\n * // Match view element which has matching classes (Object).\n * const pattern = {\n * \tname: 'p',\n * \tclasses: {\n * \t\thighlighted: true, // Match `highlighted` class.\n * \t\tmarker: true // Match `marker` class.\n * \t}\n * };\n *\n * // Match view element which has matching classes (Array).\n * const pattern = {\n * \tname: 'figure',\n * \tclasses: [\n * \t\t'image', // Match `image` class.\n * \t\t/^image-side-(left|right)$/ // Match `image-side-left` or `image-side-right` class.\n * \t]\n * };\n *\n * // Match view element which has matching classes (key-value pairs).\n * const pattern = {\n * \tname: 'figure',\n * \tclasses: [\n * \t\t{\n * \t\t\tkey: 'image', // Match `image` class.\n * \t\t\tvalue: true\n * \t\t},\n * \t\t{\n * \t\t\tkey: /^image-side-(left|right)$/, // Match `image-side-left` or `image-side-right` class.\n * \t\t\tvalue: true\n * \t\t}\n * \t]\n * };\n * ```\n *\n * Pattern can combine multiple properties allowing for more complex view element matching:\n *\n * ```ts\n * const pattern = {\n * \tname: 'span',\n * \tattributes: [ 'title' ],\n * \tstyles: {\n * \t\t'font-weight': 'bold'\n * \t},\n * \tclasses: 'highlighted'\n * };\n * ```\n *\n * If `MatcherPattern` is given as a `Function`, the function takes a view element as a first and only parameter and\n * the function should decide whether that element matches. If so, it should return what part of the view element has been matched.\n * Otherwise, the function should return `null`. The returned result will be included in `match` property of the object\n * returned by {@link ~Matcher#match} call.\n *\n * ```ts\n * // Match an empty <div> element.\n * const pattern = element => {\n * \tif ( element.name == 'div' && element.childCount > 0 ) {\n * \t\t// Return which part of the element was matched.\n * \t\treturn { name: true };\n * \t}\n *\n * \treturn null;\n * };\n *\n * // Match a <p> element with big font (\"heading-like\" element).\n * const pattern = element => {\n * \tif ( element.name == 'p' ) {\n * \t\tconst fontSize = element.getStyle( 'font-size' );\n * \t\tconst size = fontSize.match( /(\\d+)/px );\n *\n * \t\tif ( size && Number( size[ 1 ] ) > 26 ) {\n * \t\t\treturn { name: true, attributes: [ 'font-size' ] };\n * \t\t}\n * \t}\n *\n * \treturn null;\n * };\n * ```\n *\n * `MatcherPattern` is defined in a way that it is a superset of {@link module:engine/view/elementdefinition~ElementDefinition},\n * that is, every `ElementDefinition` also can be used as a `MatcherPattern`.\n */\nexport type MatcherPattern = string | RegExp | MatcherFunctionPattern | MatcherObjectPattern;\n\n/**\n * A function describing `MatcherPattern`. See {@link ~MatcherPattern} for examples and other options.\n */\nexport type MatcherFunctionPattern = ( element: Element ) => Match | null;\n\n/**\n * An object describing `MatcherPattern`. See {@link ~MatcherPattern} for examples and other options.\n */\nexport interface MatcherObjectPattern {\n\n\t/**\n\t * View element name to match.\n\t */\n\tname?: string | RegExp;\n\n\tkey?: string;\n\n\t/**\n\t * View element's classes to match.\n\t */\n\tclasses?: ClassPatterns;\n\n\t/**\n\t * View element's styles to match.\n\t */\n\tstyles?: StylePatterns;\n\n\t/**\n\t * View element's attributes to match.\n\t */\n\tattributes?: AttributePatterns;\n}\n\n/**\n * An object representing matched element parts.\n */\nexport interface Match {\n\n\t/**\n\t * True if name of the element was matched.\n\t */\n\tname?: boolean;\n\n\t/**\n\t * Array with matched attribute names.\n\t */\n\tattributes?: Array<string>;\n\n\t/**\n\t * Array with matched class names.\n\t */\n\tclasses?: Array<string>;\n\n\t/**\n\t * Array with matched style names.\n\t */\n\tstyles?: Array<string>;\n}\n\n/**\n * The result of {@link ~Matcher#match}.\n */\nexport interface MatchResult {\n\n\t/**\n\t * Matched view element.\n\t */\n\telement: Element;\n\n\t/**\n\t * Pattern that was used to find matched element.\n\t */\n\tpattern: MatcherFunctionPattern | MatcherObjectPattern;\n\n\t/**\n\t * An object representing matched element parts.\n\t */\n\tmatch: Match;\n}\n\nexport type PropertyPatterns<ValuePattern = string | RegExp> =\n\ttrue |\n\tstring |\n\tRegExp |\n\tRecord<string, true | ValuePattern> |\n\tArray<string | RegExp | { key: string | RegExp; value: true | ValuePattern }>;\n\nexport type AttributePatterns = PropertyPatterns;\nexport type StylePatterns = PropertyPatterns;\nexport type ClassPatterns = PropertyPatterns<never>;\n\n/**\n * The key-value matcher pattern is missing key or value. Both must be present.\n * Refer the documentation: {@link module:engine/view/matcher~MatcherPattern}.\n *\n * @param pattern Pattern with missing properties.\n * @error matcher-pattern-missing-key-or-value\n */\n\n/**\n * The key-value matcher pattern for `attributes` option is using deprecated `style` key.\n *\n * Use `styles` matcher pattern option instead:\n *\n * ```ts\n * // Instead of:\n * const pattern = {\n * \tattributes: {\n * \t\tkey1: 'value1',\n * \t\tkey2: 'value2',\n * \t\tstyle: /^border.*$/\n * \t}\n * }\n *\n * // Use:\n * const pattern = {\n * \tattributes: {\n * \t\tkey1: 'value1',\n * \t\tkey2: 'value2'\n * \t},\n * \tstyles: /^border.*$/\n * }\n * ```\n *\n * Refer to the {@glink updating/guides/update-to-29##update-to-ckeditor-5-v2910 Migration to v29.1.0} guide\n * and {@link module:engine/view/matcher~MatcherPattern} documentation.\n *\n * @param pattern Pattern with missing properties.\n * @error matcher-pattern-deprecated-attributes-style-key\n */\n\n/**\n * The key-value matcher pattern for `attributes` option is using deprecated `class` key.\n *\n * Use `classes` matcher pattern option instead:\n *\n * ```ts\n * // Instead of:\n * const pattern = {\n * \tattributes: {\n * \t\tkey1: 'value1',\n * \t\tkey2: 'value2',\n * \t\tclass: 'foobar'\n * \t}\n * }\n *\n * // Use:\n * const pattern = {\n * \tattributes: {\n * \t\tkey1: 'value1',\n * \t\tkey2: 'value2'\n * \t},\n * \tclasses: 'foobar'\n * }\n * ```\n *\n * Refer to the {@glink updating/guides/update-to-29##update-to-ckeditor-5-v2910 Migration to v29.1.0} guide\n * and the {@link module:engine/view/matcher~MatcherPattern} documentation.\n *\n * @param pattern Pattern with missing properties.\n * @error matcher-pattern-deprecated-attributes-class-key\n */\n","import baseGetTag from './_baseGetTag.js';\nimport isObjectLike from './isObjectLike.js';\n\n/** `Object#toString` result references. */\nvar symbolTag = '[object Symbol]';\n\n/**\n * Checks if `value` is classified as a `Symbol` primitive or object.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is a symbol, else `false`.\n * @example\n *\n * _.isSymbol(Symbol.iterator);\n * // => true\n *\n * _.isSymbol('abc');\n * // => false\n */\nfunction isSymbol(value) {\n return typeof value == 'symbol' ||\n (isObjectLike(value) && baseGetTag(value) == symbolTag);\n}\n\nexport default isSymbol;\n","import isArray from './isArray.js';\nimport isSymbol from './isSymbol.js';\n\n/** Used to match property names within property paths. */\nvar reIsDeepProp = /\\.|\\[(?:[^[\\]]*|([\"'])(?:(?!\\1)[^\\\\]|\\\\.)*?\\1)\\]/,\n reIsPlainProp = /^\\w*$/;\n\n/**\n * Checks if `value` is a property name and not a property path.\n *\n * @private\n * @param {*} value The value to check.\n * @param {Object} [object] The object to query keys on.\n * @returns {boolean} Returns `true` if `value` is a property name, else `false`.\n */\nfunction isKey(value, object) {\n if (isArray(value)) {\n return false;\n }\n var type = typeof value;\n if (type == 'number' || type == 'symbol' || type == 'boolean' ||\n value == null || isSymbol(value)) {\n return true;\n }\n return reIsPlainProp.test(value) || !reIsDeepProp.test(value) ||\n (object != null && value in Object(object));\n}\n\nexport default isKey;\n","import MapCache from './_MapCache.js';\n\n/** Error message constants. */\nvar FUNC_ERROR_TEXT = 'Expected a function';\n\n/**\n * Creates a function that memoizes the result of `func`. If `resolver` is\n * provided, it determines the cache key for storing the result based on the\n * arguments provided to the memoized function. By default, the first argument\n * provided to the memoized function is used as the map cache key. The `func`\n * is invoked with the `this` binding of the memoized function.\n *\n * **Note:** The cache is exposed as the `cache` property on the memoized\n * function. Its creation may be customized by replacing the `_.memoize.Cache`\n * constructor with one whose instances implement the\n * [`Map`](http://ecma-international.org/ecma-262/7.0/#sec-properties-of-the-map-prototype-object)\n * method interface of `clear`, `delete`, `get`, `has`, and `set`.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Function\n * @param {Function} func The function to have its output memoized.\n * @param {Function} [resolver] The function to resolve the cache key.\n * @returns {Function} Returns the new memoized function.\n * @example\n *\n * var object = { 'a': 1, 'b': 2 };\n * var other = { 'c': 3, 'd': 4 };\n *\n * var values = _.memoize(_.values);\n * values(object);\n * // => [1, 2]\n *\n * values(other);\n * // => [3, 4]\n *\n * object.a = 2;\n * values(object);\n * // => [1, 2]\n *\n * // Modify the result cache.\n * values.cache.set(object, ['a', 'b']);\n * values(object);\n * // => ['a', 'b']\n *\n * // Replace `_.memoize.Cache`.\n * _.memoize.Cache = WeakMap;\n */\nfunction memoize(func, resolver) {\n if (typeof func != 'function' || (resolver != null && typeof resolver != 'function')) {\n throw new TypeError(FUNC_ERROR_TEXT);\n }\n var memoized = function() {\n var args = arguments,\n key = resolver ? resolver.apply(this, args) : args[0],\n cache = memoized.cache;\n\n if (cache.has(key)) {\n return cache.get(key);\n }\n var result = func.apply(this, args);\n memoized.cache = cache.set(key, result) || cache;\n return result;\n };\n memoized.cache = new (memoize.Cache || MapCache);\n return memoized;\n}\n\n// Expose `MapCache`.\nmemoize.Cache = MapCache;\n\nexport default memoize;\n","import memoize from './memoize.js';\n\n/** Used as the maximum memoize cache size. */\nvar MAX_MEMOIZE_SIZE = 500;\n\n/**\n * A specialized version of `_.memoize` which clears the memoized function's\n * cache when it exceeds `MAX_MEMOIZE_SIZE`.\n *\n * @private\n * @param {Function} func The function to have its output memoized.\n * @returns {Function} Returns the new memoized function.\n */\nfunction memoizeCapped(func) {\n var result = memoize(func, function(key) {\n if (cache.size === MAX_MEMOIZE_SIZE) {\n cache.clear();\n }\n return key;\n });\n\n var cache = result.cache;\n return result;\n}\n\nexport default memoizeCapped;\n","import memoizeCapped from './_memoizeCapped.js';\n\n/** Used to match property names within property paths. */\nvar rePropName = /[^.[\\]]+|\\[(?:(-?\\d+(?:\\.\\d+)?)|([\"'])((?:(?!\\2)[^\\\\]|\\\\.)*?)\\2)\\]|(?=(?:\\.|\\[\\])(?:\\.|\\[\\]|$))/g;\n\n/** Used to match backslashes in property paths. */\nvar reEscapeChar = /\\\\(\\\\)?/g;\n\n/**\n * Converts `string` to a property path array.\n *\n * @private\n * @param {string} string The string to convert.\n * @returns {Array} Returns the property path array.\n */\nvar stringToPath = memoizeCapped(function(string) {\n var result = [];\n if (string.charCodeAt(0) === 46 /* . */) {\n result.push('');\n }\n string.replace(rePropName, function(match, number, quote, subString) {\n result.push(quote ? subString.replace(reEscapeChar, '$1') : (number || match));\n });\n return result;\n});\n\nexport default stringToPath;\n","/**\n * A specialized version of `_.map` for arrays without support for iteratee\n * shorthands.\n *\n * @private\n * @param {Array} [array] The array to iterate over.\n * @param {Function} iteratee The function invoked per iteration.\n * @returns {Array} Returns the new mapped array.\n */\nfunction arrayMap(array, iteratee) {\n var index = -1,\n length = array == null ? 0 : array.length,\n result = Array(length);\n\n while (++index < length) {\n result[index] = iteratee(array[index], index, array);\n }\n return result;\n}\n\nexport default arrayMap;\n","import Symbol from './_Symbol.js';\nimport arrayMap from './_arrayMap.js';\nimport isArray from './isArray.js';\nimport isSymbol from './isSymbol.js';\n\n/** Used as references for various `Number` constants. */\nvar INFINITY = 1 / 0;\n\n/** Used to convert symbols to primitives and strings. */\nvar symbolProto = Symbol ? Symbol.prototype : undefined,\n symbolToString = symbolProto ? symbolProto.toString : undefined;\n\n/**\n * The base implementation of `_.toString` which doesn't convert nullish\n * values to empty strings.\n *\n * @private\n * @param {*} value The value to process.\n * @returns {string} Returns the string.\n */\nfunction baseToString(value) {\n // Exit early for strings to avoid a performance hit in some environments.\n if (typeof value == 'string') {\n return value;\n }\n if (isArray(value)) {\n // Recursively convert values (susceptible to call stack limits).\n return arrayMap(value, baseToString) + '';\n }\n if (isSymbol(value)) {\n return symbolToString ? symbolToString.call(value) : '';\n }\n var result = (value + '');\n return (result == '0' && (1 / value) == -INFINITY) ? '-0' : result;\n}\n\nexport default baseToString;\n","import baseToString from './_baseToString.js';\n\n/**\n * Converts `value` to a string. An empty string is returned for `null`\n * and `undefined` values. The sign of `-0` is preserved.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to convert.\n * @returns {string} Returns the converted string.\n * @example\n *\n * _.toString(null);\n * // => ''\n *\n * _.toString(-0);\n * // => '-0'\n *\n * _.toString([1, 2, 3]);\n * // => '1,2,3'\n */\nfunction toString(value) {\n return value == null ? '' : baseToString(value);\n}\n\nexport default toString;\n","import isArray from './isArray.js';\nimport isKey from './_isKey.js';\nimport stringToPath from './_stringToPath.js';\nimport toString from './toString.js';\n\n/**\n * Casts `value` to a path array if it's not one.\n *\n * @private\n * @param {*} value The value to inspect.\n * @param {Object} [object] The object to query keys on.\n * @returns {Array} Returns the cast property path array.\n */\nfunction castPath(value, object) {\n if (isArray(value)) {\n return value;\n }\n return isKey(value, object) ? [value] : stringToPath(toString(value));\n}\n\nexport default castPath;\n","/**\n * Gets the last element of `array`.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Array\n * @param {Array} array The array to query.\n * @returns {*} Returns the last element of `array`.\n * @example\n *\n * _.last([1, 2, 3]);\n * // => 3\n */\nfunction last(array) {\n var length = array == null ? 0 : array.length;\n return length ? array[length - 1] : undefined;\n}\n\nexport default last;\n","import isSymbol from './isSymbol.js';\n\n/** Used as references for various `Number` constants. */\nvar INFINITY = 1 / 0;\n\n/**\n * Converts `value` to a string key if it's not a string or symbol.\n *\n * @private\n * @param {*} value The value to inspect.\n * @returns {string|symbol} Returns the key.\n */\nfunction toKey(value) {\n if (typeof value == 'string' || isSymbol(value)) {\n return value;\n }\n var result = (value + '');\n return (result == '0' && (1 / value) == -INFINITY) ? '-0' : result;\n}\n\nexport default toKey;\n","import castPath from './_castPath.js';\nimport toKey from './_toKey.js';\n\n/**\n * The base implementation of `_.get` without support for default values.\n *\n * @private\n * @param {Object} object The object to query.\n * @param {Array|string} path The path of the property to get.\n * @returns {*} Returns the resolved value.\n */\nfunction baseGet(object, path) {\n path = castPath(path, object);\n\n var index = 0,\n length = path.length;\n\n while (object != null && index < length) {\n object = object[toKey(path[index++])];\n }\n return (index && index == length) ? object : undefined;\n}\n\nexport default baseGet;\n","/**\n * The base implementation of `_.slice` without an iteratee call guard.\n *\n * @private\n * @param {Array} array The array to slice.\n * @param {number} [start=0] The start position.\n * @param {number} [end=array.length] The end position.\n * @returns {Array} Returns the slice of `array`.\n */\nfunction baseSlice(array, start, end) {\n var index = -1,\n length = array.length;\n\n if (start < 0) {\n start = -start > length ? 0 : (length + start);\n }\n end = end > length ? length : end;\n if (end < 0) {\n end += length;\n }\n length = start > end ? 0 : ((end - start) >>> 0);\n start >>>= 0;\n\n var result = Array(length);\n while (++index < length) {\n result[index] = array[index + start];\n }\n return result;\n}\n\nexport default baseSlice;\n","import baseGet from './_baseGet.js';\nimport baseSlice from './_baseSlice.js';\n\n/**\n * Gets the parent value at `path` of `object`.\n *\n * @private\n * @param {Object} object The object to query.\n * @param {Array} path The path to get the parent value of.\n * @returns {*} Returns the parent value.\n */\nfunction parent(object, path) {\n return path.length < 2 ? object : baseGet(object, baseSlice(path, 0, -1));\n}\n\nexport default parent;\n","import castPath from './_castPath.js';\nimport last from './last.js';\nimport parent from './_parent.js';\nimport toKey from './_toKey.js';\n\n/**\n * The base implementation of `_.unset`.\n *\n * @private\n * @param {Object} object The object to modify.\n * @param {Array|string} path The property path to unset.\n * @returns {boolean} Returns `true` if the property is deleted, else `false`.\n */\nfunction baseUnset(object, path) {\n path = castPath(path, object);\n object = parent(object, path);\n return object == null || delete object[toKey(last(path))];\n}\n\nexport default baseUnset;\n","import baseUnset from './_baseUnset.js';\n\n/**\n * Removes the property at `path` of `object`.\n *\n * **Note:** This method mutates `object`.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Object\n * @param {Object} object The object to modify.\n * @param {Array|string} path The path of the property to unset.\n * @returns {boolean} Returns `true` if the property is deleted, else `false`.\n * @example\n *\n * var object = { 'a': [{ 'b': { 'c': 7 } }] };\n * _.unset(object, 'a[0].b.c');\n * // => true\n *\n * console.log(object);\n * // => { 'a': [{ 'b': {} }] };\n *\n * _.unset(object, ['a', '0', 'b', 'c']);\n * // => true\n *\n * console.log(object);\n * // => { 'a': [{ 'b': {} }] };\n */\nfunction unset(object, path) {\n return object == null ? true : baseUnset(object, path);\n}\n\nexport default unset;\n","import baseGet from './_baseGet.js';\n\n/**\n * Gets the value at `path` of `object`. If the resolved value is\n * `undefined`, the `defaultValue` is returned in its place.\n *\n * @static\n * @memberOf _\n * @since 3.7.0\n * @category Object\n * @param {Object} object The object to query.\n * @param {Array|string} path The path of the property to get.\n * @param {*} [defaultValue] The value returned for `undefined` resolved values.\n * @returns {*} Returns the resolved value.\n * @example\n *\n * var object = { 'a': [{ 'b': { 'c': 3 } }] };\n *\n * _.get(object, 'a[0].b.c');\n * // => 3\n *\n * _.get(object, ['a', '0', 'b', 'c']);\n * // => 3\n *\n * _.get(object, 'a.b.c', 'default');\n * // => 'default'\n */\nfunction get(object, path, defaultValue) {\n var result = object == null ? undefined : baseGet(object, path);\n return result === undefined ? defaultValue : result;\n}\n\nexport default get;\n","import baseAssignValue from './_baseAssignValue.js';\nimport eq from './eq.js';\n\n/**\n * This function is like `assignValue` except that it doesn't assign\n * `undefined` values.\n *\n * @private\n * @param {Object} object The object to modify.\n * @param {string} key The key of the property to assign.\n * @param {*} value The value to assign.\n */\nfunction assignMergeValue(object, key, value) {\n if ((value !== undefined && !eq(object[key], value)) ||\n (value === undefined && !(key in object))) {\n baseAssignValue(object, key, value);\n }\n}\n\nexport default assignMergeValue;\n","import createBaseFor from './_createBaseFor.js';\n\n/**\n * The base implementation of `baseForOwn` which iterates over `object`\n * properties returned by `keysFunc` and invokes `iteratee` for each property.\n * Iteratee functions may exit iteration early by explicitly returning `false`.\n *\n * @private\n * @param {Object} object The object to iterate over.\n * @param {Function} iteratee The function invoked per iteration.\n * @param {Function} keysFunc The function to get the keys of `object`.\n * @returns {Object} Returns `object`.\n */\nvar baseFor = createBaseFor();\n\nexport default baseFor;\n","/**\n * Creates a base function for methods like `_.forIn` and `_.forOwn`.\n *\n * @private\n * @param {boolean} [fromRight] Specify iterating from right to left.\n * @returns {Function} Returns the new base function.\n */\nfunction createBaseFor(fromRight) {\n return function(object, iteratee, keysFunc) {\n var index = -1,\n iterable = Object(object),\n props = keysFunc(object),\n length = props.length;\n\n while (length--) {\n var key = props[fromRight ? length : ++index];\n if (iteratee(iterable[key], key, iterable) === false) {\n break;\n }\n }\n return object;\n };\n}\n\nexport default createBaseFor;\n","import isArrayLike from './isArrayLike.js';\nimport isObjectLike from './isObjectLike.js';\n\n/**\n * This method is like `_.isArrayLike` except that it also checks if `value`\n * is an object.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to check.\n * @returns {boolean} Returns `true` if `value` is an array-like object,\n * else `false`.\n * @example\n *\n * _.isArrayLikeObject([1, 2, 3]);\n * // => true\n *\n * _.isArrayLikeObject(document.body.children);\n * // => true\n *\n * _.isArrayLikeObject('abc');\n * // => false\n *\n * _.isArrayLikeObject(_.noop);\n * // => false\n */\nfunction isArrayLikeObject(value) {\n return isObjectLike(value) && isArrayLike(value);\n}\n\nexport default isArrayLikeObject;\n","/**\n * Gets the value at `key`, unless `key` is \"__proto__\" or \"constructor\".\n *\n * @private\n * @param {Object} object The object to query.\n * @param {string} key The key of the property to get.\n * @returns {*} Returns the property value.\n */\nfunction safeGet(object, key) {\n if (key === 'constructor' && typeof object[key] === 'function') {\n return;\n }\n\n if (key == '__proto__') {\n return;\n }\n\n return object[key];\n}\n\nexport default safeGet;\n","import copyObject from './_copyObject.js';\nimport keysIn from './keysIn.js';\n\n/**\n * Converts `value` to a plain object flattening inherited enumerable string\n * keyed properties of `value` to own properties of the plain object.\n *\n * @static\n * @memberOf _\n * @since 3.0.0\n * @category Lang\n * @param {*} value The value to convert.\n * @returns {Object} Returns the converted plain object.\n * @example\n *\n * function Foo() {\n * this.b = 2;\n * }\n *\n * Foo.prototype.c = 3;\n *\n * _.assign({ 'a': 1 }, new Foo);\n * // => { 'a': 1, 'b': 2 }\n *\n * _.assign({ 'a': 1 }, _.toPlainObject(new Foo));\n * // => { 'a': 1, 'b': 2, 'c': 3 }\n */\nfunction toPlainObject(value) {\n return copyObject(value, keysIn(value));\n}\n\nexport default toPlainObject;\n","import assignMergeValue from './_assignMergeValue.js';\nimport cloneBuffer from './_cloneBuffer.js';\nimport cloneTypedArray from './_cloneTypedArray.js';\nimport copyArray from './_copyArray.js';\nimport initCloneObject from './_initCloneObject.js';\nimport isArguments from './isArguments.js';\nimport isArray from './isArray.js';\nimport isArrayLikeObject from './isArrayLikeObject.js';\nimport isBuffer from './isBuffer.js';\nimport isFunction from './isFunction.js';\nimport isObject from './isObject.js';\nimport isPlainObject from './isPlainObject.js';\nimport isTypedArray from './isTypedArray.js';\nimport safeGet from './_safeGet.js';\nimport toPlainObject from './toPlainObject.js';\n\n/**\n * A specialized version of `baseMerge` for arrays and objects which performs\n * deep merges and tracks traversed objects enabling objects with circular\n * references to be merged.\n *\n * @private\n * @param {Object} object The destination object.\n * @param {Object} source The source object.\n * @param {string} key The key of the value to merge.\n * @param {number} srcIndex The index of `source`.\n * @param {Function} mergeFunc The function to merge values.\n * @param {Function} [customizer] The function to customize assigned values.\n * @param {Object} [stack] Tracks traversed source values and their merged\n * counterparts.\n */\nfunction baseMergeDeep(object, source, key, srcIndex, mergeFunc, customizer, stack) {\n var objValue = safeGet(object, key),\n srcValue = safeGet(source, key),\n stacked = stack.get(srcValue);\n\n if (stacked) {\n assignMergeValue(object, key, stacked);\n return;\n }\n var newValue = customizer\n ? customizer(objValue, srcValue, (key + ''), object, source, stack)\n : undefined;\n\n var isCommon = newValue === undefined;\n\n if (isCommon) {\n var isArr = isArray(srcValue),\n isBuff = !isArr && isBuffer(srcValue),\n isTyped = !isArr && !isBuff && isTypedArray(srcValue);\n\n newValue = srcValue;\n if (isArr || isBuff || isTyped) {\n if (isArray(objValue)) {\n newValue = objValue;\n }\n else if (isArrayLikeObject(objValue)) {\n newValue = copyArray(objValue);\n }\n else if (isBuff) {\n isCommon = false;\n newValue = cloneBuffer(srcValue, true);\n }\n else if (isTyped) {\n isCommon = false;\n newValue = cloneTypedArray(srcValue, true);\n }\n else {\n newValue = [];\n }\n }\n else if (isPlainObject(srcValue) || isArguments(srcValue)) {\n newValue = objValue;\n if (isArguments(objValue)) {\n newValue = toPlainObject(objValue);\n }\n else if (!isObject(objValue) || isFunction(objValue)) {\n newValue = initCloneObject(srcValue);\n }\n }\n else {\n isCommon = false;\n }\n }\n if (isCommon) {\n // Recursively merge objects and arrays (susceptible to call stack limits).\n stack.set(srcValue, newValue);\n mergeFunc(newValue, srcValue, srcIndex, customizer, stack);\n stack['delete'](srcValue);\n }\n assignMergeValue(object, key, newValue);\n}\n\nexport default baseMergeDeep;\n","import Stack from './_Stack.js';\nimport assignMergeValue from './_assignMergeValue.js';\nimport baseFor from './_baseFor.js';\nimport baseMergeDeep from './_baseMergeDeep.js';\nimport isObject from './isObject.js';\nimport keysIn from './keysIn.js';\nimport safeGet from './_safeGet.js';\n\n/**\n * The base implementation of `_.merge` without support for multiple sources.\n *\n * @private\n * @param {Object} object The destination object.\n * @param {Object} source The source object.\n * @param {number} srcIndex The index of `source`.\n * @param {Function} [customizer] The function to customize merged values.\n * @param {Object} [stack] Tracks traversed source values and their merged\n * counterparts.\n */\nfunction baseMerge(object, source, srcIndex, customizer, stack) {\n if (object === source) {\n return;\n }\n baseFor(source, function(srcValue, key) {\n stack || (stack = new Stack);\n if (isObject(srcValue)) {\n baseMergeDeep(object, source, key, srcIndex, baseMerge, customizer, stack);\n }\n else {\n var newValue = customizer\n ? customizer(safeGet(object, key), srcValue, (key + ''), object, source, stack)\n : undefined;\n\n if (newValue === undefined) {\n newValue = srcValue;\n }\n assignMergeValue(object, key, newValue);\n }\n }, keysIn);\n}\n\nexport default baseMerge;\n","/**\n * This method returns the first argument it receives.\n *\n * @static\n * @since 0.1.0\n * @memberOf _\n * @category Util\n * @param {*} value Any value.\n * @returns {*} Returns `value`.\n * @example\n *\n * var object = { 'a': 1 };\n *\n * console.log(_.identity(object) === object);\n * // => true\n */\nfunction identity(value) {\n return value;\n}\n\nexport default identity;\n","/**\n * A faster alternative to `Function#apply`, this function invokes `func`\n * with the `this` binding of `thisArg` and the arguments of `args`.\n *\n * @private\n * @param {Function} func The function to invoke.\n * @param {*} thisArg The `this` binding of `func`.\n * @param {Array} args The arguments to invoke `func` with.\n * @returns {*} Returns the result of `func`.\n */\nfunction apply(func, thisArg, args) {\n switch (args.length) {\n case 0: return func.call(thisArg);\n case 1: return func.call(thisArg, args[0]);\n case 2: return func.call(thisArg, args[0], args[1]);\n case 3: return func.call(thisArg, args[0], args[1], args[2]);\n }\n return func.apply(thisArg, args);\n}\n\nexport default apply;\n","import apply from './_apply.js';\n\n/* Built-in method references for those with the same name as other `lodash` methods. */\nvar nativeMax = Math.max;\n\n/**\n * A specialized version of `baseRest` which transforms the rest array.\n *\n * @private\n * @param {Function} func The function to apply a rest parameter to.\n * @param {number} [start=func.length-1] The start position of the rest parameter.\n * @param {Function} transform The rest array transform.\n * @returns {Function} Returns the new function.\n */\nfunction overRest(func, start, transform) {\n start = nativeMax(start === undefined ? (func.length - 1) : start, 0);\n return function() {\n var args = arguments,\n index = -1,\n length = nativeMax(args.length - start, 0),\n array = Array(length);\n\n while (++index < length) {\n array[index] = args[start + index];\n }\n index = -1;\n var otherArgs = Array(start + 1);\n while (++index < start) {\n otherArgs[index] = args[index];\n }\n otherArgs[start] = transform(array);\n return apply(func, this, otherArgs);\n };\n}\n\nexport default overRest;\n","/**\n * Creates a function that returns `value`.\n *\n * @static\n * @memberOf _\n * @since 2.4.0\n * @category Util\n * @param {*} value The value to return from the new function.\n * @returns {Function} Returns the new constant function.\n * @example\n *\n * var objects = _.times(2, _.constant({ 'a': 1 }));\n *\n * console.log(objects);\n * // => [{ 'a': 1 }, { 'a': 1 }]\n *\n * console.log(objects[0] === objects[1]);\n * // => true\n */\nfunction constant(value) {\n return function() {\n return value;\n };\n}\n\nexport default constant;\n","import constant from './constant.js';\nimport defineProperty from './_defineProperty.js';\nimport identity from './identity.js';\n\n/**\n * The base implementation of `setToString` without support for hot loop shorting.\n *\n * @private\n * @param {Function} func The function to modify.\n * @param {Function} string The `toString` result.\n * @returns {Function} Returns `func`.\n */\nvar baseSetToString = !defineProperty ? identity : function(func, string) {\n return defineProperty(func, 'toString', {\n 'configurable': true,\n 'enumerable': false,\n 'value': constant(string),\n 'writable': true\n });\n};\n\nexport default baseSetToString;\n","/** Used to detect hot functions by number of calls within a span of milliseconds. */\nvar HOT_COUNT = 800,\n HOT_SPAN = 16;\n\n/* Built-in method references for those with the same name as other `lodash` methods. */\nvar nativeNow = Date.now;\n\n/**\n * Creates a function that'll short out and invoke `identity` instead\n * of `func` when it's called `HOT_COUNT` or more times in `HOT_SPAN`\n * milliseconds.\n *\n * @private\n * @param {Function} func The function to restrict.\n * @returns {Function} Returns the new shortable function.\n */\nfunction shortOut(func) {\n var count = 0,\n lastCalled = 0;\n\n return function() {\n var stamp = nativeNow(),\n remaining = HOT_SPAN - (stamp - lastCalled);\n\n lastCalled = stamp;\n if (remaining > 0) {\n if (++count >= HOT_COUNT) {\n return arguments[0];\n }\n } else {\n count = 0;\n }\n return func.apply(undefined, arguments);\n };\n}\n\nexport default shortOut;\n","import baseSetToString from './_baseSetToString.js';\nimport shortOut from './_shortOut.js';\n\n/**\n * Sets the `toString` method of `func` to return `string`.\n *\n * @private\n * @param {Function} func The function to modify.\n * @param {Function} string The `toString` result.\n * @returns {Function} Returns `func`.\n */\nvar setToString = shortOut(baseSetToString);\n\nexport default setToString;\n","import identity from './identity.js';\nimport overRest from './_overRest.js';\nimport setToString from './_setToString.js';\n\n/**\n * The base implementation of `_.rest` which doesn't validate or coerce arguments.\n *\n * @private\n * @param {Function} func The function to apply a rest parameter to.\n * @param {number} [start=func.length-1] The start position of the rest parameter.\n * @returns {Function} Returns the new function.\n */\nfunction baseRest(func, start) {\n return setToString(overRest(func, start, identity), func + '');\n}\n\nexport default baseRest;\n","import eq from './eq.js';\nimport isArrayLike from './isArrayLike.js';\nimport isIndex from './_isIndex.js';\nimport isObject from './isObject.js';\n\n/**\n * Checks if the given arguments are from an iteratee call.\n *\n * @private\n * @param {*} value The potential iteratee value argument.\n * @param {*} index The potential iteratee index or key argument.\n * @param {*} object The potential iteratee object argument.\n * @returns {boolean} Returns `true` if the arguments are from an iteratee call,\n * else `false`.\n */\nfunction isIterateeCall(value, index, object) {\n if (!isObject(object)) {\n return false;\n }\n var type = typeof index;\n if (type == 'number'\n ? (isArrayLike(object) && isIndex(index, object.length))\n : (type == 'string' && index in object)\n ) {\n return eq(object[index], value);\n }\n return false;\n}\n\nexport default isIterateeCall;\n","import baseRest from './_baseRest.js';\nimport isIterateeCall from './_isIterateeCall.js';\n\n/**\n * Creates a function like `_.assign`.\n *\n * @private\n * @param {Function} assigner The function to assign values.\n * @returns {Function} Returns the new assigner function.\n */\nfunction createAssigner(assigner) {\n return baseRest(function(object, sources) {\n var index = -1,\n length = sources.length,\n customizer = length > 1 ? sources[length - 1] : undefined,\n guard = length > 2 ? sources[2] : undefined;\n\n customizer = (assigner.length > 3 && typeof customizer == 'function')\n ? (length--, customizer)\n : undefined;\n\n if (guard && isIterateeCall(sources[0], sources[1], guard)) {\n customizer = length < 3 ? undefined : customizer;\n length = 1;\n }\n object = Object(object);\n while (++index < length) {\n var source = sources[index];\n if (source) {\n assigner(object, source, index, customizer);\n }\n }\n return object;\n });\n}\n\nexport default createAssigner;\n","import baseMerge from './_baseMerge.js';\nimport createAssigner from './_createAssigner.js';\n\n/**\n * This method is like `_.assign` except that it recursively merges own and\n * inherited enumerable string keyed properties of source objects into the\n * destination object. Source properties that resolve to `undefined` are\n * skipped if a destination value exists. Array and plain object properties\n * are merged recursively. Other objects and value types are overridden by\n * assignment. Source objects are applied from left to right. Subsequent\n * sources overwrite property assignments of previous sources.\n *\n * **Note:** This method mutates `object`.\n *\n * @static\n * @memberOf _\n * @since 0.5.0\n * @category Object\n * @param {Object} object The destination object.\n * @param {...Object} [sources] The source objects.\n * @returns {Object} Returns `object`.\n * @example\n *\n * var object = {\n * 'a': [{ 'b': 2 }, { 'd': 4 }]\n * };\n *\n * var other = {\n * 'a': [{ 'c': 3 }, { 'e': 5 }]\n * };\n *\n * _.merge(object, other);\n * // => { 'a': [{ 'b': 2, 'c': 3 }, { 'd': 4, 'e': 5 }] }\n */\nvar merge = createAssigner(function(object, source, srcIndex) {\n baseMerge(object, source, srcIndex);\n});\n\nexport default merge;\n","import assignValue from './_assignValue.js';\nimport castPath from './_castPath.js';\nimport isIndex from './_isIndex.js';\nimport isObject from './isObject.js';\nimport toKey from './_toKey.js';\n\n/**\n * The base implementation of `_.set`.\n *\n * @private\n * @param {Object} object The object to modify.\n * @param {Array|string} path The path of the property to set.\n * @param {*} value The value to set.\n * @param {Function} [customizer] The function to customize path creation.\n * @returns {Object} Returns `object`.\n */\nfunction baseSet(object, path, value, customizer) {\n if (!isObject(object)) {\n return object;\n }\n path = castPath(path, object);\n\n var index = -1,\n length = path.length,\n lastIndex = length - 1,\n nested = object;\n\n while (nested != null && ++index < length) {\n var key = toKey(path[index]),\n newValue = value;\n\n if (key === '__proto__' || key === 'constructor' || key === 'prototype') {\n return object;\n }\n\n if (index != lastIndex) {\n var objValue = nested[key];\n newValue = customizer ? customizer(objValue, key, nested) : undefined;\n if (newValue === undefined) {\n newValue = isObject(objValue)\n ? objValue\n : (isIndex(path[index + 1]) ? [] : {});\n }\n }\n assignValue(nested, key, newValue);\n nested = nested[key];\n }\n return object;\n}\n\nexport default baseSet;\n","import baseSet from './_baseSet.js';\n\n/**\n * Sets the value at `path` of `object`. If a portion of `path` doesn't exist,\n * it's created. Arrays are created for missing index properties while objects\n * are created for all other missing properties. Use `_.setWith` to customize\n * `path` creation.\n *\n * **Note:** This method mutates `object`.\n *\n * @static\n * @memberOf _\n * @since 3.7.0\n * @category Object\n * @param {Object} object The object to modify.\n * @param {Array|string} path The path of the property to set.\n * @param {*} value The value to set.\n * @returns {Object} Returns `object`.\n * @example\n *\n * var object = { 'a': [{ 'b': { 'c': 3 } }] };\n *\n * _.set(object, 'a[0].b.c', 4);\n * console.log(object.a[0].b.c);\n * // => 4\n *\n * _.set(object, ['x', '0', 'y', 'z'], 5);\n * console.log(object.x[0].y.z);\n * // => 5\n */\nfunction set(object, path, value) {\n return object == null ? object : baseSet(object, path, value);\n}\n\nexport default set;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/stylesmap\n */\n\nimport { get, isObject, merge, set, unset } from 'lodash-es';\n\n/**\n * Styles map. Allows handling (adding, removing, retrieving) a set of style rules (usually, of an element).\n *\n * The styles map is capable of normalizing style names so e.g. the following operations are possible:\n */\nexport default class StylesMap {\n\t/**\n\t * Keeps an internal representation of styles map. Normalized styles are kept as object tree to allow unified modification and\n\t * value access model using lodash's get, set, unset, etc methods.\n\t *\n\t * When no style processor rules are defined it acts as simple key-value storage.\n\t */\n\tprivate _styles: Record<string, string>;\n\n\t/**\n\t * An instance of the {@link module:engine/view/stylesmap~StylesProcessor}.\n\t */\n\tprivate readonly _styleProcessor: StylesProcessor;\n\n\t/**\n\t * Creates Styles instance.\n\t */\n\tconstructor( styleProcessor: StylesProcessor ) {\n\t\tthis._styles = {};\n\t\tthis._styleProcessor = styleProcessor;\n\t}\n\n\t/**\n\t * Returns true if style map has no styles set.\n\t */\n\tpublic get isEmpty(): boolean {\n\t\tconst entries = Object.entries( this._styles );\n\t\tconst from = Array.from( entries );\n\n\t\treturn !from.length;\n\t}\n\n\t/**\n\t * Number of styles defined.\n\t */\n\tpublic get size(): number {\n\t\tif ( this.isEmpty ) {\n\t\t\treturn 0;\n\t\t}\n\n\t\treturn this.getStyleNames().length;\n\t}\n\n\t/**\n\t * Set styles map to a new value.\n\t *\n\t * ```ts\n\t * styles.setTo( 'border:1px solid blue;margin-top:1px;' );\n\t * ```\n\t */\n\tpublic setTo( inlineStyle: string ): void {\n\t\tthis.clear();\n\n\t\tconst parsedStyles = Array.from( parseInlineStyles( inlineStyle ).entries() );\n\n\t\tfor ( const [ key, value ] of parsedStyles ) {\n\t\t\tthis._styleProcessor.toNormalizedForm( key, value, this._styles );\n\t\t}\n\t}\n\n\t/**\n\t * Checks if a given style is set.\n\t *\n\t * ```ts\n\t * styles.setTo( 'margin-left:1px;' );\n\t *\n\t * styles.has( 'margin-left' ); // -> true\n\t * styles.has( 'padding' ); // -> false\n\t * ```\n\t *\n\t * **Note**: This check supports normalized style names.\n\t *\n\t * ```ts\n\t * // Enable 'margin' shorthand processing:\n\t * editor.data.addStyleProcessorRules( addMarginRules );\n\t *\n\t * styles.setTo( 'margin:2px;' );\n\t *\n\t * styles.has( 'margin' ); // -> true\n\t * styles.has( 'margin-top' ); // -> true\n\t * styles.has( 'margin-left' ); // -> true\n\t *\n\t * styles.remove( 'margin-top' );\n\t *\n\t * styles.has( 'margin' ); // -> false\n\t * styles.has( 'margin-top' ); // -> false\n\t * styles.has( 'margin-left' ); // -> true\n\t * ```\n\t *\n\t * @param name Style name.\n\t */\n\tpublic has( name: string ): boolean {\n\t\tif ( this.isEmpty ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst styles = this._styleProcessor.getReducedForm( name, this._styles );\n\n\t\tconst propertyDescriptor = styles.find( ( [ property ] ) => property === name );\n\n\t\t// Only return a value if it is set;\n\t\treturn Array.isArray( propertyDescriptor );\n\t}\n\n\t/**\n\t * Sets a given style.\n\t *\n\t * Can insert one by one:\n\t *\n\t * ```ts\n\t * styles.set( 'color', 'blue' );\n\t * styles.set( 'margin-right', '1em' );\n\t * ```\n\t *\n\t * ***Note**:* This method uses {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules\n\t * enabled style processor rules} to normalize passed values.\n\t *\n\t * ```ts\n\t * // Enable 'margin' shorthand processing:\n\t * editor.data.addStyleProcessorRules( addMarginRules );\n\t *\n\t * styles.set( 'margin', '2px' );\n\t * ```\n\t *\n\t * The above code will set margin to:\n\t *\n\t * ```ts\n\t * styles.getNormalized( 'margin' );\n\t * // -> { top: '2px', right: '2px', bottom: '2px', left: '2px' }\n\t * ```\n\t *\n\t * Which makes it possible to retrieve a \"sub-value\":\n\t *\n\t * ```ts\n\t * styles.get( 'margin-left' ); // -> '2px'\n\t * ```\n\t *\n\t * Or modify it:\n\t *\n\t * ```ts\n\t * styles.remove( 'margin-left' );\n\t *\n\t * styles.getNormalized( 'margin' ); // -> { top: '1px', bottom: '1px', right: '1px' }\n\t * styles.toString(); // -> 'margin-bottom:1px;margin-right:1px;margin-top:1px;'\n\t * ```\n\t *\n\t * This method also allows to set normalized values directly (if a particular styles processor rule was enabled):\n\t *\n\t * ```ts\n\t * styles.set( 'border-color', { top: 'blue' } );\n\t * styles.set( 'margin', { right: '2em' } );\n\t *\n\t * styles.toString(); // -> 'border-color-top:blue;margin-right:2em;'\n\t * ```\n\t *\n\t * @label KEY_VALUE\n\t * @param name Style property name.\n\t * @param value Value to set.\n\t */\n\tpublic set( name: string, value: StyleValue ): void;\n\n\t/**\n\t * Sets many styles at once:\n\t *\n\t * ```ts\n\t * styles.set( {\n\t * \tcolor: 'blue',\n\t * \t'margin-right': '1em'\n\t * } );\n\t * ```\n\t *\n\t * It is equivalent to:\n\t *\n\t * ```ts\n\t * styles.set( 'color', 'blue' );\n\t * styles.set( 'margin-right', '1em' );\n\t * ```\n\t *\n\t * See {@link #set:KEY_VALUE}\n\t *\n\t * @label CONFIG_OBJECT\n\t */\n\tpublic set( styles: Styles ): void;\n\n\tpublic set( nameOrObject: string | Styles, valueOrObject?: StyleValue ): void {\n\t\tif ( isObject( nameOrObject ) ) {\n\t\t\tfor ( const [ key, value ] of Object.entries( nameOrObject ) ) {\n\t\t\t\tthis._styleProcessor.toNormalizedForm( key, value, this._styles );\n\t\t\t}\n\t\t} else {\n\t\t\tthis._styleProcessor.toNormalizedForm( nameOrObject, valueOrObject!, this._styles );\n\t\t}\n\t}\n\n\t/**\n\t * Removes given style.\n\t *\n\t * ```ts\n\t * styles.setTo( 'background:#f00;margin-right:2px;' );\n\t *\n\t * styles.remove( 'background' );\n\t *\n\t * styles.toString(); // -> 'margin-right:2px;'\n\t * ```\n\t *\n\t * ***Note**:* This method uses {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules\n\t * enabled style processor rules} to normalize passed values.\n\t *\n\t * ```ts\n\t * // Enable 'margin' shorthand processing:\n\t * editor.data.addStyleProcessorRules( addMarginRules );\n\t *\n\t * styles.setTo( 'margin:1px' );\n\t *\n\t * styles.remove( 'margin-top' );\n\t * styles.remove( 'margin-right' );\n\t *\n\t * styles.toString(); // -> 'margin-bottom:1px;margin-left:1px;'\n\t * ```\n\t *\n\t * @param name Style name.\n\t */\n\tpublic remove( name: string ): void {\n\t\tconst path = toPath( name );\n\n\t\tunset( this._styles, path );\n\t\tdelete this._styles[ name ];\n\n\t\tthis._cleanEmptyObjectsOnPath( path );\n\t}\n\n\t/**\n\t * Returns a normalized style object or a single value.\n\t *\n\t * ```ts\n\t * // Enable 'margin' shorthand processing:\n\t * editor.data.addStyleProcessorRules( addMarginRules );\n\t *\n\t * const styles = new Styles();\n\t * styles.setTo( 'margin:1px 2px 3em;' );\n\t *\n\t * styles.getNormalized( 'margin' );\n\t * // will log:\n\t * // {\n\t * // top: '1px',\n\t * // right: '2px',\n\t * // bottom: '3em',\n\t * // left: '2px' // normalized value from margin shorthand\n\t * // }\n\t *\n\t * styles.getNormalized( 'margin-left' ); // -> '2px'\n\t * ```\n\t *\n\t * **Note**: This method will only return normalized styles if a style processor was defined.\n\t *\n\t * @param name Style name.\n\t */\n\tpublic getNormalized( name?: string ): StyleValue {\n\t\treturn this._styleProcessor.getNormalized( name, this._styles );\n\t}\n\n\t/**\n\t * Returns a normalized style string. Styles are sorted by name.\n\t *\n\t * ```ts\n\t * styles.set( 'margin' , '1px' );\n\t * styles.set( 'background', '#f00' );\n\t *\n\t * styles.toString(); // -> 'background:#f00;margin:1px;'\n\t * ```\n\t *\n\t * **Note**: This method supports normalized styles if defined.\n\t *\n\t * ```ts\n\t * // Enable 'margin' shorthand processing:\n\t * editor.data.addStyleProcessorRules( addMarginRules );\n\t *\n\t * styles.set( 'margin' , '1px' );\n\t * styles.set( 'background', '#f00' );\n\t * styles.remove( 'margin-top' );\n\t * styles.remove( 'margin-right' );\n\t *\n\t * styles.toString(); // -> 'background:#f00;margin-bottom:1px;margin-left:1px;'\n\t * ```\n\t */\n\tpublic toString(): string {\n\t\tif ( this.isEmpty ) {\n\t\t\treturn '';\n\t\t}\n\n\t\treturn this._getStylesEntries()\n\t\t\t.map( arr => arr.join( ':' ) )\n\t\t\t.sort()\n\t\t\t.join( ';' ) + ';';\n\t}\n\n\t/**\n\t * Returns property as a value string or undefined if property is not set.\n\t *\n\t * ```ts\n\t * // Enable 'margin' shorthand processing:\n\t * editor.data.addStyleProcessorRules( addMarginRules );\n\t *\n\t * const styles = new Styles();\n\t * styles.setTo( 'margin:1px;' );\n\t * styles.set( 'margin-bottom', '3em' );\n\t *\n\t * styles.getAsString( 'margin' ); // -> 'margin: 1px 1px 3em;'\n\t * ```\n\t *\n\t * Note, however, that all sub-values must be set for the longhand property name to return a value:\n\t *\n\t * ```ts\n\t * const styles = new Styles();\n\t * styles.setTo( 'margin:1px;' );\n\t * styles.remove( 'margin-bottom' );\n\t *\n\t * styles.getAsString( 'margin' ); // -> undefined\n\t * ```\n\t *\n\t * In the above scenario, it is not possible to return a `margin` value, so `undefined` is returned.\n\t * Instead, you should use:\n\t *\n\t * ```ts\n\t * const styles = new Styles();\n\t * styles.setTo( 'margin:1px;' );\n\t * styles.remove( 'margin-bottom' );\n\t *\n\t * for ( const styleName of styles.getStyleNames() ) {\n\t * \tconsole.log( styleName, styles.getAsString( styleName ) );\n\t * }\n\t * // 'margin-top', '1px'\n\t * // 'margin-right', '1px'\n\t * // 'margin-left', '1px'\n\t * ```\n\t *\n\t * In general, it is recommend to iterate over style names like in the example above. This way, you will always get all\n\t * the currently set style values. So, if all the 4 margin values would be set\n\t * the for-of loop above would yield only `'margin'`, `'1px'`:\n\t *\n\t * ```ts\n\t * const styles = new Styles();\n\t * styles.setTo( 'margin:1px;' );\n\t *\n\t * for ( const styleName of styles.getStyleNames() ) {\n\t * \tconsole.log( styleName, styles.getAsString( styleName ) );\n\t * }\n\t * // 'margin', '1px'\n\t * ```\n\t *\n\t * **Note**: To get a normalized version of a longhand property use the {@link #getNormalized `#getNormalized()`} method.\n\t */\n\tpublic getAsString( propertyName: string ): string | undefined {\n\t\tif ( this.isEmpty ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( this._styles[ propertyName ] && !isObject( this._styles[ propertyName ] ) ) {\n\t\t\t// Try return styles set directly - values that are not parsed.\n\t\t\treturn this._styles[ propertyName ];\n\t\t}\n\n\t\tconst styles = this._styleProcessor.getReducedForm( propertyName, this._styles );\n\n\t\tconst propertyDescriptor = styles.find( ( [ property ] ) => property === propertyName );\n\n\t\t// Only return a value if it is set;\n\t\tif ( Array.isArray( propertyDescriptor ) ) {\n\t\t\treturn propertyDescriptor[ 1 ];\n\t\t}\n\t}\n\n\t/**\n\t * Returns all style properties names as they would appear when using {@link #toString `#toString()`}.\n\t *\n\t * When `expand` is set to true and there's a shorthand style property set, it will also return all equivalent styles:\n\t *\n\t * ```ts\n\t * stylesMap.setTo( 'margin: 1em' )\n\t * ```\n\t *\n\t * will be expanded to:\n\t *\n\t * ```ts\n\t * [ 'margin', 'margin-top', 'margin-right', 'margin-bottom', 'margin-left' ]\n\t * ```\n\t *\n\t * @param expand Expand shorthand style properties and all return equivalent style representations.\n\t */\n\tpublic getStyleNames( expand = false ): Array<string> {\n\t\tif ( this.isEmpty ) {\n\t\t\treturn [];\n\t\t}\n\n\t\tif ( expand ) {\n\t\t\treturn this._styleProcessor.getStyleNames( this._styles );\n\t\t}\n\n\t\tconst entries = this._getStylesEntries();\n\n\t\treturn entries.map( ( [ key ] ) => key );\n\t}\n\n\t/**\n\t * Removes all styles.\n\t */\n\tpublic clear(): void {\n\t\tthis._styles = {};\n\t}\n\n\t/**\n\t * Returns normalized styles entries for further processing.\n\t */\n\tprivate _getStylesEntries(): Array<PropertyDescriptor> {\n\t\tconst parsed: Array<PropertyDescriptor> = [];\n\n\t\tconst keys = Object.keys( this._styles );\n\n\t\tfor ( const key of keys ) {\n\t\t\tparsed.push( ...this._styleProcessor.getReducedForm( key, this._styles ) );\n\t\t}\n\n\t\treturn parsed;\n\t}\n\n\t/**\n\t * Removes empty objects upon removing an entry from internal object.\n\t */\n\tprivate _cleanEmptyObjectsOnPath( path: string ): void {\n\t\tconst pathParts = path.split( '.' );\n\t\tconst isChildPath = pathParts.length > 1;\n\n\t\tif ( !isChildPath ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst parentPath = pathParts.splice( 0, pathParts.length - 1 ).join( '.' );\n\n\t\tconst parentObject = get( this._styles, parentPath );\n\n\t\tif ( !parentObject ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst isParentEmpty = !Array.from( Object.keys( parentObject ) ).length;\n\n\t\tif ( isParentEmpty ) {\n\t\t\tthis.remove( parentPath );\n\t\t}\n\t}\n}\n\n/**\n * Style processor is responsible for writing and reading a normalized styles object.\n */\nexport class StylesProcessor {\n\tprivate readonly _normalizers: Map<string, Normalizer>;\n\tprivate readonly _extractors: Map<string, Extractor>;\n\tprivate readonly _reducers: Map<string, Reducer>;\n\tprivate readonly _consumables: Map<string, Array<string>>;\n\n\t/**\n\t * Creates StylesProcessor instance.\n\t *\n\t * @internal\n\t */\n\tconstructor() {\n\t\tthis._normalizers = new Map();\n\t\tthis._extractors = new Map();\n\t\tthis._reducers = new Map();\n\t\tthis._consumables = new Map();\n\t}\n\n\t/**\n\t * Parse style string value to a normalized object and appends it to styles object.\n\t *\n\t * ```ts\n\t * const styles = {};\n\t *\n\t * stylesProcessor.toNormalizedForm( 'margin', '1px', styles );\n\t *\n\t * // styles will consist: { margin: { top: '1px', right: '1px', bottom: '1px', left: '1px; } }\n\t * ```\n\t *\n\t * **Note**: To define normalizer callbacks use {@link #setNormalizer}.\n\t *\n\t * @param name Name of style property.\n\t * @param propertyValue Value of style property.\n\t * @param styles Object holding normalized styles.\n\t */\n\tpublic toNormalizedForm( name: string, propertyValue: StyleValue, styles: Styles ): void {\n\t\tif ( isObject( propertyValue ) ) {\n\t\t\tappendStyleValue( styles, toPath( name ), propertyValue );\n\n\t\t\treturn;\n\t\t}\n\n\t\tif ( this._normalizers.has( name ) ) {\n\t\t\tconst normalizer = this._normalizers.get( name )!;\n\n\t\t\tconst { path, value } = normalizer( propertyValue );\n\n\t\t\tappendStyleValue( styles, path, value );\n\t\t} else {\n\t\t\tappendStyleValue( styles, name, propertyValue );\n\t\t}\n\t}\n\n\t/**\n\t * Returns a normalized version of a style property.\n\t *\n\t * ```ts\n\t * const styles = {\n\t * \tmargin: { top: '1px', right: '1px', bottom: '1px', left: '1px; },\n\t * \tbackground: { color: '#f00' }\n\t * };\n\t *\n\t * stylesProcessor.getNormalized( 'background' );\n\t * // will return: { color: '#f00' }\n\t *\n\t * stylesProcessor.getNormalized( 'margin-top' );\n\t * // will return: '1px'\n\t * ```\n\t *\n\t * **Note**: In some cases extracting single value requires defining an extractor callback {@link #setExtractor}.\n\t *\n\t * @param name Name of style property.\n\t * @param styles Object holding normalized styles.\n\t */\n\tpublic getNormalized( name: string | undefined, styles: Styles ): StyleValue {\n\t\tif ( !name ) {\n\t\t\treturn merge( {}, styles );\n\t\t}\n\n\t\t// Might be empty string.\n\t\tif ( styles[ name ] !== undefined ) {\n\t\t\treturn styles[ name ];\n\t\t}\n\n\t\tif ( this._extractors.has( name ) ) {\n\t\t\tconst extractor = this._extractors.get( name )!;\n\n\t\t\tif ( typeof extractor === 'string' ) {\n\t\t\t\treturn get( styles, extractor );\n\t\t\t}\n\n\t\t\tconst value = extractor( name, styles );\n\n\t\t\tif ( value ) {\n\t\t\t\treturn value;\n\t\t\t}\n\t\t}\n\n\t\treturn get( styles, toPath( name ) );\n\t}\n\n\t/**\n\t * Returns a reduced form of style property form normalized object.\n\t *\n\t * For default margin reducer, the below code:\n\t *\n\t * ```ts\n\t * stylesProcessor.getReducedForm( 'margin', {\n\t * \tmargin: { top: '1px', right: '1px', bottom: '2px', left: '1px; }\n\t * } );\n\t * ```\n\t *\n\t * will return:\n\t *\n\t * ```ts\n\t * [\n\t * \t[ 'margin', '1px 1px 2px' ]\n\t * ]\n\t * ```\n\t *\n\t * because it might be represented as a shorthand 'margin' value. However if one of margin long hand values is missing it should return:\n\t *\n\t * ```ts\n\t * [\n\t * \t[ 'margin-top', '1px' ],\n\t * \t[ 'margin-right', '1px' ],\n\t * \t[ 'margin-bottom', '2px' ]\n\t * \t// the 'left' value is missing - cannot use 'margin' shorthand.\n\t * ]\n\t * ```\n\t *\n\t * **Note**: To define reducer callbacks use {@link #setReducer}.\n\t *\n\t * @param name Name of style property.\n\t */\n\tpublic getReducedForm( name: string, styles: Styles ): Array<PropertyDescriptor> {\n\t\tconst normalizedValue = this.getNormalized( name, styles );\n\n\t\t// Might be empty string.\n\t\tif ( normalizedValue === undefined ) {\n\t\t\treturn [];\n\t\t}\n\n\t\tif ( this._reducers.has( name ) ) {\n\t\t\tconst reducer = this._reducers.get( name )!;\n\n\t\t\treturn reducer( normalizedValue );\n\t\t}\n\n\t\treturn [ [ name, normalizedValue as string ] ];\n\t}\n\n\t/**\n\t * Return all style properties. Also expand shorthand properties (e.g. `margin`, `background`) if respective extractor is available.\n\t *\n\t * @param styles Object holding normalized styles.\n\t */\n\tpublic getStyleNames( styles: Styles ): Array<string> {\n\t\t// Find all extractable styles that have a value.\n\t\tconst expandedStyleNames = Array.from( this._consumables.keys() ).filter( name => {\n\t\t\tconst style = this.getNormalized( name, styles );\n\n\t\t\tif ( style && typeof style == 'object' ) {\n\t\t\t\treturn Object.keys( style ).length;\n\t\t\t}\n\n\t\t\treturn style;\n\t\t} );\n\n\t\t// For simple styles (for example `color`) we don't have a map of those styles\n\t\t// but they are 1 to 1 with normalized object keys.\n\t\tconst styleNamesKeysSet = new Set( [\n\t\t\t...expandedStyleNames,\n\t\t\t...Object.keys( styles )\n\t\t] );\n\n\t\treturn Array.from( styleNamesKeysSet.values() );\n\t}\n\n\t/**\n\t * Returns related style names.\n\t *\n\t * ```ts\n\t * stylesProcessor.getRelatedStyles( 'margin' );\n\t * // will return: [ 'margin-top', 'margin-right', 'margin-bottom', 'margin-left' ];\n\t *\n\t * stylesProcessor.getRelatedStyles( 'margin-top' );\n\t * // will return: [ 'margin' ];\n\t * ```\n\t *\n\t * **Note**: To define new style relations load an existing style processor or use\n\t * {@link module:engine/view/stylesmap~StylesProcessor#setStyleRelation `StylesProcessor.setStyleRelation()`}.\n\t */\n\tpublic getRelatedStyles( name: string ): Array<string> {\n\t\treturn this._consumables.get( name ) || [];\n\t}\n\n\t/**\n\t * Adds a normalizer method for a style property.\n\t *\n\t * A normalizer returns describing how the value should be normalized.\n\t *\n\t * For instance 'margin' style is a shorthand for four margin values:\n\t *\n\t * - 'margin-top'\n\t * - 'margin-right'\n\t * - 'margin-bottom'\n\t * - 'margin-left'\n\t *\n\t * and can be written in various ways if some values are equal to others. For instance `'margin: 1px 2em;'` is a shorthand for\n\t * `'margin-top: 1px;margin-right: 2em;margin-bottom: 1px;margin-left: 2em'`.\n\t *\n\t * A normalizer should parse various margin notations as a single object:\n\t *\n\t * ```ts\n\t * const styles = {\n\t * \tmargin: {\n\t * \t\ttop: '1px',\n\t * \t\tright: '2em',\n\t * \t\tbottom: '1px',\n\t * \t\tleft: '2em'\n\t * \t}\n\t * };\n\t * ```\n\t *\n\t * Thus a normalizer for 'margin' style should return an object defining style path and value to store:\n\t *\n\t * ```ts\n\t * const returnValue = {\n\t * \tpath: 'margin',\n\t * \tvalue: {\n\t * \t\ttop: '1px',\n\t * \t\tright: '2em',\n\t * \t\tbottom: '1px',\n\t * \t\tleft: '2em'\n\t * \t}\n\t * };\n\t * ```\n\t *\n\t * Additionally to fully support all margin notations there should be also defined 4 normalizers for longhand margin notations. Below\n\t * is an example for 'margin-top' style property normalizer:\n\t *\n\t * ```ts\n\t * stylesProcessor.setNormalizer( 'margin-top', valueString => {\n\t * \treturn {\n\t * \t\tpath: 'margin.top',\n\t * \t\tvalue: valueString\n\t * \t}\n\t * } );\n\t * ```\n\t */\n\tpublic setNormalizer( name: string, callback: Normalizer ): void {\n\t\tthis._normalizers.set( name, callback );\n\t}\n\n\t/**\n\t * Adds a extractor callback for a style property.\n\t *\n\t * Most normalized style values are stored as one level objects. It is assumed that `'margin-top'` style will be stored as:\n\t *\n\t * ```ts\n\t * const styles = {\n\t * \tmargin: {\n\t * \t\ttop: 'value'\n\t * \t}\n\t * }\n\t * ```\n\t *\n\t * However, some styles can have conflicting notations and thus it might be harder to extract a style value from shorthand. For instance\n\t * the 'border-top-style' can be defined using `'border-top:solid'`, `'border-style:solid none none none'` or by `'border:solid'`\n\t * shorthands. The default border styles processors stores styles as:\n\t *\n\t * ```ts\n\t * const styles = {\n\t * \tborder: {\n\t * \t\tstyle: {\n\t * \t\t\ttop: 'solid'\n\t * \t\t}\n\t * \t}\n\t * }\n\t * ```\n\t *\n\t * as it is better to modify border style independently from other values. On the other part the output of the border might be\n\t * desired as `border-top`, `border-left`, etc notation.\n\t *\n\t * In the above example an extractor should return a side border value that combines style, color and width:\n\t *\n\t * ```ts\n\t * styleProcessor.setExtractor( 'border-top', styles => {\n\t * \treturn {\n\t * \t\tcolor: styles.border.color.top,\n\t * \t\tstyle: styles.border.style.top,\n\t * \t\twidth: styles.border.width.top\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * @param callbackOrPath Callback that return a requested value or path string for single values.\n\t */\n\tpublic setExtractor( name: string, callbackOrPath: Extractor ): void {\n\t\tthis._extractors.set( name, callbackOrPath );\n\t}\n\n\t/**\n\t * Adds a reducer callback for a style property.\n\t *\n\t * Reducer returns a minimal notation for given style name. For longhand properties it is not required to write a reducer as\n\t * by default the direct value from style path is taken.\n\t *\n\t * For shorthand styles a reducer should return minimal style notation either by returning single name-value tuple or multiple tuples\n\t * if a shorthand cannot be used. For instance for a margin shorthand a reducer might return:\n\t *\n\t * ```ts\n\t * const marginShortHandTuple = [\n\t * \t[ 'margin', '1px 1px 2px' ]\n\t * ];\n\t * ```\n\t *\n\t * or a longhand tuples for defined values:\n\t *\n\t * ```ts\n\t * // Considering margin.bottom and margin.left are undefined.\n\t * const marginLonghandsTuples = [\n\t * \t[ 'margin-top', '1px' ],\n\t * \t[ 'margin-right', '1px' ]\n\t * ];\n\t * ```\n\t *\n\t * A reducer obtains a normalized style value:\n\t *\n\t * ```ts\n\t * // Simplified reducer that always outputs 4 values which are always present:\n\t * stylesProcessor.setReducer( 'margin', margin => {\n\t * \treturn [\n\t * \t\t[ 'margin', `${ margin.top } ${ margin.right } ${ margin.bottom } ${ margin.left }` ]\n\t * \t]\n\t * } );\n\t * ```\n\t */\n\tpublic setReducer( name: string, callback: Reducer ): void {\n\t\tthis._reducers.set( name, callback );\n\t}\n\n\t/**\n\t * Defines a style shorthand relation to other style notations.\n\t *\n\t * ```ts\n\t * stylesProcessor.setStyleRelation( 'margin', [\n\t * \t'margin-top',\n\t * \t'margin-right',\n\t * \t'margin-bottom',\n\t * \t'margin-left'\n\t * ] );\n\t * ```\n\t *\n\t * This enables expanding of style names for shorthands. For instance, if defined,\n\t * {@link module:engine/conversion/viewconsumable~ViewConsumable view consumable} items are automatically created\n\t * for long-hand margin style notation alongside the `'margin'` item.\n\t *\n\t * This means that when an element being converted has a style `margin`, a converter for `margin-left` will work just\n\t * fine since the view consumable will contain a consumable `margin-left` item (thanks to the relation) and\n\t * `element.getStyle( 'margin-left' )` will work as well assuming that the style processor was correctly configured.\n\t * However, once `margin-left` is consumed, `margin` will not be consumable anymore.\n\t */\n\tpublic setStyleRelation( shorthandName: string, styleNames: Array<string> ): void {\n\t\tthis._mapStyleNames( shorthandName, styleNames );\n\n\t\tfor ( const alsoName of styleNames ) {\n\t\t\tthis._mapStyleNames( alsoName, [ shorthandName ] );\n\t\t}\n\t}\n\n\t/**\n\t * Set two-way binding of style names.\n\t */\n\tprivate _mapStyleNames( name: string, styleNames: Array<string> ) {\n\t\tif ( !this._consumables.has( name ) ) {\n\t\t\tthis._consumables.set( name, [] );\n\t\t}\n\n\t\tthis._consumables.get( name )!.push( ...styleNames );\n\t}\n}\n\n/**\n * Parses inline styles and puts property - value pairs into styles map.\n *\n * @param stylesString Styles to parse.\n * @returns Map of parsed properties and values.\n */\nfunction parseInlineStyles( stylesString: string ): Map<string, string> {\n\t// `null` if no quote was found in input string or last found quote was a closing quote. See below.\n\tlet quoteType = null;\n\tlet propertyNameStart = 0;\n\tlet propertyValueStart = 0;\n\tlet propertyName = null;\n\n\tconst stylesMap = new Map();\n\n\t// Do not set anything if input string is empty.\n\tif ( stylesString === '' ) {\n\t\treturn stylesMap;\n\t}\n\n\t// Fix inline styles that do not end with `;` so they are compatible with algorithm below.\n\tif ( stylesString.charAt( stylesString.length - 1 ) != ';' ) {\n\t\tstylesString = stylesString + ';';\n\t}\n\n\t// Seek the whole string for \"special characters\".\n\tfor ( let i = 0; i < stylesString.length; i++ ) {\n\t\tconst char = stylesString.charAt( i );\n\n\t\tif ( quoteType === null ) {\n\t\t\t// No quote found yet or last found quote was a closing quote.\n\t\t\tswitch ( char ) {\n\t\t\t\tcase ':':\n\t\t\t\t\t// Most of time colon means that property name just ended.\n\t\t\t\t\t// Sometimes however `:` is found inside property value (for example in background image url).\n\t\t\t\t\tif ( !propertyName ) {\n\t\t\t\t\t\t// Treat this as end of property only if property name is not already saved.\n\t\t\t\t\t\t// Save property name.\n\t\t\t\t\t\tpropertyName = stylesString.substr( propertyNameStart, i - propertyNameStart );\n\t\t\t\t\t\t// Save this point as the start of property value.\n\t\t\t\t\t\tpropertyValueStart = i + 1;\n\t\t\t\t\t}\n\n\t\t\t\t\tbreak;\n\n\t\t\t\tcase '\"':\n\t\t\t\tcase '\\'':\n\t\t\t\t\t// Opening quote found (this is an opening quote, because `quoteType` is `null`).\n\t\t\t\t\tquoteType = char;\n\n\t\t\t\t\tbreak;\n\n\t\t\t\tcase ';': {\n\t\t\t\t\t// Property value just ended.\n\t\t\t\t\t// Use previously stored property value start to obtain property value.\n\t\t\t\t\tconst propertyValue = stylesString.substr( propertyValueStart, i - propertyValueStart );\n\n\t\t\t\t\tif ( propertyName ) {\n\t\t\t\t\t\t// Save parsed part.\n\t\t\t\t\t\tstylesMap.set( propertyName.trim(), propertyValue.trim() );\n\t\t\t\t\t}\n\n\t\t\t\t\tpropertyName = null;\n\n\t\t\t\t\t// Save this point as property name start. Property name starts immediately after previous property value ends.\n\t\t\t\t\tpropertyNameStart = i + 1;\n\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t} else if ( char === quoteType ) {\n\t\t\t// If a quote char is found and it is a closing quote, mark this fact by `null`-ing `quoteType`.\n\t\t\tquoteType = null;\n\t\t}\n\t}\n\n\treturn stylesMap;\n}\n\n/**\n * Return lodash compatible path from style name.\n */\nfunction toPath( name: string ): string {\n\treturn name.replace( '-', '.' );\n}\n\n/**\n * Appends style definition to the styles object.\n */\nfunction appendStyleValue( stylesObject: Styles, nameOrPath: string, valueOrObject: StyleValue ) {\n\tlet valueToSet = valueOrObject;\n\n\tif ( isObject( valueOrObject ) ) {\n\t\tvalueToSet = merge( {}, get( stylesObject, nameOrPath ), valueOrObject );\n\t}\n\n\tset( stylesObject, nameOrPath, valueToSet );\n}\n\n/**\n * A CSS style property descriptor that contains tuplet of two strings:\n *\n * - first string describes property name\n * - second string describes property value\n *\n * ```ts\n * const marginDescriptor = [ 'margin', '2px 3em' ];\n * const marginTopDescriptor = [ 'margin-top', '2px' ];\n * ```\n */\nexport type PropertyDescriptor = [ name: string, value: string ];\n\n/**\n * An object describing values associated with the sides of a box, for instance margins, paddings,\n * border widths, border colors, etc.\n *\n * ```ts\n * const margin = {\n * \ttop: '1px',\n * \tright: '3px',\n * \tbottom: '3px',\n * \tleft: '7px'\n * };\n *\n * const borderColor = {\n * \ttop: 'red',\n * \tright: 'blue',\n * \tbottom: 'blue',\n * \tleft: 'red'\n * };\n * ```\n */\nexport type BoxSides = {\n\n\t/**\n\t * Top side value.\n\t */\n\ttop: undefined | string;\n\n\t/**\n\t * Left side value.\n\t */\n\tleft: undefined | string;\n\n\t/**\n\t * Right side value.\n\t */\n\tright: undefined | string;\n\n\t/**\n\t * Bottom side value.\n\t */\n\tbottom: undefined | string;\n};\n\n/**\n * Object holding styles as key-value pairs.\n */\nexport interface Styles {\n\t[ name: string ]: StyleValue;\n}\n\n/**\n * The value of style.\n */\nexport type StyleValue = string | Array<string> | Styles | BoxSides;\n\n/**\n * A normalizer method for a style property.\n *\n * @see ~StylesProcessor#setNormalizer\n */\nexport type Normalizer = ( name: string ) => { path: string; value: StyleValue };\n\n/**\n * An extractor callback for a style property or path string for single values.\n *\n * @see ~StylesProcessor#setExtractor\n */\nexport type Extractor = string | ( ( name: string, styles: Styles ) => StyleValue | undefined );\n\n/**\n * A reducer callback for a style property.\n *\n * @see ~StylesProcessor#setReducer\n */\nexport type Reducer = ( value: StyleValue ) => Array<PropertyDescriptor>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/element\n */\n\nimport Node from './node';\nimport Text from './text';\nimport TextProxy from './textproxy';\nimport { isIterable, toArray, toMap, type ArrayOrItem } from '@ckeditor/ckeditor5-utils';\nimport { default as Matcher, type MatcherPattern } from './matcher';\nimport { default as StylesMap, type StyleValue } from './stylesmap';\n\nimport type Document from './document';\nimport type Item from './item';\n\n// @if CK_DEBUG_ENGINE // const { convertMapToTags } = require( '../dev-utils/utils' );\n\n/**\n * View element.\n *\n * The editing engine does not define a fixed semantics of its elements (it is \"DTD-free\").\n * This is why the type of the {@link module:engine/view/element~Element} need to\n * be defined by the feature developer. When creating an element you should use one of the following methods:\n *\n * * {@link module:engine/view/downcastwriter~DowncastWriter#createContainerElement `downcastWriter#createContainerElement()`}\n * in order to create a {@link module:engine/view/containerelement~ContainerElement},\n * * {@link module:engine/view/downcastwriter~DowncastWriter#createAttributeElement `downcastWriter#createAttributeElement()`}\n * in order to create a {@link module:engine/view/attributeelement~AttributeElement},\n * * {@link module:engine/view/downcastwriter~DowncastWriter#createEmptyElement `downcastWriter#createEmptyElement()`}\n * in order to create a {@link module:engine/view/emptyelement~EmptyElement}.\n * * {@link module:engine/view/downcastwriter~DowncastWriter#createUIElement `downcastWriter#createUIElement()`}\n * in order to create a {@link module:engine/view/uielement~UIElement}.\n * * {@link module:engine/view/downcastwriter~DowncastWriter#createEditableElement `downcastWriter#createEditableElement()`}\n * in order to create a {@link module:engine/view/editableelement~EditableElement}.\n *\n * Note that for view elements which are not created from the model, like elements from mutations, paste or\n * {@link module:engine/controller/datacontroller~DataController#set data.set} it is not possible to define the type of the element.\n * In such cases the {@link module:engine/view/upcastwriter~UpcastWriter#createElement `UpcastWriter#createElement()`} method\n * should be used to create generic view elements.\n */\nexport default class Element extends Node {\n\t/**\n\t * Name of the element.\n\t */\n\tpublic readonly name: string;\n\n\t/**\n\t * A list of attribute names that should be rendered in the editing pipeline even though filtering mechanisms\n\t * implemented in the {@link module:engine/view/domconverter~DomConverter} (for instance,\n\t * {@link module:engine/view/domconverter~DomConverter#shouldRenderAttribute}) would filter them out.\n\t *\n\t * These attributes can be specified as an option when the element is created by\n\t * the {@link module:engine/view/downcastwriter~DowncastWriter}. To check whether an unsafe an attribute should\n\t * be permitted, use the {@link #shouldRenderUnsafeAttribute} method.\n\t *\n\t * @internal\n\t */\n\tpublic readonly _unsafeAttributesToRender: Array<string> = [];\n\n\t/**\n\t * Map of attributes, where attributes names are keys and attributes values are values.\n\t */\n\tprivate readonly _attrs: Map<string, string>;\n\n\t/**\n\t * Array of child nodes.\n\t */\n\tprivate readonly _children: Array<Node>;\n\n\t/**\n\t * Set of classes associated with element instance.\n\t */\n\tprivate readonly _classes: Set<string>;\n\n\t/**\n\t * Normalized styles.\n\t */\n\tprivate readonly _styles: StylesMap;\n\n\t/**\n\t * Map of custom properties.\n\t * Custom properties can be added to element instance, will be cloned but not rendered into DOM.\n\t */\n\tprivate readonly _customProperties = new Map<string | symbol, unknown>();\n\n\t/**\n\t * Creates a view element.\n\t *\n\t * Attributes can be passed in various formats:\n\t *\n\t * ```ts\n\t * new Element( viewDocument, 'div', { class: 'editor', contentEditable: 'true' } ); // object\n\t * new Element( viewDocument, 'div', [ [ 'class', 'editor' ], [ 'contentEditable', 'true' ] ] ); // map-like iterator\n\t * new Element( viewDocument, 'div', mapOfAttributes ); // map\n\t * ```\n\t *\n\t * @internal\n\t * @param document The document instance to which this element belongs.\n\t * @param name Node name.\n\t * @param attrs Collection of attributes.\n\t * @param children A list of nodes to be inserted into created element.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\tname: string,\n\t\tattrs?: ElementAttributes,\n\t\tchildren?: Node | Iterable<Node>\n\t) {\n\t\tsuper( document );\n\n\t\tthis.name = name;\n\n\t\tthis._attrs = parseAttributes( attrs );\n\t\tthis._children = [];\n\n\t\tif ( children ) {\n\t\t\tthis._insertChild( 0, children );\n\t\t}\n\n\t\tthis._classes = new Set();\n\n\t\tif ( this._attrs.has( 'class' ) ) {\n\t\t\t// Remove class attribute and handle it by class set.\n\t\t\tconst classString = this._attrs.get( 'class' );\n\t\t\tparseClasses( this._classes, classString! );\n\t\t\tthis._attrs.delete( 'class' );\n\t\t}\n\n\t\tthis._styles = new StylesMap( this.document.stylesProcessor );\n\n\t\tif ( this._attrs.has( 'style' ) ) {\n\t\t\t// Remove style attribute and handle it by styles map.\n\t\t\tthis._styles.setTo( this._attrs.get( 'style' )! );\n\n\t\t\tthis._attrs.delete( 'style' );\n\t\t}\n\t}\n\n\t/**\n\t * Number of element's children.\n\t */\n\tpublic get childCount(): number {\n\t\treturn this._children.length;\n\t}\n\n\t/**\n\t * Is `true` if there are no nodes inside this element, `false` otherwise.\n\t */\n\tpublic get isEmpty(): boolean {\n\t\treturn this._children.length === 0;\n\t}\n\n\t/**\n\t * Gets child at the given index.\n\t *\n\t * @param index Index of child.\n\t * @returns Child node.\n\t */\n\tpublic getChild( index: number ): Node | undefined {\n\t\treturn this._children[ index ];\n\t}\n\n\t/**\n\t * Gets index of the given child node. Returns `-1` if child node is not found.\n\t *\n\t * @param node Child node.\n\t * @returns Index of the child node.\n\t */\n\tpublic getChildIndex( node: Node ): number {\n\t\treturn this._children.indexOf( node );\n\t}\n\n\t/**\n\t * Gets child nodes iterator.\n\t *\n\t * @returns Child nodes iterator.\n\t */\n\tpublic getChildren(): IterableIterator<Node> {\n\t\treturn this._children[ Symbol.iterator ]();\n\t}\n\n\t/**\n\t * Returns an iterator that contains the keys for attributes. Order of inserting attributes is not preserved.\n\t *\n\t * @returns Keys for attributes.\n\t */\n\tpublic* getAttributeKeys(): IterableIterator<string> {\n\t\tif ( this._classes.size > 0 ) {\n\t\t\tyield 'class';\n\t\t}\n\n\t\tif ( !this._styles.isEmpty ) {\n\t\t\tyield 'style';\n\t\t}\n\n\t\tyield* this._attrs.keys();\n\t}\n\n\t/**\n\t * Returns iterator that iterates over this element's attributes.\n\t *\n\t * Attributes are returned as arrays containing two items. First one is attribute key and second is attribute value.\n\t * This format is accepted by native `Map` object and also can be passed in `Node` constructor.\n\t */\n\tpublic* getAttributes(): IterableIterator<[ string, string ]> {\n\t\tyield* this._attrs.entries();\n\n\t\tif ( this._classes.size > 0 ) {\n\t\t\tyield [ 'class', this.getAttribute( 'class' )! ];\n\t\t}\n\n\t\tif ( !this._styles.isEmpty ) {\n\t\t\tyield [ 'style', this.getAttribute( 'style' )! ];\n\t\t}\n\t}\n\n\t/**\n\t * Gets attribute by key. If attribute is not present - returns undefined.\n\t *\n\t * @param key Attribute key.\n\t * @returns Attribute value.\n\t */\n\tpublic getAttribute( key: string ): string | undefined {\n\t\tif ( key == 'class' ) {\n\t\t\tif ( this._classes.size > 0 ) {\n\t\t\t\treturn [ ...this._classes ].join( ' ' );\n\t\t\t}\n\n\t\t\treturn undefined;\n\t\t}\n\n\t\tif ( key == 'style' ) {\n\t\t\tconst inlineStyle = this._styles.toString();\n\n\t\t\treturn inlineStyle == '' ? undefined : inlineStyle;\n\t\t}\n\n\t\treturn this._attrs.get( key );\n\t}\n\n\t/**\n\t * Returns a boolean indicating whether an attribute with the specified key exists in the element.\n\t *\n\t * @param key Attribute key.\n\t * @returns `true` if attribute with the specified key exists in the element, `false` otherwise.\n\t */\n\tpublic hasAttribute( key: string ): boolean {\n\t\tif ( key == 'class' ) {\n\t\t\treturn this._classes.size > 0;\n\t\t}\n\n\t\tif ( key == 'style' ) {\n\t\t\treturn !this._styles.isEmpty;\n\t\t}\n\n\t\treturn this._attrs.has( key );\n\t}\n\n\t/**\n\t * Checks if this element is similar to other element.\n\t * Both elements should have the same name and attributes to be considered as similar. Two similar elements\n\t * can contain different set of children nodes.\n\t */\n\tpublic isSimilar( otherElement: Item ): boolean {\n\t\tif ( !( otherElement instanceof Element ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// If exactly the same Element is provided - return true immediately.\n\t\tif ( this === otherElement ) {\n\t\t\treturn true;\n\t\t}\n\n\t\t// Check element name.\n\t\tif ( this.name != otherElement.name ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Check number of attributes, classes and styles.\n\t\tif ( this._attrs.size !== otherElement._attrs.size || this._classes.size !== otherElement._classes.size ||\n\t\t\tthis._styles.size !== otherElement._styles.size ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Check if attributes are the same.\n\t\tfor ( const [ key, value ] of this._attrs ) {\n\t\t\tif ( !otherElement._attrs.has( key ) || otherElement._attrs.get( key ) !== value ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\t// Check if classes are the same.\n\t\tfor ( const className of this._classes ) {\n\t\t\tif ( !otherElement._classes.has( className ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\t// Check if styles are the same.\n\t\tfor ( const property of this._styles.getStyleNames() ) {\n\t\t\tif (\n\t\t\t\t!otherElement._styles.has( property ) ||\n\t\t\t\totherElement._styles.getAsString( property ) !== this._styles.getAsString( property )\n\t\t\t) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Returns true if class is present.\n\t * If more then one class is provided - returns true only when all classes are present.\n\t *\n\t * ```ts\n\t * element.hasClass( 'foo' ); // Returns true if 'foo' class is present.\n\t * element.hasClass( 'foo', 'bar' ); // Returns true if 'foo' and 'bar' classes are both present.\n\t * ```\n\t */\n\tpublic hasClass( ...className: Array<string> ): boolean {\n\t\tfor ( const name of className ) {\n\t\t\tif ( !this._classes.has( name ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Returns iterator that contains all class names.\n\t */\n\tpublic getClassNames(): Iterable<string> {\n\t\treturn this._classes.keys();\n\t}\n\n\t/**\n\t * Returns style value for the given property mae.\n\t * If the style does not exist `undefined` is returned.\n\t *\n\t * **Note**: This method can work with normalized style names if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#getAsString `StylesMap#getAsString()`} for details.\n\t *\n\t * For an element with style set to `'margin:1px'`:\n\t *\n\t * ```ts\n\t * // Enable 'margin' shorthand processing:\n\t * editor.data.addStyleProcessorRules( addMarginRules );\n\t *\n\t * const element = view.change( writer => {\n\t * \tconst element = writer.createElement();\n\t * \twriter.setStyle( 'margin', '1px' );\n\t * \twriter.setStyle( 'margin-bottom', '3em' );\n\t *\n\t * \treturn element;\n\t * } );\n\t *\n\t * element.getStyle( 'margin' ); // -> 'margin: 1px 1px 3em;'\n\t * ```\n\t */\n\tpublic getStyle( property: string ): string | undefined {\n\t\treturn this._styles.getAsString( property );\n\t}\n\n\t/**\n\t * Returns a normalized style object or single style value.\n\t *\n\t * For an element with style set to: margin:1px 2px 3em;\n\t *\n\t * ```ts\n\t * element.getNormalizedStyle( 'margin' ) );\n\t * ```\n\t *\n\t * will return:\n\t *\n\t * ```ts\n\t * {\n\t * \ttop: '1px',\n\t * \tright: '2px',\n\t * \tbottom: '3em',\n\t * \tleft: '2px' // a normalized value from margin shorthand\n\t * }\n\t * ```\n\t *\n\t * and reading for single style value:\n\t *\n\t * ```ts\n\t * styles.getNormalizedStyle( 'margin-left' );\n\t * ```\n\t *\n\t * Will return a `2px` string.\n\t *\n\t * **Note**: This method will return normalized values only if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#getNormalized `StylesMap#getNormalized()`} for details.\n\t *\n\t * @param property Name of CSS property\n\t */\n\tpublic getNormalizedStyle( property: string ): StyleValue {\n\t\treturn this._styles.getNormalized( property );\n\t}\n\n\t/**\n\t * Returns iterator that contains all style names.\n\t *\n\t * @param expand Expand shorthand style properties and return all equivalent style representations.\n\t */\n\tpublic getStyleNames( expand?: boolean ): Iterable<string> {\n\t\treturn this._styles.getStyleNames( expand );\n\t}\n\n\t/**\n\t * Returns true if style keys are present.\n\t * If more then one style property is provided - returns true only when all properties are present.\n\t *\n\t * ```ts\n\t * element.hasStyle( 'color' ); // Returns true if 'border-top' style is present.\n\t * element.hasStyle( 'color', 'border-top' ); // Returns true if 'color' and 'border-top' styles are both present.\n\t * ```\n\t */\n\tpublic hasStyle( ...property: Array<string> ): boolean {\n\t\tfor ( const name of property ) {\n\t\t\tif ( !this._styles.has( name ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Returns ancestor element that match specified pattern.\n\t * Provided patterns should be compatible with {@link module:engine/view/matcher~Matcher Matcher} as it is used internally.\n\t *\n\t * @see module:engine/view/matcher~Matcher\n\t * @param patterns Patterns used to match correct ancestor. See {@link module:engine/view/matcher~Matcher}.\n\t * @returns Found element or `null` if no matching ancestor was found.\n\t */\n\tpublic findAncestor( ...patterns: Array<MatcherPattern | ( ( element: Element ) => boolean )> ): Element | null {\n\t\tconst matcher = new Matcher( ...patterns as any );\n\t\tlet parent = this.parent;\n\n\t\twhile ( parent && !parent.is( 'documentFragment' ) ) {\n\t\t\tif ( matcher.match( parent ) ) {\n\t\t\t\treturn parent;\n\t\t\t}\n\n\t\t\tparent = parent.parent;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns the custom property value for the given key.\n\t */\n\tpublic getCustomProperty( key: string | symbol ): unknown {\n\t\treturn this._customProperties.get( key );\n\t}\n\n\t/**\n\t * Returns an iterator which iterates over this element's custom properties.\n\t * Iterator provides `[ key, value ]` pairs for each stored property.\n\t */\n\tpublic* getCustomProperties(): IterableIterator<[ string | symbol, unknown ]> {\n\t\tyield* this._customProperties.entries();\n\t}\n\n\t/**\n\t * Returns identity string based on element's name, styles, classes and other attributes.\n\t * Two elements that {@link #isSimilar are similar} will have same identity string.\n\t * It has the following format:\n\t *\n\t * ```ts\n\t * 'name class=\"class1,class2\" style=\"style1:value1;style2:value2\" attr1=\"val1\" attr2=\"val2\"'\n\t * ```\n \t *\n\t * For example:\n\t *\n\t * ```ts\n\t * const element = writer.createContainerElement( 'foo', {\n\t * \tbanana: '10',\n\t * \tapple: '20',\n\t * \tstyle: 'color: red; border-color: white;',\n\t * \tclass: 'baz'\n\t * } );\n\t *\n\t * // returns 'foo class=\"baz\" style=\"border-color:white;color:red\" apple=\"20\" banana=\"10\"'\n\t * element.getIdentity();\n\t * ```\n\t *\n\t * **Note**: Classes, styles and other attributes are sorted alphabetically.\n\t */\n\tpublic getIdentity(): string {\n\t\tconst classes = Array.from( this._classes ).sort().join( ',' );\n\t\tconst styles = this._styles.toString();\n\t\tconst attributes = Array.from( this._attrs ).map( i => `${ i[ 0 ] }=\"${ i[ 1 ] }\"` ).sort().join( ' ' );\n\n\t\treturn this.name +\n\t\t\t( classes == '' ? '' : ` class=\"${ classes }\"` ) +\n\t\t\t( !styles ? '' : ` style=\"${ styles }\"` ) +\n\t\t\t( attributes == '' ? '' : ` ${ attributes }` );\n\t}\n\n\t/**\n\t * Decides whether an unsafe attribute is whitelisted and should be rendered in the editing pipeline even though filtering mechanisms\n\t * like {@link module:engine/view/domconverter~DomConverter#shouldRenderAttribute} say it should not.\n\t *\n\t * Unsafe attribute names can be specified when creating an element via {@link module:engine/view/downcastwriter~DowncastWriter}.\n\t *\n\t * @param attributeName The name of the attribute to be checked.\n\t */\n\tpublic shouldRenderUnsafeAttribute( attributeName: string ): boolean {\n\t\treturn this._unsafeAttributesToRender.includes( attributeName );\n\t}\n\n\t/**\n\t * Clones provided element.\n\t *\n\t * @internal\n\t * @param deep If set to `true` clones element and all its children recursively. When set to `false`,\n\t * element will be cloned without any children.\n\t * @returns Clone of this element.\n\t */\n\tpublic _clone( deep = false ): this {\n\t\tconst childrenClone: Array<Node> = [];\n\n\t\tif ( deep ) {\n\t\t\tfor ( const child of this.getChildren() ) {\n\t\t\t\tchildrenClone.push( child._clone( deep ) );\n\t\t\t}\n\t\t}\n\n\t\t// ContainerElement and AttributeElement should be also cloned properly.\n\t\tconst cloned = new ( this.constructor as any )( this.document, this.name, this._attrs, childrenClone );\n\n\t\t// Classes and styles are cloned separately - this solution is faster than adding them back to attributes and\n\t\t// parse once again in constructor.\n\t\tcloned._classes = new Set( this._classes );\n\t\tcloned._styles.set( this._styles.getNormalized() );\n\n\t\t// Clone custom properties.\n\t\tcloned._customProperties = new Map( this._customProperties );\n\n\t\t// Clone filler offset method.\n\t\t// We can't define this method in a prototype because it's behavior which\n\t\t// is changed by e.g. toWidget() function from ckeditor5-widget. Perhaps this should be one of custom props.\n\t\tcloned.getFillerOffset = this.getFillerOffset;\n\n\t\t// Clone unsafe attributes list.\n\t\tcloned._unsafeAttributesToRender = this._unsafeAttributesToRender;\n\n\t\treturn cloned;\n\t}\n\n\t/**\n\t * {@link module:engine/view/element~Element#_insertChild Insert} a child node or a list of child nodes at the end of this node\n\t * and sets the parent of these nodes to this element.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#insert\n\t * @internal\n\t * @param items Items to be inserted.\n\t * @fires change\n\t * @returns Number of appended nodes.\n\t */\n\tpublic _appendChild( items: Item | string | Iterable<Item | string> ): number {\n\t\treturn this._insertChild( this.childCount, items );\n\t}\n\n\t/**\n\t * Inserts a child node or a list of child nodes on the given index and sets the parent of these nodes to\n\t * this element.\n\t *\n\t * @internal\n\t * @see module:engine/view/downcastwriter~DowncastWriter#insert\n\t * @param index Position where nodes should be inserted.\n\t * @param items Items to be inserted.\n\t * @fires change\n\t * @returns Number of inserted nodes.\n\t */\n\tpublic _insertChild( index: number, items: Item | string | Iterable<Item | string> ): number {\n\t\tthis._fireChange( 'children', this );\n\t\tlet count = 0;\n\n\t\tconst nodes = normalize( this.document, items );\n\n\t\tfor ( const node of nodes ) {\n\t\t\t// If node that is being added to this element is already inside another element, first remove it from the old parent.\n\t\t\tif ( node.parent !== null ) {\n\t\t\t\tnode._remove();\n\t\t\t}\n\n\t\t\t( node as any ).parent = this;\n\t\t\t( node as any ).document = this.document;\n\n\t\t\tthis._children.splice( index, 0, node );\n\t\t\tindex++;\n\t\t\tcount++;\n\t\t}\n\n\t\treturn count;\n\t}\n\n\t/**\n\t * Removes number of child nodes starting at the given index and set the parent of these nodes to `null`.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#remove\n\t * @internal\n\t * @param index Number of the first node to remove.\n\t * @param howMany Number of nodes to remove.\n\t * @fires change\n\t * @returns The array of removed nodes.\n\t */\n\tpublic _removeChildren( index: number, howMany: number = 1 ): Array<Node> {\n\t\tthis._fireChange( 'children', this );\n\n\t\tfor ( let i = index; i < index + howMany; i++ ) {\n\t\t\t( this._children[ i ] as any ).parent = null;\n\t\t}\n\n\t\treturn this._children.splice( index, howMany );\n\t}\n\n\t/**\n\t * Adds or overwrite attribute with a specified key and value.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#setAttribute\n\t * @internal\n\t * @param key Attribute key.\n\t * @param value Attribute value.\n\t * @fires change\n\t */\n\tpublic _setAttribute( key: string, value: unknown ): void {\n\t\tconst stringValue = String( value );\n\n\t\tthis._fireChange( 'attributes', this );\n\n\t\tif ( key == 'class' ) {\n\t\t\tparseClasses( this._classes, stringValue );\n\t\t} else if ( key == 'style' ) {\n\t\t\tthis._styles.setTo( stringValue );\n\t\t} else {\n\t\t\tthis._attrs.set( key, stringValue );\n\t\t}\n\t}\n\n\t/**\n\t * Removes attribute from the element.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#removeAttribute\n\t * @internal\n\t * @param key Attribute key.\n\t * @returns Returns true if an attribute existed and has been removed.\n\t * @fires change\n\t */\n\tpublic _removeAttribute( key: string ): boolean {\n\t\tthis._fireChange( 'attributes', this );\n\n\t\t// Remove class attribute.\n\t\tif ( key == 'class' ) {\n\t\t\tif ( this._classes.size > 0 ) {\n\t\t\t\tthis._classes.clear();\n\n\t\t\t\treturn true;\n\t\t\t}\n\n\t\t\treturn false;\n\t\t}\n\n\t\t// Remove style attribute.\n\t\tif ( key == 'style' ) {\n\t\t\tif ( !this._styles.isEmpty ) {\n\t\t\t\tthis._styles.clear();\n\n\t\t\t\treturn true;\n\t\t\t}\n\n\t\t\treturn false;\n\t\t}\n\n\t\t// Remove other attributes.\n\t\treturn this._attrs.delete( key );\n\t}\n\n\t/**\n\t * Adds specified class.\n\t *\n\t * ```ts\n\t * element._addClass( 'foo' ); // Adds 'foo' class.\n\t * element._addClass( [ 'foo', 'bar' ] ); // Adds 'foo' and 'bar' classes.\n\t * ```\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#addClass\n\t * @internal\n\t * @fires change\n\t */\n\tpublic _addClass( className: ArrayOrItem<string> ): void {\n\t\tthis._fireChange( 'attributes', this );\n\n\t\tfor ( const name of toArray( className ) ) {\n\t\t\tthis._classes.add( name );\n\t\t}\n\t}\n\n\t/**\n\t * Removes specified class.\n\t *\n\t * ```ts\n\t * element._removeClass( 'foo' ); // Removes 'foo' class.\n\t * element._removeClass( [ 'foo', 'bar' ] ); // Removes both 'foo' and 'bar' classes.\n\t * ```\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#removeClass\n\t * @internal\n\t * @fires change\n\t */\n\tpublic _removeClass( className: string | Array<string> ): void {\n\t\tthis._fireChange( 'attributes', this );\n\n\t\tfor ( const name of toArray( className ) ) {\n\t\t\tthis._classes.delete( name );\n\t\t}\n\t}\n\n\t/**\n\t * Adds style to the element.\n\t *\n\t * ```ts\n\t * element._setStyle( 'color', 'red' );\n\t * ```\n\t *\n\t * **Note**: This method can work with normalized style names if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#set `StylesMap#set()`} for details.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#setStyle\n\t * @label KEY_VALUE\n\t * @internal\n\t * @param property Property name.\n\t * @param value Value to set.\n\t * @fires change\n\t */\n\tpublic _setStyle( property: string, value: string ): void;\n\n\t/**\n\t * Adds style to the element.\n\t *\n\t * ```ts\n\t * element._setStyle( {\n\t * \tcolor: 'red',\n\t * \tposition: 'fixed'\n\t * } );\n\t * ```\n\t *\n\t * **Note**: This method can work with normalized style names if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#set `StylesMap#set()`} for details.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#setStyle\n\t * @label OBJECT\n\t * @internal\n\t * @param properties Object with key - value pairs.\n\t * @fires change\n\t */\n\tpublic _setStyle( properties: Record<string, string> ): void;\n\n\tpublic _setStyle( property: string | Record<string, string>, value?: string ): void {\n\t\tthis._fireChange( 'attributes', this );\n\n\t\tif ( typeof property != 'string' ) {\n\t\t\tthis._styles.set( property );\n\t\t} else {\n\t\t\tthis._styles.set( property, value! );\n\t\t}\n\t}\n\n\t/**\n\t * Removes specified style.\n\t *\n\t * ```ts\n\t * element._removeStyle( 'color' ); // Removes 'color' style.\n\t * element._removeStyle( [ 'color', 'border-top' ] ); // Removes both 'color' and 'border-top' styles.\n\t * ```\n\t *\n\t * **Note**: This method can work with normalized style names if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#remove `StylesMap#remove()`} for details.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#removeStyle\n\t * @internal\n\t * @fires change\n\t */\n\tpublic _removeStyle( property: ArrayOrItem<string> ): void {\n\t\tthis._fireChange( 'attributes', this );\n\n\t\tfor ( const name of toArray( property ) ) {\n\t\t\tthis._styles.remove( name );\n\t\t}\n\t}\n\n\t/**\n\t * Sets a custom property. Unlike attributes, custom properties are not rendered to the DOM,\n\t * so they can be used to add special data to elements.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#setCustomProperty\n\t * @internal\n\t */\n\tpublic _setCustomProperty( key: string | symbol, value: unknown ): void {\n\t\tthis._customProperties.set( key, value );\n\t}\n\n\t/**\n\t * Removes the custom property stored under the given key.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#removeCustomProperty\n\t * @internal\n\t * @returns Returns true if property was removed.\n\t */\n\tpublic _removeCustomProperty( key: string | symbol ): boolean {\n\t\treturn this._customProperties.delete( key );\n\t}\n\n\t/**\n\t * Returns block {@link module:engine/view/filler filler} offset or `null` if block filler is not needed.\n\t */\n\tpublic getFillerOffset?(): number | null;\n\n\t// @if CK_DEBUG_ENGINE // public printTree( level = 0 ): string {\n\t// @if CK_DEBUG_ENGINE // \tlet string = '';\n\n\t// @if CK_DEBUG_ENGINE // \tstring += '\\t'.repeat( level ) + `<${ this.name }${ convertMapToTags( this.getAttributes() ) }>`;\n\n\t// @if CK_DEBUG_ENGINE // \tfor ( const child of this.getChildren() as any ) {\n\t// @if CK_DEBUG_ENGINE // \t\tif ( child.is( '$text' ) ) {\n\t// @if CK_DEBUG_ENGINE // \t\t\tstring += '\\n' + '\\t'.repeat( level + 1 ) + child.data;\n\t// @if CK_DEBUG_ENGINE // \t\t} else {\n\t// @if CK_DEBUG_ENGINE // \t\t\tstring += '\\n' + child.printTree( level + 1 );\n\t// @if CK_DEBUG_ENGINE // \t\t}\n\t// @if CK_DEBUG_ENGINE // \t}\n\n\t// @if CK_DEBUG_ENGINE // \tif ( this.childCount ) {\n\t// @if CK_DEBUG_ENGINE // \t\tstring += '\\n' + '\\t'.repeat( level );\n\t// @if CK_DEBUG_ENGINE // \t}\n\n\t// @if CK_DEBUG_ENGINE // \tstring += `</${ this.name }>`;\n\n\t// @if CK_DEBUG_ENGINE // \treturn string;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logTree(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( this.printTree() );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'element' || type === 'view:element' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'node' || type === 'view:node';\n\t} else {\n\t\treturn name === this.name && ( type === 'element' || type === 'view:element' );\n\t}\n};\n\n/**\n * Collection of attributes.\n */\nexport type ElementAttributes = Record<string, unknown> | Iterable<[ string, unknown ]> | null;\n\n/**\n * Parses attributes provided to the element constructor before they are applied to an element. If attributes are passed\n * as an object (instead of `Iterable`), the object is transformed to the map. Attributes with `null` value are removed.\n * Attributes with non-`String` value are converted to `String`.\n *\n * @param attrs Attributes to parse.\n * @returns Parsed attributes.\n */\nfunction parseAttributes( attrs?: ElementAttributes ) {\n\tconst attrsMap = toMap( attrs );\n\n\tfor ( const [ key, value ] of attrsMap ) {\n\t\tif ( value === null ) {\n\t\t\tattrsMap.delete( key );\n\t\t} else if ( typeof value != 'string' ) {\n\t\t\tattrsMap.set( key, String( value ) );\n\t\t}\n\t}\n\n\treturn attrsMap as Map<string, string>;\n}\n\n/**\n * Parses class attribute and puts all classes into classes set.\n * Classes set s cleared before insertion.\n *\n * @param classesSet Set to insert parsed classes.\n * @param classesString String with classes to parse.\n */\nfunction parseClasses( classesSet: Set<string>, classesString: string ) {\n\tconst classArray = classesString.split( /\\s+/ );\n\tclassesSet.clear();\n\tclassArray.forEach( name => classesSet.add( name ) );\n}\n\n/**\n * Converts strings to Text and non-iterables to arrays.\n */\nfunction normalize( document: Document, nodes: string | Item | Iterable<string | Item> ): Array<Node> {\n\t// Separate condition because string is iterable.\n\tif ( typeof nodes == 'string' ) {\n\t\treturn [ new Text( document, nodes ) ];\n\t}\n\n\tif ( !isIterable( nodes ) ) {\n\t\tnodes = [ nodes ];\n\t}\n\n\t// Array.from to enable .map() on non-arrays.\n\treturn Array.from( nodes )\n\t\t.map( node => {\n\t\t\tif ( typeof node == 'string' ) {\n\t\t\t\treturn new Text( document, node );\n\t\t\t}\n\n\t\t\tif ( node instanceof TextProxy ) {\n\t\t\t\treturn new Text( document, node.data );\n\t\t\t}\n\n\t\t\treturn node;\n\t\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/containerelement\n */\n\nimport Element, { type ElementAttributes } from './element';\nimport type Document from './document';\nimport type Node from './node';\n\n/**\n * Containers are elements which define document structure. They define boundaries for\n * {@link module:engine/view/attributeelement~AttributeElement attributes}. They are mostly used for block elements like `<p>` or `<div>`.\n *\n * Editing engine does not define a fixed HTML DTD. This is why a feature developer needs to choose between various\n * types (container element, {@link module:engine/view/attributeelement~AttributeElement attribute element},\n * {@link module:engine/view/emptyelement~EmptyElement empty element}, etc) when developing a feature.\n *\n * The container element should be your default choice when writing a converter, unless:\n *\n * * this element represents a model text attribute (then use {@link module:engine/view/attributeelement~AttributeElement}),\n * * this is an empty element like `<img>` (then use {@link module:engine/view/emptyelement~EmptyElement}),\n * * this is a root element,\n * * this is a nested editable element (then use {@link module:engine/view/editableelement~EditableElement}).\n *\n * To create a new container element instance use the\n * {@link module:engine/view/downcastwriter~DowncastWriter#createContainerElement `DowncastWriter#createContainerElement()`}\n * method.\n */\nexport default class ContainerElement extends Element {\n\t/**\n\t * Creates a container element.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#createContainerElement\n\t * @see module:engine/view/element~Element\n\t * @internal\n\t * @param document The document instance to which this element belongs.\n\t * @param name Node name.\n\t * @param attrs Collection of attributes.\n\t * @param children A list of nodes to be inserted into created element.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\tname: string,\n\t\tattrs?: ElementAttributes,\n\t\tchildren?: Node | Iterable<Node>\n\t) {\n\t\tsuper( document, name, attrs, children );\n\n\t\tthis.getFillerOffset = getFillerOffset;\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nContainerElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'containerElement' || type === 'view:containerElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'element' || type === 'view:element' ||\n\t\t\ttype === 'node' || type === 'view:node';\n\t} else {\n\t\treturn name === this.name && (\n\t\t\ttype === 'containerElement' || type === 'view:containerElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'element' || type === 'view:element'\n\t\t);\n\t}\n};\n\n/**\n * Returns block {@link module:engine/view/filler filler} offset or `null` if block filler is not needed.\n *\n * @returns Block filler offset or `null` if block filler is not needed.\n */\nexport function getFillerOffset( this: ContainerElement ): number | null {\n\tconst children = [ ...this.getChildren() ];\n\tconst lastChild = children[ this.childCount - 1 ];\n\n\t// Block filler is required after a `<br>` if it's the last element in its container. See #1422.\n\tif ( lastChild && lastChild.is( 'element', 'br' ) ) {\n\t\treturn this.childCount;\n\t}\n\n\tfor ( const child of children ) {\n\t\t// If there's any non-UI element don't render the bogus.\n\t\tif ( !child.is( 'uiElement' ) ) {\n\t\t\treturn null;\n\t\t}\n\t}\n\n\t// If there are only UI elements render the bogus at the end of the element.\n\treturn this.childCount;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/editableelement\n */\n\nimport ContainerElement from './containerelement';\nimport { ObservableMixin } from '@ckeditor/ckeditor5-utils';\nimport type { ViewSelectionChangeEvent } from './selection';\nimport type { ElementAttributes } from './element';\nimport type Document from './document';\nimport type Node from './node';\n\n/**\n * Editable element which can be a {@link module:engine/view/rooteditableelement~RootEditableElement root}\n * or nested editable area in the editor.\n *\n * Editable is automatically read-only when its {@link module:engine/view/document~Document Document} is read-only.\n *\n * The constructor of this class shouldn't be used directly. To create new `EditableElement` use the\n * {@link module:engine/view/downcastwriter~DowncastWriter#createEditableElement `downcastWriter#createEditableElement()`} method.\n */\nexport default class EditableElement extends ObservableMixin( ContainerElement ) {\n\t/**\n\t * Whether the editable is in read-write or read-only mode.\n\t *\n\t * @observable\n\t */\n\tdeclare public isReadOnly: boolean;\n\n\t/**\n\t * Whether the editable is focused.\n\t *\n\t * This property updates when {@link module:engine/view/document~Document#isFocused document.isFocused} or view\n\t * selection is changed.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public isFocused: boolean;\n\n\t/**\n\t * Placeholder of editable element.\n\t *\n\t * ```ts\n\t * editor.editing.view.document.getRoot( 'main' ).placeholder = 'New placeholder';\n\t * ```\n\t *\n\t * @observable\n\t */\n\tdeclare public placeholder?: string;\n\n\t/**\n\t * Creates an editable element.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#createEditableElement\n\t * @internal\n\t * @param document The document instance to which this element belongs.\n\t * @param name Node name.\n\t * @param attrs Collection of attributes.\n\t * @param children A list of nodes to be inserted into created element.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\tchildren?: Node | Iterable<Node>\n\t) {\n\t\tsuper( document, name, attributes, children );\n\n\t\tthis.set( 'isReadOnly', false );\n\t\tthis.set( 'isFocused', false );\n\t\tthis.set( 'placeholder', undefined );\n\n\t\tthis.bind( 'isReadOnly' ).to( document );\n\n\t\tthis.bind( 'isFocused' ).to(\n\t\t\tdocument,\n\t\t\t'isFocused',\n\t\t\tisFocused => isFocused && document.selection.editableElement == this\n\t\t);\n\n\t\t// Update focus state based on selection changes.\n\t\tthis.listenTo<ViewSelectionChangeEvent>( document.selection, 'change', () => {\n\t\t\tthis.isFocused = document.isFocused && document.selection.editableElement == this;\n\t\t} );\n\t}\n\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nEditableElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'editableElement' || type === 'view:editableElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'containerElement' || type === 'view:containerElement' ||\n\t\t\ttype === 'element' || type === 'view:element' ||\n\t\t\ttype === 'node' || type === 'view:node';\n\t} else {\n\t\treturn name === this.name && (\n\t\t\ttype === 'editableElement' || type === 'view:editableElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'containerElement' || type === 'view:containerElement' ||\n\t\t\ttype === 'element' || type === 'view:element'\n\t\t);\n\t}\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/rooteditableelement\n */\n\nimport EditableElement from './editableelement';\n\nimport type Document from './document';\n\nconst rootNameSymbol = Symbol( 'rootName' );\n\n/**\n * Class representing a single root in the data view. A root can be either {@link ~RootEditableElement#isReadOnly editable or read-only},\n * but in both cases it is called \"an editable\". Roots can contain other {@link module:engine/view/editableelement~EditableElement\n * editable elements} making them \"nested editables\".\n */\nexport default class RootEditableElement extends EditableElement {\n\t/**\n\t * Creates root editable element.\n\t *\n\t * @param document The document instance to which this element belongs.\n\t * @param name Node name.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\tname: string\n\t) {\n\t\tsuper( document, name );\n\n\t\tthis.rootName = 'main';\n\t}\n\n\t/**\n\t * Name of this root inside {@link module:engine/view/document~Document} that is an owner of this root. If no\n\t * other name is set, `main` name is used.\n\t *\n\t * @readonly\n\t */\n\tpublic get rootName(): string {\n\t\treturn this.getCustomProperty( rootNameSymbol ) as string;\n\t}\n\n\tpublic set rootName( rootName: string ) {\n\t\tthis._setCustomProperty( rootNameSymbol, rootName );\n\t}\n\n\t/**\n\t * Overrides old element name and sets new one.\n\t * This is needed because view roots are created before they are attached to the DOM.\n\t * The name of the root element is temporary at this stage. It has to be changed when the\n\t * view root element is attached to the DOM element.\n\t *\n\t * @internal\n\t * @param name The new name of element.\n\t */\n\tpublic set _name( name: string ) {\n\t\t( this as any ).name = name;\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nRootEditableElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'rootElement' || type === 'view:rootElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'editableElement' || type === 'view:editableElement' ||\n\t\t\ttype === 'containerElement' || type === 'view:containerElement' ||\n\t\t\ttype === 'element' || type === 'view:element' ||\n\t\t\ttype === 'node' || type === 'view:node';\n\t} else {\n\t\treturn name === this.name && (\n\t\t\ttype === 'rootElement' || type === 'view:rootElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'editableElement' || type === 'view:editableElement' ||\n\t\t\ttype === 'containerElement' || type === 'view:containerElement' ||\n\t\t\ttype === 'element' || type === 'view:element'\n\t\t);\n\t}\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/treewalker\n */\n\nimport Element from './element';\nimport Text from './text';\nimport TextProxy from './textproxy';\nimport Position from './position';\nimport type Item from './item';\nimport type DocumentFragment from './documentfragment';\nimport type Range from './range';\nimport type Node from './node';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Position iterator class. It allows to iterate forward and backward over the document.\n */\nexport default class TreeWalker implements IterableIterator<TreeWalkerValue> {\n\t/**\n\t * Walking direction. Defaults `'forward'`.\n\t */\n\tpublic readonly direction: TreeWalkerDirection;\n\n\t/**\n\t * Iterator boundaries.\n\t *\n\t * When the iterator is walking `'forward'` on the end of boundary or is walking `'backward'`\n\t * on the start of boundary, then `{ done: true }` is returned.\n\t *\n\t * If boundaries are not defined they are set before first and after last child of the root node.\n\t */\n\tpublic readonly boundaries: Range | null;\n\n\t/**\n\t * Flag indicating whether all characters from {@link module:engine/view/text~Text} should be returned as one\n\t * {@link module:engine/view/text~Text} or one by one as {@link module:engine/view/textproxy~TextProxy}.\n\t */\n\tpublic readonly singleCharacters: boolean;\n\n\t/**\n\t * Flag indicating whether iterator should enter elements or not. If the iterator is shallow child nodes of any\n\t * iterated node will not be returned along with `elementEnd` tag.\n\t */\n\tpublic readonly shallow: boolean;\n\n\t/**\n\t * Flag indicating whether iterator should ignore `elementEnd` tags. If set to `true`, walker will not\n\t * return a parent node of the start position. Each {@link module:engine/view/element~Element} will be returned once.\n\t * When set to `false` each element might be returned twice: for `'elementStart'` and `'elementEnd'`.\n\t */\n\tpublic readonly ignoreElementEnd: boolean;\n\n\t/**\n\t * Iterator position. If start position is not defined then position depends on {@link #direction}. If direction is\n\t * `'forward'` position starts form the beginning, when direction is `'backward'` position starts from the end.\n\t */\n\tprivate _position: Position;\n\n\t/**\n\t * Start boundary parent.\n\t */\n\tprivate readonly _boundaryStartParent: Node | DocumentFragment | null;\n\n\t/**\n\t * End boundary parent.\n\t */\n\tprivate readonly _boundaryEndParent: Node | DocumentFragment | null;\n\n\t/**\n\t * Creates a range iterator. All parameters are optional, but you have to specify either `boundaries` or `startPosition`.\n\t *\n\t * @param options Object with configuration.\n\t */\n\tconstructor( options: TreeWalkerOptions = {} ) {\n\t\tif ( !options.boundaries && !options.startPosition ) {\n\t\t\t/**\n\t\t\t * Neither boundaries nor starting position have been defined.\n\t\t\t *\n\t\t\t * @error view-tree-walker-no-start-position\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'view-tree-walker-no-start-position',\n\t\t\t\tnull\n\t\t\t);\n\t\t}\n\n\t\tif ( options.direction && options.direction != 'forward' && options.direction != 'backward' ) {\n\t\t\t/**\n\t\t\t * Only `backward` and `forward` direction allowed.\n\t\t\t *\n\t\t\t * @error view-tree-walker-unknown-direction\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-tree-walker-unknown-direction', options.startPosition, { direction: options.direction } );\n\t\t}\n\n\t\tthis.boundaries = options.boundaries || null;\n\n\t\tif ( options.startPosition ) {\n\t\t\tthis._position = Position._createAt( options.startPosition );\n\t\t} else {\n\t\t\tthis._position = Position._createAt( options.boundaries![ options.direction == 'backward' ? 'end' : 'start' ] );\n\t\t}\n\n\t\tthis.direction = options.direction || 'forward';\n\t\tthis.singleCharacters = !!options.singleCharacters;\n\t\tthis.shallow = !!options.shallow;\n\t\tthis.ignoreElementEnd = !!options.ignoreElementEnd;\n\n\t\tthis._boundaryStartParent = this.boundaries ? this.boundaries.start.parent : null;\n\t\tthis._boundaryEndParent = this.boundaries ? this.boundaries.end.parent : null;\n\t}\n\n\t/**\n\t * Iterable interface.\n\t */\n\tpublic [ Symbol.iterator ](): IterableIterator<TreeWalkerValue> {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Iterator position. If start position is not defined then position depends on {@link #direction}. If direction is\n\t * `'forward'` position starts form the beginning, when direction is `'backward'` position starts from the end.\n\t */\n\tpublic get position(): Position {\n\t\treturn this._position;\n\t}\n\n\t/**\n\t * Moves {@link #position} in the {@link #direction} skipping values as long as the callback function returns `true`.\n\t *\n\t * For example:\n\t *\n\t * ```ts\n\t * walker.skip( value => value.type == 'text' ); // <p>{}foo</p> -> <p>foo[]</p>\n\t * walker.skip( value => true ); // Move the position to the end: <p>{}foo</p> -> <p>foo</p>[]\n\t * walker.skip( value => false ); // Do not move the position.\n\t * ```\n\t *\n\t * @param skip Callback function. Gets {@link module:engine/view/treewalker~TreeWalkerValue} and should\n\t * return `true` if the value should be skipped or `false` if not.\n\t */\n\tpublic skip( skip: ( value: TreeWalkerValue ) => boolean ): void {\n\t\tlet nextResult: IteratorResult<TreeWalkerValue>;\n\t\tlet prevPosition: Position;\n\n\t\tdo {\n\t\t\tprevPosition = this.position;\n\t\t\tnextResult = this.next();\n\t\t} while ( !nextResult.done && skip( nextResult.value ) );\n\n\t\tif ( !nextResult.done ) {\n\t\t\tthis._position = prevPosition;\n\t\t}\n\t}\n\n\t/**\n\t * Gets the next tree walker's value.\n\t *\n\t * @returns Object implementing iterator interface, returning\n\t * information about taken step.\n\t */\n\tpublic next(): IteratorResult<TreeWalkerValue, undefined> {\n\t\tif ( this.direction == 'forward' ) {\n\t\t\treturn this._next();\n\t\t} else {\n\t\t\treturn this._previous();\n\t\t}\n\t}\n\n\t/**\n\t * Makes a step forward in view. Moves the {@link #position} to the next position and returns the encountered value.\n\t */\n\tprivate _next(): IteratorResult<TreeWalkerValue, undefined> {\n\t\tlet position = this.position.clone();\n\t\tconst previousPosition = this.position;\n\t\tconst parent = position.parent;\n\n\t\t// We are at the end of the root.\n\t\tif ( parent.parent === null && position.offset === ( parent as any ).childCount ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// We reached the walker boundary.\n\t\tif ( parent === this._boundaryEndParent && position.offset == this.boundaries!.end.offset ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// Get node just after current position.\n\t\tlet node;\n\n\t\t// Text is a specific parent because it contains string instead of child nodes.\n\t\tif ( parent instanceof Text ) {\n\t\t\tif ( position.isAtEnd ) {\n\t\t\t\t// Prevent returning \"elementEnd\" for Text node. Skip that value and return the next walker step.\n\t\t\t\tthis._position = Position._createAfter( parent );\n\n\t\t\t\treturn this._next();\n\t\t\t}\n\n\t\t\tnode = parent.data[ position.offset ];\n\t\t} else {\n\t\t\tnode = ( parent as Element | DocumentFragment ).getChild( position.offset );\n\t\t}\n\n\t\tif ( node instanceof Element ) {\n\t\t\tif ( !this.shallow ) {\n\t\t\t\tposition = new Position( node, 0 );\n\t\t\t} else {\n\t\t\t\t// We are past the walker boundaries.\n\t\t\t\tif ( this.boundaries && this.boundaries.end.isBefore( position ) ) {\n\t\t\t\t\treturn { done: true, value: undefined };\n\t\t\t\t}\n\n\t\t\t\tposition.offset++;\n\t\t\t}\n\n\t\t\tthis._position = position;\n\n\t\t\treturn this._formatReturnValue( 'elementStart', node, previousPosition, position, 1 );\n\t\t}\n\n\t\tif ( node instanceof Text ) {\n\t\t\tif ( this.singleCharacters ) {\n\t\t\t\tposition = new Position( node, 0 );\n\t\t\t\tthis._position = position;\n\n\t\t\t\treturn this._next();\n\t\t\t}\n\n\t\t\tlet charactersCount = node.data.length;\n\t\t\tlet item;\n\n\t\t\t// If text stick out of walker range, we need to cut it and wrap in TextProxy.\n\t\t\tif ( node == this._boundaryEndParent ) {\n\t\t\t\tcharactersCount = this.boundaries!.end.offset;\n\t\t\t\titem = new TextProxy( node, 0, charactersCount );\n\t\t\t\tposition = Position._createAfter( item );\n\t\t\t} else {\n\t\t\t\titem = new TextProxy( node, 0, node.data.length );\n\t\t\t\t// If not just keep moving forward.\n\t\t\t\tposition.offset++;\n\t\t\t}\n\n\t\t\tthis._position = position;\n\n\t\t\treturn this._formatReturnValue( 'text', item, previousPosition, position, charactersCount );\n\t\t}\n\n\t\tif ( typeof node == 'string' ) {\n\t\t\tlet textLength;\n\n\t\t\tif ( this.singleCharacters ) {\n\t\t\t\ttextLength = 1;\n\t\t\t} else {\n\t\t\t\t// Check if text stick out of walker range.\n\t\t\t\tconst endOffset = parent === this._boundaryEndParent ? this.boundaries!.end.offset : ( parent as Text ).data.length;\n\n\t\t\t\ttextLength = endOffset - position.offset;\n\t\t\t}\n\n\t\t\tconst textProxy = new TextProxy( parent as Text, position.offset, textLength );\n\n\t\t\tposition.offset += textLength;\n\t\t\tthis._position = position;\n\n\t\t\treturn this._formatReturnValue( 'text', textProxy, previousPosition, position, textLength );\n\t\t}\n\n\t\t// `node` is not set, we reached the end of current `parent`.\n\t\tposition = Position._createAfter( parent as any );\n\t\tthis._position = position;\n\n\t\tif ( this.ignoreElementEnd ) {\n\t\t\treturn this._next();\n\t\t}\n\n\t\treturn this._formatReturnValue( 'elementEnd', parent as any, previousPosition, position );\n\t}\n\n\t/**\n\t * Makes a step backward in view. Moves the {@link #position} to the previous position and returns the encountered value.\n\t */\n\tprivate _previous(): IteratorResult<TreeWalkerValue, undefined> {\n\t\tlet position = this.position.clone();\n\t\tconst previousPosition = this.position;\n\t\tconst parent = position.parent;\n\n\t\t// We are at the beginning of the root.\n\t\tif ( parent.parent === null && position.offset === 0 ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// We reached the walker boundary.\n\t\tif ( parent == this._boundaryStartParent && position.offset == this.boundaries!.start.offset ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// Get node just before current position.\n\t\tlet node;\n\n\t\t// Text {@link module:engine/view/text~Text} element is a specific parent because contains string instead of child nodes.\n\t\tif ( parent instanceof Text ) {\n\t\t\tif ( position.isAtStart ) {\n\t\t\t\t// Prevent returning \"elementStart\" for Text node. Skip that value and return the next walker step.\n\t\t\t\tthis._position = Position._createBefore( parent );\n\n\t\t\t\treturn this._previous();\n\t\t\t}\n\n\t\t\tnode = parent.data[ position.offset - 1 ];\n\t\t} else {\n\t\t\tnode = ( parent as Element | DocumentFragment ).getChild( position.offset - 1 );\n\t\t}\n\n\t\tif ( node instanceof Element ) {\n\t\t\tif ( this.shallow ) {\n\t\t\t\tposition.offset--;\n\t\t\t\tthis._position = position;\n\n\t\t\t\treturn this._formatReturnValue( 'elementStart', node, previousPosition, position, 1 );\n\t\t\t}\n\n\t\t\tposition = new Position( node, node.childCount );\n\t\t\tthis._position = position;\n\n\t\t\tif ( this.ignoreElementEnd ) {\n\t\t\t\treturn this._previous();\n\t\t\t}\n\n\t\t\treturn this._formatReturnValue( 'elementEnd', node, previousPosition, position );\n\t\t}\n\n\t\tif ( node instanceof Text ) {\n\t\t\tif ( this.singleCharacters ) {\n\t\t\t\tposition = new Position( node, node.data.length );\n\t\t\t\tthis._position = position;\n\n\t\t\t\treturn this._previous();\n\t\t\t}\n\n\t\t\tlet charactersCount = node.data.length;\n\t\t\tlet item;\n\n\t\t\t// If text stick out of walker range, we need to cut it and wrap in TextProxy.\n\t\t\tif ( node == this._boundaryStartParent ) {\n\t\t\t\tconst offset = this.boundaries!.start.offset;\n\n\t\t\t\titem = new TextProxy( node, offset, node.data.length - offset );\n\t\t\t\tcharactersCount = item.data.length;\n\t\t\t\tposition = Position._createBefore( item );\n\t\t\t} else {\n\t\t\t\titem = new TextProxy( node, 0, node.data.length );\n\t\t\t\t// If not just keep moving backward.\n\t\t\t\tposition.offset--;\n\t\t\t}\n\n\t\t\tthis._position = position;\n\n\t\t\treturn this._formatReturnValue( 'text', item, previousPosition, position, charactersCount );\n\t\t}\n\n\t\tif ( typeof node == 'string' ) {\n\t\t\tlet textLength;\n\n\t\t\tif ( !this.singleCharacters ) {\n\t\t\t\t// Check if text stick out of walker range.\n\t\t\t\tconst startOffset = parent === this._boundaryStartParent ? this.boundaries!.start.offset : 0;\n\n\t\t\t\ttextLength = position.offset - startOffset;\n\t\t\t} else {\n\t\t\t\ttextLength = 1;\n\t\t\t}\n\n\t\t\tposition.offset -= textLength;\n\n\t\t\tconst textProxy = new TextProxy( parent as Text, position.offset, textLength );\n\n\t\t\tthis._position = position;\n\n\t\t\treturn this._formatReturnValue( 'text', textProxy, previousPosition, position, textLength );\n\t\t}\n\n\t\t// `node` is not set, we reached the beginning of current `parent`.\n\t\tposition = Position._createBefore( parent as any );\n\t\tthis._position = position;\n\n\t\treturn this._formatReturnValue( 'elementStart', parent as Element, previousPosition, position, 1 );\n\t}\n\n\t/**\n\t * Format returned data and adjust `previousPosition` and `nextPosition` if reach the bound of the {@link module:engine/view/text~Text}.\n\t *\n\t * @param type Type of step.\n\t * @param item Item between old and new position.\n\t * @param previousPosition Previous position of iterator.\n\t * @param nextPosition Next position of iterator.\n\t * @param length Length of the item.\n\t */\n\tprivate _formatReturnValue(\n\t\ttype: TreeWalkerValueType,\n\t\titem: Item,\n\t\tpreviousPosition: Position,\n\t\tnextPosition: Position,\n\t\tlength?: number\n\t): IteratorYieldResult<TreeWalkerValue> {\n\t\t// Text is a specific parent, because contains string instead of children.\n\t\t// Walker doesn't enter to the Text except situations when walker is iterating over every single character,\n\t\t// or the bound starts/ends inside the Text. So when the position is at the beginning or at the end of the Text\n\t\t// we move it just before or just after Text.\n\t\tif ( item instanceof TextProxy ) {\n\t\t\t// Position is at the end of Text.\n\t\t\tif ( item.offsetInText + item.data.length == item.textNode.data.length ) {\n\t\t\t\tif ( this.direction == 'forward' && !( this.boundaries && this.boundaries.end.isEqual( this.position ) ) ) {\n\t\t\t\t\tnextPosition = Position._createAfter( item.textNode );\n\t\t\t\t\t// When we change nextPosition of returned value we need also update walker current position.\n\t\t\t\t\tthis._position = nextPosition;\n\t\t\t\t} else {\n\t\t\t\t\tpreviousPosition = Position._createAfter( item.textNode );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Position is at the begining ot the text.\n\t\t\tif ( item.offsetInText === 0 ) {\n\t\t\t\tif ( this.direction == 'backward' && !( this.boundaries && this.boundaries.start.isEqual( this.position ) ) ) {\n\t\t\t\t\tnextPosition = Position._createBefore( item.textNode );\n\t\t\t\t\t// When we change nextPosition of returned value we need also update walker current position.\n\t\t\t\t\tthis._position = nextPosition;\n\t\t\t\t} else {\n\t\t\t\t\tpreviousPosition = Position._createBefore( item.textNode );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn {\n\t\t\tdone: false,\n\t\t\tvalue: {\n\t\t\t\ttype,\n\t\t\t\titem,\n\t\t\t\tpreviousPosition,\n\t\t\t\tnextPosition,\n\t\t\t\tlength\n\t\t\t}\n\t\t};\n\t}\n}\n\n/**\n * Type of the step made by {@link module:engine/view/treewalker~TreeWalker}.\n * Possible values: `'elementStart'` if walker is at the beginning of a node, `'elementEnd'` if walker is at the end\n * of node, or `'text'` if walker traversed over single and multiple characters.\n * For {@link module:engine/view/text~Text} `elementStart` and `elementEnd` is not returned.\n */\nexport type TreeWalkerValueType = 'elementStart' | 'elementEnd' | 'text';\n\n/**\n * Object returned by {@link module:engine/view/treewalker~TreeWalker} when traversing tree view.\n */\nexport interface TreeWalkerValue {\n\n\t/**\n\t * Type of the step made by {@link module:engine/view/treewalker~TreeWalker}.\n\t */\n\ttype: TreeWalkerValueType;\n\n\t/**\n\t * Item between the old and the new positions of the tree walker.\n\t */\n\titem: Item;\n\n\t/**\n\t * Previous position of the iterator.\n\t * * Forward iteration: For `'elementEnd'` it is the last position inside the element. For all other types it is the\n\t * position before the item.\n\t * * Backward iteration: For `'elementStart'` it is the first position inside the element. For all other types it is\n\t * the position after item.\n\t * * If the position is at the beginning or at the end of the {@link module:engine/view/text~Text} it is always moved from the\n\t * inside of the text to its parent just before or just after that text.\n\t */\n\tpreviousPosition: Position;\n\n\t/**\n\t * Next position of the iterator.\n\t * * Forward iteration: For `'elementStart'` it is the first position inside the element. For all other types it is\n\t * the position after the item.\n\t * * Backward iteration: For `'elementEnd'` it is last position inside element. For all other types it is the position\n\t * before the item.\n\t * * If the position is at the beginning or at the end of the {@link module:engine/view/text~Text} it is always moved from the\n\t * inside of the text to its parent just before or just after that text.\n\t */\n\tnextPosition: Position;\n\n\t/**\n\t * Length of the item. For `'elementStart'` it is `1`. For `'text'` it is\n\t * the length of that text. For `'elementEnd'` it is `undefined`.\n\t */\n\tlength?: number;\n}\n\n/**\n * Tree walking direction.\n */\nexport type TreeWalkerDirection = 'forward' | 'backward';\n\n/**\n * The configuration of {@link ~TreeWalker}.\n */\nexport interface TreeWalkerOptions {\n\n\t/**\n\t * Walking direction.\n\t *\n\t * @default 'forward'\n\t */\n\tdirection?: TreeWalkerDirection;\n\n\t/**\n\t * Range to define boundaries of the iterator.\n\t */\n\tboundaries?: Range | null;\n\n\t/**\n\t * Starting position.\n\t */\n\tstartPosition?: Position;\n\n\t/**\n\t * Flag indicating whether all characters from\n\t * {@link module:engine/view/text~Text} should be returned as one {@link module:engine/view/text~Text} (`false`) or one by one as\n\t * {@link module:engine/view/textproxy~TextProxy} (`true`).\n\t */\n\tsingleCharacters?: boolean;\n\n\t/**\n\t * Flag indicating whether iterator should enter elements or not. If the\n\t * iterator is shallow child nodes of any iterated node will not be returned along with `elementEnd` tag.\n\t */\n\tshallow?: boolean;\n\n\t/**\n\t * Flag indicating whether iterator should ignore `elementEnd`\n\t * tags. If the option is true walker will not return a parent node of start position. If this option is `true`\n\t * each {@link module:engine/view/element~Element} will be returned once, while if the option is `false` they might be returned\n\t * twice: for `'elementStart'` and `'elementEnd'`.\n\t */\n\tignoreElementEnd?: boolean;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/position\n */\n\nimport TypeCheckable from './typecheckable';\n\nimport { CKEditorError, compareArrays } from '@ckeditor/ckeditor5-utils';\n\nimport EditableElement from './editableelement';\n\n// To check if component is loaded more than once.\nimport '@ckeditor/ckeditor5-utils/src/version';\n\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\nimport type Item from './item';\nimport type Node from './node';\nimport { default as TreeWalker, type TreeWalkerValue, type TreeWalkerOptions } from './treewalker';\n\n/**\n * Position in the view tree. Position is represented by its parent node and an offset in this parent.\n *\n * In order to create a new position instance use the `createPosition*()` factory methods available in:\n *\n * * {@link module:engine/view/view~View}\n * * {@link module:engine/view/downcastwriter~DowncastWriter}\n * * {@link module:engine/view/upcastwriter~UpcastWriter}\n */\nexport default class Position extends TypeCheckable {\n\t/**\n\t * Position parent.\n\t */\n\tpublic readonly parent: Node | DocumentFragment;\n\n\t/**\n\t * Position offset.\n\t */\n\tpublic offset: number;\n\n\t/**\n\t * Creates a position.\n\t *\n\t * @param parent Position parent.\n\t * @param offset Position offset.\n\t */\n\tconstructor( parent: Node | DocumentFragment, offset: number ) {\n\t\tsuper();\n\n\t\tthis.parent = parent;\n\t\tthis.offset = offset;\n\t}\n\n\t/**\n\t * Node directly after the position. Equals `null` when there is no node after position or position is located\n\t * inside text node.\n\t */\n\tpublic get nodeAfter(): Node | null {\n\t\tif ( this.parent.is( '$text' ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn ( this.parent as Element ).getChild( this.offset ) || null;\n\t}\n\n\t/**\n\t * Node directly before the position. Equals `null` when there is no node before position or position is located\n\t * inside text node.\n\t */\n\tpublic get nodeBefore(): Node | null {\n\t\tif ( this.parent.is( '$text' ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn ( this.parent as Element ).getChild( this.offset - 1 ) || null;\n\t}\n\n\t/**\n\t * Is `true` if position is at the beginning of its {@link module:engine/view/position~Position#parent parent}, `false` otherwise.\n\t */\n\tpublic get isAtStart(): boolean {\n\t\treturn this.offset === 0;\n\t}\n\n\t/**\n\t * Is `true` if position is at the end of its {@link module:engine/view/position~Position#parent parent}, `false` otherwise.\n\t */\n\tpublic get isAtEnd(): boolean {\n\t\tconst endOffset = this.parent.is( '$text' ) ? this.parent.data.length : ( this.parent as any ).childCount;\n\n\t\treturn this.offset === endOffset;\n\t}\n\n\t/**\n\t * Position's root, that is the root of the position's parent element.\n\t */\n\tpublic get root(): Node | DocumentFragment {\n\t\treturn this.parent.root;\n\t}\n\n\t/**\n\t * {@link module:engine/view/editableelement~EditableElement EditableElement} instance that contains this position, or `null` if\n\t * position is not inside an editable element.\n\t */\n\tpublic get editableElement(): EditableElement | null {\n\t\tlet editable = this.parent;\n\n\t\twhile ( !( editable instanceof EditableElement ) ) {\n\t\t\tif ( editable.parent ) {\n\t\t\t\teditable = editable.parent;\n\t\t\t} else {\n\t\t\t\treturn null;\n\t\t\t}\n\t\t}\n\n\t\treturn editable;\n\t}\n\n\t/**\n\t * Returns a new instance of Position with offset incremented by `shift` value.\n\t *\n\t * @param shift How position offset should get changed. Accepts negative values.\n\t * @returns Shifted position.\n\t */\n\tpublic getShiftedBy( shift: number ): Position {\n\t\tconst shifted = Position._createAt( this );\n\n\t\tconst offset = shifted.offset + shift;\n\t\tshifted.offset = offset < 0 ? 0 : offset;\n\n\t\treturn shifted;\n\t}\n\n\t/**\n\t * Gets the farthest position which matches the callback using\n\t * {@link module:engine/view/treewalker~TreeWalker TreeWalker}.\n\t *\n\t * For example:\n\t *\n\t * ```ts\n\t * getLastMatchingPosition( value => value.type == 'text' ); // <p>{}foo</p> -> <p>foo[]</p>\n\t * getLastMatchingPosition( value => value.type == 'text', { direction: 'backward' } ); // <p>foo[]</p> -> <p>{}foo</p>\n\t * getLastMatchingPosition( value => false ); // Do not move the position.\n\t * ```\n\t *\n\t * @param skip Callback function. Gets {@link module:engine/view/treewalker~TreeWalkerValue} and should\n\t * return `true` if the value should be skipped or `false` if not.\n\t * @param options Object with configuration options. See {@link module:engine/view/treewalker~TreeWalker}.\n\t * @returns The position after the last item which matches the `skip` callback test.\n\t */\n\tpublic getLastMatchingPosition( skip: ( value: TreeWalkerValue ) => boolean, options: TreeWalkerOptions = {} ): Position {\n\t\toptions.startPosition = this;\n\n\t\tconst treeWalker = new TreeWalker( options );\n\t\ttreeWalker.skip( skip );\n\n\t\treturn treeWalker.position;\n\t}\n\n\t/**\n\t * Returns ancestors array of this position, that is this position's parent and it's ancestors.\n\t *\n\t * @returns Array with ancestors.\n\t */\n\tpublic getAncestors(): Array<Node | DocumentFragment> {\n\t\tif ( this.parent.is( 'documentFragment' ) ) {\n\t\t\treturn [ this.parent ];\n\t\t} else {\n\t\t\treturn this.parent.getAncestors( { includeSelf: true } );\n\t\t}\n\t}\n\n\t/**\n\t * Returns a {@link module:engine/view/node~Node} or {@link module:engine/view/documentfragment~DocumentFragment}\n\t * which is a common ancestor of both positions.\n\t */\n\tpublic getCommonAncestor( position: Position ): Node | DocumentFragment | null {\n\t\tconst ancestorsA = this.getAncestors();\n\t\tconst ancestorsB = position.getAncestors();\n\n\t\tlet i = 0;\n\n\t\twhile ( ancestorsA[ i ] == ancestorsB[ i ] && ancestorsA[ i ] ) {\n\t\t\ti++;\n\t\t}\n\n\t\treturn i === 0 ? null : ancestorsA[ i - 1 ];\n\t}\n\n\t/**\n\t * Checks whether this position equals given position.\n\t *\n\t * @param otherPosition Position to compare with.\n\t * @returns True if positions are same.\n\t */\n\tpublic isEqual( otherPosition: Position ): boolean {\n\t\treturn ( this.parent == otherPosition.parent && this.offset == otherPosition.offset );\n\t}\n\n\t/**\n\t * Checks whether this position is located before given position. When method returns `false` it does not mean that\n\t * this position is after give one. Two positions may be located inside separate roots and in that situation this\n\t * method will still return `false`.\n\t *\n\t * @see module:engine/view/position~Position#isAfter\n\t * @see module:engine/view/position~Position#compareWith\n\t * @param otherPosition Position to compare with.\n\t * @returns Returns `true` if this position is before given position.\n\t */\n\tpublic isBefore( otherPosition: Position ): boolean {\n\t\treturn this.compareWith( otherPosition ) == 'before';\n\t}\n\n\t/**\n\t * Checks whether this position is located after given position. When method returns `false` it does not mean that\n\t * this position is before give one. Two positions may be located inside separate roots and in that situation this\n\t * method will still return `false`.\n\t *\n\t * @see module:engine/view/position~Position#isBefore\n\t * @see module:engine/view/position~Position#compareWith\n\t * @param otherPosition Position to compare with.\n\t * @returns Returns `true` if this position is after given position.\n\t */\n\tpublic isAfter( otherPosition: Position ): boolean {\n\t\treturn this.compareWith( otherPosition ) == 'after';\n\t}\n\n\t/**\n\t * Checks whether this position is before, after or in same position that other position. Two positions may be also\n\t * different when they are located in separate roots.\n\t *\n\t * @param otherPosition Position to compare with.\n\t */\n\tpublic compareWith( otherPosition: Position ): PositionRelation {\n\t\tif ( this.root !== otherPosition.root ) {\n\t\t\treturn 'different';\n\t\t}\n\n\t\tif ( this.isEqual( otherPosition ) ) {\n\t\t\treturn 'same';\n\t\t}\n\n\t\t// Get path from root to position's parent element.\n\t\tconst thisPath = this.parent.is( 'node' ) ? this.parent.getPath() : [];\n\t\tconst otherPath = otherPosition.parent.is( 'node' ) ? otherPosition.parent.getPath() : [];\n\n\t\t// Add the positions' offsets to the parents offsets.\n\t\tthisPath.push( this.offset );\n\t\totherPath.push( otherPosition.offset );\n\n\t\t// Compare both path arrays to find common ancestor.\n\t\tconst result = compareArrays( thisPath, otherPath );\n\n\t\tswitch ( result ) {\n\t\t\tcase 'prefix':\n\t\t\t\treturn 'before';\n\n\t\t\tcase 'extension':\n\t\t\t\treturn 'after';\n\n\t\t\tdefault:\n\t\t\t\t// Cast to number to avoid having 'same' as a type of `result`.\n\t\t\t\treturn thisPath[ result as number ] < otherPath[ result as number ] ? 'before' : 'after';\n\t\t}\n\t}\n\n\t/**\n\t * Creates a {@link module:engine/view/treewalker~TreeWalker TreeWalker} instance with this positions as a start position.\n\t *\n\t * @param options Object with configuration options. See {@link module:engine/view/treewalker~TreeWalker}\n\t */\n\tpublic getWalker( options: TreeWalkerOptions = {} ): TreeWalker {\n\t\toptions.startPosition = this;\n\n\t\treturn new TreeWalker( options );\n\t}\n\n\t/**\n\t * Clones this position.\n\t */\n\tpublic clone(): Position {\n\t\treturn new Position( this.parent, this.offset );\n\t}\n\n\t/**\n\t * Creates position at the given location. The location can be specified as:\n\t *\n\t * * a {@link module:engine/view/position~Position position},\n\t * * parent element and offset (offset defaults to `0`),\n\t * * parent element and `'end'` (sets position at the end of that element),\n\t * * {@link module:engine/view/item~Item view item} and `'before'` or `'after'` (sets position before or after given view item).\n\t *\n\t * This method is a shortcut to other constructors such as:\n\t *\n\t * * {@link module:engine/view/position~Position._createBefore},\n\t * * {@link module:engine/view/position~Position._createAfter}.\n\t *\n\t * @internal\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/view/item~Item view item}.\n\t */\n\tpublic static _createAt( itemOrPosition: Item | Position, offset?: PositionOffset ): Position {\n\t\tif ( itemOrPosition instanceof Position ) {\n\t\t\treturn new this( itemOrPosition.parent, itemOrPosition.offset );\n\t\t} else {\n\t\t\tconst node = itemOrPosition;\n\n\t\t\tif ( offset == 'end' ) {\n\t\t\t\toffset = node.is( '$text' ) ? node.data.length : ( node as any ).childCount;\n\t\t\t} else if ( offset == 'before' ) {\n\t\t\t\treturn this._createBefore( node );\n\t\t\t} else if ( offset == 'after' ) {\n\t\t\t\treturn this._createAfter( node );\n\t\t\t} else if ( offset !== 0 && !offset ) {\n\t\t\t\t/**\n\t\t\t\t * {@link module:engine/view/view~View#createPositionAt `View#createPositionAt()`}\n\t\t\t\t * requires the offset to be specified when the first parameter is a view item.\n\t\t\t\t *\n\t\t\t\t * @error view-createpositionat-offset-required\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'view-createpositionat-offset-required', node );\n\t\t\t}\n\n\t\t\treturn new Position( node as any, offset as number );\n\t\t}\n\t}\n\n\t/**\n\t * Creates a new position after given view item.\n\t *\n\t * @internal\n\t * @param item View item after which the position should be located.\n\t */\n\tpublic static _createAfter( item: Item ): Position {\n\t\t// TextProxy is not a instance of Node so we need do handle it in specific way.\n\t\tif ( item.is( '$textProxy' ) ) {\n\t\t\treturn new Position( item.textNode, item.offsetInText + item.data.length );\n\t\t}\n\n\t\tif ( !item.parent ) {\n\t\t\t/**\n\t\t\t * You can not make a position after a root.\n\t\t\t *\n\t\t\t * @error view-position-after-root\n\t\t\t * @param {module:engine/view/node~Node} root\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-position-after-root', item, { root: item } );\n\t\t}\n\n\t\treturn new Position( item.parent, ( item.index as number ) + 1 );\n\t}\n\n\t/**\n\t * Creates a new position before given view item.\n\t *\n\t * @internal\n\t * @param item View item before which the position should be located.\n\t */\n\tpublic static _createBefore( item: Item ): Position {\n\t\t// TextProxy is not a instance of Node so we need do handle it in specific way.\n\t\tif ( item.is( '$textProxy' ) ) {\n\t\t\treturn new Position( item.textNode, item.offsetInText );\n\t\t}\n\n\t\tif ( !item.parent ) {\n\t\t\t/**\n\t\t\t * You cannot make a position before a root.\n\t\t\t *\n\t\t\t * @error view-position-before-root\n\t\t\t * @param {module:engine/view/node~Node} root\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-position-before-root', item, { root: item } );\n\t\t}\n\n\t\treturn new Position( item.parent, item.index as number );\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nPosition.prototype.is = function( type: string ): boolean {\n\treturn type === 'position' || type === 'view:position';\n};\n\n/**\n * A flag indicating whether this position is `'before'` or `'after'` or `'same'` as given position.\n * If positions are in different roots `'different'` flag is returned.\n */\nexport type PositionRelation = 'before' | 'after' | 'same' | 'different';\n\n/**\n * Offset or one of the flags.\n */\nexport type PositionOffset = number | 'before' | 'after' | 'end';\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/range\n */\n\nimport TypeCheckable from './typecheckable';\nimport Position from './position';\n\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\nimport type Item from './item';\nimport type Node from './node';\nimport { default as TreeWalker, type TreeWalkerValue, type TreeWalkerOptions } from './treewalker';\n\n/**\n * Range in the view tree. A range is represented by its start and end {@link module:engine/view/position~Position positions}.\n *\n * In order to create a new position instance use the `createPosition*()` factory methods available in:\n *\n * * {@link module:engine/view/view~View}\n * * {@link module:engine/view/downcastwriter~DowncastWriter}\n * * {@link module:engine/view/upcastwriter~UpcastWriter}\n */\nexport default class Range extends TypeCheckable implements Iterable<TreeWalkerValue> {\n\t/**\n\t * Start position.\n\t */\n\tpublic readonly start: Position;\n\n\t/**\n\t * End position.\n\t */\n\tpublic readonly end: Position;\n\n\t/**\n\t * Creates a range spanning from `start` position to `end` position.\n\t *\n\t * **Note:** Constructor creates it's own {@link module:engine/view/position~Position} instances basing on passed values.\n\t *\n\t * @param start Start position.\n\t * @param end End position. If not set, range will be collapsed at the `start` position.\n\t */\n\tconstructor( start: Position, end: Position | null = null ) {\n\t\tsuper();\n\n\t\tthis.start = start.clone();\n\t\tthis.end = end ? end.clone() : start.clone();\n\t}\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * Iterates over all {@link module:engine/view/item~Item view items} that are in this range and returns\n\t * them together with additional information like length or {@link module:engine/view/position~Position positions},\n\t * grouped as {@link module:engine/view/treewalker~TreeWalkerValue}.\n\t *\n\t * This iterator uses {@link module:engine/view/treewalker~TreeWalker TreeWalker} with `boundaries` set to this range and\n\t * `ignoreElementEnd` option\n\t * set to `true`.\n\t */\n\tpublic* [ Symbol.iterator ](): IterableIterator<TreeWalkerValue> {\n\t\tyield* new TreeWalker( { boundaries: this, ignoreElementEnd: true } );\n\t}\n\n\t/**\n\t * Returns whether the range is collapsed, that is it start and end positions are equal.\n\t */\n\tpublic get isCollapsed(): boolean {\n\t\treturn this.start.isEqual( this.end );\n\t}\n\n\t/**\n\t * Returns whether this range is flat, that is if {@link module:engine/view/range~Range#start start} position and\n\t * {@link module:engine/view/range~Range#end end} position are in the same {@link module:engine/view/position~Position#parent parent}.\n\t */\n\tpublic get isFlat(): boolean {\n\t\treturn this.start.parent === this.end.parent;\n\t}\n\n\t/**\n\t * Range root element.\n\t */\n\tpublic get root(): Node | DocumentFragment {\n\t\treturn this.start.root;\n\t}\n\n\t/**\n\t * Creates a maximal range that has the same content as this range but is expanded in both ways (at the beginning\n\t * and at the end).\n\t *\n\t * For example:\n\t *\n\t * ```html\n\t * <p>Foo</p><p><b>{Bar}</b></p> -> <p>Foo</p>[<p><b>Bar</b>]</p>\n\t * <p><b>foo</b>{bar}<span></span></p> -> <p><b>foo[</b>bar<span></span>]</p>\n\t * ```\n\t *\n\t * Note that in the sample above:\n\t *\n\t * - `<p>` have type of {@link module:engine/view/containerelement~ContainerElement},\n\t * - `<b>` have type of {@link module:engine/view/attributeelement~AttributeElement},\n\t * - `<span>` have type of {@link module:engine/view/uielement~UIElement}.\n\t *\n\t * @returns Enlarged range.\n\t */\n\tpublic getEnlarged(): Range {\n\t\tlet start = this.start.getLastMatchingPosition( enlargeTrimSkip, { direction: 'backward' } );\n\t\tlet end = this.end.getLastMatchingPosition( enlargeTrimSkip );\n\n\t\t// Fix positions, in case if they are in Text node.\n\t\tif ( start.parent.is( '$text' ) && start.isAtStart ) {\n\t\t\tstart = Position._createBefore( start.parent );\n\t\t}\n\n\t\tif ( end.parent.is( '$text' ) && end.isAtEnd ) {\n\t\t\tend = Position._createAfter( end.parent );\n\t\t}\n\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Creates a minimum range that has the same content as this range but is trimmed in both ways (at the beginning\n\t * and at the end).\n\t *\n\t * For example:\n\t *\n\t * ```html\n\t * <p>Foo</p>[<p><b>Bar</b>]</p> -> <p>Foo</p><p><b>{Bar}</b></p>\n\t * <p><b>foo[</b>bar<span></span>]</p> -> <p><b>foo</b>{bar}<span></span></p>\n\t * ```\n\t *\n\t * Note that in the sample above:\n\t *\n\t * - `<p>` have type of {@link module:engine/view/containerelement~ContainerElement},\n\t * - `<b>` have type of {@link module:engine/view/attributeelement~AttributeElement},\n\t * - `<span>` have type of {@link module:engine/view/uielement~UIElement}.\n\t *\n\t * @returns Shrunk range.\n\t */\n\tpublic getTrimmed(): Range {\n\t\tlet start = this.start.getLastMatchingPosition( enlargeTrimSkip );\n\n\t\tif ( start.isAfter( this.end ) || start.isEqual( this.end ) ) {\n\t\t\treturn new Range( start, start );\n\t\t}\n\n\t\tlet end = this.end.getLastMatchingPosition( enlargeTrimSkip, { direction: 'backward' } );\n\t\tconst nodeAfterStart = start.nodeAfter;\n\t\tconst nodeBeforeEnd = end.nodeBefore;\n\n\t\t// Because TreeWalker prefers positions next to text node, we need to move them manually into these text nodes.\n\t\tif ( nodeAfterStart && nodeAfterStart.is( '$text' ) ) {\n\t\t\tstart = new Position( nodeAfterStart, 0 );\n\t\t}\n\n\t\tif ( nodeBeforeEnd && nodeBeforeEnd.is( '$text' ) ) {\n\t\t\tend = new Position( nodeBeforeEnd, nodeBeforeEnd.data.length );\n\t\t}\n\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Two ranges are equal if their start and end positions are equal.\n\t *\n\t * @param otherRange Range to compare with.\n\t * @returns `true` if ranges are equal, `false` otherwise\n\t */\n\tpublic isEqual( otherRange: Range ): boolean {\n\t\treturn this == otherRange || ( this.start.isEqual( otherRange.start ) && this.end.isEqual( otherRange.end ) );\n\t}\n\n\t/**\n\t * Checks whether this range contains given {@link module:engine/view/position~Position position}.\n\t *\n\t * @param position Position to check.\n\t * @returns `true` if given {@link module:engine/view/position~Position position} is contained in this range, `false` otherwise.\n\t */\n\tpublic containsPosition( position: Position ): boolean {\n\t\treturn position.isAfter( this.start ) && position.isBefore( this.end );\n\t}\n\n\t/**\n\t * Checks whether this range contains given {@link module:engine/view/range~Range range}.\n\t *\n\t * @param otherRange Range to check.\n\t * @param loose Whether the check is loose or strict. If the check is strict (`false`), compared range cannot\n\t * start or end at the same position as this range boundaries. If the check is loose (`true`), compared range can start, end or\n\t * even be equal to this range. Note that collapsed ranges are always compared in strict mode.\n\t * @returns `true` if given {@link module:engine/view/range~Range range} boundaries are contained by this range, `false`\n\t * otherwise.\n\t */\n\tpublic containsRange( otherRange: Range, loose: boolean = false ): boolean {\n\t\tif ( otherRange.isCollapsed ) {\n\t\t\tloose = false;\n\t\t}\n\n\t\tconst containsStart = this.containsPosition( otherRange.start ) || ( loose && this.start.isEqual( otherRange.start ) );\n\t\tconst containsEnd = this.containsPosition( otherRange.end ) || ( loose && this.end.isEqual( otherRange.end ) );\n\n\t\treturn containsStart && containsEnd;\n\t}\n\n\t/**\n\t * Computes which part(s) of this {@link module:engine/view/range~Range range} is not a part of given\n\t * {@link module:engine/view/range~Range range}.\n\t * Returned array contains zero, one or two {@link module:engine/view/range~Range ranges}.\n\t *\n\t * Examples:\n\t *\n\t * ```ts\n\t * let foo = downcastWriter.createText( 'foo' );\n\t * let img = downcastWriter.createContainerElement( 'img' );\n\t * let bar = downcastWriter.createText( 'bar' );\n\t * let p = downcastWriter.createContainerElement( 'p', null, [ foo, img, bar ] );\n\t *\n\t * let range = view.createRange( view.createPositionAt( foo, 2 ), view.createPositionAt( bar, 1 ); // \"o\", img, \"b\" are in range.\n\t * let otherRange = view.createRange( // \"oo\", img, \"ba\" are in range.\n\t * \tview.createPositionAt( foo, 1 ),\n\t * \tview.createPositionAt( bar, 2 )\n\t * );\n\t * let transformed = range.getDifference( otherRange );\n\t * // transformed array has no ranges because `otherRange` contains `range`\n\t *\n\t * otherRange = view.createRange( view.createPositionAt( foo, 1 ), view.createPositionAt( p, 2 ); // \"oo\", img are in range.\n\t * transformed = range.getDifference( otherRange );\n\t * // transformed array has one range: from ( p, 2 ) to ( bar, 1 )\n\t *\n\t * otherRange = view.createRange( view.createPositionAt( p, 1 ), view.createPositionAt( p, 2 ) ); // img is in range.\n\t * transformed = range.getDifference( otherRange );\n\t * // transformed array has two ranges: from ( foo, 1 ) to ( p, 1 ) and from ( p, 2 ) to ( bar, 1 )\n\t * ```\n\t *\n\t * @param otherRange Range to differentiate against.\n\t * @returns The difference between ranges.\n\t */\n\tpublic getDifference( otherRange: Range ): Array<Range> {\n\t\tconst ranges: Array<Range> = [];\n\n\t\tif ( this.isIntersecting( otherRange ) ) {\n\t\t\t// Ranges intersect.\n\n\t\t\tif ( this.containsPosition( otherRange.start ) ) {\n\t\t\t\t// Given range start is inside this range. This means that we have to\n\t\t\t\t// add shrunken range - from the start to the middle of this range.\n\t\t\t\tranges.push( new Range( this.start, otherRange.start ) );\n\t\t\t}\n\n\t\t\tif ( this.containsPosition( otherRange.end ) ) {\n\t\t\t\t// Given range end is inside this range. This means that we have to\n\t\t\t\t// add shrunken range - from the middle of this range to the end.\n\t\t\t\tranges.push( new Range( otherRange.end, this.end ) );\n\t\t\t}\n\t\t} else {\n\t\t\t// Ranges do not intersect, return the original range.\n\t\t\tranges.push( this.clone() );\n\t\t}\n\n\t\treturn ranges;\n\t}\n\n\t/**\n\t * Returns an intersection of this {@link module:engine/view/range~Range range} and given {@link module:engine/view/range~Range range}.\n\t * Intersection is a common part of both of those ranges. If ranges has no common part, returns `null`.\n\t *\n\t * Examples:\n\t *\n\t * ```ts\n\t * let foo = downcastWriter.createText( 'foo' );\n\t * let img = downcastWriter.createContainerElement( 'img' );\n\t * let bar = downcastWriter.createText( 'bar' );\n\t * let p = downcastWriter.createContainerElement( 'p', null, [ foo, img, bar ] );\n\t *\n\t * let range = view.createRange( view.createPositionAt( foo, 2 ), view.createPositionAt( bar, 1 ); // \"o\", img, \"b\" are in range.\n\t * let otherRange = view.createRange( view.createPositionAt( foo, 1 ), view.createPositionAt( p, 2 ); // \"oo\", img are in range.\n\t * let transformed = range.getIntersection( otherRange ); // range from ( foo, 1 ) to ( p, 2 ).\n\t *\n\t * otherRange = view.createRange( view.createPositionAt( bar, 1 ), view.createPositionAt( bar, 3 ); \"ar\" is in range.\n\t * transformed = range.getIntersection( otherRange ); // null - no common part.\n\t * ```\n\t *\n\t * @param otherRange Range to check for intersection.\n\t * @returns A common part of given ranges or `null` if ranges have no common part.\n\t */\n\tpublic getIntersection( otherRange: Range ): Range | null {\n\t\tif ( this.isIntersecting( otherRange ) ) {\n\t\t\t// Ranges intersect, so a common range will be returned.\n\t\t\t// At most, it will be same as this range.\n\t\t\tlet commonRangeStart = this.start;\n\t\t\tlet commonRangeEnd = this.end;\n\n\t\t\tif ( this.containsPosition( otherRange.start ) ) {\n\t\t\t\t// Given range start is inside this range. This means thaNt we have to\n\t\t\t\t// shrink common range to the given range start.\n\t\t\t\tcommonRangeStart = otherRange.start;\n\t\t\t}\n\n\t\t\tif ( this.containsPosition( otherRange.end ) ) {\n\t\t\t\t// Given range end is inside this range. This means that we have to\n\t\t\t\t// shrink common range to the given range end.\n\t\t\t\tcommonRangeEnd = otherRange.end;\n\t\t\t}\n\n\t\t\treturn new Range( commonRangeStart, commonRangeEnd );\n\t\t}\n\n\t\t// Ranges do not intersect, so they do not have common part.\n\t\treturn null;\n\t}\n\n\t/**\n\t * Creates a {@link module:engine/view/treewalker~TreeWalker TreeWalker} instance with this range as a boundary.\n\t *\n\t * @param options Object with configuration options. See {@link module:engine/view/treewalker~TreeWalker}.\n\t */\n\tpublic getWalker( options: TreeWalkerOptions = {} ): TreeWalker {\n\t\toptions.boundaries = this;\n\n\t\treturn new TreeWalker( options );\n\t}\n\n\t/**\n\t * Returns a {@link module:engine/view/node~Node} or {@link module:engine/view/documentfragment~DocumentFragment}\n\t * which is a common ancestor of range's both ends (in which the entire range is contained).\n\t */\n\tpublic getCommonAncestor(): Node | DocumentFragment | null {\n\t\treturn this.start.getCommonAncestor( this.end );\n\t}\n\n\t/**\n\t * Returns an {@link module:engine/view/element~Element Element} contained by the range.\n\t * The element will be returned when it is the **only** node within the range and **fullycontained**\n\t * at the same time.\n\t */\n\tpublic getContainedElement(): Element | null {\n\t\tif ( this.isCollapsed ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tlet nodeAfterStart = this.start.nodeAfter;\n\t\tlet nodeBeforeEnd = this.end.nodeBefore;\n\n\t\t// Handle the situation when the range position is at the beginning / at the end of a text node.\n\t\t// In such situation `.nodeAfter` and `.nodeBefore` are `null` but the range still might be spanning\n\t\t// over one element.\n\t\t//\n\t\t// <p>Foo{<span class=\"widget\"></span>}bar</p> vs <p>Foo[<span class=\"widget\"></span>]bar</p>\n\t\t//\n\t\t// These are basically the same range, only the difference is if the range position is at\n\t\t// at the end/at the beginning of a text node or just before/just after the text node.\n\t\t//\n\t\tif ( this.start.parent.is( '$text' ) && this.start.isAtEnd && this.start.parent.nextSibling ) {\n\t\t\tnodeAfterStart = this.start.parent.nextSibling;\n\t\t}\n\n\t\tif ( this.end.parent.is( '$text' ) && this.end.isAtStart && this.end.parent.previousSibling ) {\n\t\t\tnodeBeforeEnd = this.end.parent.previousSibling;\n\t\t}\n\n\t\tif ( nodeAfterStart && nodeAfterStart.is( 'element' ) && nodeAfterStart === nodeBeforeEnd ) {\n\t\t\treturn nodeAfterStart;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Clones this range.\n\t */\n\tpublic clone(): Range {\n\t\treturn new Range( this.start, this.end );\n\t}\n\n\t/**\n\t * Returns an iterator that iterates over all {@link module:engine/view/item~Item view items} that are in this range and returns\n\t * them.\n\t *\n\t * This method uses {@link module:engine/view/treewalker~TreeWalker} with `boundaries` set to this range and `ignoreElementEnd` option\n\t * set to `true`. However it returns only {@link module:engine/view/item~Item items},\n\t * not {@link module:engine/view/treewalker~TreeWalkerValue}.\n\t *\n\t * You may specify additional options for the tree walker. See {@link module:engine/view/treewalker~TreeWalker} for\n\t * a full list of available options.\n\t *\n\t * @param options Object with configuration options. See {@link module:engine/view/treewalker~TreeWalker}.\n\t */\n\tpublic* getItems( options: TreeWalkerOptions = {} ): IterableIterator<Item> {\n\t\toptions.boundaries = this;\n\t\toptions.ignoreElementEnd = true;\n\n\t\tconst treeWalker = new TreeWalker( options );\n\n\t\tfor ( const value of treeWalker ) {\n\t\t\tyield value.item;\n\t\t}\n\t}\n\n\t/**\n\t * Returns an iterator that iterates over all {@link module:engine/view/position~Position positions} that are boundaries or\n\t * contained in this range.\n\t *\n\t * This method uses {@link module:engine/view/treewalker~TreeWalker} with `boundaries` set to this range. However it returns only\n\t * {@link module:engine/view/position~Position positions}, not {@link module:engine/view/treewalker~TreeWalkerValue}.\n\t *\n\t * You may specify additional options for the tree walker. See {@link module:engine/view/treewalker~TreeWalker} for\n\t * a full list of available options.\n\t *\n\t * @param options Object with configuration options. See {@link module:engine/view/treewalker~TreeWalker}.\n\t */\n\tpublic* getPositions( options: TreeWalkerOptions = {} ): IterableIterator<Position> {\n\t\toptions.boundaries = this;\n\n\t\tconst treeWalker = new TreeWalker( options );\n\n\t\tyield treeWalker.position;\n\n\t\tfor ( const value of treeWalker ) {\n\t\t\tyield value.nextPosition;\n\t\t}\n\t}\n\n\t/**\n\t * Checks and returns whether this range intersects with the given range.\n\t *\n\t * @param otherRange Range to compare with.\n\t * @returns True if ranges intersect.\n\t */\n\tpublic isIntersecting( otherRange: Range ): boolean {\n\t\treturn this.start.isBefore( otherRange.end ) && this.end.isAfter( otherRange.start );\n\t}\n\n\t/**\n\t * Creates a range from the given parents and offsets.\n\t *\n\t * @internal\n\t * @param startElement Start position parent element.\n\t * @param startOffset Start position offset.\n\t * @param endElement End position parent element.\n\t * @param endOffset End position offset.\n\t * @returns Created range.\n\t */\n\tpublic static _createFromParentsAndOffsets(\n\t\tstartElement: Element | DocumentFragment,\n\t\tstartOffset: number,\n\t\tendElement: Element | DocumentFragment,\n\t\tendOffset: number\n\t): Range {\n\t\treturn new this(\n\t\t\tnew Position( startElement, startOffset ),\n\t\t\tnew Position( endElement, endOffset )\n\t\t);\n\t}\n\n\t/**\n\t * Creates a new range, spreading from specified {@link module:engine/view/position~Position position} to a position moved by\n\t * given `shift`. If `shift` is a negative value, shifted position is treated as the beginning of the range.\n\t *\n\t * @internal\n\t * @param position Beginning of the range.\n\t * @param shift How long the range should be.\n\t */\n\tpublic static _createFromPositionAndShift( position: Position, shift: number ): Range {\n\t\tconst start = position;\n\t\tconst end = position.getShiftedBy( shift );\n\n\t\treturn shift > 0 ? new this( start, end ) : new this( end, start );\n\t}\n\n\t/**\n\t * Creates a range inside an {@link module:engine/view/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * @internal\n\t * @param element Element which is a parent for the range.\n\t */\n\tpublic static _createIn( element: Element | DocumentFragment ): Range {\n\t\treturn this._createFromParentsAndOffsets( element, 0, element, element.childCount );\n\t}\n\n\t/**\n\t * Creates a range that starts before given {@link module:engine/view/item~Item view item} and ends after it.\n\t *\n\t * @internal\n\t */\n\tpublic static _createOn( item: Item ): Range {\n\t\tconst size = item.is( '$textProxy' ) ? item.offsetSize : 1;\n\n\t\treturn this._createFromPositionAndShift( Position._createBefore( item ), size );\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nRange.prototype.is = function( type: string ): boolean {\n\treturn type === 'range' || type === 'view:range';\n};\n\n/**\n * Function used by getEnlarged and getTrimmed methods.\n */\nfunction enlargeTrimSkip( value: TreeWalkerValue ): boolean {\n\tif ( value.item.is( 'attributeElement' ) || value.item.is( 'uiElement' ) ) {\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/selection\n */\n\nimport TypeCheckable from './typecheckable';\nimport Range from './range';\nimport Position, { type PositionOffset } from './position';\nimport Node from './node';\nimport DocumentSelection from './documentselection';\n\nimport {\n\tCKEditorError,\n\tEmitterMixin,\n\tcount,\n\tisIterable\n} from '@ckeditor/ckeditor5-utils';\n\nimport type Element from './element';\nimport type Item from './item';\nimport type EditableElement from './editableelement';\n\n/**\n * Class representing an arbirtary selection in the view.\n * See also {@link module:engine/view/documentselection~DocumentSelection}.\n *\n * New selection instances can be created via the constructor or one these methods:\n *\n * * {@link module:engine/view/view~View#createSelection `View#createSelection()`},\n * * {@link module:engine/view/upcastwriter~UpcastWriter#createSelection `UpcastWriter#createSelection()`}.\n *\n * A selection can consist of {@link module:engine/view/range~Range ranges} that can be set by using\n * the {@link module:engine/view/selection~Selection#setTo `Selection#setTo()`} method.\n */\nexport default class Selection extends EmitterMixin( TypeCheckable ) {\n\t/**\n\t * Stores all ranges that are selected.\n\t */\n\tprivate _ranges: Array<Range>;\n\n\t/**\n\t * Specifies whether the last added range was added as a backward or forward range.\n\t */\n\tprivate _lastRangeBackward: boolean;\n\n\t/**\n\t * Specifies whether selection instance is fake.\n\t */\n\tprivate _isFake: boolean;\n\n\t/**\n\t * Fake selection's label.\n\t */\n\tprivate _fakeSelectionLabel: string;\n\n\t/**\n\t * Creates new selection instance.\n\t *\n\t * **Note**: The selection constructor is available as a factory method:\n\t *\n\t * * {@link module:engine/view/view~View#createSelection `View#createSelection()`},\n\t * * {@link module:engine/view/upcastwriter~UpcastWriter#createSelection `UpcastWriter#createSelection()`}.\n\t *\n\t * ```ts\n\t * // Creates empty selection without ranges.\n\t * const selection = writer.createSelection();\n\t *\n\t * // Creates selection at the given range.\n\t * const range = writer.createRange( start, end );\n\t * const selection = writer.createSelection( range );\n\t *\n\t * // Creates selection at the given ranges\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( star2, end2 ) ];\n\t * const selection = writer.createSelection( ranges );\n\t *\n\t * // Creates selection from the other selection.\n\t * const otherSelection = writer.createSelection();\n\t * const selection = writer.createSelection( otherSelection );\n\t *\n\t * // Creates selection from the document selection.\n\t * const selection = writer.createSelection( editor.editing.view.document.selection );\n\t *\n\t * // Creates selection at the given position.\n\t * const position = writer.createPositionFromPath( root, path );\n\t * const selection = writer.createSelection( position );\n\t *\n\t * // Creates collapsed selection at the position of given item and offset.\n\t * const paragraph = writer.createContainerElement( 'paragraph' );\n\t * const selection = writer.createSelection( paragraph, offset );\n\t *\n\t * // Creates a range inside an {@link module:engine/view/element~Element element} which starts before the\n\t * // first child of that element and ends after the last child of that element.\n\t * const selection = writer.createSelection( paragraph, 'in' );\n\t *\n\t * // Creates a range on an {@link module:engine/view/item~Item item} which starts before the item and ends\n\t * // just after the item.\n\t * const selection = writer.createSelection( paragraph, 'on' );\n\t * ```\n\t *\n\t * `Selection`'s constructor allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = writer.createSelection( range, { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = writer.createSelection( range, { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * @internal\n\t */\n\tconstructor(\n\t\t...args: [] | [\n\t\t\tselectable: Node,\n\t\t\tplaceOrOffset: PlaceOrOffset,\n\t\t\toptions?: SelectionOptions\n\t\t] | [\n\t\t\tselectable?: Exclude<Selectable, Node>,\n\t\t\toptions?: SelectionOptions\n\t\t]\n\t) {\n\t\tsuper();\n\n\t\tthis._ranges = [];\n\t\tthis._lastRangeBackward = false;\n\t\tthis._isFake = false;\n\t\tthis._fakeSelectionLabel = '';\n\n\t\tif ( args.length ) {\n\t\t\tthis.setTo( ...args );\n\t\t}\n\t}\n\n\t/**\n\t * Returns true if selection instance is marked as `fake`.\n\t *\n\t * @see #setTo\n\t */\n\tpublic get isFake(): boolean {\n\t\treturn this._isFake;\n\t}\n\n\t/**\n\t * Returns fake selection label.\n\t *\n\t * @see #setTo\n\t */\n\tpublic get fakeSelectionLabel(): string {\n\t\treturn this._fakeSelectionLabel;\n\t}\n\n\t/**\n\t * Selection anchor. Anchor may be described as a position where the selection starts. Together with\n\t * {@link #focus focus} they define the direction of selection, which is important\n\t * when expanding/shrinking selection. Anchor is always the start or end of the most recent added range.\n\t * It may be a bit unintuitive when there are multiple ranges in selection.\n\t *\n\t * @see #focus\n\t */\n\tpublic get anchor(): Position | null {\n\t\tif ( !this._ranges.length ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst range = this._ranges[ this._ranges.length - 1 ];\n\t\tconst anchor = this._lastRangeBackward ? range.end : range.start;\n\n\t\treturn anchor.clone();\n\t}\n\n\t/**\n\t * Selection focus. Focus is a position where the selection ends.\n\t *\n\t * @see #anchor\n\t */\n\tpublic get focus(): Position | null {\n\t\tif ( !this._ranges.length ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst range = this._ranges[ this._ranges.length - 1 ];\n\t\tconst focus = this._lastRangeBackward ? range.start : range.end;\n\n\t\treturn focus.clone();\n\t}\n\n\t/**\n\t * Returns whether the selection is collapsed. Selection is collapsed when there is exactly one range which is\n\t * collapsed.\n\t */\n\tpublic get isCollapsed(): boolean {\n\t\treturn this.rangeCount === 1 && this._ranges[ 0 ].isCollapsed;\n\t}\n\n\t/**\n\t * Returns number of ranges in selection.\n\t */\n\tpublic get rangeCount(): number {\n\t\treturn this._ranges.length;\n\t}\n\n\t/**\n\t * Specifies whether the {@link #focus} precedes {@link #anchor}.\n\t */\n\tpublic get isBackward(): boolean {\n\t\treturn !this.isCollapsed && this._lastRangeBackward;\n\t}\n\n\t/**\n\t * {@link module:engine/view/editableelement~EditableElement EditableElement} instance that contains this selection, or `null`\n\t * if the selection is not inside an editable element.\n\t */\n\tpublic get editableElement(): EditableElement | null {\n\t\tif ( this.anchor ) {\n\t\t\treturn this.anchor.editableElement;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns an iterable that contains copies of all ranges added to the selection.\n\t */\n\tpublic* getRanges(): IterableIterator<Range> {\n\t\tfor ( const range of this._ranges ) {\n\t\t\tyield range.clone();\n\t\t}\n\t}\n\n\t/**\n\t * Returns copy of the first range in the selection. First range is the one which\n\t * {@link module:engine/view/range~Range#start start} position {@link module:engine/view/position~Position#isBefore is before} start\n\t * position of all other ranges (not to confuse with the first range added to the selection).\n\t * Returns `null` if no ranges are added to selection.\n\t */\n\tpublic getFirstRange(): Range | null {\n\t\tlet first = null;\n\n\t\tfor ( const range of this._ranges ) {\n\t\t\tif ( !first || range.start.isBefore( first.start ) ) {\n\t\t\t\tfirst = range;\n\t\t\t}\n\t\t}\n\n\t\treturn first ? first.clone() : null;\n\t}\n\n\t/**\n\t * Returns copy of the last range in the selection. Last range is the one which {@link module:engine/view/range~Range#end end}\n\t * position {@link module:engine/view/position~Position#isAfter is after} end position of all other ranges (not to confuse\n\t * with the last range added to the selection). Returns `null` if no ranges are added to selection.\n\t */\n\tpublic getLastRange(): Range | null {\n\t\tlet last = null;\n\n\t\tfor ( const range of this._ranges ) {\n\t\t\tif ( !last || range.end.isAfter( last.end ) ) {\n\t\t\t\tlast = range;\n\t\t\t}\n\t\t}\n\n\t\treturn last ? last.clone() : null;\n\t}\n\n\t/**\n\t * Returns copy of the first position in the selection. First position is the position that\n\t * {@link module:engine/view/position~Position#isBefore is before} any other position in the selection ranges.\n\t * Returns `null` if no ranges are added to selection.\n\t */\n\tpublic getFirstPosition(): Position | null {\n\t\tconst firstRange = this.getFirstRange();\n\n\t\treturn firstRange ? firstRange.start.clone() : null;\n\t}\n\n\t/**\n\t * Returns copy of the last position in the selection. Last position is the position that\n\t * {@link module:engine/view/position~Position#isAfter is after} any other position in the selection ranges.\n\t * Returns `null` if no ranges are added to selection.\n\t */\n\tpublic getLastPosition(): Position | null {\n\t\tconst lastRange = this.getLastRange();\n\n\t\treturn lastRange ? lastRange.end.clone() : null;\n\t}\n\n\t/**\n\t * Checks whether, this selection is equal to given selection. Selections are equal if they have same directions,\n\t * same number of ranges and all ranges from one selection equal to a range from other selection.\n\t *\n\t * @param otherSelection Selection to compare with.\n\t * @returns `true` if selections are equal, `false` otherwise.\n\t */\n\tpublic isEqual( otherSelection: Selection | DocumentSelection ): boolean {\n\t\tif ( this.isFake != otherSelection.isFake ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( this.isFake && this.fakeSelectionLabel != otherSelection.fakeSelectionLabel ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( this.rangeCount != otherSelection.rangeCount ) {\n\t\t\treturn false;\n\t\t} else if ( this.rangeCount === 0 ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tif ( !this.anchor!.isEqual( otherSelection.anchor! ) || !this.focus!.isEqual( otherSelection.focus! ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tfor ( const thisRange of this._ranges ) {\n\t\t\tlet found = false;\n\n\t\t\tfor ( const otherRange of ( otherSelection as any )._ranges ) {\n\t\t\t\tif ( thisRange.isEqual( otherRange ) ) {\n\t\t\t\t\tfound = true;\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( !found ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Checks whether this selection is similar to given selection. Selections are similar if they have same directions, same\n\t * number of ranges, and all {@link module:engine/view/range~Range#getTrimmed trimmed} ranges from one selection are\n\t * equal to any trimmed range from other selection.\n\t *\n\t * @param otherSelection Selection to compare with.\n\t * @returns `true` if selections are similar, `false` otherwise.\n\t */\n\tpublic isSimilar( otherSelection: Selection | DocumentSelection ): boolean {\n\t\tif ( this.isBackward != otherSelection.isBackward ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst numOfRangesA = count( this.getRanges() );\n\t\tconst numOfRangesB = count( otherSelection.getRanges() );\n\n\t\t// If selections have different number of ranges, they cannot be similar.\n\t\tif ( numOfRangesA != numOfRangesB ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// If both selections have no ranges, they are similar.\n\t\tif ( numOfRangesA == 0 ) {\n\t\t\treturn true;\n\t\t}\n\n\t\t// Check if each range in one selection has a similar range in other selection.\n\t\tfor ( let rangeA of this.getRanges() ) {\n\t\t\trangeA = rangeA.getTrimmed();\n\n\t\t\tlet found = false;\n\n\t\t\tfor ( let rangeB of otherSelection.getRanges() ) {\n\t\t\t\trangeB = rangeB.getTrimmed();\n\n\t\t\t\tif ( rangeA.start.isEqual( rangeB.start ) && rangeA.end.isEqual( rangeB.end ) ) {\n\t\t\t\t\tfound = true;\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// For `rangeA`, neither range in `otherSelection` was similar. So selections are not similar.\n\t\t\tif ( !found ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\t// There were no ranges that weren't matched. Selections are similar.\n\t\treturn true;\n\t}\n\n\t/**\n\t * Returns the selected element. {@link module:engine/view/element~Element Element} is considered as selected if there is only\n\t * one range in the selection, and that range contains exactly one element.\n\t * Returns `null` if there is no selected element.\n\t */\n\tpublic getSelectedElement(): Element | null {\n\t\tif ( this.rangeCount !== 1 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn this.getFirstRange()!.getContainedElement();\n\t}\n\n\t/**\n\t * Sets this selection's ranges and direction to the specified location based on the given\n\t * {@link module:engine/view/selection~Selectable selectable}.\n\t *\n\t * ```ts\n\t * // Sets selection to the given range.\n\t * const range = writer.createRange( start, end );\n\t * selection.setTo( range );\n\t *\n\t * // Sets selection to given ranges.\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( star2, end2 ) ];\n\t * selection.setTo( range );\n\t *\n\t * // Sets selection to the other selection.\n\t * const otherSelection = writer.createSelection();\n\t * selection.setTo( otherSelection );\n\t *\n\t * // Sets selection to contents of DocumentSelection.\n\t * selection.setTo( editor.editing.view.document.selection );\n\t *\n\t * // Sets collapsed selection at the given position.\n\t * const position = writer.createPositionAt( root, path );\n\t * selection.setTo( position );\n\t *\n\t * // Sets collapsed selection at the position of given item and offset.\n\t * selection.setTo( paragraph, offset );\n\t * ```\n\t *\n\t * Creates a range inside an {@link module:engine/view/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * ```ts\n\t * selection.setTo( paragraph, 'in' );\n\t * ```\n\t *\n\t * Creates a range on an {@link module:engine/view/item~Item item} which starts before the item and ends just after the item.\n\t *\n\t * ```ts\n\t * selection.setTo( paragraph, 'on' );\n\t *\n\t * // Clears selection. Removes all ranges.\n\t * selection.setTo( null );\n\t * ```\n\t *\n\t * `Selection#setTo()` method allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Sets selection as backward.\n\t * selection.setTo( range, { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * selection.setTo( range, { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * @fires change\n\t */\n\tpublic setTo(\n\t\t...args: [\n\t\t\tselectable: Node,\n\t\t\tplaceOrOffset: PlaceOrOffset,\n\t\t\toptions?: SelectionOptions\n\t\t] | [\n\t\t\tselectable?: Exclude<Selectable, Node>,\n\t\t\toptions?: SelectionOptions\n\t\t]\n\t): void {\n\t\tlet [ selectable, placeOrOffset, options ] = args;\n\n\t\tif ( typeof placeOrOffset == 'object' ) {\n\t\t\toptions = placeOrOffset;\n\t\t\tplaceOrOffset = undefined;\n\t\t}\n\n\t\tif ( selectable === null ) {\n\t\t\tthis._setRanges( [] );\n\t\t\tthis._setFakeOptions( options );\n\t\t} else if ( selectable instanceof Selection || selectable instanceof DocumentSelection ) {\n\t\t\tthis._setRanges( selectable.getRanges(), selectable.isBackward );\n\t\t\tthis._setFakeOptions( { fake: selectable.isFake, label: selectable.fakeSelectionLabel } );\n\t\t} else if ( selectable instanceof Range ) {\n\t\t\tthis._setRanges( [ selectable ], options && options.backward );\n\t\t\tthis._setFakeOptions( options );\n\t\t} else if ( selectable instanceof Position ) {\n\t\t\tthis._setRanges( [ new Range( selectable ) ] );\n\t\t\tthis._setFakeOptions( options );\n\t\t} else if ( selectable instanceof Node ) {\n\t\t\tconst backward = !!options && !!options.backward;\n\t\t\tlet range;\n\n\t\t\tif ( placeOrOffset === undefined ) {\n\t\t\t\t/**\n\t\t\t\t * selection.setTo requires the second parameter when the first parameter is a node.\n\t\t\t\t *\n\t\t\t\t * @error view-selection-setto-required-second-parameter\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'view-selection-setto-required-second-parameter', this );\n\t\t\t} else if ( placeOrOffset == 'in' ) {\n\t\t\t\trange = Range._createIn( selectable as Element );\n\t\t\t} else if ( placeOrOffset == 'on' ) {\n\t\t\t\trange = Range._createOn( selectable );\n\t\t\t} else {\n\t\t\t\trange = new Range( Position._createAt( selectable, placeOrOffset ) );\n\t\t\t}\n\n\t\t\tthis._setRanges( [ range ], backward );\n\t\t\tthis._setFakeOptions( options );\n\t\t} else if ( isIterable( selectable ) ) {\n\t\t\t// We assume that the selectable is an iterable of ranges.\n\t\t\t// Array.from() is used to prevent setting ranges to the old iterable\n\t\t\tthis._setRanges( selectable, options && options.backward );\n\t\t\tthis._setFakeOptions( options );\n\t\t} else {\n\t\t\t/**\n\t\t\t * Cannot set selection to given place.\n\t\t\t *\n\t\t\t * @error view-selection-setto-not-selectable\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-selection-setto-not-selectable', this );\n\t\t}\n\n\t\tthis.fire<ViewSelectionChangeEvent>( 'change' );\n\t}\n\n\t/**\n\t * Moves {@link #focus} to the specified location.\n\t *\n\t * The location can be specified in the same form as {@link module:engine/view/view~View#createPositionAt view.createPositionAt()}\n\t * parameters.\n\t *\n\t * @fires change\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/view/item~Item view item}.\n\t */\n\tpublic setFocus( itemOrPosition: Item | Position, offset?: PositionOffset ): void {\n\t\tif ( this.anchor === null ) {\n\t\t\t/**\n\t\t\t * Cannot set selection focus if there are no ranges in selection.\n\t\t\t *\n\t\t\t * @error view-selection-setfocus-no-ranges\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-selection-setfocus-no-ranges', this );\n\t\t}\n\n\t\tconst newFocus = Position._createAt( itemOrPosition, offset );\n\n\t\tif ( newFocus.compareWith( this.focus! ) == 'same' ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst anchor = this.anchor;\n\n\t\tthis._ranges.pop();\n\n\t\tif ( newFocus.compareWith( anchor ) == 'before' ) {\n\t\t\tthis._addRange( new Range( newFocus, anchor ), true );\n\t\t} else {\n\t\t\tthis._addRange( new Range( anchor, newFocus ) );\n\t\t}\n\n\t\tthis.fire<ViewSelectionChangeEvent>( 'change' );\n\t}\n\n\t/**\n\t * Replaces all ranges that were added to the selection with given array of ranges. Last range of the array\n\t * is treated like the last added range and is used to set {@link #anchor anchor} and {@link #focus focus}.\n\t * Accepts a flag describing in which way the selection is made.\n\t *\n\t * @param newRanges Iterable object of ranges to set.\n\t * @param isLastBackward Flag describing if last added range was selected forward - from start to end\n\t * (`false`) or backward - from end to start (`true`). Defaults to `false`.\n\t */\n\tprivate _setRanges( newRanges: Iterable<Range>, isLastBackward: boolean = false ) {\n\t\t// New ranges should be copied to prevent removing them by setting them to `[]` first.\n\t\t// Only applies to situations when selection is set to the same selection or same selection's ranges.\n\t\tnewRanges = Array.from( newRanges );\n\n\t\tthis._ranges = [];\n\n\t\tfor ( const range of newRanges ) {\n\t\t\tthis._addRange( range );\n\t\t}\n\n\t\tthis._lastRangeBackward = !!isLastBackward;\n\t}\n\n\t/**\n\t * Sets this selection instance to be marked as `fake`. A fake selection does not render as browser native selection\n\t * over selected elements and is hidden to the user. This way, no native selection UI artifacts are displayed to\n\t * the user and selection over elements can be represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM (and be\n\t * properly handled by screen readers).\n\t */\n\tprivate _setFakeOptions( options: SelectionOptions = {} ) {\n\t\tthis._isFake = !!options.fake;\n\t\tthis._fakeSelectionLabel = options.fake ? options.label || '' : '';\n\t}\n\n\t/**\n\t * Adds a range to the selection. Added range is copied. This means that passed range is not saved in the\n\t * selection instance and you can safely operate on it.\n\t *\n\t * Accepts a flag describing in which way the selection is made - passed range might be selected from\n\t * {@link module:engine/view/range~Range#start start} to {@link module:engine/view/range~Range#end end}\n\t * or from {@link module:engine/view/range~Range#end end} to {@link module:engine/view/range~Range#start start}.\n\t * The flag is used to set {@link #anchor anchor} and {@link #focus focus} properties.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-selection-range-intersects` if added range intersects\n\t * with ranges already stored in Selection instance.\n\t */\n\tprivate _addRange( range: Range, isBackward: boolean = false ): void {\n\t\tif ( !( range instanceof Range ) ) {\n\t\t\t/**\n\t\t\t * Selection range set to an object that is not an instance of {@link module:engine/view/range~Range}.\n\t\t\t *\n\t\t\t * @error view-selection-add-range-not-range\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'view-selection-add-range-not-range',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\tthis._pushRange( range );\n\t\tthis._lastRangeBackward = !!isBackward;\n\t}\n\n\t/**\n\t * Adds range to selection - creates copy of given range so it can be safely used and modified.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-selection-range-intersects` if added range intersects\n\t * with ranges already stored in selection instance.\n\t */\n\tprivate _pushRange( range: Range ): void {\n\t\tfor ( const storedRange of this._ranges ) {\n\t\t\tif ( range.isIntersecting( storedRange ) ) {\n\t\t\t\t/**\n\t\t\t\t * Trying to add a range that intersects with another range from selection.\n\t\t\t\t *\n\t\t\t\t * @error view-selection-range-intersects\n\t\t\t\t * @param {module:engine/view/range~Range} addedRange Range that was added to the selection.\n\t\t\t\t * @param {module:engine/view/range~Range} intersectingRange Range from selection that intersects with `addedRange`.\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'view-selection-range-intersects',\n\t\t\t\t\tthis,\n\t\t\t\t\t{ addedRange: range, intersectingRange: storedRange }\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\n\t\tthis._ranges.push( new Range( range.start, range.end ) );\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nSelection.prototype.is = function( type: string ): boolean {\n\treturn type === 'selection' || type === 'view:selection';\n};\n\n/**\n * Additional options for {@link ~Selection}.\n */\nexport interface SelectionOptions {\n\n\t/**\n\t * Sets this selection instance to be backward.\n\t */\n\tbackward?: boolean;\n\n\t/**\n\t * Sets this selection instance to be marked as `fake`.\n\t */\n\tfake?: boolean;\n\n\t/**\n\t * Label for the fake selection.\n\t */\n\tlabel?: string;\n}\n\n/**\n * The place or offset of the selection.\n */\nexport type PlaceOrOffset = number | 'before' | 'end' | 'after' | 'on' | 'in';\n\n/**\n * Fired whenever selection ranges are changed through {@link ~Selection Selection API}.\n *\n * @eventName ~Selection#change\n */\nexport type ViewSelectionChangeEvent = {\n\tname: 'change';\n\targs: [];\n};\n\n/**\n * An entity that is used to set selection.\n *\n * See also {@link module:engine/view/selection~Selection#setTo}\n */\nexport type Selectable = Selection | DocumentSelection | Position | Iterable<Range> | Range | Node | null;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/documentselection\n */\n\nimport TypeCheckable from './typecheckable';\nimport Selection, {\n\ttype PlaceOrOffset,\n\ttype Selectable,\n\ttype SelectionOptions,\n\ttype ViewSelectionChangeEvent\n} from './selection';\n\nimport { EmitterMixin } from '@ckeditor/ckeditor5-utils';\n\nimport type EditableElement from './editableelement';\nimport type Element from './element';\nimport type Node from './node';\nimport type Item from './item';\nimport type { default as Position, PositionOffset } from './position';\nimport type Range from './range';\n\n/**\n * Class representing the document selection in the view.\n *\n * Its instance is available in {@link module:engine/view/document~Document#selection `Document#selection`}.\n *\n * It is similar to {@link module:engine/view/selection~Selection} but\n * it has a read-only API and can be modified only by the writer available in\n * the {@link module:engine/view/view~View#change `View#change()`} block\n * (so via {@link module:engine/view/downcastwriter~DowncastWriter#setSelection `DowncastWriter#setSelection()`}).\n */\nexport default class DocumentSelection extends EmitterMixin( TypeCheckable ) {\n\t/**\n\t * Selection is used internally (`DocumentSelection` is a proxy to that selection).\n\t */\n\tprivate readonly _selection: Selection;\n\n\t/**\n\t * Creates new DocumentSelection instance.\n\t *\n\t * ```ts\n\t * // Creates collapsed selection at the position of given item and offset.\n\t * const paragraph = writer.createContainerElement( 'paragraph' );\n\t * const selection = new DocumentSelection( paragraph, offset );\n\t *\n\t * // Creates a range inside an {@link module:engine/view/element~Element element} which starts before the\n\t * // first child of that element and ends after the last child of that element.\n\t * const selection = new DocumentSelection( paragraph, 'in' );\n\t *\n\t * // Creates a range on an {@link module:engine/view/item~Item item} which starts before the item and ends\n\t * // just after the item.\n\t * const selection = new DocumentSelection( paragraph, 'on' );\n\t * ```\n\t *\n\t * `Selection`'s constructor allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = new DocumentSelection( element, 'in', { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = new DocumentSelection( element, 'in', { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #constructor:SELECTABLE `constructor( selectable, options )`}.\n\t *\n\t * @label NODE_OFFSET\n\t */\n\tpublic constructor( selectable: Node, placeOrOffset: PlaceOrOffset, options?: SelectionOptions );\n\n\t/**\n\t * Creates new DocumentSelection instance.\n\t *\n\t * ```ts\n\t * // Creates empty selection without ranges.\n\t * const selection = new DocumentSelection();\n\t *\n\t * // Creates selection at the given range.\n\t * const range = writer.createRange( start, end );\n\t * const selection = new DocumentSelection( range );\n\t *\n\t * // Creates selection at the given ranges\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( start2, end2 ) ];\n\t * const selection = new DocumentSelection( ranges );\n\t *\n\t * // Creates selection from the other selection.\n\t * const otherSelection = writer.createSelection();\n\t * const selection = new DocumentSelection( otherSelection );\n\t *\n\t * // Creates selection at the given position.\n\t * const position = writer.createPositionAt( root, offset );\n\t * const selection = new DocumentSelection( position );\n\t * ```\n\t *\n\t * `Selection`'s constructor allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = new DocumentSelection( range, { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = new DocumentSelection( range, { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #constructor:NODE_OFFSET `constructor( node, placeOrOffset, options )`}.\n\t *\n\t * @label SELECTABLE\n\t */\n\tpublic constructor( selectable?: Exclude<Selectable, Node>, options?: SelectionOptions );\n\n\tconstructor( ...args: ConstructorParameters<typeof Selection> ) {\n\t\tsuper();\n\n\t\tthis._selection = new Selection();\n\n\t\t// Delegate change event to be fired on DocumentSelection instance.\n\t\tthis._selection.delegate( 'change' ).to( this );\n\n\t\t// Set selection data.\n\t\tif ( args.length ) {\n\t\t\tthis._selection.setTo( ...args );\n\t\t}\n\t}\n\n\t/**\n\t * Returns true if selection instance is marked as `fake`.\n\t *\n\t * @see #_setTo\n\t */\n\tpublic get isFake(): boolean {\n\t\treturn this._selection.isFake;\n\t}\n\n\t/**\n\t * Returns fake selection label.\n\t *\n\t * @see #_setTo\n\t */\n\tpublic get fakeSelectionLabel(): string {\n\t\treturn this._selection.fakeSelectionLabel;\n\t}\n\n\t/**\n\t * Selection anchor. Anchor may be described as a position where the selection starts. Together with\n\t * {@link #focus focus} they define the direction of selection, which is important\n\t * when expanding/shrinking selection. Anchor is always the start or end of the most recent added range.\n\t * It may be a bit unintuitive when there are multiple ranges in selection.\n\t *\n\t * @see #focus\n\t */\n\tpublic get anchor(): Position | null {\n\t\treturn this._selection.anchor;\n\t}\n\n\t/**\n\t * Selection focus. Focus is a position where the selection ends.\n\t *\n\t * @see #anchor\n\t */\n\tpublic get focus(): Position | null {\n\t\treturn this._selection.focus;\n\t}\n\n\t/**\n\t * Returns whether the selection is collapsed. Selection is collapsed when there is exactly one range which is\n\t * collapsed.\n\t */\n\tpublic get isCollapsed(): boolean {\n\t\treturn this._selection.isCollapsed;\n\t}\n\n\t/**\n\t * Returns number of ranges in selection.\n\t */\n\tpublic get rangeCount(): number {\n\t\treturn this._selection.rangeCount;\n\t}\n\n\t/**\n\t * Specifies whether the {@link #focus} precedes {@link #anchor}.\n\t */\n\tpublic get isBackward(): boolean {\n\t\treturn this._selection.isBackward;\n\t}\n\n\t/**\n\t * {@link module:engine/view/editableelement~EditableElement EditableElement} instance that contains this selection, or `null`\n\t * if the selection is not inside an editable element.\n\t */\n\tpublic get editableElement(): EditableElement | null {\n\t\treturn this._selection.editableElement;\n\t}\n\n\t/**\n\t * Used for the compatibility with the {@link module:engine/view/selection~Selection#isEqual} method.\n\t *\n\t * @internal\n\t */\n\tpublic get _ranges(): Array<Range> {\n\t\treturn ( this._selection as any )._ranges;\n\t}\n\n\t/**\n\t * Returns an iterable that contains copies of all ranges added to the selection.\n\t */\n\tpublic* getRanges(): IterableIterator<Range> {\n\t\tyield* this._selection.getRanges();\n\t}\n\n\t/**\n\t * Returns copy of the first range in the selection. First range is the one which\n\t * {@link module:engine/view/range~Range#start start} position {@link module:engine/view/position~Position#isBefore is before} start\n\t * position of all other ranges (not to confuse with the first range added to the selection).\n\t * Returns `null` if no ranges are added to selection.\n\t */\n\tpublic getFirstRange(): Range | null {\n\t\treturn this._selection.getFirstRange();\n\t}\n\n\t/**\n\t * Returns copy of the last range in the selection. Last range is the one which {@link module:engine/view/range~Range#end end}\n\t * position {@link module:engine/view/position~Position#isAfter is after} end position of all other ranges (not to confuse\n\t * with the last range added to the selection). Returns `null` if no ranges are added to selection.\n\t */\n\tpublic getLastRange(): Range | null {\n\t\treturn this._selection.getLastRange();\n\t}\n\n\t/**\n\t * Returns copy of the first position in the selection. First position is the position that\n\t * {@link module:engine/view/position~Position#isBefore is before} any other position in the selection ranges.\n\t * Returns `null` if no ranges are added to selection.\n\t */\n\tpublic getFirstPosition(): Position | null {\n\t\treturn this._selection.getFirstPosition();\n\t}\n\n\t/**\n\t * Returns copy of the last position in the selection. Last position is the position that\n\t * {@link module:engine/view/position~Position#isAfter is after} any other position in the selection ranges.\n\t * Returns `null` if no ranges are added to selection.\n\t */\n\tpublic getLastPosition(): Position | null {\n\t\treturn this._selection.getLastPosition();\n\t}\n\n\t/**\n\t * Returns the selected element. {@link module:engine/view/element~Element Element} is considered as selected if there is only\n\t * one range in the selection, and that range contains exactly one element.\n\t * Returns `null` if there is no selected element.\n\t */\n\tpublic getSelectedElement(): Element | null {\n\t\treturn this._selection.getSelectedElement();\n\t}\n\n\t/**\n\t * Checks whether, this selection is equal to given selection. Selections are equal if they have same directions,\n\t * same number of ranges and all ranges from one selection equal to a range from other selection.\n\t *\n\t * @param otherSelection Selection to compare with.\n\t * @returns `true` if selections are equal, `false` otherwise.\n\t */\n\tpublic isEqual( otherSelection: Selection | DocumentSelection ): boolean {\n\t\treturn this._selection.isEqual( otherSelection );\n\t}\n\n\t/**\n\t * Checks whether this selection is similar to given selection. Selections are similar if they have same directions, same\n\t * number of ranges, and all {@link module:engine/view/range~Range#getTrimmed trimmed} ranges from one selection are\n\t * equal to any trimmed range from other selection.\n\t *\n\t * @param otherSelection Selection to compare with.\n\t * @returns `true` if selections are similar, `false` otherwise.\n\t */\n\tpublic isSimilar( otherSelection: Selection | DocumentSelection ): boolean {\n\t\treturn this._selection.isSimilar( otherSelection );\n\t}\n\n\t/**\n\t * Sets this selection's ranges and direction to the specified location based on the given\n\t * {@link module:engine/view/selection~Selectable selectable}.\n\t *\n\t * ```ts\n\t * // Sets selection to the given range.\n\t * const range = writer.createRange( start, end );\n\t * documentSelection._setTo( range );\n\t *\n\t * // Sets selection to given ranges.\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( start2, end2 ) ];\n\t * documentSelection._setTo( range );\n\t *\n\t * // Sets selection to the other selection.\n\t * const otherSelection = writer.createSelection();\n\t * documentSelection._setTo( otherSelection );\n\t *\n\t * // Sets collapsed selection at the given position.\n\t * const position = writer.createPositionAt( root, offset );\n\t * documentSelection._setTo( position );\n\t *\n\t * // Sets collapsed selection at the position of given item and offset.\n\t * documentSelection._setTo( paragraph, offset );\n\t * ```\n\t *\n\t * Creates a range inside an {@link module:engine/view/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * ```ts\n\t * documentSelection._setTo( paragraph, 'in' );\n\t * ```\n\t *\n\t * Creates a range on an {@link module:engine/view/item~Item item} which starts before the item and ends just after the item.\n\t *\n\t * ```ts\n\t * documentSelection._setTo( paragraph, 'on' );\n\t *\n\t * // Clears selection. Removes all ranges.\n\t * documentSelection._setTo( null );\n\t * ```\n\t *\n\t * `Selection#_setTo()` method allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Sets selection as backward.\n\t * documentSelection._setTo( range, { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to des cribe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * documentSelection._setTo( range, { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * @internal\n\t * @fires change\n\t */\n\tpublic _setTo( ...args: Parameters<Selection[ 'setTo' ]> ): void {\n\t\tthis._selection.setTo( ...args );\n\t}\n\n\t/**\n\t * Moves {@link #focus} to the specified location.\n\t *\n\t * The location can be specified in the same form as {@link module:engine/view/view~View#createPositionAt view.createPositionAt()}\n\t * parameters.\n\t *\n\t * @internal\n\t * @fires change\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/view/item~Item view item}.\n\t */\n\tpublic _setFocus( itemOrPosition: Item | Position, offset?: PositionOffset ): void {\n\t\tthis._selection.setFocus( itemOrPosition, offset );\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nDocumentSelection.prototype.is = function( type: string ): boolean {\n\treturn type === 'selection' ||\n\t\ttype == 'documentSelection' ||\n\t\ttype == 'view:selection' ||\n\t\ttype == 'view:documentSelection';\n};\n\n/**\n * Fired whenever selection ranges are changed through {@link ~DocumentSelection Selection API}.\n *\n * @eventName ~DocumentSelection#change\n */\nexport type ViewDocumentSelectionChangeEvent = ViewSelectionChangeEvent;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/bubblingeventinfo\n */\n\nimport { EventInfo } from '@ckeditor/ckeditor5-utils';\nimport type Document from '../document';\nimport type Node from '../node';\nimport type Range from '../range';\n\n/**\n * The event object passed to bubbling event callbacks. It is used to provide information about the event as well as a tool to\n * manipulate it.\n */\nexport default class BubblingEventInfo<TName extends string = string, TReturn = unknown> extends EventInfo<TName, TReturn> {\n\t/**\n\t * The view range that the bubbling should start from.\n\t */\n\tpublic readonly startRange: Range;\n\n\t/**\n\t * The current event phase.\n\t */\n\tprivate _eventPhase: EventPhase;\n\n\t/**\n\t * The current bubbling target.\n\t */\n\tprivate _currentTarget: Document | Node | null;\n\n\t/**\n\t * @param source The emitter.\n\t * @param name The event name.\n\t * @param startRange The view range that the bubbling should start from.\n\t */\n\tconstructor( source: object, name: TName, startRange: Range ) {\n\t\tsuper( source, name );\n\n\t\tthis.startRange = startRange;\n\t\tthis._eventPhase = 'none';\n\t\tthis._currentTarget = null;\n\t}\n\n\t/**\n\t * The current event phase.\n\t */\n\tpublic get eventPhase(): EventPhase {\n\t\treturn this._eventPhase;\n\t}\n\n\t/**\n\t * The current bubbling target.\n\t */\n\tpublic get currentTarget(): Document | Node | null {\n\t\treturn this._currentTarget;\n\t}\n}\n\n/**\n * The phase the event is in.\n */\nexport type EventPhase = 'none' | 'capturing' | 'atTarget' | 'bubbling';\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/bubblingemittermixin\n */\n\nimport {\n\tCKEditorError,\n\tEmitterMixin,\n\tEventInfo,\n\ttoArray,\n\ttype ArrayOrItem,\n\ttype Emitter,\n\ttype GetEventInfo,\n\ttype GetNameOrEventInfo,\n\ttype BaseEvent,\n\ttype CallbackOptions,\n\ttype Constructor,\n\ttype Mixed\n} from '@ckeditor/ckeditor5-utils';\n\nimport BubblingEventInfo, { type EventPhase } from './bubblingeventinfo';\nimport type Document from '../document';\nimport type Node from '../node';\nimport type Range from '../range';\nimport type Element from '../element';\nimport type DocumentSelection from '../documentselection';\n\nconst contextsSymbol = Symbol( 'bubbling contexts' );\n\n/**\n * Bubbling emitter mixin for the view document as described in the {@link ~BubblingEmitter} interface.\n *\n * This function creates a class that inherits from the provided `base` and implements `Emitter` interface.\n * The base class must implement {@link module:utils/emittermixin~Emitter} interface.\n *\n * ```ts\n * class BaseClass extends EmitterMixin() {\n * \t// ...\n * }\n *\n * class MyClass extends BubblingEmitterMixin( BaseClass ) {\n * \t// This class derives from `BaseClass` and implements the `BubblingEmitter` interface.\n * }\n * ```\n */\nexport default function BubblingEmitterMixin<Base extends Constructor<Emitter>>( base: Base ): Mixed<Base, BubblingEmitter> {\n\tabstract class Mixin extends base implements BubblingEmitter {\n\t\tpublic abstract get selection(): DocumentSelection;\n\n\t\tpublic override fire<TEvent extends BaseEvent>(\n\t\t\teventOrInfo: GetNameOrEventInfo<TEvent>,\n\t\t\t...eventArgs: TEvent[ 'args' ]\n\t\t): GetEventInfo<TEvent>[ 'return' ] {\n\t\t\ttry {\n\t\t\t\tconst eventInfo = eventOrInfo instanceof EventInfo ? eventOrInfo : new EventInfo( this, eventOrInfo );\n\t\t\t\tconst eventContexts = getBubblingContexts( this );\n\n\t\t\t\tif ( !eventContexts.size ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tupdateEventInfo( eventInfo, 'capturing', this );\n\n\t\t\t\t// The capture phase of the event.\n\t\t\t\tif ( fireListenerFor( eventContexts, '$capture', eventInfo, ...eventArgs ) ) {\n\t\t\t\t\treturn eventInfo.return;\n\t\t\t\t}\n\n\t\t\t\tconst startRange = ( eventInfo as BubblingEventInfo ).startRange || this.selection.getFirstRange();\n\t\t\t\tconst selectedElement = startRange ? startRange.getContainedElement() : null;\n\t\t\t\tconst isCustomContext = selectedElement ? Boolean( getCustomContext( eventContexts, selectedElement ) ) : false;\n\n\t\t\t\tlet node: Node | null = selectedElement || getDeeperRangeParent( startRange );\n\n\t\t\t\tupdateEventInfo( eventInfo, 'atTarget', node );\n\n\t\t\t\t// For the not yet bubbling event trigger for $text node if selection can be there and it's not a custom context selected.\n\t\t\t\tif ( !isCustomContext ) {\n\t\t\t\t\tif ( fireListenerFor( eventContexts, '$text', eventInfo, ...eventArgs ) ) {\n\t\t\t\t\t\treturn eventInfo.return;\n\t\t\t\t\t}\n\n\t\t\t\t\tupdateEventInfo( eventInfo, 'bubbling', node );\n\t\t\t\t}\n\n\t\t\t\twhile ( node ) {\n\t\t\t\t\t// Root node handling.\n\t\t\t\t\tif ( node.is( 'rootElement' ) ) {\n\t\t\t\t\t\tif ( fireListenerFor( eventContexts, '$root', eventInfo, ...eventArgs ) ) {\n\t\t\t\t\t\t\treturn eventInfo.return;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Element node handling.\n\t\t\t\t\telse if ( node.is( 'element' ) ) {\n\t\t\t\t\t\tif ( fireListenerFor( eventContexts, node.name, eventInfo, ...eventArgs ) ) {\n\t\t\t\t\t\t\treturn eventInfo.return;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\t// Check custom contexts (i.e., a widget).\n\t\t\t\t\tif ( fireListenerFor( eventContexts, node, eventInfo, ...eventArgs ) ) {\n\t\t\t\t\t\treturn eventInfo.return;\n\t\t\t\t\t}\n\n\t\t\t\t\tnode = node.parent as Node;\n\n\t\t\t\t\tupdateEventInfo( eventInfo, 'bubbling', node );\n\t\t\t\t}\n\n\t\t\t\tupdateEventInfo( eventInfo, 'bubbling', this );\n\n\t\t\t\t// Document context.\n\t\t\t\tfireListenerFor( eventContexts, '$document', eventInfo, ...eventArgs );\n\n\t\t\t\treturn eventInfo.return;\n\t\t\t} catch ( err: any ) {\n\t\t\t\t// @if CK_DEBUG // throw err;\n\t\t\t\t/* istanbul ignore next -- @preserve */\n\t\t\t\tCKEditorError.rethrowUnexpectedError( err, this );\n\t\t\t}\n\t\t}\n\n\t\tpublic _addEventListener(\n\t\t\tthis: Document,\n\t\t\tevent: string,\n\t\t\tcallback: ( ev: EventInfo, ...args: Array<any> ) => void,\n\t\t\toptions: BubblingCallbackOptions\n\t\t) {\n\t\t\tconst contexts = toArray( options.context || '$document' );\n\t\t\tconst eventContexts = getBubblingContexts( this );\n\n\t\t\tfor ( const context of contexts ) {\n\t\t\t\tlet emitter = eventContexts.get( context );\n\n\t\t\t\tif ( !emitter ) {\n\t\t\t\t\temitter = new ( EmitterMixin() )();\n\t\t\t\t\teventContexts.set( context, emitter! );\n\t\t\t\t}\n\n\t\t\t\tthis.listenTo( emitter!, event, callback, options );\n\t\t\t}\n\t\t}\n\n\t\tpublic _removeEventListener( this: Document, event: string, callback: Function ): void {\n\t\t\tconst eventContexts = getBubblingContexts( this );\n\n\t\t\tfor ( const emitter of eventContexts.values() ) {\n\t\t\t\tthis.stopListening( emitter, event, callback );\n\t\t\t}\n\t\t}\n\t}\n\n\treturn Mixin as any;\n}\n\n// Backward compatibility with `mix`.\n{\n\tconst mixin = ( BubblingEmitterMixin as any )( Object );\n\n\t[ 'fire', '_addEventListener', '_removeEventListener' ].forEach( key => {\n\t\t( BubblingEmitterMixin as any )[ key ] = mixin.prototype[ key ];\n\t} );\n}\n\n/**\n * Update the event info bubbling fields.\n *\n * @param eventInfo The event info object to update.\n * @param eventPhase The current event phase.\n * @param currentTarget The current bubbling target.\n */\nfunction updateEventInfo(\n\teventInfo: EventInfo,\n\teventPhase: EventPhase,\n\tcurrentTarget: unknown\n) {\n\tif ( eventInfo instanceof BubblingEventInfo ) {\n\t\t( eventInfo as any )._eventPhase = eventPhase;\n\t\t( eventInfo as any )._currentTarget = currentTarget;\n\t}\n}\n\n/**\n * Fires the listener for the specified context. Returns `true` if event was stopped.\n *\n * @param eventInfo The `EventInfo` object.\n * @param eventArgs Additional arguments to be passed to the callbacks.\n * @returns True if event stop was called.\n */\nfunction fireListenerFor(\n\teventContexts: BubblingEventContexts,\n\tcontext: string | Node,\n\teventInfo: EventInfo,\n\t...eventArgs: Array<unknown>\n) {\n\tconst emitter = typeof context == 'string' ? eventContexts.get( context ) : getCustomContext( eventContexts, context );\n\n\tif ( !emitter ) {\n\t\treturn false;\n\t}\n\n\temitter.fire( eventInfo, ...eventArgs );\n\n\treturn eventInfo.stop.called;\n}\n\n/**\n * Returns an emitter for a specified view node.\n */\nfunction getCustomContext( eventContexts: BubblingEventContexts, node: Node ): Emitter | null {\n\tfor ( const [ context, emitter ] of eventContexts ) {\n\t\tif ( typeof context == 'function' && context( node ) ) {\n\t\t\treturn emitter;\n\t\t}\n\t}\n\n\treturn null;\n}\n\n/**\n * Returns bubbling contexts map for the source (emitter).\n */\nfunction getBubblingContexts( source: { [ x: string ]: any; [ contextsSymbol ]?: BubblingEventContexts } ) {\n\tif ( !source[ contextsSymbol ] ) {\n\t\tsource[ contextsSymbol ] = new Map();\n\t}\n\n\treturn source[ contextsSymbol ];\n}\n\n/**\n * Returns the deeper parent element for the range.\n */\nfunction getDeeperRangeParent( range: Range ): Node | null {\n\tif ( !range ) {\n\t\treturn null;\n\t}\n\n\tconst startParent = range.start.parent as Element;\n\tconst endParent = range.end.parent as Element;\n\n\tconst startPath = startParent.getPath();\n\tconst endPath = endParent.getPath();\n\n\treturn startPath.length > endPath.length ? startParent : endParent;\n}\n\n/**\n * Bubbling emitter for the view document.\n *\n * Bubbling emitter is triggering events in the context of specified {@link module:engine/view/element~Element view element} name,\n * predefined `'$text'`, `'$root'`, `'$document'` and `'$capture'` contexts, and context matchers provided as a function.\n *\n * Before bubbling starts, listeners for `'$capture'` context are triggered. Then the bubbling starts from the deeper selection\n * position (by firing event on the `'$text'` context) and propagates the view document tree up to the `'$root'` and finally\n * the listeners at `'$document'` context are fired (this is the default context).\n *\n * Examples:\n *\n * ```ts\n * // Listeners registered in the context of the view element names:\n * this.listenTo( viewDocument, 'enter', ( evt, data ) => {\n * \t// ...\n * }, { context: 'blockquote' } );\n *\n * this.listenTo( viewDocument, 'enter', ( evt, data ) => {\n * \t// ...\n * }, { context: 'li' } );\n *\n * // Listeners registered in the context of the '$text' and '$root' nodes.\n * this.listenTo( view.document, 'arrowKey', ( evt, data ) => {\n * \t// ...\n * }, { context: '$text', priority: 'high' } );\n *\n * this.listenTo( view.document, 'arrowKey', ( evt, data ) => {\n * \t// ...\n * }, { context: '$root' } );\n *\n * // Listeners registered in the context of custom callback function.\n * this.listenTo( view.document, 'arrowKey', ( evt, data ) => {\n * \t// ...\n * }, { context: isWidget } );\n *\n * this.listenTo( view.document, 'arrowKey', ( evt, data ) => {\n * \t// ...\n * }, { context: isWidget, priority: 'high' } );\n * ```\n *\n * Example flow for selection in text:\n *\n * ```xml\n * <blockquote><p>Foo[]bar</p></blockquote>\n * ```\n *\n * Fired events on contexts:\n * 1. `'$capture'`\n * 2. `'$text'`\n * 3. `'p'`\n * 4. `'blockquote'`\n * 5. `'$root'`\n * 6. `'$document'`\n *\n * Example flow for selection on element (i.e., Widget):\n *\n * ```xml\n * <blockquote><p>Foo[<widget/>]bar</p></blockquote>\n * ```\n *\n * Fired events on contexts:\n * 1. `'$capture'`\n * 2. *widget* (custom matcher)\n * 3. `'p'`\n * 4. `'blockquote'`\n * 5. `'$root'`\n * 6. `'$document'`\n *\n * There could be multiple listeners registered for the same context and at different priority levels:\n *\n * ```html\n * <p>Foo[]bar</p>\n * ```\n *\n * 1. `'$capture'` at priorities:\n * 1. `'highest'`\n * 2. `'high'`\n * 3. `'normal'`\n * 4. `'low'`\n * 5. `'lowest'`\n * 2. `'$text'` at priorities:\n * 1. `'highest'`\n * 2. `'high'`\n * 3. `'normal'`\n * 4. `'low'`\n * 5. `'lowest'`\n * 3. `'p'` at priorities:\n * 1. `'highest'`\n * 2. `'high'`\n * 3. `'normal'`\n * 4. `'low'`\n * 5. `'lowest'`\n * 4. `'$root'` at priorities:\n * 1. `'highest'`\n * 2. `'high'`\n * 3. `'normal'`\n * 4. `'low'`\n * 5. `'lowest'`\n * 5. `'$document'` at priorities:\n * 1. `'highest'`\n * 2. `'high'`\n * 3. `'normal'`\n * 4. `'low'`\n * 5. `'lowest'`\n */\nexport type BubblingEmitter = Emitter;\n\n/**\n * A context matcher function.\n *\n * Should return true for nodes that that match the custom context.\n */\nexport type BubblingEventContextFunction = ( node: Node ) => boolean;\n\n/**\n * Helper type that allows describing bubbling event. Extends `TEvent` so that:\n *\n * * the event is called with {@link module:engine/view/observer/bubblingeventinfo~BubblingEventInfo}`\n * instead of {@link module:utils/eventinfo~EventInfo}, and\n * * {@link ~BubblingCallbackOptions} can be specified as additional options.\n *\n * @typeParam TEvent The event description to extend.\n */\nexport type BubblingEvent<TEvent extends BaseEvent> = TEvent & {\n\teventInfo: BubblingEventInfo<TEvent[ 'name' ], ( TEvent extends { return: infer TReturn } ? TReturn : unknown )>;\n\tcallbackOptions: BubblingCallbackOptions;\n};\n\n/**\n * Additional options for registering a callback.\n */\nexport interface BubblingCallbackOptions extends CallbackOptions {\n\n\t/**\n\t * Specifies the context in which the event should be triggered to call the callback.\n\t *\n\t * @see ~BubblingEmitter\n\t */\n\tcontext?: ArrayOrItem<string | BubblingEventContextFunction>;\n}\n\ntype BubblingEventContexts = Map<string | BubblingEventContextFunction, Emitter>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/document\n */\n\nimport DocumentSelection from './documentselection';\nimport BubblingEmitterMixin from './observer/bubblingemittermixin';\n\nimport { Collection, ObservableMixin } from '@ckeditor/ckeditor5-utils';\n\nimport type { StylesProcessor } from './stylesmap';\nimport type RootEditableElement from './rooteditableelement';\nimport type DowncastWriter from './downcastwriter';\n\n// @if CK_DEBUG_ENGINE // const { logDocument } = require( '../dev-utils/utils' );\n\n/**\n * Document class creates an abstract layer over the content editable area, contains a tree of view elements and\n * {@link module:engine/view/documentselection~DocumentSelection view selection} associated with this document.\n */\nexport default class Document extends BubblingEmitterMixin( ObservableMixin() ) {\n\t/**\n\t * Selection done on this document.\n\t */\n\tpublic readonly selection: DocumentSelection;\n\n\t/**\n\t * Roots of the view tree. Collection of the {@link module:engine/view/element~Element view elements}.\n\t *\n\t * View roots are created as a result of binding between {@link module:engine/view/document~Document#roots} and\n\t * {@link module:engine/model/document~Document#roots} and this is handled by\n\t * {@link module:engine/controller/editingcontroller~EditingController}, so to create view root we need to create\n\t * model root using {@link module:engine/model/document~Document#createRoot}.\n\t */\n\tpublic readonly roots: Collection<RootEditableElement>;\n\n\t/**\n\t * The styles processor instance used by this document when normalizing styles.\n\t */\n\tpublic readonly stylesProcessor: StylesProcessor;\n\n\t/**\n\t * Defines whether document is in read-only mode.\n\t *\n\t * When document is read-ony then all roots are read-only as well and caret placed inside this root is hidden.\n\t *\n\t * @observable\n\t */\n\tdeclare public isReadOnly: boolean;\n\n\t/**\n\t * True if document is focused.\n\t *\n\t * This property is updated by the {@link module:engine/view/observer/focusobserver~FocusObserver}.\n\t * If the {@link module:engine/view/observer/focusobserver~FocusObserver} is disabled this property will not change.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public isFocused: boolean;\n\n\t/**\n\t * `true` while the user is making a selection in the document (e.g. holding the mouse button and moving the cursor).\n\t * When they stop selecting, the property goes back to `false`.\n\t *\n\t * This property is updated by the {@link module:engine/view/observer/selectionobserver~SelectionObserver}.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public isSelecting: boolean;\n\n\t/**\n\t * True if composition is in progress inside the document.\n\t *\n\t * This property is updated by the {@link module:engine/view/observer/compositionobserver~CompositionObserver}.\n\t * If the {@link module:engine/view/observer/compositionobserver~CompositionObserver} is disabled this property will not change.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public isComposing: boolean;\n\n\t/**\n\t * Post-fixer callbacks registered to the view document.\n\t */\n\tprivate readonly _postFixers = new Set<ViewDocumentPostFixer>();\n\n\t/**\n\t * Creates a Document instance.\n\t *\n\t * @param stylesProcessor The styles processor instance.\n\t */\n\tconstructor( stylesProcessor: StylesProcessor ) {\n\t\tsuper();\n\n\t\tthis.selection = new DocumentSelection();\n\t\tthis.roots = new Collection( { idProperty: 'rootName' } );\n\t\tthis.stylesProcessor = stylesProcessor;\n\n\t\tthis.set( 'isReadOnly', false );\n\t\tthis.set( 'isFocused', false );\n\t\tthis.set( 'isSelecting', false );\n\t\tthis.set( 'isComposing', false );\n\t}\n\n\t/**\n\t * Gets a {@link module:engine/view/document~Document#roots view root element} with the specified name. If the name is not\n\t * specific \"main\" root is returned.\n\t *\n\t * @param name Name of the root.\n\t * @returns The view root element with the specified name or null when there is no root of given name.\n\t */\n\tpublic getRoot( name: string = 'main' ): RootEditableElement | null {\n\t\treturn this.roots.get( name );\n\t}\n\n\t/**\n\t * Allows registering post-fixer callbacks. A post-fixers mechanism allows to update the view tree just before it is rendered\n\t * to the DOM.\n\t *\n\t * Post-fixers are executed right after all changes from the outermost change block were applied but\n\t * before the {@link module:engine/view/view~View#event:render render event} is fired. If a post-fixer callback made\n\t * a change, it should return `true`. When this happens, all post-fixers are fired again to check if something else should\n\t * not be fixed in the new document tree state.\n\t *\n\t * View post-fixers are useful when you want to apply some fixes whenever the view structure changes. Keep in mind that\n\t * changes executed in a view post-fixer should not break model-view mapping.\n\t *\n\t * The types of changes which should be safe:\n\t *\n\t * * adding or removing attribute from elements,\n\t * * changes inside of {@link module:engine/view/uielement~UIElement UI elements},\n\t * * {@link module:engine/controller/editingcontroller~EditingController#reconvertItem marking some of the model elements to be\n\t * re-converted}.\n\t *\n\t * Try to avoid changes which touch view structure:\n\t *\n\t * * you should not add or remove nor wrap or unwrap any view elements,\n\t * * you should not change the editor data model in a view post-fixer.\n\t *\n\t * As a parameter, a post-fixer callback receives a {@link module:engine/view/downcastwriter~DowncastWriter downcast writer}.\n\t *\n\t * Typically, a post-fixer will look like this:\n\t *\n\t * ```ts\n\t * editor.editing.view.document.registerPostFixer( writer => {\n\t * \tif ( checkSomeCondition() ) {\n\t * \t\twriter.doSomething();\n\t *\n\t * \t\t// Let other post-fixers know that something changed.\n\t * \t\treturn true;\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * Note that nothing happens right after you register a post-fixer (e.g. execute such a code in the console).\n\t * That is because adding a post-fixer does not execute it.\n\t * The post-fixer will be executed as soon as any change in the document needs to cause its rendering.\n\t * If you want to re-render the editor's view after registering the post-fixer then you should do it manually by calling\n\t * {@link module:engine/view/view~View#forceRender `view.forceRender()`}.\n\t *\n\t * If you need to register a callback which is executed when DOM elements are already updated,\n\t * use {@link module:engine/view/view~View#event:render render event}.\n\t */\n\tpublic registerPostFixer( postFixer: ViewDocumentPostFixer ): void {\n\t\tthis._postFixers.add( postFixer );\n\t}\n\n\t/**\n\t * Destroys this instance. Makes sure that all observers are destroyed and listeners removed.\n\t */\n\tpublic destroy(): void {\n\t\tthis.roots.map( root => root.destroy() );\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Performs post-fixer loops. Executes post-fixer callbacks as long as none of them has done any changes to the model.\n\t *\n\t * @internal\n\t */\n\tpublic _callPostFixers( writer: DowncastWriter ): void {\n\t\tlet wasFixed = false;\n\n\t\tdo {\n\t\t\tfor ( const callback of this._postFixers ) {\n\t\t\t\twasFixed = callback( writer );\n\n\t\t\t\tif ( wasFixed ) {\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t} while ( wasFixed );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public log( version: any ): void {\n\t// @if CK_DEBUG_ENGINE // \tlogDocument( this, version );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n/**\n * Document PostFixer.\n *\n * @see module:engine/view/document~Document#registerPostFixer\n */\nexport type ViewDocumentPostFixer = ( writer: DowncastWriter ) => boolean;\n\n/**\n * Enum representing type of the change.\n *\n * Possible values:\n *\n * * `children` - for child list changes,\n * * `attributes` - for element attributes changes,\n * * `text` - for text nodes changes.\n *\n * @typedef {String} module:engine/view/document~ChangeType\n */\nexport type ChangeType = 'children' | 'attributes' | 'text';\n\n/**\n * Event fired whenever document content layout changes. It is fired whenever content is\n * {@link module:engine/view/view~View#event:render rendered}, but should be also fired by observers in case of\n * other actions which may change layout, for instance when image loads.\n *\n * @eventName ~Document#layoutChanged\n */\nexport type ViewDocumentLayoutChangedEvent = {\n\tname: 'layoutChanged';\n\targs: [];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/attributeelement\n */\n\nimport Element, { type ElementAttributes } from './element';\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type DocumentFragment from './documentfragment';\nimport type Document from './document';\nimport type Node from './node';\n\n// Default attribute priority.\nconst DEFAULT_PRIORITY = 10;\n\n/**\n * Attribute elements are used to represent formatting elements in the view (think `<b>`, `<span style=\"font-size: 2em\">`, etc.).\n * Most often they are created when downcasting model text attributes.\n *\n * Editing engine does not define a fixed HTML DTD. This is why a feature developer needs to choose between various\n * types (container element, {@link module:engine/view/attributeelement~AttributeElement attribute element},\n * {@link module:engine/view/emptyelement~EmptyElement empty element}, etc) when developing a feature.\n *\n * To create a new attribute element instance use the\n * {@link module:engine/view/downcastwriter~DowncastWriter#createAttributeElement `DowncastWriter#createAttributeElement()`} method.\n */\nexport default class AttributeElement extends Element {\n\tpublic static readonly DEFAULT_PRIORITY: number = DEFAULT_PRIORITY;\n\n\t/**\n\t * Element priority. Decides in what order elements are wrapped by {@link module:engine/view/downcastwriter~DowncastWriter}.\n\t *\n\t * @internal\n\t * @readonly\n\t */\n\tpublic _priority: number = DEFAULT_PRIORITY;\n\n\t/**\n\t * Element identifier. If set, it is used by {@link module:engine/view/element~Element#isSimilar},\n\t * and then two elements are considered similar if, and only if they have the same `_id`.\n\t *\n\t * @internal\n\t * @readonly\n\t */\n\tpublic _id: string | number | null = null;\n\n\t/**\n\t * Keeps all the attribute elements that have the same {@link module:engine/view/attributeelement~AttributeElement#id ids}\n\t * and still exist in the view tree.\n\t *\n\t * This property is managed by {@link module:engine/view/downcastwriter~DowncastWriter}.\n\t */\n\tprivate readonly _clonesGroup: Set<AttributeElement> | null = null;\n\n\t/**\n\t * Creates an attribute element.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#createAttributeElement\n\t * @see module:engine/view/element~Element\n\t * @protected\n\t * @param document The document instance to which this element belongs.\n\t * @param name Node name.\n\t * @param attrs Collection of attributes.\n\t * @param children A list of nodes to be inserted into created element.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\tname: string,\n\t\tattrs?: ElementAttributes,\n\t\tchildren?: Node | Iterable<Node>\n\t) {\n\t\tsuper( document, name, attrs, children );\n\n\t\tthis.getFillerOffset = getFillerOffset;\n\t}\n\n\t/**\n\t * Element priority. Decides in what order elements are wrapped by {@link module:engine/view/downcastwriter~DowncastWriter}.\n\t */\n\tpublic get priority(): number {\n\t\treturn this._priority;\n\t}\n\n\t/**\n\t * Element identifier. If set, it is used by {@link module:engine/view/element~Element#isSimilar},\n\t * and then two elements are considered similar if, and only if they have the same `id`.\n\t */\n\tpublic get id(): string | number | null {\n\t\treturn this._id;\n\t}\n\n\t/**\n\t * Returns all {@link module:engine/view/attributeelement~AttributeElement attribute elements} that has the\n\t * same {@link module:engine/view/attributeelement~AttributeElement#id id} and are in the view tree (were not removed).\n\t *\n\t * Note: If this element has been removed from the tree, returned set will not include it.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError attribute-element-get-elements-with-same-id-no-id}\n\t * if this element has no `id`.\n\t *\n\t * @returns Set containing all the attribute elements\n\t * with the same `id` that were added and not removed from the view tree.\n\t */\n\tpublic getElementsWithSameId(): Set<AttributeElement> {\n\t\tif ( this.id === null ) {\n\t\t\t/**\n\t\t\t * Cannot get elements with the same id for an attribute element without id.\n\t\t\t *\n\t\t\t * @error attribute-element-get-elements-with-same-id-no-id\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'attribute-element-get-elements-with-same-id-no-id',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\treturn new Set( this._clonesGroup );\n\t}\n\n\t/**\n\t * Checks if this element is similar to other element.\n\t *\n\t * If none of elements has set {@link module:engine/view/attributeelement~AttributeElement#id}, then both elements\n\t * should have the same name, attributes and priority to be considered as similar. Two similar elements can contain\n\t * different set of children nodes.\n\t *\n\t * If at least one element has {@link module:engine/view/attributeelement~AttributeElement#id} set, then both\n\t * elements have to have the same {@link module:engine/view/attributeelement~AttributeElement#id} value to be\n\t * considered similar.\n\t *\n\t * Similarity is important for {@link module:engine/view/downcastwriter~DowncastWriter}. For example:\n\t *\n\t * * two following similar elements can be merged together into one, longer element,\n\t * * {@link module:engine/view/downcastwriter~DowncastWriter#unwrap} checks similarity of passed element and processed element to\n\t * decide whether processed element should be unwrapped,\n\t * * etc.\n\t */\n\tpublic override isSimilar( otherElement: Element ): boolean {\n\t\t// If any element has an `id` set, just compare the ids.\n\t\tif ( this.id !== null || ( otherElement as any ).id !== null ) {\n\t\t\treturn this.id === ( otherElement as any ).id;\n\t\t}\n\n\t\treturn super.isSimilar( otherElement ) && this.priority == ( otherElement as any ).priority;\n\t}\n\n\t/**\n\t * Clones provided element with priority.\n\t *\n\t * @internal\n\t * @param deep If set to `true` clones element and all its children recursively. When set to `false`,\n\t * element will be cloned without any children.\n\t * @returns Clone of this element.\n\t */\n\tpublic override _clone( deep: boolean = false ): this {\n\t\tconst cloned = super._clone( deep );\n\n\t\t// Clone priority too.\n\t\tcloned._priority = this._priority;\n\n\t\t// And id too.\n\t\tcloned._id = this._id;\n\n\t\treturn cloned;\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nAttributeElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'attributeElement' || type === 'view:attributeElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'element' || type === 'view:element' ||\n\t\t\ttype === 'node' || type === 'view:node';\n\t} else {\n\t\treturn name === this.name && (\n\t\t\ttype === 'attributeElement' || type === 'view:attributeElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'element' || type === 'view:element'\n\t\t);\n\t}\n};\n\n/**\n * Returns block {@link module:engine/view/filler~Filler filler} offset or `null` if block filler is not needed.\n *\n * @returns Block filler offset or `null` if block filler is not needed.\n */\nfunction getFillerOffset( this: AttributeElement ): number | null {\n\t// <b>foo</b> does not need filler.\n\tif ( nonUiChildrenCount( this ) ) {\n\t\treturn null;\n\t}\n\n\tlet element = this.parent;\n\n\t// <p><b></b></p> needs filler -> <p><b><br></b></p>\n\twhile ( element && element.is( 'attributeElement' ) ) {\n\t\tif ( nonUiChildrenCount( element ) > 1 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\telement = element.parent;\n\t}\n\n\tif ( !element || nonUiChildrenCount( element ) > 1 ) {\n\t\treturn null;\n\t}\n\n\t// Render block filler at the end of element (after all ui elements).\n\treturn this.childCount;\n}\n\n/**\n * Returns total count of children that are not {@link module:engine/view/uielement~UIElement UIElements}.\n */\nfunction nonUiChildrenCount( element: Element | DocumentFragment ): number {\n\treturn Array.from( element.getChildren() ).filter( element => !element.is( 'uiElement' ) ).length;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/emptyelement\n */\n\nimport Element, { type ElementAttributes } from './element';\nimport Node from './node';\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type Document from './document';\nimport type Item from './item';\n\n/**\n * Empty element class. It is used to represent elements that cannot contain any child nodes (for example `<img>` elements).\n *\n * To create a new empty element use the\n * {@link module:engine/view/downcastwriter~DowncastWriter#createEmptyElement `downcastWriter#createEmptyElement()`} method.\n */\nexport default class EmptyElement extends Element {\n\t/**\n\t * Creates new instance of EmptyElement.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-emptyelement-cannot-add` when third parameter is passed,\n\t * to inform that usage of EmptyElement is incorrect (adding child nodes to EmptyElement is forbidden).\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#createEmptyElement\n\t * @internal\n\t * @param document The document instance to which this element belongs.\n\t * @param name Node name.\n\t * @param attrs Collection of attributes.\n\t * @param children A list of nodes to be inserted into created element.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\tchildren?: Node | Iterable<Node>\n\t) {\n\t\tsuper( document, name, attributes, children );\n\n\t\tthis.getFillerOffset = getFillerOffset;\n\t}\n\n\t/**\n\t * Overrides {@link module:engine/view/element~Element#_insertChild} method.\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-emptyelement-cannot-add` to prevent\n\t * adding any child nodes to EmptyElement.\n\t *\n\t * @internal\n\t */\n\tpublic override _insertChild( index: number, items: Item | Iterable<Item> ): number {\n\t\tif ( items && ( items instanceof Node || Array.from( items as Iterable<Item> ).length > 0 ) ) {\n\t\t\t/**\n\t\t\t * Cannot add children to {@link module:engine/view/emptyelement~EmptyElement}.\n\t\t\t *\n\t\t\t * @error view-emptyelement-cannot-add\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'view-emptyelement-cannot-add',\n\t\t\t\t[ this, items ]\n\t\t\t);\n\t\t}\n\n\t\treturn 0;\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nEmptyElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'emptyElement' || type === 'view:emptyElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'element' || type === 'view:element' ||\n\t\t\ttype === 'node' || type === 'view:node';\n\t} else {\n\t\treturn name === this.name && (\n\t\t\ttype === 'emptyElement' || type === 'view:emptyElement' ||\n\t\t\ttype === 'element' || type === 'view:element'\n\t\t);\n\t}\n};\n\n/**\n * Returns `null` because block filler is not needed for EmptyElements.\n */\nfunction getFillerOffset() {\n\treturn null;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/uielement\n */\n\nimport Element, { type ElementAttributes } from './element';\nimport Node from './node';\nimport { CKEditorError, keyCodes } from '@ckeditor/ckeditor5-utils';\n\nimport type View from './view';\nimport type Document from './document';\nimport type DomConverter from './domconverter';\nimport type Item from './item';\nimport type { ViewDocumentArrowKeyEvent } from './observer/arrowkeysobserver';\nimport type { KeyEventData } from './observer/keyobserver';\n\ntype DomDocument = globalThis.Document;\ntype DomElement = globalThis.HTMLElement;\n\n/**\n * UI element class. It should be used to represent editing UI which needs to be injected into the editing view\n * If possible, you should keep your UI outside the editing view. However, if that is not possible,\n * UI elements can be used.\n *\n * How a UI element is rendered is in your control (you pass a callback to\n * {@link module:engine/view/downcastwriter~DowncastWriter#createUIElement `downcastWriter#createUIElement()`}).\n * The editor will ignore your UI element the selection cannot be placed in it, it is skipped (invisible) when\n * the user modifies the selection by using arrow keys and the editor does not listen to any mutations which\n * happen inside your UI elements.\n *\n * The limitation is that you cannot convert a model element to a UI element. UI elements need to be\n * created for {@link module:engine/model/markercollection~Marker markers} or as additinal elements\n * inside normal {@link module:engine/view/containerelement~ContainerElement container elements}.\n *\n * To create a new UI element use the\n * {@link module:engine/view/downcastwriter~DowncastWriter#createUIElement `downcastWriter#createUIElement()`} method.\n */\nexport default class UIElement extends Element {\n\t/**\n\t * Creates new instance of UIElement.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-uielement-cannot-add` when third parameter is passed,\n\t * to inform that usage of UIElement is incorrect (adding child nodes to UIElement is forbidden).\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#createUIElement\n\t * @internal\n\t * @param document The document instance to which this element belongs.\n\t * @param name Node name.\n\t * @param attrs Collection of attributes.\n\t * @param children A list of nodes to be inserted into created element.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\tname: string,\n\t\tattrs?: ElementAttributes,\n\t\tchildren?: Node | Iterable<Node>\n\t) {\n\t\tsuper( document, name, attrs, children );\n\n\t\tthis.getFillerOffset = getFillerOffset;\n\t}\n\n\t/**\n\t * Overrides {@link module:engine/view/element~Element#_insertChild} method.\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-uielement-cannot-add` to prevent adding any child nodes\n\t * to UIElement.\n\t *\n\t * @internal\n\t */\n\tpublic override _insertChild( index: number, items: Item | Iterable<Item> ): number {\n\t\tif ( items && ( items instanceof Node || Array.from( items as Iterable<Item> ).length > 0 ) ) {\n\t\t\t/**\n\t\t\t * Cannot add children to {@link module:engine/view/uielement~UIElement}.\n\t\t\t *\n\t\t\t * @error view-uielement-cannot-add\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-uielement-cannot-add', [ this, items ] );\n\t\t}\n\n\t\treturn 0;\n\t}\n\n\t/**\n\t * Renders this {@link module:engine/view/uielement~UIElement} to DOM. This method is called by\n\t * {@link module:engine/view/domconverter~DomConverter}.\n\t * Do not use inheritance to create custom rendering method, replace `render()` method instead:\n\t *\n\t * ```ts\n\t * const myUIElement = downcastWriter.createUIElement( 'span' );\n\t * myUIElement.render = function( domDocument, domConverter ) {\n\t * \tconst domElement = this.toDomElement( domDocument );\n\t *\n\t * \tdomConverter.setContentOf( domElement, '<b>this is ui element</b>' );\n\t *\n\t * \treturn domElement;\n\t * };\n\t * ```\n\t *\n\t * If changes in your UI element should trigger some editor UI update you should call\n\t * the {@link module:ui/editorui/editorui~EditorUI#update `editor.ui.update()`} method\n\t * after rendering your UI element.\n\t *\n\t * @param domConverter Instance of the DomConverter used to optimize the output.\n\t */\n\tpublic render( domDocument: DomDocument, domConverter: DomConverter ): DomElement {\n\t\t// Provide basic, default output.\n\t\treturn this.toDomElement( domDocument );\n\t}\n\n\t/**\n\t * Creates DOM element based on this view UIElement.\n\t * Note that each time this method is called new DOM element is created.\n\t */\n\tpublic toDomElement( domDocument: DomDocument ): DomElement {\n\t\tconst domElement = domDocument.createElement( this.name );\n\n\t\tfor ( const key of this.getAttributeKeys() ) {\n\t\t\tdomElement.setAttribute( key, this.getAttribute( key )! );\n\t\t}\n\n\t\treturn domElement;\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nUIElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'uiElement' || type === 'view:uiElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'element' || type === 'view:element' ||\n\t\t\ttype === 'node' || type === 'view:node';\n\t} else {\n\t\treturn name === this.name && (\n\t\t\ttype === 'uiElement' || type === 'view:uiElement' ||\n\t\t\ttype === 'element' || type === 'view:element'\n\t\t);\n\t}\n};\n\n/**\n * This function injects UI element handling to the given {@link module:engine/view/document~Document document}.\n *\n * A callback is added to {@link module:engine/view/document~Document#event:keydown document keydown event}.\n * The callback handles the situation when right arrow key is pressed and selection is collapsed before a UI element.\n * Without this handler, it would be impossible to \"jump over\" UI element using right arrow key.\n *\n * @param view View controller to which the quirks handling will be injected.\n */\nexport function injectUiElementHandling( view: View ): void {\n\tview.document.on<ViewDocumentArrowKeyEvent>( 'arrowKey', ( evt, data ) =>\n\t\tjumpOverUiElement( evt, data, view.domConverter ), { priority: 'low' } );\n}\n\n/**\n * Returns `null` because block filler is not needed for UIElements.\n */\nfunction getFillerOffset() {\n\treturn null;\n}\n\n/**\n * Selection cannot be placed in a `UIElement`. Whenever it is placed there, it is moved before it. This\n * causes a situation when it is impossible to jump over `UIElement` using right arrow key, because the selection\n * ends up in ui element (in DOM) and is moved back to the left. This handler fixes this situation.\n */\nfunction jumpOverUiElement( evt: unknown, data: KeyEventData, domConverter: DomConverter ) {\n\tif ( data.keyCode == keyCodes.arrowright ) {\n\t\tconst domSelection = data.domTarget.ownerDocument.defaultView!.getSelection()!;\n\t\tconst domSelectionCollapsed = domSelection.rangeCount == 1 && domSelection.getRangeAt( 0 ).collapsed;\n\n\t\t// Jump over UI element if selection is collapsed or shift key is pressed. These are the cases when selection would extend.\n\t\tif ( domSelectionCollapsed || data.shiftKey ) {\n\t\t\tconst domParent = domSelection.focusNode!;\n\t\t\tconst domOffset = domSelection.focusOffset;\n\n\t\t\tconst viewPosition = domConverter.domPositionToView( domParent, domOffset );\n\n\t\t\t// In case if dom element is not converted to view or is not mapped or something. Happens for example in some tests.\n\t\t\tif ( viewPosition === null ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Skip all following ui elements.\n\t\t\tlet jumpedOverAnyUiElement = false;\n\n\t\t\tconst nextViewPosition = viewPosition.getLastMatchingPosition( value => {\n\t\t\t\tif ( value.item.is( 'uiElement' ) ) {\n\t\t\t\t\t// Remember that there was at least one ui element.\n\t\t\t\t\tjumpedOverAnyUiElement = true;\n\t\t\t\t}\n\n\t\t\t\t// Jump over ui elements, jump over empty attribute elements, move up from inside of attribute element.\n\t\t\t\tif ( value.item.is( 'uiElement' ) || value.item.is( 'attributeElement' ) ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\t// Don't jump over text or don't get out of container element.\n\t\t\t\treturn false;\n\t\t\t} );\n\n\t\t\t// If anything has been skipped, fix position.\n\t\t\t// This `if` could be possibly omitted but maybe it is better not to mess with DOM selection if not needed.\n\t\t\tif ( jumpedOverAnyUiElement ) {\n\t\t\t\tconst newDomPosition = domConverter.viewPositionToDom( nextViewPosition )!;\n\n\t\t\t\tif ( domSelectionCollapsed ) {\n\t\t\t\t\t// Selection was collapsed, so collapse it at further position.\n\t\t\t\t\tdomSelection.collapse( newDomPosition.parent, newDomPosition.offset );\n\t\t\t\t} else {\n\t\t\t\t\t// Selection was not collapse, so extend it instead of collapsing.\n\t\t\t\t\tdomSelection.extend( newDomPosition.parent, newDomPosition.offset );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* eslint-disable @typescript-eslint/no-unused-vars */\n\n/**\n * @module engine/view/rawelement\n */\n\nimport Element, { type ElementAttributes } from './element';\nimport Node from './node';\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type Document from './document';\nimport type DomConverter from './domconverter';\nimport type Item from './item';\n\ntype DomElement = globalThis.HTMLElement;\n\n/**\n * The raw element class.\n *\n * The raw elements work as data containers (\"wrappers\", \"sandboxes\") but their children are not managed or\n * even recognized by the editor. This encapsulation allows integrations to maintain custom DOM structures\n * in the editor content without, for instance, worrying about compatibility with other editor features.\n * Raw elements are a perfect tool for integration with external frameworks and data sources.\n *\n * Unlike {@link module:engine/view/uielement~UIElement UI elements}, raw elements act like real editor\n * content (similar to {@link module:engine/view/containerelement~ContainerElement} or\n * {@link module:engine/view/emptyelement~EmptyElement}), they are considered by the editor selection and\n * {@link module:widget/utils~toWidget they can work as widgets}.\n *\n * To create a new raw element, use the\n * {@link module:engine/view/downcastwriter~DowncastWriter#createRawElement `downcastWriter#createRawElement()`} method.\n */\nexport default class RawElement extends Element {\n\t/**\n\t * Creates a new instance of a raw element.\n\t *\n\t * Throws the `view-rawelement-cannot-add` {@link module:utils/ckeditorerror~CKEditorError CKEditorError} when the `children`\n\t * parameter is passed to inform that the usage of `RawElement` is incorrect (adding child nodes to `RawElement` is forbidden).\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#createRawElement\n\t * @internal\n\t * @param document The document instance to which this element belongs.\n\t * @param name Node name.\n\t * @param attrs Collection of attributes.\n\t * @param children A list of nodes to be inserted into created element.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\tname: string,\n\t\tattrs?: ElementAttributes,\n\t\tchildren?: Node | Iterable<Node>\n\t) {\n\t\tsuper( document, name, attrs, children );\n\n\t\t// Returns `null` because filler is not needed for raw elements.\n\t\tthis.getFillerOffset = getFillerOffset;\n\t}\n\n\t/**\n\t * Overrides the {@link module:engine/view/element~Element#_insertChild} method.\n\t * Throws the `view-rawelement-cannot-add` {@link module:utils/ckeditorerror~CKEditorError CKEditorError} to prevent\n\t * adding any child nodes to a raw element.\n\t *\n\t * @internal\n\t */\n\tpublic override _insertChild( index: number, items: Item | Iterable<Item> ): number {\n\t\tif ( items && ( items instanceof Node || Array.from( items as Iterable<Item> ).length > 0 ) ) {\n\t\t\t/**\n\t\t\t * Cannot add children to a {@link module:engine/view/rawelement~RawElement} instance.\n\t\t\t *\n\t\t\t * @error view-rawelement-cannot-add\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'view-rawelement-cannot-add',\n\t\t\t\t[ this, items ]\n\t\t\t);\n\t\t}\n\n\t\treturn 0;\n\t}\n\n\t/**\n\t * This allows rendering the children of a {@link module:engine/view/rawelement~RawElement} on the DOM level.\n\t * This method is called by the {@link module:engine/view/domconverter~DomConverter} with the raw DOM element\n\t * passed as an argument, leaving the number and shape of the children up to the integrator.\n\t *\n\t * This method **must be defined** for the raw element to work:\n\t *\n\t * ```ts\n\t * const myRawElement = downcastWriter.createRawElement( 'div' );\n\t *\n\t * myRawElement.render = function( domElement, domConverter ) {\n\t * \tdomConverter.setContentOf( domElement, '<b>This is the raw content of myRawElement.</b>' );\n\t * };\n\t * ```\n\t *\n\t * @param domElement The native DOM element representing the raw view element.\n\t * @param domConverter Instance of the DomConverter used to optimize the output.\n\t */\n\tpublic render( domElement: DomElement, domConverter: DomConverter ): void {}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nRawElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'rawElement' || type === 'view:rawElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === this.name || type === 'view:' + this.name ||\n\t\t\ttype === 'element' || type === 'view:element' ||\n\t\t\ttype === 'node' || type === 'view:node';\n\t} else {\n\t\treturn name === this.name && (\n\t\t\ttype === 'rawElement' || type === 'view:rawElement' ||\n\t\t\ttype === 'element' || type === 'view:element'\n\t\t);\n\t}\n};\n\n/**\n * Returns `null` because block filler is not needed for raw elements.\n */\nfunction getFillerOffset() {\n\treturn null;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/documentfragment\n */\n\nimport TypeCheckable from './typecheckable';\nimport Text from './text';\nimport TextProxy from './textproxy';\n\nimport { EmitterMixin, isIterable } from '@ckeditor/ckeditor5-utils';\n\nimport type { default as Document, ChangeType } from './document';\n\nimport type Item from './item';\nimport type Node from './node';\n\n/**\n * Document fragment.\n *\n * To create a new document fragment instance use the\n * {@link module:engine/view/upcastwriter~UpcastWriter#createDocumentFragment `UpcastWriter#createDocumentFragment()`}\n * method.\n */\nexport default class DocumentFragment extends EmitterMixin( TypeCheckable ) implements Iterable<Node> {\n\t/**\n\t * The document to which this document fragment belongs.\n\t */\n\tpublic readonly document: Document;\n\n\t/**\n\t * Array of child nodes.\n\t */\n\tprivate readonly _children: Array<Node> = [];\n\n\t/**\n\t * Map of custom properties.\n\t * Custom properties can be added to document fragment instance.\n\t */\n\tprivate readonly _customProperties = new Map<string | symbol, unknown>();\n\n\t/**\n\t * Creates new DocumentFragment instance.\n\t *\n\t * @internal\n\t * @param document The document to which this document fragment belongs.\n\t * @param children A list of nodes to be inserted into the created document fragment.\n\t */\n\tconstructor( document: Document, children?: Node | Iterable<Node> ) {\n\t\tsuper();\n\n\t\tthis.document = document;\n\n\t\tif ( children ) {\n\t\t\tthis._insertChild( 0, children );\n\t\t}\n\t}\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * Iterates over nodes added to this document fragment.\n\t */\n\tpublic [ Symbol.iterator ](): Iterator<Node> {\n\t\treturn this._children[ Symbol.iterator ]();\n\t}\n\n\t/**\n\t * Number of child nodes in this document fragment.\n\t */\n\tpublic get childCount(): number {\n\t\treturn this._children.length;\n\t}\n\n\t/**\n\t * Is `true` if there are no nodes inside this document fragment, `false` otherwise.\n\t */\n\tpublic get isEmpty(): boolean {\n\t\treturn this.childCount === 0;\n\t}\n\n\t/**\n\t * Artificial root of `DocumentFragment`. Returns itself. Added for compatibility reasons.\n\t */\n\tpublic get root(): this {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Artificial parent of `DocumentFragment`. Returns `null`. Added for compatibility reasons.\n\t */\n\tpublic get parent(): null {\n\t\treturn null;\n\t}\n\n\t/**\n\t * Artificial element name. Returns `undefined`. Added for compatibility reasons.\n\t */\n\tpublic get name(): undefined {\n\t\treturn undefined;\n\t}\n\n\t/**\n\t * Returns the custom property value for the given key.\n\t */\n\tpublic getCustomProperty( key: string | symbol ): unknown {\n\t\treturn this._customProperties.get( key );\n\t}\n\n\t/**\n\t * Returns an iterator which iterates over this document fragment's custom properties.\n\t * Iterator provides `[ key, value ]` pairs for each stored property.\n\t */\n\tpublic* getCustomProperties(): Iterable<[ string | symbol, unknown ]> {\n\t\tyield* this._customProperties.entries();\n\t}\n\n\t/**\n\t * {@link module:engine/view/documentfragment~DocumentFragment#_insertChild Insert} a child node or a list of child nodes at the end\n\t * and sets the parent of these nodes to this fragment.\n\t *\n\t * @internal\n\t * @param items Items to be inserted.\n\t * @returns Number of appended nodes.\n\t */\n\tpublic _appendChild( items: Item | string | Iterable<Item | string> ): number {\n\t\treturn this._insertChild( this.childCount, items );\n\t}\n\n\t/**\n\t * Gets child at the given index.\n\t *\n\t * @param index Index of child.\n\t * @returns Child node.\n\t */\n\tpublic getChild( index: number ): Node {\n\t\treturn this._children[ index ];\n\t}\n\n\t/**\n\t * Gets index of the given child node. Returns `-1` if child node is not found.\n\t *\n\t * @param node Child node.\n\t * @returns Index of the child node.\n\t */\n\tpublic getChildIndex( node: Node ): number {\n\t\treturn this._children.indexOf( node );\n\t}\n\n\t/**\n\t * Gets child nodes iterator.\n\t *\n\t * @returns Child nodes iterator.\n\t */\n\tpublic getChildren(): IterableIterator<Node> {\n\t\treturn this._children[ Symbol.iterator ]();\n\t}\n\n\t/**\n\t * Inserts a child node or a list of child nodes on the given index and sets the parent of these nodes to\n\t * this fragment.\n\t *\n\t * @internal\n\t * @param index Position where nodes should be inserted.\n\t * @param items Items to be inserted.\n\t * @returns Number of inserted nodes.\n\t */\n\tpublic _insertChild( index: number, items: Item | string | Iterable<Item | string> ): number {\n\t\tthis._fireChange( 'children', this );\n\t\tlet count = 0;\n\n\t\tconst nodes = normalize( this.document, items );\n\n\t\tfor ( const node of nodes ) {\n\t\t\t// If node that is being added to this element is already inside another element, first remove it from the old parent.\n\t\t\tif ( node.parent !== null ) {\n\t\t\t\tnode._remove();\n\t\t\t}\n\n\t\t\t( node as any ).parent = this;\n\n\t\t\tthis._children.splice( index, 0, node );\n\t\t\tindex++;\n\t\t\tcount++;\n\t\t}\n\n\t\treturn count;\n\t}\n\n\t/**\n\t * Removes number of child nodes starting at the given index and set the parent of these nodes to `null`.\n\t *\n\t * @internal\n\t * @param index Number of the first node to remove.\n\t * @param howMany Number of nodes to remove.\n\t * @returns The array of removed nodes.\n\t */\n\tpublic _removeChildren( index: number, howMany: number = 1 ): Array<Node> {\n\t\tthis._fireChange( 'children', this );\n\n\t\tfor ( let i = index; i < index + howMany; i++ ) {\n\t\t\t( this._children[ i ] as any ).parent = null;\n\t\t}\n\n\t\treturn this._children.splice( index, howMany );\n\t}\n\n\t/**\n\t * Fires `change` event with given type of the change.\n\t *\n\t * @internal\n\t * @param type Type of the change.\n\t * @param node Changed node.\n\t */\n\tpublic _fireChange( type: ChangeType, node: Node | DocumentFragment ): void {\n\t\tthis.fire( 'change:' + type, node );\n\t}\n\n\t/**\n\t * Sets a custom property. They can be used to add special data to elements.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#setCustomProperty\n\t * @internal\n\t */\n\tpublic _setCustomProperty( key: string | symbol, value: unknown ): void {\n\t\tthis._customProperties.set( key, value );\n\t}\n\n\t/**\n\t * Removes the custom property stored under the given key.\n\t *\n\t * @see module:engine/view/downcastwriter~DowncastWriter#removeCustomProperty\n\t * @internal\n\t * @returns Returns true if property was removed.\n\t */\n\tpublic _removeCustomProperty( key: string | symbol ): boolean {\n\t\treturn this._customProperties.delete( key );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public printTree(): string {\n\t// @if CK_DEBUG_ENGINE // \tlet string = 'ViewDocumentFragment: [';\n\n\t// @if CK_DEBUG_ENGINE // \tfor ( const child of this.getChildren() as any ) {\n\t// @if CK_DEBUG_ENGINE // \t\tif ( child.is( '$text' ) ) {\n\t// @if CK_DEBUG_ENGINE // \t\t\tstring += '\\n' + '\\t'.repeat( 1 ) + child.data;\n\t// @if CK_DEBUG_ENGINE // \t\t} else {\n\t// @if CK_DEBUG_ENGINE // \t\t\tstring += '\\n' + child.printTree( 1 );\n\t// @if CK_DEBUG_ENGINE // \t\t}\n\t// @if CK_DEBUG_ENGINE // \t}\n\n\t// @if CK_DEBUG_ENGINE // \tstring += '\\n]';\n\n\t// @if CK_DEBUG_ENGINE // \treturn string;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logTree(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( this.printTree() );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nDocumentFragment.prototype.is = function( type: string ): boolean {\n\treturn type === 'documentFragment' || type === 'view:documentFragment';\n};\n\n/**\n * Converts strings to Text and non-iterables to arrays.\n */\nfunction normalize( document: Document, nodes: Item | string | Iterable<Item | string> ): Array<Node> {\n\t// Separate condition because string is iterable.\n\tif ( typeof nodes == 'string' ) {\n\t\treturn [ new Text( document, nodes ) ];\n\t}\n\n\tif ( !isIterable( nodes ) ) {\n\t\tnodes = [ nodes ];\n\t}\n\n\t// Array.from to enable .map() on non-arrays.\n\treturn Array.from( nodes )\n\t\t.map( node => {\n\t\t\tif ( typeof node == 'string' ) {\n\t\t\t\treturn new Text( document, node );\n\t\t\t}\n\n\t\t\tif ( node instanceof TextProxy ) {\n\t\t\t\treturn new Text( document, node.data );\n\t\t\t}\n\n\t\t\treturn node;\n\t\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/downcastwriter\n */\n\nimport Position, { type PositionOffset } from './position';\nimport Range from './range';\nimport Selection, {\n\ttype PlaceOrOffset,\n\ttype Selectable,\n\ttype SelectionOptions\n} from './selection';\nimport ContainerElement from './containerelement';\nimport AttributeElement from './attributeelement';\nimport EmptyElement from './emptyelement';\nimport UIElement from './uielement';\nimport RawElement from './rawelement';\nimport { CKEditorError, isIterable } from '@ckeditor/ckeditor5-utils';\nimport DocumentFragment from './documentfragment';\nimport Text from './text';\nimport EditableElement from './editableelement';\nimport { isPlainObject } from 'lodash-es';\n\nimport type Document from './document';\nimport type Node from './node';\nimport type { default as Element, ElementAttributes } from './element';\nimport type DomConverter from './domconverter';\nimport type Item from './item';\nimport type { SlotFilter } from '../conversion/downcasthelpers';\n\ntype DomDocument = globalThis.Document;\ntype DomElement = globalThis.HTMLElement;\n\n/**\n * View downcast writer.\n *\n * It provides a set of methods used to manipulate view nodes.\n *\n * Do not create an instance of this writer manually. To modify a view structure, use\n * the {@link module:engine/view/view~View#change `View#change()`} block.\n *\n * The `DowncastWriter` is designed to work with semantic views which are the views that were/are being downcasted from the model.\n * To work with ordinary views (e.g. parsed from a pasted content) use the\n * {@link module:engine/view/upcastwriter~UpcastWriter upcast writer}.\n *\n * Read more about changing the view in the {@glink framework/architecture/editing-engine#changing-the-view Changing the view}\n * section of the {@glink framework/architecture/editing-engine Editing engine architecture} guide.\n */\nexport default class DowncastWriter {\n\t/**\n\t * The view document instance in which this writer operates.\n\t */\n\tpublic readonly document: Document;\n\n\t/**\n\t * Holds references to the attribute groups that share the same {@link module:engine/view/attributeelement~AttributeElement#id id}.\n\t * The keys are `id`s, the values are `Set`s holding {@link module:engine/view/attributeelement~AttributeElement}s.\n\t */\n\tprivate readonly _cloneGroups = new Map<string | number, Set<AttributeElement>>();\n\n\t/**\n\t * The slot factory used by the `elementToStructure` downcast helper.\n\t */\n\tprivate _slotFactory: ( ( writer: DowncastWriter, modeOrFilter: string | SlotFilter ) => Element ) | null = null;\n\n\t/**\n\t * @param document The view document instance.\n\t */\n\tconstructor( document: Document ) {\n\t\tthis.document = document;\n\t}\n\n\t/**\n\t * Sets {@link module:engine/view/documentselection~DocumentSelection selection's} ranges and direction to the\n\t * specified location based on the given {@link module:engine/view/selection~Selectable selectable}.\n\t *\n\t * Usage:\n\t *\n\t * ```ts\n\t * // Sets collapsed selection at the position of given item and offset.\n\t * const paragraph = writer.createContainerElement( 'p' );\n\t * writer.setSelection( paragraph, offset );\n\t * ```\n\t *\n\t * Creates a range inside an {@link module:engine/view/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * ```ts\n\t * writer.setSelection( paragraph, 'in' );\n\t * ```\n\t *\n\t * Creates a range on the {@link module:engine/view/item~Item item} which starts before the item and ends just after the item.\n\t *\n\t * ```ts\n\t * writer.setSelection( paragraph, 'on' );\n\t * ```\n\t *\n\t * `DowncastWriter#setSelection()` allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Sets selection as backward.\n\t * writer.setSelection( element, 'in', { backward: true } );\n\t *\n\t * // Sets selection as fake.\n\t * // Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * // This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * // represented in other way, for example by applying proper CSS class.\n\t * writer.setSelection( element, 'in', { fake: true } );\n\t *\n\t * // Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * // (and be properly handled by screen readers).\n\t * writer.setSelection( element, 'in', { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #setSelection:SELECTABLE `setSelection( selectable, options )`}.\n\t *\n\t * @label NODE_OFFSET\n\t */\n\tpublic setSelection( selectable: Node, placeOrOffset: PlaceOrOffset, options?: SelectionOptions ): void;\n\n\t/**\n\t * Sets {@link module:engine/view/documentselection~DocumentSelection selection's} ranges and direction to the\n\t * specified location based on the given {@link module:engine/view/selection~Selectable selectable}.\n\t *\n\t * Usage:\n\t *\n\t * ```ts\n\t * // Sets selection to the given range.\n\t * const range = writer.createRange( start, end );\n\t * writer.setSelection( range );\n\t *\n\t * // Sets backward selection to the given range.\n\t * const range = writer.createRange( start, end );\n\t * writer.setSelection( range );\n\t *\n\t * // Sets selection to given ranges.\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( start2, end2 ) ];\n\t * writer.setSelection( range );\n\t *\n\t * // Sets selection to the other selection.\n\t * const otherSelection = writer.createSelection();\n\t * writer.setSelection( otherSelection );\n\t *\n\t * // Sets collapsed selection at the given position.\n\t * const position = writer.createPositionFromPath( root, path );\n\t * writer.setSelection( position );\n\t *\n\t * // Removes all ranges.\n\t * writer.setSelection( null );\n\t * ```\n\t *\n\t * `DowncastWriter#setSelection()` allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Sets selection as backward.\n\t * writer.setSelection( range, { backward: true } );\n\t *\n\t * // Sets selection as fake.\n\t * // Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * // This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * // represented in other way, for example by applying proper CSS class.\n\t * writer.setSelection( range, { fake: true } );\n\t *\n\t * // Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * // (and be properly handled by screen readers).\n\t * writer.setSelection( range, { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #setSelection:NODE_OFFSET `setSelection( node, placeOrOffset, options )`}.\n\t *\n\t * @label SELECTABLE\n\t */\n\tpublic setSelection( selectable: Exclude<Selectable, Node>, options?: SelectionOptions ): void;\n\n\tpublic setSelection( ...args: Parameters<Selection[ 'setTo' ]> ): void {\n\t\tthis.document.selection._setTo( ...args );\n\t}\n\n\t/**\n\t * Moves {@link module:engine/view/documentselection~DocumentSelection#focus selection's focus} to the specified location.\n\t *\n\t * The location can be specified in the same form as {@link module:engine/view/view~View#createPositionAt view.createPositionAt()}\n\t * parameters.\n\t *\n\t * @param Offset or one of the flags. Used only when the first parameter is a {@link module:engine/view/item~Item view item}.\n\t */\n\tpublic setSelectionFocus( itemOrPosition: Item | Position, offset?: PositionOffset ): void {\n\t\tthis.document.selection._setFocus( itemOrPosition, offset );\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/documentfragment~DocumentFragment} instance.\n\t *\n\t * @param children A list of nodes to be inserted into the created document fragment.\n\t * @returns The created document fragment.\n\t */\n\tpublic createDocumentFragment( children?: Node | Iterable<Node> ): DocumentFragment {\n\t\treturn new DocumentFragment( this.document, children );\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/text~Text text node}.\n\t *\n\t * ```ts\n\t * writer.createText( 'foo' );\n\t * ```\n\t *\n\t * @param data The text's data.\n\t * @returns The created text node.\n\t */\n\tpublic createText( data: string ): Text {\n\t\treturn new Text( this.document, data );\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/attributeelement~AttributeElement}.\n\t *\n\t * ```ts\n\t * writer.createAttributeElement( 'strong' );\n\t * writer.createAttributeElement( 'a', { href: 'foo.bar' } );\n\t *\n\t * // Make `<a>` element contain other attributes element so the `<a>` element is not broken.\n\t * writer.createAttributeElement( 'a', { href: 'foo.bar' }, { priority: 5 } );\n\t *\n\t * // Set `id` of a marker element so it is not joined or merged with \"normal\" elements.\n\t * writer.createAttributeElement( 'span', { class: 'my-marker' }, { id: 'marker:my' } );\n\t * ```\n\t *\n\t * @param name Name of the element.\n\t * @param attributes Element's attributes.\n\t * @param options Element's options.\n\t * @param options.priority Element's {@link module:engine/view/attributeelement~AttributeElement#priority priority}.\n\t * @param options.id Element's {@link module:engine/view/attributeelement~AttributeElement#id id}.\n\t * @param options.renderUnsafeAttributes A list of attribute names that should be rendered in the editing\n\t * pipeline even though they would normally be filtered out by unsafe attribute detection mechanisms.\n\t * @returns Created element.\n\t */\n\tpublic createAttributeElement(\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\toptions: {\n\t\t\tpriority?: number;\n\t\t\tid?: number | string;\n\t\t\trenderUnsafeAttributes?: Array<string>;\n\t\t} = {}\n\t): AttributeElement {\n\t\tconst attributeElement = new AttributeElement( this.document, name, attributes );\n\n\t\tif ( typeof options.priority === 'number' ) {\n\t\t\tattributeElement._priority = options.priority;\n\t\t}\n\n\t\tif ( options.id ) {\n\t\t\tattributeElement._id = options.id;\n\t\t}\n\n\t\tif ( options.renderUnsafeAttributes ) {\n\t\t\tattributeElement._unsafeAttributesToRender.push( ...options.renderUnsafeAttributes );\n\t\t}\n\n\t\treturn attributeElement;\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/containerelement~ContainerElement}.\n\t *\n\t * ```ts\n\t * writer.createContainerElement( 'p' );\n\t *\n\t * // Create element with custom attributes.\n\t * writer.createContainerElement( 'div', { id: 'foo-bar', 'data-baz': '123' } );\n\t *\n\t * // Create element with custom styles.\n\t * writer.createContainerElement( 'p', { style: 'font-weight: bold; padding-bottom: 10px' } );\n\t *\n\t * // Create element with custom classes.\n\t * writer.createContainerElement( 'p', { class: 'foo bar baz' } );\n\t *\n\t * // Create element with specific options.\n\t * writer.createContainerElement( 'span', { class: 'placeholder' }, { renderUnsafeAttributes: [ 'foo' ] } );\n\t * ```\n\t *\n\t * @label WITHOUT_CHILDREN\n\t * @param name Name of the element.\n\t * @param attributes Elements attributes.\n\t * @param options Element's options.\n\t * @param options.renderUnsafeAttributes A list of attribute names that should be rendered in the editing\n\t * pipeline even though they would normally be filtered out by unsafe attribute detection mechanisms.\n\t * @returns Created element.\n\t */\n\tpublic createContainerElement(\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\toptions?: { renderUnsafeAttributes?: Array<string> }\n\t): ContainerElement;\n\n\t/**\n\t * Creates a new {@link module:engine/view/containerelement~ContainerElement} with children.\n\t *\n\t * ```ts\n\t * // Create element with children.\n\t * writer.createContainerElement( 'figure', { class: 'image' }, [\n\t * \twriter.createEmptyElement( 'img' ),\n\t * \twriter.createContainerElement( 'figcaption' )\n\t * ] );\n\t *\n\t * // Create element with specific options.\n\t * writer.createContainerElement( 'figure', { class: 'image' }, [\n\t * \twriter.createEmptyElement( 'img' ),\n\t * \twriter.createContainerElement( 'figcaption' )\n\t * ], { renderUnsafeAttributes: [ 'foo' ] } );\n\t * ```\n\t *\n\t * @label WITH_CHILDREN\n\t * @param name Name of the element.\n\t * @param attributes Elements attributes.\n\t * @param children A node or a list of nodes to be inserted into the created element.\n\t * If no children were specified, element's `options` can be passed in this argument.\n\t * @param options Element's options.\n\t * @param options.renderUnsafeAttributes A list of attribute names that should be rendered in the editing\n\t * pipeline even though they would normally be filtered out by unsafe attribute detection mechanisms.\n\t * @returns Created element.\n\t */\n\tpublic createContainerElement(\n\t\tname: string,\n\t\tattributes: ElementAttributes,\n\t\tchildren: Node | Iterable<Node>,\n\t\toptions?: { renderUnsafeAttributes?: Array<string> }\n\t): ContainerElement;\n\n\tpublic createContainerElement(\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\tchildrenOrOptions: Node | Iterable<Node> | { renderUnsafeAttributes?: Array<string> } = {},\n\t\toptions: { renderUnsafeAttributes?: Array<string> } = {}\n\t): ContainerElement {\n\t\tlet children = null;\n\n\t\tif ( isPlainObject( childrenOrOptions ) ) {\n\t\t\toptions = childrenOrOptions as { renderUnsafeAttributes?: Array<string> };\n\t\t} else {\n\t\t\tchildren = childrenOrOptions;\n\t\t}\n\n\t\tconst containerElement = new ContainerElement( this.document, name, attributes, children as Node | Iterable<Node> );\n\n\t\tif ( options.renderUnsafeAttributes ) {\n\t\t\tcontainerElement._unsafeAttributesToRender.push( ...options.renderUnsafeAttributes );\n\t\t}\n\n\t\treturn containerElement;\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/editableelement~EditableElement}.\n\t *\n\t * ```ts\n\t * writer.createEditableElement( 'div' );\n\t * writer.createEditableElement( 'div', { id: 'foo-1234' } );\n\t * ```\n\t *\n\t * Note: The editable element is to be used in the editing pipeline. Usually, together with\n\t * {@link module:widget/utils~toWidgetEditable `toWidgetEditable()`}.\n\t *\n\t * @param name Name of the element.\n\t * @param attributes Elements attributes.\n\t * @param options Element's options.\n\t * @param options.renderUnsafeAttributes A list of attribute names that should be rendered in the editing\n\t * pipeline even though they would normally be filtered out by unsafe attribute detection mechanisms.\n\t * @returns Created element.\n\t */\n\tpublic createEditableElement(\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\toptions: {\n\t\t\trenderUnsafeAttributes?: Array<string>;\n\t\t} = {}\n\t): EditableElement {\n\t\tconst editableElement = new EditableElement( this.document, name, attributes );\n\n\t\tif ( options.renderUnsafeAttributes ) {\n\t\t\teditableElement._unsafeAttributesToRender.push( ...options.renderUnsafeAttributes );\n\t\t}\n\n\t\treturn editableElement;\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/emptyelement~EmptyElement}.\n\t *\n\t * ```ts\n\t * writer.createEmptyElement( 'img' );\n\t * writer.createEmptyElement( 'img', { id: 'foo-1234' } );\n\t * ```\n\t *\n\t * @param name Name of the element.\n\t * @param attributes Elements attributes.\n\t * @param options Element's options.\n\t * @param options.renderUnsafeAttributes A list of attribute names that should be rendered in the editing\n\t * pipeline even though they would normally be filtered out by unsafe attribute detection mechanisms.\n\t * @returns Created element.\n\t */\n\tpublic createEmptyElement(\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\toptions: {\n\t\t\trenderUnsafeAttributes?: Array<string>;\n\t\t} = {}\n\t): EmptyElement {\n\t\tconst emptyElement = new EmptyElement( this.document, name, attributes );\n\n\t\tif ( options.renderUnsafeAttributes ) {\n\t\t\temptyElement._unsafeAttributesToRender.push( ...options.renderUnsafeAttributes );\n\t\t}\n\n\t\treturn emptyElement;\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/uielement~UIElement}.\n\t *\n\t * ```ts\n\t * writer.createUIElement( 'span' );\n\t * writer.createUIElement( 'span', { id: 'foo-1234' } );\n\t * ```\n\t *\n\t * A custom render function can be provided as the third parameter:\n\t *\n\t * ```ts\n\t * writer.createUIElement( 'span', null, function( domDocument ) {\n\t * \tconst domElement = this.toDomElement( domDocument );\n\t * \tdomElement.innerHTML = '<b>this is ui element</b>';\n\t *\n\t * \treturn domElement;\n\t * } );\n\t * ```\n\t *\n\t * Unlike {@link #createRawElement raw elements}, UI elements are by no means editor content, for instance,\n\t * they are ignored by the editor selection system.\n\t *\n\t * You should not use UI elements as data containers. Check out {@link #createRawElement} instead.\n\t *\n\t * @param name The name of the element.\n\t * @param attributes Element attributes.\n\t * @param renderFunction A custom render function.\n\t * @returns The created element.\n\t */\n\tpublic createUIElement(\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\trenderFunction?: ( this: UIElement, domDocument: DomDocument, domConverter: DomConverter ) => DomElement\n\t): UIElement {\n\t\tconst uiElement = new UIElement( this.document, name, attributes );\n\n\t\tif ( renderFunction ) {\n\t\t\tuiElement.render = renderFunction;\n\t\t}\n\n\t\treturn uiElement;\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/rawelement~RawElement}.\n\t *\n\t * ```ts\n\t * writer.createRawElement( 'span', { id: 'foo-1234' }, function( domElement ) {\n\t * \tdomElement.innerHTML = '<b>This is the raw content of the raw element.</b>';\n\t * } );\n\t * ```\n\t *\n\t * Raw elements work as data containers (\"wrappers\", \"sandboxes\") but their children are not managed or\n\t * even recognized by the editor. This encapsulation allows integrations to maintain custom DOM structures\n\t * in the editor content without, for instance, worrying about compatibility with other editor features.\n\t * Raw elements are a perfect tool for integration with external frameworks and data sources.\n\t *\n\t * Unlike {@link #createUIElement UI elements}, raw elements act like \"real\" editor content (similar to\n\t * {@link module:engine/view/containerelement~ContainerElement} or {@link module:engine/view/emptyelement~EmptyElement}),\n\t * and they are considered by the editor selection.\n\t *\n\t * You should not use raw elements to render the UI in the editor content. Check out {@link #createUIElement `#createUIElement()`}\n\t * instead.\n\t *\n\t * @param name The name of the element.\n\t * @param attributes Element attributes.\n\t * @param renderFunction A custom render function.\n\t * @param options Element's options.\n\t * @param options.renderUnsafeAttributes A list of attribute names that should be rendered in the editing\n\t * pipeline even though they would normally be filtered out by unsafe attribute detection mechanisms.\n\t * @returns The created element.\n\t */\n\tpublic createRawElement(\n\t\tname: string,\n\t\tattributes?: ElementAttributes,\n\t\trenderFunction?: ( domElement: DomElement, domConverter: DomConverter ) => void,\n\t\toptions: {\n\t\t\trenderUnsafeAttributes?: Array<string>;\n\t\t} = {}\n\t): RawElement {\n\t\tconst rawElement = new RawElement( this.document, name, attributes );\n\n\t\tif ( renderFunction ) {\n\t\t\trawElement.render = renderFunction;\n\t\t}\n\n\t\tif ( options.renderUnsafeAttributes ) {\n\t\t\trawElement._unsafeAttributesToRender.push( ...options.renderUnsafeAttributes );\n\t\t}\n\n\t\treturn rawElement;\n\t}\n\n\t/**\n\t * Adds or overwrites the element's attribute with a specified key and value.\n\t *\n\t * ```ts\n\t * writer.setAttribute( 'href', 'http://ckeditor.com', linkElement );\n\t * ```\n\t *\n\t * @param key The attribute key.\n\t * @param value The attribute value.\n\t */\n\tpublic setAttribute( key: string, value: unknown, element: Element ): void {\n\t\telement._setAttribute( key, value );\n\t}\n\n\t/**\n\t * Removes attribute from the element.\n\t *\n\t * ```ts\n\t * writer.removeAttribute( 'href', linkElement );\n\t * ```\n\t *\n\t * @param key Attribute key.\n\t */\n\tpublic removeAttribute( key: string, element: Element ): void {\n\t\telement._removeAttribute( key );\n\t}\n\n\t/**\n\t * Adds specified class to the element.\n\t *\n\t * ```ts\n\t * writer.addClass( 'foo', linkElement );\n\t * writer.addClass( [ 'foo', 'bar' ], linkElement );\n\t * ```\n\t */\n\tpublic addClass( className: string | Array<string>, element: Element ): void {\n\t\telement._addClass( className );\n\t}\n\n\t/**\n\t * Removes specified class from the element.\n\t *\n\t * ```ts\n\t * writer.removeClass( 'foo', linkElement );\n\t * writer.removeClass( [ 'foo', 'bar' ], linkElement );\n\t * ```\n\t */\n\tpublic removeClass( className: string | Array<string>, element: Element ): void {\n\t\telement._removeClass( className );\n\t}\n\n\t/**\n\t * Adds style to the element.\n\t *\n\t * ```ts\n\t * writer.setStyle( 'color', 'red', element );\n\t * ```\n\t *\n\t * **Note**: The passed style can be normalized if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#set `StylesMap#set()`} for details.\n\t *\n\t * @label KEY_VALUE\n\t * @param property Property name.\n\t * @param value Value to set.\n\t * @param element Element to set styles on.\n\t */\n\tpublic setStyle( property: string, value: string, element: Element ): void;\n\n\t/**\n\t * Adds many styles to the element.\n\t *\n\t * ```ts\n\t * writer.setStyle( {\n\t * \tcolor: 'red',\n\t * \tposition: 'fixed'\n\t * }, element );\n\t * ```\n\t *\n\t * **Note**: The passed style can be normalized if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#set `StylesMap#set()`} for details.\n\t *\n\t * @label OBJECT\n\t * @param property Object with key - value pairs.\n\t * @param element Element to set styles on.\n\t */\n\tpublic setStyle( property: Record<string, string>, element: Element ): void;\n\n\tpublic setStyle(\n\t\tproperty: string | Record<string, string>,\n\t\tvalue: string | Element,\n\t\telement?: Element\n\t): void\n\t{\n\t\tif ( isPlainObject( property ) && element === undefined ) {\n\t\t\t( value as Element )._setStyle( property as Record<string, string> );\n\t\t} else {\n\t\t\telement!._setStyle( property as string, value as string );\n\t\t}\n\t}\n\n\t/**\n\t * Removes specified style from the element.\n\t *\n\t * ```ts\n\t * writer.removeStyle( 'color', element ); // Removes 'color' style.\n\t * writer.removeStyle( [ 'color', 'border-top' ], element ); // Removes both 'color' and 'border-top' styles.\n\t * ```\n\t *\n\t * **Note**: This method can work with normalized style names if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#remove `StylesMap#remove()`} for details.\n\t */\n\tpublic removeStyle( property: string | Array<string>, element: Element ): void {\n\t\telement._removeStyle( property );\n\t}\n\n\t/**\n\t * Sets a custom property on element. Unlike attributes, custom properties are not rendered to the DOM,\n\t * so they can be used to add special data to elements.\n\t */\n\tpublic setCustomProperty( key: string | symbol, value: unknown, element: Element | DocumentFragment ): void {\n\t\telement._setCustomProperty( key, value );\n\t}\n\n\t/**\n\t * Removes a custom property stored under the given key.\n\t *\n\t * @returns Returns true if property was removed.\n\t */\n\tpublic removeCustomProperty( key: string | symbol, element: Element | DocumentFragment ): boolean {\n\t\treturn element._removeCustomProperty( key );\n\t}\n\n\t/**\n\t * Breaks attribute elements at the provided position or at the boundaries of a provided range. It breaks attribute elements\n\t * up to their first ancestor that is a container element.\n\t *\n\t * In following examples `<p>` is a container, `<b>` and `<u>` are attribute elements:\n\t *\n\t * ```html\n\t * <p>foo<b><u>bar{}</u></b></p> -> <p>foo<b><u>bar</u></b>[]</p>\n\t * <p>foo<b><u>{}bar</u></b></p> -> <p>foo{}<b><u>bar</u></b></p>\n\t * <p>foo<b><u>b{}ar</u></b></p> -> <p>foo<b><u>b</u></b>[]<b><u>ar</u></b></p>\n\t * <p><b>fo{o</b><u>ba}r</u></p> -> <p><b>fo</b><b>o</b><u>ba</u><u>r</u></b></p>\n\t * ```\n\t *\n\t * **Note:** {@link module:engine/view/documentfragment~DocumentFragment DocumentFragment} is treated like a container.\n\t *\n\t * **Note:** The difference between {@link module:engine/view/downcastwriter~DowncastWriter#breakAttributes breakAttributes()} and\n\t * {@link module:engine/view/downcastwriter~DowncastWriter#breakContainer breakContainer()} is that `breakAttributes()` breaks all\n\t * {@link module:engine/view/attributeelement~AttributeElement attribute elements} that are ancestors of a given `position`,\n\t * up to the first encountered {@link module:engine/view/containerelement~ContainerElement container element}.\n\t * `breakContainer()` assumes that a given `position` is directly in the container element and breaks that container element.\n\t *\n\t * Throws the `view-writer-invalid-range-container` {@link module:utils/ckeditorerror~CKEditorError CKEditorError}\n\t * when the {@link module:engine/view/range~Range#start start}\n\t * and {@link module:engine/view/range~Range#end end} positions of a passed range are not placed inside same parent container.\n\t *\n\t * Throws the `view-writer-cannot-break-empty-element` {@link module:utils/ckeditorerror~CKEditorError CKEditorError}\n\t * when trying to break attributes inside an {@link module:engine/view/emptyelement~EmptyElement EmptyElement}.\n\t *\n\t * Throws the `view-writer-cannot-break-ui-element` {@link module:utils/ckeditorerror~CKEditorError CKEditorError}\n\t * when trying to break attributes inside a {@link module:engine/view/uielement~UIElement UIElement}.\n\t *\n\t * @see module:engine/view/attributeelement~AttributeElement\n\t * @see module:engine/view/containerelement~ContainerElement\n\t * @see module:engine/view/downcastwriter~DowncastWriter#breakContainer\n\t * @param positionOrRange The position where to break attribute elements.\n\t * @returns The new position or range, after breaking the attribute elements.\n\t */\n\tpublic breakAttributes( positionOrRange: Position | Range ): Position | Range {\n\t\tif ( positionOrRange instanceof Position ) {\n\t\t\treturn this._breakAttributes( positionOrRange );\n\t\t} else {\n\t\t\treturn this._breakAttributesRange( positionOrRange );\n\t\t}\n\t}\n\n\t/**\n\t * Breaks a {@link module:engine/view/containerelement~ContainerElement container view element} into two, at the given position.\n\t * The position has to be directly inside the container element and cannot be in the root. It does not break the conrainer view element\n\t * if the position is at the beginning or at the end of its parent element.\n\t *\n\t * ```html\n\t * <p>foo^bar</p> -> <p>foo</p><p>bar</p>\n\t * <div><p>foo</p>^<p>bar</p></div> -> <div><p>foo</p></div><div><p>bar</p></div>\n\t * <p>^foobar</p> -> ^<p>foobar</p>\n\t * <p>foobar^</p> -> <p>foobar</p>^\n\t * ```\n\t *\n\t * **Note:** The difference between {@link module:engine/view/downcastwriter~DowncastWriter#breakAttributes breakAttributes()} and\n\t * {@link module:engine/view/downcastwriter~DowncastWriter#breakContainer breakContainer()} is that `breakAttributes()` breaks all\n\t * {@link module:engine/view/attributeelement~AttributeElement attribute elements} that are ancestors of a given `position`,\n\t * up to the first encountered {@link module:engine/view/containerelement~ContainerElement container element}.\n\t * `breakContainer()` assumes that the given `position` is directly in the container element and breaks that container element.\n\t *\n\t * @see module:engine/view/attributeelement~AttributeElement\n\t * @see module:engine/view/containerelement~ContainerElement\n\t * @see module:engine/view/downcastwriter~DowncastWriter#breakAttributes\n\t * @param position The position where to break the element.\n\t * @returns The position between broken elements. If an element has not been broken,\n\t * the returned position is placed either before or after it.\n\t */\n\tpublic breakContainer( position: Position ): Position {\n\t\tconst element = position.parent;\n\n\t\tif ( !( element.is( 'containerElement' ) ) ) {\n\t\t\t/**\n\t\t\t * Trying to break an element which is not a container element.\n\t\t\t *\n\t\t\t * @error view-writer-break-non-container-element\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-writer-break-non-container-element', this.document );\n\t\t}\n\n\t\tif ( !element.parent ) {\n\t\t\t/**\n\t\t\t * Trying to break root element.\n\t\t\t *\n\t\t\t * @error view-writer-break-root\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-writer-break-root', this.document );\n\t\t}\n\n\t\tif ( position.isAtStart ) {\n\t\t\treturn Position._createBefore( element );\n\t\t} else if ( !position.isAtEnd ) {\n\t\t\tconst newElement = element._clone( false );\n\n\t\t\tthis.insert( Position._createAfter( element ), newElement as any );\n\n\t\t\tconst sourceRange = new Range( position, Position._createAt( element, 'end' ) );\n\t\t\tconst targetPosition = new Position( newElement, 0 );\n\n\t\t\tthis.move( sourceRange, targetPosition );\n\t\t}\n\n\t\treturn Position._createAfter( element );\n\t}\n\n\t/**\n\t * Merges {@link module:engine/view/attributeelement~AttributeElement attribute elements}. It also merges text nodes if needed.\n\t * Only {@link module:engine/view/attributeelement~AttributeElement#isSimilar similar} attribute elements can be merged.\n\t *\n\t * In following examples `<p>` is a container and `<b>` is an attribute element:\n\t *\n\t * ```html\n\t * <p>foo[]bar</p> -> <p>foo{}bar</p>\n\t * <p><b>foo</b>[]<b>bar</b></p> -> <p><b>foo{}bar</b></p>\n\t * <p><b foo=\"bar\">a</b>[]<b foo=\"baz\">b</b></p> -> <p><b foo=\"bar\">a</b>[]<b foo=\"baz\">b</b></p>\n\t * ```\n\t *\n\t * It will also take care about empty attributes when merging:\n\t *\n\t * ```html\n\t * <p><b>[]</b></p> -> <p>[]</p>\n\t * <p><b>foo</b><i>[]</i><b>bar</b></p> -> <p><b>foo{}bar</b></p>\n\t * ```\n\t *\n\t * **Note:** Difference between {@link module:engine/view/downcastwriter~DowncastWriter#mergeAttributes mergeAttributes} and\n\t * {@link module:engine/view/downcastwriter~DowncastWriter#mergeContainers mergeContainers} is that `mergeAttributes` merges two\n\t * {@link module:engine/view/attributeelement~AttributeElement attribute elements} or {@link module:engine/view/text~Text text nodes}\n\t * while `mergeContainer` merges two {@link module:engine/view/containerelement~ContainerElement container elements}.\n\t *\n\t * @see module:engine/view/attributeelement~AttributeElement\n\t * @see module:engine/view/containerelement~ContainerElement\n\t * @see module:engine/view/downcastwriter~DowncastWriter#mergeContainers\n\t * @param position Merge position.\n\t * @returns Position after merge.\n\t */\n\tpublic mergeAttributes( position: Position ): Position {\n\t\tconst positionOffset = position.offset;\n\t\tconst positionParent = position.parent;\n\n\t\t// When inside text node - nothing to merge.\n\t\tif ( positionParent.is( '$text' ) ) {\n\t\t\treturn position;\n\t\t}\n\n\t\t// When inside empty attribute - remove it.\n\t\tif ( positionParent.is( 'attributeElement' ) && positionParent.childCount === 0 ) {\n\t\t\tconst parent = positionParent.parent;\n\t\t\tconst offset = positionParent.index;\n\n\t\t\tpositionParent._remove();\n\t\t\tthis._removeFromClonedElementsGroup( positionParent );\n\n\t\t\treturn this.mergeAttributes( new Position( parent!, offset! ) );\n\t\t}\n\n\t\tconst nodeBefore = ( positionParent as Element ).getChild( positionOffset - 1 );\n\t\tconst nodeAfter = ( positionParent as Element ).getChild( positionOffset );\n\n\t\t// Position should be placed between two nodes.\n\t\tif ( !nodeBefore || !nodeAfter ) {\n\t\t\treturn position;\n\t\t}\n\n\t\t// When position is between two text nodes.\n\t\tif ( nodeBefore.is( '$text' ) && nodeAfter.is( '$text' ) ) {\n\t\t\treturn mergeTextNodes( nodeBefore, nodeAfter );\n\t\t}\n\t\t// When position is between two same attribute elements.\n\t\telse if ( nodeBefore.is( 'attributeElement' ) && nodeAfter.is( 'attributeElement' ) && nodeBefore.isSimilar( nodeAfter ) ) {\n\t\t\t// Move all children nodes from node placed after selection and remove that node.\n\t\t\tconst count = nodeBefore.childCount;\n\t\t\tnodeBefore._appendChild( nodeAfter.getChildren() );\n\n\t\t\tnodeAfter._remove();\n\t\t\tthis._removeFromClonedElementsGroup( nodeAfter );\n\n\t\t\t// New position is located inside the first node, before new nodes.\n\t\t\t// Call this method recursively to merge again if needed.\n\t\t\treturn this.mergeAttributes( new Position( nodeBefore, count ) );\n\t\t}\n\n\t\treturn position;\n\t}\n\n\t/**\n\t * Merges two {@link module:engine/view/containerelement~ContainerElement container elements} that are before and after given position.\n\t * Precisely, the element after the position is removed and it's contents are moved to element before the position.\n\t *\n\t * ```html\n\t * <p>foo</p>^<p>bar</p> -> <p>foo^bar</p>\n\t * <div>foo</div>^<p>bar</p> -> <div>foo^bar</div>\n\t * ```\n\t *\n\t * **Note:** Difference between {@link module:engine/view/downcastwriter~DowncastWriter#mergeAttributes mergeAttributes} and\n\t * {@link module:engine/view/downcastwriter~DowncastWriter#mergeContainers mergeContainers} is that `mergeAttributes` merges two\n\t * {@link module:engine/view/attributeelement~AttributeElement attribute elements} or {@link module:engine/view/text~Text text nodes}\n\t * while `mergeContainer` merges two {@link module:engine/view/containerelement~ContainerElement container elements}.\n\t *\n\t * @see module:engine/view/attributeelement~AttributeElement\n\t * @see module:engine/view/containerelement~ContainerElement\n\t * @see module:engine/view/downcastwriter~DowncastWriter#mergeAttributes\n\t * @param position Merge position.\n\t * @returns Position after merge.\n\t */\n\tpublic mergeContainers( position: Position ): Position {\n\t\tconst prev = position.nodeBefore;\n\t\tconst next = position.nodeAfter;\n\n\t\tif ( !prev || !next || !prev.is( 'containerElement' ) || !next.is( 'containerElement' ) ) {\n\t\t\t/**\n\t\t\t * Element before and after given position cannot be merged.\n\t\t\t *\n\t\t\t * @error view-writer-merge-containers-invalid-position\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-writer-merge-containers-invalid-position', this.document );\n\t\t}\n\n\t\tconst lastChild = prev.getChild( prev.childCount - 1 );\n\t\tconst newPosition = lastChild instanceof Text ? Position._createAt( lastChild, 'end' ) : Position._createAt( prev, 'end' );\n\n\t\tthis.move( Range._createIn( next ), Position._createAt( prev, 'end' ) );\n\t\tthis.remove( Range._createOn( next ) );\n\n\t\treturn newPosition;\n\t}\n\n\t/**\n\t * Inserts a node or nodes at specified position. Takes care about breaking attributes before insertion\n\t * and merging them afterwards.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-insert-invalid-node` when nodes to insert\n\t * contains instances that are not {@link module:engine/view/text~Text Texts},\n\t * {@link module:engine/view/attributeelement~AttributeElement AttributeElements},\n\t * {@link module:engine/view/containerelement~ContainerElement ContainerElements},\n\t * {@link module:engine/view/emptyelement~EmptyElement EmptyElements},\n\t * {@link module:engine/view/rawelement~RawElement RawElements} or\n\t * {@link module:engine/view/uielement~UIElement UIElements}.\n\t *\n\t * @param position Insertion position.\n\t * @param nodes Node or nodes to insert.\n\t * @returns Range around inserted nodes.\n\t */\n\tpublic insert( position: Position, nodes: Node | Iterable<Node> ): Range {\n\t\tnodes = isIterable( nodes ) ? [ ...nodes ] : [ nodes ];\n\n\t\t// Check if nodes to insert are instances of AttributeElements, ContainerElements, EmptyElements, UIElements or Text.\n\t\tvalidateNodesToInsert( nodes, this.document );\n\n\t\t// Group nodes in batches of nodes that require or do not require breaking an AttributeElements.\n\t\tconst nodeGroups = ( nodes as Array<Node> ).reduce( ( groups: Array<{ breakAttributes: boolean; nodes: Array<Node> }>, node ) => {\n\t\t\tconst lastGroup = groups[ groups.length - 1 ];\n\n\t\t\t// Break attributes on nodes that do exist in the model tree so they can have attributes, other elements\n\t\t\t// can't have an attribute in model and won't get wrapped with an AttributeElement while down-casted.\n\t\t\tconst breakAttributes = !node.is( 'uiElement' );\n\n\t\t\tif ( !lastGroup || lastGroup.breakAttributes != breakAttributes ) {\n\t\t\t\tgroups.push( {\n\t\t\t\t\tbreakAttributes,\n\t\t\t\t\tnodes: [ node ]\n\t\t\t\t} );\n\t\t\t} else {\n\t\t\t\tlastGroup.nodes.push( node );\n\t\t\t}\n\n\t\t\treturn groups;\n\t\t}, [] );\n\n\t\t// Insert nodes in batches.\n\t\tlet start = null;\n\t\tlet end = position;\n\n\t\tfor ( const { nodes, breakAttributes } of nodeGroups ) {\n\t\t\tconst range = this._insertNodes( end, nodes, breakAttributes );\n\n\t\t\tif ( !start ) {\n\t\t\t\tstart = range.start;\n\t\t\t}\n\n\t\t\tend = range.end;\n\t\t}\n\n\t\t// When no nodes were inserted - return collapsed range.\n\t\tif ( !start ) {\n\t\t\treturn new Range( position );\n\t\t}\n\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Removes provided range from the container.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-invalid-range-container` when\n\t * {@link module:engine/view/range~Range#start start} and {@link module:engine/view/range~Range#end end} positions are not placed inside\n\t * same parent container.\n\t *\n\t * @param rangeOrItem Range to remove from container\n\t * or an {@link module:engine/view/item~Item item} to remove. If range is provided, after removing, it will be updated\n\t * to a collapsed range showing the new position.\n\t * @returns Document fragment containing removed nodes.\n\t */\n\tpublic remove( rangeOrItem: Range | Item ): DocumentFragment {\n\t\tconst range = rangeOrItem instanceof Range ? rangeOrItem : Range._createOn( rangeOrItem );\n\n\t\tvalidateRangeContainer( range, this.document );\n\n\t\t// If range is collapsed - nothing to remove.\n\t\tif ( range.isCollapsed ) {\n\t\t\treturn new DocumentFragment( this.document );\n\t\t}\n\n\t\t// Break attributes at range start and end.\n\t\tconst { start: breakStart, end: breakEnd } = this._breakAttributesRange( range, true );\n\t\tconst parentContainer = breakStart.parent as Element;\n\n\t\tconst count = breakEnd.offset - breakStart.offset;\n\n\t\t// Remove nodes in range.\n\t\tconst removed = parentContainer._removeChildren( breakStart.offset, count );\n\n\t\tfor ( const node of removed ) {\n\t\t\tthis._removeFromClonedElementsGroup( node );\n\t\t}\n\n\t\t// Merge after removing.\n\t\tconst mergePosition = this.mergeAttributes( breakStart );\n\t\t( range as any ).start = mergePosition;\n\t\t( range as any ).end = mergePosition.clone();\n\n\t\t// Return removed nodes.\n\t\treturn new DocumentFragment( this.document, removed );\n\t}\n\n\t/**\n\t * Removes matching elements from given range.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-invalid-range-container` when\n\t * {@link module:engine/view/range~Range#start start} and {@link module:engine/view/range~Range#end end} positions are not placed inside\n\t * same parent container.\n\t *\n\t * @param range Range to clear.\n\t * @param element Element to remove.\n\t */\n\tpublic clear( range: Range, element: Element ): void {\n\t\tvalidateRangeContainer( range, this.document );\n\n\t\t// Create walker on given range.\n\t\t// We walk backward because when we remove element during walk it modifies range end position.\n\t\tconst walker = range.getWalker( {\n\t\t\tdirection: 'backward',\n\t\t\tignoreElementEnd: true\n\t\t} );\n\n\t\t// Let's walk.\n\t\tfor ( const current of walker ) {\n\t\t\tconst item = current.item;\n\t\t\tlet rangeToRemove;\n\n\t\t\t// When current item matches to the given element.\n\t\t\tif ( item.is( 'element' ) && element.isSimilar( item ) ) {\n\t\t\t\t// Create range on this element.\n\t\t\t\trangeToRemove = Range._createOn( item );\n\t\t\t\t// When range starts inside Text or TextProxy element.\n\t\t\t} else if ( !current.nextPosition.isAfter( range.start ) && item.is( '$textProxy' ) ) {\n\t\t\t\t// We need to check if parent of this text matches to given element.\n\t\t\t\tconst parentElement = item.getAncestors().find( ancestor => {\n\t\t\t\t\treturn ancestor.is( 'element' ) && element.isSimilar( ancestor );\n\t\t\t\t} );\n\n\t\t\t\t// If it is then create range inside this element.\n\t\t\t\tif ( parentElement ) {\n\t\t\t\t\trangeToRemove = Range._createIn( parentElement as Element );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// If we have found element to remove.\n\t\t\tif ( rangeToRemove ) {\n\t\t\t\t// We need to check if element range stick out of the given range and truncate if it is.\n\t\t\t\tif ( rangeToRemove.end.isAfter( range.end ) ) {\n\t\t\t\t\t( rangeToRemove as any ).end = range.end;\n\t\t\t\t}\n\n\t\t\t\tif ( rangeToRemove.start.isBefore( range.start ) ) {\n\t\t\t\t\t( rangeToRemove as any ).start = range.start;\n\t\t\t\t}\n\n\t\t\t\t// At the end we remove range with found element.\n\t\t\t\tthis.remove( rangeToRemove );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Moves nodes from provided range to target position.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-invalid-range-container` when\n\t * {@link module:engine/view/range~Range#start start} and {@link module:engine/view/range~Range#end end} positions are not placed inside\n\t * same parent container.\n\t *\n\t * @param sourceRange Range containing nodes to move.\n\t * @param targetPosition Position to insert.\n\t * @returns Range in target container. Inserted nodes are placed between\n\t * {@link module:engine/view/range~Range#start start} and {@link module:engine/view/range~Range#end end} positions.\n\t */\n\tpublic move( sourceRange: Range, targetPosition: Position ): Range {\n\t\tlet nodes;\n\n\t\tif ( targetPosition.isAfter( sourceRange.end ) ) {\n\t\t\ttargetPosition = this._breakAttributes( targetPosition, true );\n\n\t\t\tconst parent = targetPosition.parent as Element;\n\t\t\tconst countBefore = parent.childCount;\n\n\t\t\tsourceRange = this._breakAttributesRange( sourceRange, true );\n\n\t\t\tnodes = this.remove( sourceRange );\n\n\t\t\ttargetPosition.offset += ( parent.childCount - countBefore );\n\t\t} else {\n\t\t\tnodes = this.remove( sourceRange );\n\t\t}\n\n\t\treturn this.insert( targetPosition, nodes );\n\t}\n\n\t/**\n\t * Wraps elements within range with provided {@link module:engine/view/attributeelement~AttributeElement AttributeElement}.\n\t * If a collapsed range is provided, it will be wrapped only if it is equal to view selection.\n\t *\n\t * If a collapsed range was passed and is same as selection, the selection\n\t * will be moved to the inside of the wrapped attribute element.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError} `view-writer-invalid-range-container`\n\t * when {@link module:engine/view/range~Range#start}\n\t * and {@link module:engine/view/range~Range#end} positions are not placed inside same parent container.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError} `view-writer-wrap-invalid-attribute` when passed attribute element is not\n\t * an instance of {@link module:engine/view/attributeelement~AttributeElement AttributeElement}.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError} `view-writer-wrap-nonselection-collapsed-range` when passed range\n\t * is collapsed and different than view selection.\n\t *\n\t * @param range Range to wrap.\n\t * @param attribute Attribute element to use as wrapper.\n\t * @returns range Range after wrapping, spanning over wrapping attribute element.\n\t */\n\tpublic wrap( range: Range, attribute: AttributeElement ): Range {\n\t\tif ( !( attribute instanceof AttributeElement ) ) {\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'view-writer-wrap-invalid-attribute',\n\t\t\t\tthis.document\n\t\t\t);\n\t\t}\n\n\t\tvalidateRangeContainer( range, this.document );\n\n\t\tif ( !range.isCollapsed ) {\n\t\t\t// Non-collapsed range. Wrap it with the attribute element.\n\t\t\treturn this._wrapRange( range, attribute );\n\t\t} else {\n\t\t\t// Collapsed range. Wrap position.\n\t\t\tlet position = range.start;\n\n\t\t\tif ( position.parent.is( 'element' ) && !_hasNonUiChildren( position.parent ) ) {\n\t\t\t\tposition = position.getLastMatchingPosition( value => value.item.is( 'uiElement' ) );\n\t\t\t}\n\n\t\t\tposition = this._wrapPosition( position, attribute );\n\t\t\tconst viewSelection = this.document.selection;\n\n\t\t\t// If wrapping position is equal to view selection, move view selection inside wrapping attribute element.\n\t\t\tif ( viewSelection.isCollapsed && viewSelection.getFirstPosition()!.isEqual( range.start ) ) {\n\t\t\t\tthis.setSelection( position );\n\t\t\t}\n\n\t\t\treturn new Range( position );\n\t\t}\n\t}\n\n\t/**\n\t * Unwraps nodes within provided range from attribute element.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-invalid-range-container` when\n\t * {@link module:engine/view/range~Range#start start} and {@link module:engine/view/range~Range#end end} positions are not placed inside\n\t * same parent container.\n\t */\n\tpublic unwrap( range: Range, attribute: AttributeElement ): Range {\n\t\tif ( !( attribute instanceof AttributeElement ) ) {\n\t\t\t/**\n\t\t\t * The `attribute` passed to {@link module:engine/view/downcastwriter~DowncastWriter#unwrap `DowncastWriter#unwrap()`}\n\t\t\t * must be an instance of {@link module:engine/view/attributeelement~AttributeElement `AttributeElement`}.\n\t\t\t *\n\t\t\t * @error view-writer-unwrap-invalid-attribute\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'view-writer-unwrap-invalid-attribute',\n\t\t\t\tthis.document\n\t\t\t);\n\t\t}\n\n\t\tvalidateRangeContainer( range, this.document );\n\n\t\t// If range is collapsed - nothing to unwrap.\n\t\tif ( range.isCollapsed ) {\n\t\t\treturn range;\n\t\t}\n\n\t\t// Break attributes at range start and end.\n\t\tconst { start: breakStart, end: breakEnd } = this._breakAttributesRange( range, true );\n\t\tconst parentContainer = breakStart.parent as Element;\n\n\t\t// Unwrap children located between break points.\n\t\tconst newRange = this._unwrapChildren( parentContainer, breakStart.offset, breakEnd.offset, attribute );\n\n\t\t// Merge attributes at the both ends and return a new range.\n\t\tconst start = this.mergeAttributes( newRange.start );\n\n\t\t// If start position was merged - move end position back.\n\t\tif ( !start.isEqual( newRange.start ) ) {\n\t\t\tnewRange.end.offset--;\n\t\t}\n\n\t\tconst end = this.mergeAttributes( newRange.end );\n\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Renames element by creating a copy of renamed element but with changed name and then moving contents of the\n\t * old element to the new one. Keep in mind that this will invalidate all {@link module:engine/view/position~Position positions} which\n\t * has renamed element as {@link module:engine/view/position~Position#parent a parent}.\n\t *\n\t * New element has to be created because `Element#tagName` property in DOM is readonly.\n\t *\n\t * Since this function creates a new element and removes the given one, the new element is returned to keep reference.\n\t *\n\t * @param newName New name for element.\n\t * @param viewElement Element to be renamed.\n\t * @returns Element created due to rename.\n\t */\n\tpublic rename( newName: string, viewElement: ContainerElement ): ContainerElement {\n\t\tconst newElement = new ContainerElement( this.document, newName, viewElement.getAttributes() );\n\n\t\tthis.insert( Position._createAfter( viewElement ), newElement );\n\t\tthis.move( Range._createIn( viewElement ), Position._createAt( newElement, 0 ) );\n\t\tthis.remove( Range._createOn( viewElement ) );\n\n\t\treturn newElement;\n\t}\n\n\t/**\n\t * Cleans up memory by removing obsolete cloned elements group from the writer.\n\t *\n\t * Should be used whenever all {@link module:engine/view/attributeelement~AttributeElement attribute elements}\n\t * with the same {@link module:engine/view/attributeelement~AttributeElement#id id} are going to be removed from the view and\n\t * the group will no longer be needed.\n\t *\n\t * Cloned elements group are not removed automatically in case if the group is still needed after all its elements\n\t * were removed from the view.\n\t *\n\t * Keep in mind that group names are equal to the `id` property of the attribute element.\n\t *\n\t * @param groupName Name of the group to clear.\n\t */\n\tpublic clearClonedElementsGroup( groupName: string ): void {\n\t\tthis._cloneGroups.delete( groupName );\n\t}\n\n\t/**\n\t * Creates position at the given location. The location can be specified as:\n\t *\n\t * * a {@link module:engine/view/position~Position position},\n\t * * parent element and offset (offset defaults to `0`),\n\t * * parent element and `'end'` (sets position at the end of that element),\n\t * * {@link module:engine/view/item~Item view item} and `'before'` or `'after'` (sets position before or after given view item).\n\t *\n\t * This method is a shortcut to other constructors such as:\n\t *\n\t * * {@link #createPositionBefore},\n\t * * {@link #createPositionAfter},\n\t *\n\t * @param offset Offset or one of the flags. Used only when the first parameter is a {@link module:engine/view/item~Item view item}.\n\t */\n\tpublic createPositionAt( itemOrPosition: Item | Position, offset?: PositionOffset ): Position {\n\t\treturn Position._createAt( itemOrPosition, offset );\n\t}\n\n\t/**\n\t * Creates a new position after given view item.\n\t *\n\t * @param item View item after which the position should be located.\n\t */\n\tpublic createPositionAfter( item: Item ): Position {\n\t\treturn Position._createAfter( item );\n\t}\n\n\t/**\n\t * Creates a new position before given view item.\n\t *\n\t * @param item View item before which the position should be located.\n\t */\n\tpublic createPositionBefore( item: Item ): Position {\n\t\treturn Position._createBefore( item );\n\t}\n\n\t/**\n\t * Creates a range spanning from `start` position to `end` position.\n\t *\n\t * **Note:** This factory method creates its own {@link module:engine/view/position~Position} instances basing on passed values.\n\t *\n\t * @param start Start position.\n\t * @param end End position. If not set, range will be collapsed at `start` position.\n\t */\n\tpublic createRange( start: Position, end?: Position | null ): Range {\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Creates a range that starts before given {@link module:engine/view/item~Item view item} and ends after it.\n\t */\n\tpublic createRangeOn( item: Item ): Range {\n\t\treturn Range._createOn( item );\n\t}\n\n\t/**\n\t * Creates a range inside an {@link module:engine/view/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * @param element Element which is a parent for the range.\n\t */\n\tpublic createRangeIn( element: Element | DocumentFragment ): Range {\n\t\treturn Range._createIn( element );\n\t}\n\n\t/**\n\t * Creates new {@link module:engine/view/selection~Selection} instance.\n\t *\n\t * ```ts\n\t * // Creates collapsed selection at the position of given item and offset.\n\t * const paragraph = writer.createContainerElement( 'p' );\n\t * const selection = writer.createSelection( paragraph, offset );\n\t *\n\t * // Creates a range inside an {@link module:engine/view/element~Element element} which starts before the\n\t * // first child of that element and ends after the last child of that element.\n\t * const selection = writer.createSelection( paragraph, 'in' );\n\t *\n\t * // Creates a range on an {@link module:engine/view/item~Item item} which starts before the item and ends\n\t * // just after the item.\n\t * const selection = writer.createSelection( paragraph, 'on' );\n\t * ```\n\t *\n\t * `Selection`'s constructor allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = writer.createSelection( element, 'in', { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = writer.createSelection( element, 'in', { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #createSelection:SELECTABLE `createSelection( selectable, options )`}.\n\t *\n\t * @label NODE_OFFSET\n\t */\n\tpublic createSelection( selectable: Node, placeOrOffset: PlaceOrOffset, options?: SelectionOptions ): Selection;\n\n\t/**\n\t * Creates new {@link module:engine/view/selection~Selection} instance.\n\t *\n\t * ```ts\n\t * // Creates empty selection without ranges.\n\t * const selection = writer.createSelection();\n\t *\n\t * // Creates selection at the given range.\n\t * const range = writer.createRange( start, end );\n\t * const selection = writer.createSelection( range );\n\t *\n\t * // Creates selection at the given ranges\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( star2, end2 ) ];\n\t * const selection = writer.createSelection( ranges );\n\t *\n\t * // Creates selection from the other selection.\n\t * const otherSelection = writer.createSelection();\n\t * const selection = writer.createSelection( otherSelection );\n\t *\n\t * // Creates selection from the document selection.\n\t * const selection = writer.createSelection( editor.editing.view.document.selection );\n\t *\n\t * // Creates selection at the given position.\n\t * const position = writer.createPositionFromPath( root, path );\n\t * const selection = writer.createSelection( position );\n\t * ```\n\t *\n\t * `Selection`'s constructor allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = writer.createSelection( range, { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = writer.createSelection( range, { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #createSelection:NODE_OFFSET `createSelection( node, placeOrOffset, options )`}.\n\t *\n\t * @label SELECTABLE\n\t */\n\tpublic createSelection( selectable?: Exclude<Selectable, Node>, option?: SelectionOptions ): Selection;\n\n\tpublic createSelection( ...args: ConstructorParameters<typeof Selection> ): Selection {\n\t\treturn new Selection( ...args );\n\t}\n\n\t/**\n\t * Creates placeholders for child elements of the {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure\n\t * `elementToStructure()`} conversion helper.\n\t *\n\t * ```ts\n\t * const viewSlot = conversionApi.writer.createSlot();\n\t * const viewPosition = conversionApi.writer.createPositionAt( viewElement, 0 );\n\t *\n\t * conversionApi.writer.insert( viewPosition, viewSlot );\n\t * ```\n\t *\n\t * It could be filtered down to a specific subset of children (only `<foo>` model elements in this case):\n\t *\n\t * ```ts\n\t * const viewSlot = conversionApi.writer.createSlot( node => node.is( 'element', 'foo' ) );\n\t * const viewPosition = conversionApi.writer.createPositionAt( viewElement, 0 );\n\t *\n\t * conversionApi.writer.insert( viewPosition, viewSlot );\n\t * ```\n\t *\n\t * While providing a filtered slot, make sure to provide slots for all child nodes. A single node can not be downcasted into\n\t * multiple slots.\n\t *\n\t * **Note**: You should not change the order of nodes. View elements should be in the same order as model nodes.\n\t *\n\t * @param modeOrFilter The filter for child nodes.\n\t * @returns The slot element to be placed in to the view structure while processing\n\t * {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure `elementToStructure()`}.\n\t */\n\tpublic createSlot( modeOrFilter: 'children' | SlotFilter = 'children' ): Element {\n\t\tif ( !this._slotFactory ) {\n\t\t\t/**\n\t\t\t * The `createSlot()` method is only allowed inside the `elementToStructure` downcast helper callback.\n\t\t\t *\n\t\t\t * @error view-writer-invalid-create-slot-context\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-writer-invalid-create-slot-context', this.document );\n\t\t}\n\n\t\treturn this._slotFactory( this, modeOrFilter );\n\t}\n\n\t/**\n\t * Registers a slot factory.\n\t *\n\t * @internal\n\t * @param slotFactory The slot factory.\n\t */\n\tpublic _registerSlotFactory( slotFactory: ( writer: DowncastWriter, modeOrFilter: string | SlotFilter ) => Element ): void {\n\t\tthis._slotFactory = slotFactory;\n\t}\n\n\t/**\n\t * Clears the registered slot factory.\n\t *\n\t * @internal\n\t */\n\tpublic _clearSlotFactory(): void {\n\t\tthis._slotFactory = null;\n\t}\n\n\t/**\n\t * Inserts a node or nodes at the specified position. Takes care of breaking attributes before insertion\n\t * and merging them afterwards if requested by the breakAttributes param.\n\t *\n\t * @param position Insertion position.\n\t * @param nodes Node or nodes to insert.\n\t * @param breakAttributes Whether attributes should be broken.\n\t * @returns Range around inserted nodes.\n\t */\n\tprivate _insertNodes( position: Position, nodes: Iterable<Node>, breakAttributes: boolean ): Range {\n\t\tlet parentElement;\n\n\t\t// Break attributes on nodes that do exist in the model tree so they can have attributes, other elements\n\t\t// can't have an attribute in model and won't get wrapped with an AttributeElement while down-casted.\n\t\tif ( breakAttributes ) {\n\t\t\tparentElement = getParentContainer( position );\n\t\t} else {\n\t\t\tparentElement = position.parent.is( '$text' ) ? position.parent.parent : position.parent;\n\t\t}\n\n\t\tif ( !parentElement ) {\n\t\t\t/**\n\t\t\t * Position's parent container cannot be found.\n\t\t\t *\n\t\t\t * @error view-writer-invalid-position-container\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'view-writer-invalid-position-container',\n\t\t\t\tthis.document\n\t\t\t);\n\t\t}\n\n\t\tlet insertionPosition;\n\n\t\tif ( breakAttributes ) {\n\t\t\tinsertionPosition = this._breakAttributes( position, true );\n\t\t} else {\n\t\t\tinsertionPosition = position.parent.is( '$text' ) ? breakTextNode( position ) : position;\n\t\t}\n\n\t\tconst length = ( parentElement as Element | DocumentFragment )._insertChild( insertionPosition.offset, nodes );\n\n\t\tfor ( const node of nodes ) {\n\t\t\tthis._addToClonedElementsGroup( node );\n\t\t}\n\n\t\tconst endPosition = insertionPosition.getShiftedBy( length );\n\t\tconst start = this.mergeAttributes( insertionPosition );\n\n\t\t// If start position was merged - move end position.\n\t\tif ( !start.isEqual( insertionPosition ) ) {\n\t\t\tendPosition.offset--;\n\t\t}\n\n\t\tconst end = this.mergeAttributes( endPosition );\n\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Wraps children with provided `wrapElement`. Only children contained in `parent` element between\n\t * `startOffset` and `endOffset` will be wrapped.\n\t */\n\tprivate _wrapChildren( parent: Element, startOffset: number, endOffset: number, wrapElement: AttributeElement ) {\n\t\tlet i = startOffset;\n\t\tconst wrapPositions: Array<Position> = [];\n\n\t\twhile ( i < endOffset ) {\n\t\t\tconst child = parent.getChild( i )!;\n\t\t\tconst isText = child.is( '$text' );\n\t\t\tconst isAttribute = child.is( 'attributeElement' );\n\n\t\t\t//\n\t\t\t// (In all examples, assume that `wrapElement` is `<span class=\"foo\">` element.)\n\t\t\t//\n\t\t\t// Check if `wrapElement` can be joined with the wrapped element. One of requirements is having same name.\n\t\t\t// If possible, join elements.\n\t\t\t//\n\t\t\t// <p><span class=\"bar\">abc</span></p> --> <p><span class=\"foo bar\">abc</span></p>\n\t\t\t//\n\t\t\tif ( isAttribute && this._wrapAttributeElement( wrapElement, child ) ) {\n\t\t\t\twrapPositions.push( new Position( parent, i ) );\n\t\t\t}\n\t\t\t//\n\t\t\t// Wrap the child if it is not an attribute element or if it is an attribute element that should be inside\n\t\t\t// `wrapElement` (due to priority).\n\t\t\t//\n\t\t\t// <p>abc</p> --> <p><span class=\"foo\">abc</span></p>\n\t\t\t// <p><strong>abc</strong></p> --> <p><span class=\"foo\"><strong>abc</strong></span></p>\n\t\t\telse if ( isText || !isAttribute || shouldABeOutsideB( wrapElement, child ) ) {\n\t\t\t\t// Clone attribute.\n\t\t\t\tconst newAttribute = wrapElement._clone();\n\n\t\t\t\t// Wrap current node with new attribute.\n\t\t\t\tchild._remove();\n\t\t\t\tnewAttribute._appendChild( child );\n\n\t\t\t\tparent._insertChild( i, newAttribute );\n\t\t\t\tthis._addToClonedElementsGroup( newAttribute );\n\n\t\t\t\twrapPositions.push( new Position( parent, i ) );\n\t\t\t}\n\t\t\t//\n\t\t\t// If other nested attribute is found and it wasn't wrapped (see above), continue wrapping inside it.\n\t\t\t//\n\t\t\t// <p><a href=\"foo.html\">abc</a></p> --> <p><a href=\"foo.html\"><span class=\"foo\">abc</span></a></p>\n\t\t\t//\n\t\t\telse /* if ( isAttribute ) */ {\n\t\t\t\tthis._wrapChildren( child, 0, child.childCount, wrapElement );\n\t\t\t}\n\n\t\t\ti++;\n\t\t}\n\n\t\t// Merge at each wrap.\n\t\tlet offsetChange = 0;\n\n\t\tfor ( const position of wrapPositions ) {\n\t\t\tposition.offset -= offsetChange;\n\n\t\t\t// Do not merge with elements outside selected children.\n\t\t\tif ( position.offset == startOffset ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst newPosition = this.mergeAttributes( position );\n\n\t\t\t// If nodes were merged - other merge offsets will change.\n\t\t\tif ( !newPosition.isEqual( position ) ) {\n\t\t\t\toffsetChange++;\n\t\t\t\tendOffset--;\n\t\t\t}\n\t\t}\n\n\t\treturn Range._createFromParentsAndOffsets( parent, startOffset, parent, endOffset );\n\t}\n\n\t/**\n\t * Unwraps children from provided `unwrapElement`. Only children contained in `parent` element between\n\t * `startOffset` and `endOffset` will be unwrapped.\n\t */\n\tprivate _unwrapChildren( parent: Element, startOffset: number, endOffset: number, unwrapElement: AttributeElement ) {\n\t\tlet i = startOffset;\n\t\tconst unwrapPositions: Array<Position> = [];\n\n\t\t// Iterate over each element between provided offsets inside parent.\n\t\t// We don't use tree walker or range iterator because we will be removing and merging potentially multiple nodes,\n\t\t// so it could get messy. It is safer to it manually in this case.\n\t\twhile ( i < endOffset ) {\n\t\t\tconst child = parent.getChild( i )!;\n\n\t\t\t// Skip all text nodes. There should be no container element's here either.\n\t\t\tif ( !child.is( 'attributeElement' ) ) {\n\t\t\t\ti++;\n\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t//\n\t\t\t// (In all examples, assume that `unwrapElement` is `<span class=\"foo\">` element.)\n\t\t\t//\n\t\t\t// If the child is similar to the given attribute element, unwrap it - it will be completely removed.\n\t\t\t//\n\t\t\t// <p><span class=\"foo\">abc</span>xyz</p> --> <p>abcxyz</p>\n\t\t\t//\n\t\t\tif ( child.isSimilar( unwrapElement ) ) {\n\t\t\t\tconst unwrapped = child.getChildren();\n\t\t\t\tconst count = child.childCount;\n\n\t\t\t\t// Replace wrapper element with its children\n\t\t\t\tchild._remove();\n\t\t\t\tparent._insertChild( i, unwrapped );\n\n\t\t\t\tthis._removeFromClonedElementsGroup( child );\n\n\t\t\t\t// Save start and end position of moved items.\n\t\t\t\tunwrapPositions.push(\n\t\t\t\t\tnew Position( parent, i ),\n\t\t\t\t\tnew Position( parent, i + count )\n\t\t\t\t);\n\n\t\t\t\t// Skip elements that were unwrapped. Assuming there won't be another element to unwrap in child elements.\n\t\t\t\ti += count;\n\t\t\t\tendOffset += count - 1;\n\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t//\n\t\t\t// If the child is not similar but is an attribute element, try partial unwrapping - remove the same attributes/styles/classes.\n\t\t\t// Partial unwrapping will happen only if the elements have the same name.\n\t\t\t//\n\t\t\t// <p><span class=\"foo bar\">abc</span>xyz</p> --> <p><span class=\"bar\">abc</span>xyz</p>\n\t\t\t// <p><i class=\"foo\">abc</i>xyz</p> --> <p><i class=\"foo\">abc</i>xyz</p>\n\t\t\t//\n\t\t\tif ( this._unwrapAttributeElement( unwrapElement, child ) ) {\n\t\t\t\tunwrapPositions.push(\n\t\t\t\t\tnew Position( parent, i ),\n\t\t\t\t\tnew Position( parent, i + 1 )\n\t\t\t\t);\n\n\t\t\t\ti++;\n\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t//\n\t\t\t// If other nested attribute is found, look through it's children for elements to unwrap.\n\t\t\t//\n\t\t\t// <p><i><span class=\"foo\">abc</span></i><p> --> <p><i>abc</i><p>\n\t\t\t//\n\t\t\tthis._unwrapChildren( child, 0, child.childCount, unwrapElement );\n\n\t\t\ti++;\n\t\t}\n\n\t\t// Merge at each unwrap.\n\t\tlet offsetChange = 0;\n\n\t\tfor ( const position of unwrapPositions ) {\n\t\t\tposition.offset -= offsetChange;\n\n\t\t\t// Do not merge with elements outside selected children.\n\t\t\tif ( position.offset == startOffset || position.offset == endOffset ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst newPosition = this.mergeAttributes( position );\n\n\t\t\t// If nodes were merged - other merge offsets will change.\n\t\t\tif ( !newPosition.isEqual( position ) ) {\n\t\t\t\toffsetChange++;\n\t\t\t\tendOffset--;\n\t\t\t}\n\t\t}\n\n\t\treturn Range._createFromParentsAndOffsets( parent, startOffset, parent, endOffset );\n\t}\n\n\t/**\n\t * Helper function for `view.writer.wrap`. Wraps range with provided attribute element.\n\t * This method will also merge newly added attribute element with its siblings whenever possible.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError} `view-writer-wrap-invalid-attribute` when passed attribute element is not\n\t * an instance of {@link module:engine/view/attributeelement~AttributeElement AttributeElement}.\n\t *\n\t * @returns New range after wrapping, spanning over wrapping attribute element.\n\t */\n\tprivate _wrapRange( range: Range, attribute: AttributeElement ): Range {\n\t\t// Break attributes at range start and end.\n\t\tconst { start: breakStart, end: breakEnd } = this._breakAttributesRange( range, true );\n\t\tconst parentContainer = breakStart.parent as Element;\n\n\t\t// Wrap all children with attribute.\n\t\tconst newRange = this._wrapChildren( parentContainer, breakStart.offset, breakEnd.offset, attribute );\n\n\t\t// Merge attributes at the both ends and return a new range.\n\t\tconst start = this.mergeAttributes( newRange.start );\n\n\t\t// If start position was merged - move end position back.\n\t\tif ( !start.isEqual( newRange.start ) ) {\n\t\t\tnewRange.end.offset--;\n\t\t}\n\t\tconst end = this.mergeAttributes( newRange.end );\n\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Helper function for {@link #wrap}. Wraps position with provided attribute element.\n\t * This method will also merge newly added attribute element with its siblings whenever possible.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError} `view-writer-wrap-invalid-attribute` when passed attribute element is not\n\t * an instance of {@link module:engine/view/attributeelement~AttributeElement AttributeElement}.\n\t *\n\t * @returns New position after wrapping.\n\t */\n\tprivate _wrapPosition( position: Position, attribute: AttributeElement ): Position {\n\t\t// Return same position when trying to wrap with attribute similar to position parent.\n\t\tif ( attribute.isSimilar( position.parent as any ) ) {\n\t\t\treturn movePositionToTextNode( position.clone() );\n\t\t}\n\n\t\t// When position is inside text node - break it and place new position between two text nodes.\n\t\tif ( position.parent.is( '$text' ) ) {\n\t\t\tposition = breakTextNode( position );\n\t\t}\n\n\t\t// Create fake element that will represent position, and will not be merged with other attributes.\n\t\tconst fakeElement = this.createAttributeElement( '_wrapPosition-fake-element' );\n\t\t( fakeElement as any )._priority = Number.POSITIVE_INFINITY;\n\t\tfakeElement.isSimilar = () => false;\n\n\t\t// Insert fake element in position location.\n\t\t( position.parent as Element )._insertChild( position.offset, fakeElement );\n\n\t\t// Range around inserted fake attribute element.\n\t\tconst wrapRange = new Range( position, position.getShiftedBy( 1 ) );\n\n\t\t// Wrap fake element with attribute (it will also merge if possible).\n\t\tthis.wrap( wrapRange, attribute );\n\n\t\t// Remove fake element and place new position there.\n\t\tconst newPosition = new Position( fakeElement.parent!, fakeElement.index! );\n\t\tfakeElement._remove();\n\n\t\t// If position is placed between text nodes - merge them and return position inside.\n\t\tconst nodeBefore = newPosition.nodeBefore;\n\t\tconst nodeAfter = newPosition.nodeAfter;\n\n\t\tif ( nodeBefore instanceof Text && nodeAfter instanceof Text ) {\n\t\t\treturn mergeTextNodes( nodeBefore, nodeAfter );\n\t\t}\n\n\t\t// If position is next to text node - move position inside.\n\t\treturn movePositionToTextNode( newPosition );\n\t}\n\n\t/**\n\t * Wraps one {@link module:engine/view/attributeelement~AttributeElement AttributeElement} into another by\n\t * merging them if possible. When merging is possible - all attributes, styles and classes are moved from wrapper\n\t * element to element being wrapped.\n\t *\n\t * @param wrapper Wrapper AttributeElement.\n\t * @param toWrap AttributeElement to wrap using wrapper element.\n\t * @returns Returns `true` if elements are merged.\n\t */\n\tprivate _wrapAttributeElement( wrapper: AttributeElement, toWrap: AttributeElement ): boolean {\n\t\tif ( !canBeJoined( wrapper, toWrap ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Can't merge if name or priority differs.\n\t\tif ( wrapper.name !== toWrap.name || wrapper.priority !== toWrap.priority ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Check if attributes can be merged.\n\t\tfor ( const key of wrapper.getAttributeKeys() ) {\n\t\t\t// Classes and styles should be checked separately.\n\t\t\tif ( key === 'class' || key === 'style' ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// If some attributes are different we cannot wrap.\n\t\t\tif ( toWrap.hasAttribute( key ) && toWrap.getAttribute( key ) !== wrapper.getAttribute( key ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\t// Check if styles can be merged.\n\t\tfor ( const key of wrapper.getStyleNames() ) {\n\t\t\tif ( toWrap.hasStyle( key ) && toWrap.getStyle( key ) !== wrapper.getStyle( key ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\t// Move all attributes/classes/styles from wrapper to wrapped AttributeElement.\n\t\tfor ( const key of wrapper.getAttributeKeys() ) {\n\t\t\t// Classes and styles should be checked separately.\n\t\t\tif ( key === 'class' || key === 'style' ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// Move only these attributes that are not present - other are similar.\n\t\t\tif ( !toWrap.hasAttribute( key ) ) {\n\t\t\t\tthis.setAttribute( key, wrapper.getAttribute( key )!, toWrap );\n\t\t\t}\n\t\t}\n\n\t\tfor ( const key of wrapper.getStyleNames() ) {\n\t\t\tif ( !toWrap.hasStyle( key ) ) {\n\t\t\t\tthis.setStyle( key, wrapper.getStyle( key )!, toWrap );\n\t\t\t}\n\t\t}\n\n\t\tfor ( const key of wrapper.getClassNames() ) {\n\t\t\tif ( !toWrap.hasClass( key ) ) {\n\t\t\t\tthis.addClass( key, toWrap );\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Unwraps {@link module:engine/view/attributeelement~AttributeElement AttributeElement} from another by removing\n\t * corresponding attributes, classes and styles. All attributes, classes and styles from wrapper should be present\n\t * inside element being unwrapped.\n\t *\n\t * @param wrapper Wrapper AttributeElement.\n\t * @param toUnwrap AttributeElement to unwrap using wrapper element.\n\t * @returns Returns `true` if elements are unwrapped.\n\t **/\n\tprivate _unwrapAttributeElement( wrapper: AttributeElement, toUnwrap: AttributeElement ): boolean {\n\t\tif ( !canBeJoined( wrapper, toUnwrap ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Can't unwrap if name or priority differs.\n\t\tif ( wrapper.name !== toUnwrap.name || wrapper.priority !== toUnwrap.priority ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Check if AttributeElement has all wrapper attributes.\n\t\tfor ( const key of wrapper.getAttributeKeys() ) {\n\t\t\t// Classes and styles should be checked separately.\n\t\t\tif ( key === 'class' || key === 'style' ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// If some attributes are missing or different we cannot unwrap.\n\t\t\tif ( !toUnwrap.hasAttribute( key ) || toUnwrap.getAttribute( key ) !== wrapper.getAttribute( key ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\t// Check if AttributeElement has all wrapper classes.\n\t\tif ( !toUnwrap.hasClass( ...wrapper.getClassNames() ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Check if AttributeElement has all wrapper styles.\n\t\tfor ( const key of wrapper.getStyleNames() ) {\n\t\t\t// If some styles are missing or different we cannot unwrap.\n\t\t\tif ( !toUnwrap.hasStyle( key ) || toUnwrap.getStyle( key ) !== wrapper.getStyle( key ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\t// Remove all wrapper's attributes from unwrapped element.\n\t\tfor ( const key of wrapper.getAttributeKeys() ) {\n\t\t\t// Classes and styles should be checked separately.\n\t\t\tif ( key === 'class' || key === 'style' ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tthis.removeAttribute( key, toUnwrap );\n\t\t}\n\n\t\t// Remove all wrapper's classes from unwrapped element.\n\t\tthis.removeClass( Array.from( wrapper.getClassNames() ), toUnwrap );\n\n\t\t// Remove all wrapper's styles from unwrapped element.\n\t\tthis.removeStyle( Array.from( wrapper.getStyleNames() ), toUnwrap );\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Helper function used by other `DowncastWriter` methods. Breaks attribute elements at the boundaries of given range.\n\t *\n\t * @param range Range which `start` and `end` positions will be used to break attributes.\n\t * @param forceSplitText If set to `true`, will break text nodes even if they are directly in container element.\n\t * This behavior will result in incorrect view state, but is needed by other view writing methods which then fixes view state.\n\t * @returns New range with located at break positions.\n\t */\n\tprivate _breakAttributesRange( range: Range, forceSplitText: boolean = false ) {\n\t\tconst rangeStart = range.start;\n\t\tconst rangeEnd = range.end;\n\n\t\tvalidateRangeContainer( range, this.document );\n\n\t\t// Break at the collapsed position. Return new collapsed range.\n\t\tif ( range.isCollapsed ) {\n\t\t\tconst position = this._breakAttributes( range.start, forceSplitText );\n\n\t\t\treturn new Range( position, position );\n\t\t}\n\n\t\tconst breakEnd = this._breakAttributes( rangeEnd, forceSplitText );\n\t\tconst count = ( breakEnd.parent as Element ).childCount;\n\t\tconst breakStart = this._breakAttributes( rangeStart, forceSplitText );\n\n\t\t// Calculate new break end offset.\n\t\tbreakEnd.offset += ( breakEnd.parent as Element ).childCount - count;\n\n\t\treturn new Range( breakStart, breakEnd );\n\t}\n\n\t/**\n\t * Helper function used by other `DowncastWriter` methods. Breaks attribute elements at given position.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-cannot-break-empty-element` when break position\n\t * is placed inside {@link module:engine/view/emptyelement~EmptyElement EmptyElement}.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-cannot-break-ui-element` when break position\n\t * is placed inside {@link module:engine/view/uielement~UIElement UIElement}.\n\t *\n\t * @param position Position where to break attributes.\n\t * @param forceSplitText If set to `true`, will break text nodes even if they are directly in container element.\n\t * This behavior will result in incorrect view state, but is needed by other view writing methods which then fixes view state.\n\t * @returns New position after breaking the attributes.\n\t */\n\tprivate _breakAttributes( position: Position, forceSplitText: boolean = false ): Position {\n\t\tconst positionOffset = position.offset;\n\t\tconst positionParent = position.parent;\n\n\t\t// If position is placed inside EmptyElement - throw an exception as we cannot break inside.\n\t\tif ( position.parent.is( 'emptyElement' ) ) {\n\t\t\t/**\n\t\t\t * Cannot break an `EmptyElement` instance.\n\t\t\t *\n\t\t\t * This error is thrown if\n\t\t\t * {@link module:engine/view/downcastwriter~DowncastWriter#breakAttributes `DowncastWriter#breakAttributes()`}\n\t\t\t * was executed in an incorrect position.\n\t\t\t *\n\t\t\t * @error view-writer-cannot-break-empty-element\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-writer-cannot-break-empty-element', this.document );\n\t\t}\n\n\t\t// If position is placed inside UIElement - throw an exception as we cannot break inside.\n\t\tif ( position.parent.is( 'uiElement' ) ) {\n\t\t\t/**\n\t\t\t * Cannot break a `UIElement` instance.\n\t\t\t *\n\t\t\t * This error is thrown if\n\t\t\t * {@link module:engine/view/downcastwriter~DowncastWriter#breakAttributes `DowncastWriter#breakAttributes()`}\n\t\t\t * was executed in an incorrect position.\n\t\t\t *\n\t\t\t * @error view-writer-cannot-break-ui-element\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-writer-cannot-break-ui-element', this.document );\n\t\t}\n\n\t\t// If position is placed inside RawElement - throw an exception as we cannot break inside.\n\t\tif ( position.parent.is( 'rawElement' ) ) {\n\t\t\t/**\n\t\t\t * Cannot break a `RawElement` instance.\n\t\t\t *\n\t\t\t * This error is thrown if\n\t\t\t * {@link module:engine/view/downcastwriter~DowncastWriter#breakAttributes `DowncastWriter#breakAttributes()`}\n\t\t\t * was executed in an incorrect position.\n\t\t\t *\n\t\t\t * @error view-writer-cannot-break-raw-element\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-writer-cannot-break-raw-element', this.document );\n\t\t}\n\n\t\t// There are no attributes to break and text nodes breaking is not forced.\n\t\tif ( !forceSplitText && positionParent.is( '$text' ) && isContainerOrFragment( positionParent.parent! ) ) {\n\t\t\treturn position.clone();\n\t\t}\n\n\t\t// Position's parent is container, so no attributes to break.\n\t\tif ( isContainerOrFragment( positionParent ) ) {\n\t\t\treturn position.clone();\n\t\t}\n\n\t\t// Break text and start again in new position.\n\t\tif ( positionParent.is( '$text' ) ) {\n\t\t\treturn this._breakAttributes( breakTextNode( position ), forceSplitText );\n\t\t}\n\n\t\tconst length = ( positionParent as any ).childCount;\n\n\t\t// <p>foo<b><u>bar{}</u></b></p>\n\t\t// <p>foo<b><u>bar</u>[]</b></p>\n\t\t// <p>foo<b><u>bar</u></b>[]</p>\n\t\tif ( positionOffset == length ) {\n\t\t\tconst newPosition = new Position( positionParent.parent as any, ( positionParent as any ).index + 1 );\n\n\t\t\treturn this._breakAttributes( newPosition, forceSplitText );\n\t\t} else {\n\t\t\t// <p>foo<b><u>{}bar</u></b></p>\n\t\t\t// <p>foo<b>[]<u>bar</u></b></p>\n\t\t\t// <p>foo{}<b><u>bar</u></b></p>\n\t\t\tif ( positionOffset === 0 ) {\n\t\t\t\tconst newPosition = new Position( positionParent.parent as Element, ( positionParent as any ).index );\n\n\t\t\t\treturn this._breakAttributes( newPosition, forceSplitText );\n\t\t\t}\n\t\t\t// <p>foo<b><u>b{}ar</u></b></p>\n\t\t\t// <p>foo<b><u>b[]ar</u></b></p>\n\t\t\t// <p>foo<b><u>b</u>[]<u>ar</u></b></p>\n\t\t\t// <p>foo<b><u>b</u></b>[]<b><u>ar</u></b></p>\n\t\t\telse {\n\t\t\t\tconst offsetAfter = ( positionParent as any ).index + 1;\n\n\t\t\t\t// Break element.\n\t\t\t\tconst clonedNode = ( positionParent as any )._clone();\n\n\t\t\t\t// Insert cloned node to position's parent node.\n\t\t\t\t( positionParent.parent as any )._insertChild( offsetAfter, clonedNode );\n\t\t\t\tthis._addToClonedElementsGroup( clonedNode );\n\n\t\t\t\t// Get nodes to move.\n\t\t\t\tconst count = ( positionParent as any ).childCount - positionOffset;\n\t\t\t\tconst nodesToMove = ( positionParent as any )._removeChildren( positionOffset, count );\n\n\t\t\t\t// Move nodes to cloned node.\n\t\t\t\tclonedNode._appendChild( nodesToMove );\n\n\t\t\t\t// Create new position to work on.\n\t\t\t\tconst newPosition = new Position( ( positionParent as any ).parent, offsetAfter );\n\n\t\t\t\treturn this._breakAttributes( newPosition, forceSplitText );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Stores the information that an {@link module:engine/view/attributeelement~AttributeElement attribute element} was\n\t * added to the tree. Saves the reference to the group in the given element and updates the group, so other elements\n\t * from the group now keep a reference to the given attribute element.\n\t *\n\t * The clones group can be obtained using {@link module:engine/view/attributeelement~AttributeElement#getElementsWithSameId}.\n\t *\n\t * Does nothing if added element has no {@link module:engine/view/attributeelement~AttributeElement#id id}.\n\t *\n\t * @param element Attribute element to save.\n\t */\n\tprivate _addToClonedElementsGroup( element: Node ): void {\n\t\t// Add only if the element is in document tree.\n\t\tif ( !element.root.is( 'rootElement' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Traverse the element's children recursively to find other attribute elements that also might got inserted.\n\t\t// The loop is at the beginning so we can make fast returns later in the code.\n\t\tif ( element.is( 'element' ) ) {\n\t\t\tfor ( const child of element.getChildren() ) {\n\t\t\t\tthis._addToClonedElementsGroup( child );\n\t\t\t}\n\t\t}\n\n\t\tconst id = ( element as any ).id;\n\n\t\tif ( !id ) {\n\t\t\treturn;\n\t\t}\n\n\t\tlet group = this._cloneGroups.get( id );\n\n\t\tif ( !group ) {\n\t\t\tgroup = new Set();\n\t\t\tthis._cloneGroups.set( id, group );\n\t\t}\n\n\t\tgroup.add( element as AttributeElement );\n\t\t( element as any )._clonesGroup = group;\n\t}\n\n\t/**\n\t * Removes all the information about the given {@link module:engine/view/attributeelement~AttributeElement attribute element}\n\t * from its clones group.\n\t *\n\t * Keep in mind, that the element will still keep a reference to the group (but the group will not keep a reference to it).\n\t * This allows to reference the whole group even if the element was already removed from the tree.\n\t *\n\t * Does nothing if the element has no {@link module:engine/view/attributeelement~AttributeElement#id id}.\n\t *\n\t * @param element Attribute element to remove.\n\t */\n\tprivate _removeFromClonedElementsGroup( element: Node ) {\n\t\t// Traverse the element's children recursively to find other attribute elements that also got removed.\n\t\t// The loop is at the beginning so we can make fast returns later in the code.\n\t\tif ( element.is( 'element' ) ) {\n\t\t\tfor ( const child of element.getChildren() ) {\n\t\t\t\tthis._removeFromClonedElementsGroup( child );\n\t\t\t}\n\t\t}\n\n\t\tconst id = ( element as any ).id;\n\n\t\tif ( !id ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst group = this._cloneGroups.get( id );\n\n\t\tif ( !group ) {\n\t\t\treturn;\n\t\t}\n\n\t\tgroup.delete( element as AttributeElement );\n\t\t// Not removing group from element on purpose!\n\t\t// If other parts of code have reference to this element, they will be able to get references to other elements from the group.\n\t}\n}\n\n// Helper function for `view.writer.wrap`. Checks if given element has any children that are not ui elements.\nfunction _hasNonUiChildren( parent: Element ): boolean {\n\treturn Array.from( parent.getChildren() ).some( child => !child.is( 'uiElement' ) );\n}\n\n/**\n * The `attribute` passed to {@link module:engine/view/downcastwriter~DowncastWriter#wrap `DowncastWriter#wrap()`}\n * must be an instance of {@link module:engine/view/attributeelement~AttributeElement `AttributeElement`}.\n *\n * @error view-writer-wrap-invalid-attribute\n */\n\n/**\n * Returns first parent container of specified {@link module:engine/view/position~Position Position}.\n * Position's parent node is checked as first, then next parents are checked.\n * Note that {@link module:engine/view/documentfragment~DocumentFragment DocumentFragment} is treated like a container.\n *\n * @param position Position used as a start point to locate parent container.\n * @returns Parent container element or `undefined` if container is not found.\n */\nfunction getParentContainer( position: Position ): ContainerElement | DocumentFragment | undefined {\n\tlet parent = position.parent;\n\n\twhile ( !isContainerOrFragment( parent ) ) {\n\t\tif ( !parent ) {\n\t\t\treturn undefined;\n\t\t}\n\n\t\tparent = parent.parent as any;\n\t}\n\n\treturn ( parent as ContainerElement | DocumentFragment );\n}\n\n/**\n * Checks if first {@link module:engine/view/attributeelement~AttributeElement AttributeElement} provided to the function\n * can be wrapped outside second element. It is done by comparing elements'\n * {@link module:engine/view/attributeelement~AttributeElement#priority priorities}, if both have same priority\n * {@link module:engine/view/element~Element#getIdentity identities} are compared.\n */\nfunction shouldABeOutsideB( a: AttributeElement, b: AttributeElement ): boolean {\n\tif ( a.priority < b.priority ) {\n\t\treturn true;\n\t} else if ( a.priority > b.priority ) {\n\t\treturn false;\n\t}\n\n\t// When priorities are equal and names are different - use identities.\n\treturn a.getIdentity() < b.getIdentity();\n}\n\n/**\n * Returns new position that is moved to near text node. Returns same position if there is no text node before of after\n * specified position.\n *\n * ```html\n * <p>foo[]</p> -> <p>foo{}</p>\n * <p>[]foo</p> -> <p>{}foo</p>\n * ```\n *\n * @returns Position located inside text node or same position if there is no text nodes\n * before or after position location.\n */\nfunction movePositionToTextNode( position: Position ): Position {\n\tconst nodeBefore = position.nodeBefore;\n\n\tif ( nodeBefore && nodeBefore.is( '$text' ) ) {\n\t\treturn new Position( nodeBefore, nodeBefore.data.length );\n\t}\n\n\tconst nodeAfter = position.nodeAfter;\n\n\tif ( nodeAfter && nodeAfter.is( '$text' ) ) {\n\t\treturn new Position( nodeAfter, 0 );\n\t}\n\n\treturn position;\n}\n\n/**\n * Breaks text node into two text nodes when possible.\n *\n * ```html\n * <p>foo{}bar</p> -> <p>foo[]bar</p>\n * <p>{}foobar</p> -> <p>[]foobar</p>\n * <p>foobar{}</p> -> <p>foobar[]</p>\n * ```\n *\n * @param position Position that need to be placed inside text node.\n * @returns New position after breaking text node.\n */\nfunction breakTextNode( position: Position ): Position {\n\tif ( position.offset == ( position.parent as Text ).data.length ) {\n\t\treturn new Position( position.parent.parent as any, ( position.parent as Text ).index! + 1 );\n\t}\n\n\tif ( position.offset === 0 ) {\n\t\treturn new Position( position.parent.parent as any, ( position.parent as Text ).index! );\n\t}\n\n\t// Get part of the text that need to be moved.\n\tconst textToMove = ( position.parent as Text ).data.slice( position.offset );\n\n\t// Leave rest of the text in position's parent.\n\t( position.parent as Text )._data = ( position.parent as Text ).data.slice( 0, position.offset );\n\n\t// Insert new text node after position's parent text node.\n\t( position.parent.parent as any )._insertChild(\n\t\t( position.parent as Text ).index! + 1,\n\t\tnew Text( position.root.document, textToMove )\n\t);\n\n\t// Return new position between two newly created text nodes.\n\treturn new Position( position.parent.parent as any, ( position.parent as Text ).index! + 1 );\n}\n\n/**\n * Merges two text nodes into first node. Removes second node and returns merge position.\n *\n * @param t1 First text node to merge. Data from second text node will be moved at the end of this text node.\n * @param t2 Second text node to merge. This node will be removed after merging.\n * @returns Position after merging text nodes.\n */\nfunction mergeTextNodes( t1: Text, t2: Text ): Position {\n\t// Merge text data into first text node and remove second one.\n\tconst nodeBeforeLength = t1.data.length;\n\tt1._data += t2.data;\n\tt2._remove();\n\n\treturn new Position( t1, nodeBeforeLength );\n}\n\nconst validNodesToInsert = [ Text, AttributeElement, ContainerElement, EmptyElement, RawElement, UIElement ];\n\n/**\n * Checks if provided nodes are valid to insert.\n *\n * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-insert-invalid-node` when nodes to insert\n * contains instances that are not supported ones (see error description for valid ones.\n */\nfunction validateNodesToInsert( nodes: Iterable<Node>, errorContext: Document ): void {\n\tfor ( const node of nodes ) {\n\t\tif ( !validNodesToInsert.some( ( validNode => node instanceof validNode ) ) ) { // eslint-disable-line no-use-before-define\n\t\t\t/**\n\t\t\t * One of the nodes to be inserted is of an invalid type.\n\t\t\t *\n\t\t\t * Nodes to be inserted with {@link module:engine/view/downcastwriter~DowncastWriter#insert `DowncastWriter#insert()`} should be\n\t\t\t * of the following types:\n\t\t\t *\n\t\t\t * * {@link module:engine/view/attributeelement~AttributeElement AttributeElement},\n\t\t\t * * {@link module:engine/view/containerelement~ContainerElement ContainerElement},\n\t\t\t * * {@link module:engine/view/emptyelement~EmptyElement EmptyElement},\n\t\t\t * * {@link module:engine/view/uielement~UIElement UIElement},\n\t\t\t * * {@link module:engine/view/rawelement~RawElement RawElement},\n\t\t\t * * {@link module:engine/view/text~Text Text}.\n\t\t\t *\n\t\t\t * @error view-writer-insert-invalid-node-type\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-writer-insert-invalid-node-type', errorContext );\n\t\t}\n\n\t\tif ( !node.is( '$text' ) ) {\n\t\t\tvalidateNodesToInsert( ( node as Element ).getChildren(), errorContext );\n\t\t}\n\t}\n}\n\n/**\n * Checks if node is ContainerElement or DocumentFragment, because in most cases they should be treated the same way.\n *\n * @returns Returns `true` if node is instance of ContainerElement or DocumentFragment.\n */\nfunction isContainerOrFragment( node: Node | DocumentFragment ): boolean {\n\treturn node && ( node.is( 'containerElement' ) || node.is( 'documentFragment' ) );\n}\n\n/**\n * Checks if {@link module:engine/view/range~Range#start range start} and {@link module:engine/view/range~Range#end range end} are placed\n * inside same {@link module:engine/view/containerelement~ContainerElement container element}.\n * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `view-writer-invalid-range-container` when validation fails.\n */\nfunction validateRangeContainer( range: Range, errorContext: Document ) {\n\tconst startContainer = getParentContainer( range.start );\n\tconst endContainer = getParentContainer( range.end );\n\n\tif ( !startContainer || !endContainer || startContainer !== endContainer ) {\n\t\t/**\n\t\t * The container of the given range is invalid.\n\t\t *\n\t\t * This may happen if {@link module:engine/view/range~Range#start range start} and\n\t\t * {@link module:engine/view/range~Range#end range end} positions are not placed inside the same container element or\n\t\t * a parent container for these positions cannot be found.\n\t\t *\n\t\t * Methods like {@link module:engine/view/downcastwriter~DowncastWriter#wrap `DowncastWriter#remove()`},\n\t\t * {@link module:engine/view/downcastwriter~DowncastWriter#wrap `DowncastWriter#clean()`},\n\t\t * {@link module:engine/view/downcastwriter~DowncastWriter#wrap `DowncastWriter#wrap()`},\n\t\t * {@link module:engine/view/downcastwriter~DowncastWriter#wrap `DowncastWriter#unwrap()`} need to be called\n\t\t * on a range that has its start and end positions located in the same container element. Both positions can be\n\t\t * nested within other elements (e.g. an attribute element) but the closest container ancestor must be the same.\n\t\t *\n\t\t * @error view-writer-invalid-range-container\n\t\t */\n\t\tthrow new CKEditorError( 'view-writer-invalid-range-container', errorContext );\n\t}\n}\n\n/**\n * Checks if two attribute elements can be joined together. Elements can be joined together if, and only if\n * they do not have ids specified.\n */\nfunction canBeJoined( a: AttributeElement, b: AttributeElement ) {\n\treturn a.id === null && b.id === null;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport { keyCodes, isText, type KeystrokeInfo } from '@ckeditor/ckeditor5-utils';\nimport type View from './view';\nimport type DomEventData from './observer/domeventdata';\nimport type { ViewDocumentArrowKeyEvent } from './observer/arrowkeysobserver';\n\n/**\n * Set of utilities related to handling block and inline fillers.\n *\n * Browsers do not allow to put caret in elements which does not have height. Because of it, we need to fill all\n * empty elements which should be selectable with elements or characters called \"fillers\". Unfortunately there is no one\n * universal filler, this is why two types are uses:\n *\n * * Block filler is an element which fill block elements, like `<p>`. CKEditor uses `<br>` as a block filler during the editing,\n * as browsers do natively. So instead of an empty `<p>` there will be `<p><br></p>`. The advantage of block filler is that\n * it is transparent for the selection, so when the caret is before the `<br>` and user presses right arrow he will be\n * moved to the next paragraph, not after the `<br>`. The disadvantage is that it breaks a block, so it can not be used\n * in the middle of a line of text. The {@link module:engine/view/filler~BR_FILLER `<br>` filler} can be replaced with any other\n * character in the data output, for instance {@link module:engine/view/filler~NBSP_FILLER non-breaking space} or\n * {@link module:engine/view/filler~MARKED_NBSP_FILLER marked non-breaking space}.\n *\n * * Inline filler is a filler which does not break a line of text, so it can be used inside the text, for instance in the empty\n * `<b>` surrendered by text: `foo<b></b>bar`, if we want to put the caret there. CKEditor uses a sequence of the zero-width\n * spaces as an {@link module:engine/view/filler~INLINE_FILLER inline filler} having the predetermined\n * {@link module:engine/view/filler~INLINE_FILLER_LENGTH length}. A sequence is used, instead of a single character to\n * avoid treating random zero-width spaces as the inline filler. Disadvantage of the inline filler is that it is not\n * transparent for the selection. The arrow key moves the caret between zero-width spaces characters, so the additional\n * code is needed to handle the caret.\n *\n * Both inline and block fillers are handled by the {@link module:engine/view/renderer~Renderer renderer} and are not present in the\n * view.\n *\n * @module engine/view/filler\n */\n\n/**\n * Non-breaking space filler creator. This function creates the `&nbsp;` text node.\n * It defines how the filler is created.\n *\n * @see module:engine/view/filler~MARKED_NBSP_FILLER\n * @see module:engine/view/filler~BR_FILLER\n */\nexport const NBSP_FILLER = ( domDocument: Document ): Text => domDocument.createTextNode( '\\u00A0' );\n\n/**\n * Marked non-breaking space filler creator. This function creates the `<span data-cke-filler=\"true\">&nbsp;</span>` element.\n * It defines how the filler is created.\n *\n * @see module:engine/view/filler~NBSP_FILLER\n * @see module:engine/view/filler~BR_FILLER\n */\nexport const MARKED_NBSP_FILLER = ( domDocument: Document ): HTMLSpanElement => {\n\tconst span = domDocument.createElement( 'span' );\n\tspan.dataset.ckeFiller = 'true';\n\tspan.innerText = '\\u00A0';\n\n\treturn span;\n};\n\n/**\n * `<br>` filler creator. This function creates the `<br data-cke-filler=\"true\">` element.\n * It defines how the filler is created.\n *\n * @see module:engine/view/filler~NBSP_FILLER\n * @see module:engine/view/filler~MARKED_NBSP_FILLER\n */\nexport const BR_FILLER = ( domDocument: Document ): HTMLBRElement => {\n\tconst fillerBr = domDocument.createElement( 'br' );\n\tfillerBr.dataset.ckeFiller = 'true';\n\n\treturn fillerBr;\n};\n\n/**\n * Length of the {@link module:engine/view/filler~INLINE_FILLER INLINE_FILLER}.\n */\nexport const INLINE_FILLER_LENGTH = 7;\n\n/**\n * Inline filler which is a sequence of the word joiners.\n */\nexport const INLINE_FILLER = '\\u2060'.repeat( INLINE_FILLER_LENGTH );\n\n/**\n * Checks if the node is a text node which starts with the {@link module:engine/view/filler~INLINE_FILLER inline filler}.\n *\n * ```ts\n * startsWithFiller( document.createTextNode( INLINE_FILLER ) ); // true\n * startsWithFiller( document.createTextNode( INLINE_FILLER + 'foo' ) ); // true\n * startsWithFiller( document.createTextNode( 'foo' ) ); // false\n * startsWithFiller( document.createElement( 'p' ) ); // false\n * ```\n *\n * @param domNode DOM node.\n * @returns True if the text node starts with the {@link module:engine/view/filler~INLINE_FILLER inline filler}.\n */\nexport function startsWithFiller( domNode: Node | string ): boolean {\n\tif ( typeof domNode == 'string' ) {\n\t\treturn domNode.substr( 0, INLINE_FILLER_LENGTH ) === INLINE_FILLER;\n\t}\n\n\treturn isText( domNode ) && ( domNode.data.substr( 0, INLINE_FILLER_LENGTH ) === INLINE_FILLER );\n}\n\n/**\n * Checks if the text node contains only the {@link module:engine/view/filler~INLINE_FILLER inline filler}.\n *\n * ```ts\n * isInlineFiller( document.createTextNode( INLINE_FILLER ) ); // true\n * isInlineFiller( document.createTextNode( INLINE_FILLER + 'foo' ) ); // false\n * ```\n *\n * @param domText DOM text node.\n * @returns True if the text node contains only the {@link module:engine/view/filler~INLINE_FILLER inline filler}.\n */\nexport function isInlineFiller( domText: Text ): boolean {\n\treturn domText.data.length == INLINE_FILLER_LENGTH && startsWithFiller( domText );\n}\n\n/**\n * Get string data from the text node, removing an {@link module:engine/view/filler~INLINE_FILLER inline filler} from it,\n * if text node contains it.\n *\n * ```ts\n * getDataWithoutFiller( document.createTextNode( INLINE_FILLER + 'foo' ) ) == 'foo' // true\n * getDataWithoutFiller( document.createTextNode( 'foo' ) ) == 'foo' // true\n * ```\n *\n * @param domText DOM text node, possible with inline filler.\n * @returns Data without filler.\n */\nexport function getDataWithoutFiller( domText: Text | string ): string {\n\tconst data = typeof domText == 'string' ? domText : domText.data;\n\n\tif ( startsWithFiller( domText ) ) {\n\t\treturn data.slice( INLINE_FILLER_LENGTH );\n\t}\n\n\treturn data;\n}\n\n/**\n * Assign key observer which move cursor from the end of the inline filler to the beginning of it when\n * the left arrow is pressed, so the filler does not break navigation.\n *\n * @param view View controller instance we should inject quirks handling on.\n */\nexport function injectQuirksHandling( view: View ): void {\n\tview.document.on<ViewDocumentArrowKeyEvent>( 'arrowKey', jumpOverInlineFiller, { priority: 'low' } );\n}\n\n/**\n * Move cursor from the end of the inline filler to the beginning of it when, so the filler does not break navigation.\n */\nfunction jumpOverInlineFiller( evt: unknown, data: DomEventData & KeystrokeInfo ) {\n\tif ( data.keyCode == keyCodes.arrowleft ) {\n\t\tconst domSelection = data.domTarget.ownerDocument.defaultView!.getSelection()!;\n\n\t\tif ( domSelection.rangeCount == 1 && domSelection.getRangeAt( 0 ).collapsed ) {\n\t\t\tconst domParent = domSelection.getRangeAt( 0 ).startContainer;\n\t\t\tconst domOffset = domSelection.getRangeAt( 0 ).startOffset;\n\n\t\t\tif ( startsWithFiller( domParent ) && domOffset <= INLINE_FILLER_LENGTH ) {\n\t\t\t\tdomSelection.collapse( domParent, 0 );\n\t\t\t}\n\t\t}\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./renderer.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/renderer\n */\n\nimport ViewText from './text';\nimport ViewPosition from './position';\nimport { INLINE_FILLER, INLINE_FILLER_LENGTH, startsWithFiller, isInlineFiller } from './filler';\n\nimport {\n\tCKEditorError,\n\tObservableMixin,\n\tdiff,\n\tenv,\n\tfastDiff,\n\tinsertAt,\n\tisComment,\n\tisNode,\n\tisText,\n\tremove,\n\ttype DiffResult,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport type { ChangeType } from './document';\nimport type DocumentSelection from './documentselection';\nimport type DomConverter from './domconverter';\nimport type ViewElement from './element';\nimport type ViewNode from './node';\n\nimport '../../theme/renderer.css';\n\ntype DomText = globalThis.Text;\ntype DomNode = globalThis.Node;\ntype DomDocument = globalThis.Document;\ntype DomElement = globalThis.HTMLElement;\ntype DomSelection = globalThis.Selection;\n\n/**\n * Renderer is responsible for updating the DOM structure and the DOM selection based on\n * the {@link module:engine/view/renderer~Renderer#markToSync information about updated view nodes}.\n * In other words, it renders the view to the DOM.\n *\n * Its main responsibility is to make only the necessary, minimal changes to the DOM. However, unlike in many\n * virtual DOM implementations, the primary reason for doing minimal changes is not the performance but ensuring\n * that native editing features such as text composition, autocompletion, spell checking, selection's x-index are\n * affected as little as possible.\n *\n * Renderer uses {@link module:engine/view/domconverter~DomConverter} to transform view nodes and positions\n * to and from the DOM.\n */\nexport default class Renderer extends ObservableMixin() {\n\t/**\n\t * Set of DOM Documents instances.\n\t */\n\tpublic readonly domDocuments: Set<DomDocument> = new Set();\n\n\t/**\n\t * Converter instance.\n\t */\n\tpublic readonly domConverter: DomConverter;\n\n\t/**\n\t * Set of nodes which attributes changed and may need to be rendered.\n\t */\n\tpublic readonly markedAttributes: Set<ViewElement> = new Set();\n\n\t/**\n\t * Set of elements which child lists changed and may need to be rendered.\n\t */\n\tpublic readonly markedChildren: Set<ViewElement> = new Set();\n\n\t/**\n\t * Set of text nodes which text data changed and may need to be rendered.\n\t */\n\tpublic readonly markedTexts: Set<ViewNode> = new Set();\n\n\t/**\n\t * View selection. Renderer updates DOM selection based on the view selection.\n\t */\n\tpublic readonly selection: DocumentSelection;\n\n\t/**\n\t * Indicates if the view document is focused and selection can be rendered. Selection will not be rendered if\n\t * this is set to `false`.\n\t *\n\t * @observable\n\t */\n\tdeclare public readonly isFocused: boolean;\n\n\t/**\n\t * Indicates whether the user is making a selection in the document (e.g. holding the mouse button and moving the cursor).\n\t * When they stop selecting, the property goes back to `false`.\n\t *\n\t * Note: In some browsers, the renderer will stop rendering the selection and inline fillers while the user is making\n\t * a selection to avoid glitches in DOM selection\n\t * (https://github.com/ckeditor/ckeditor5/issues/10562, https://github.com/ckeditor/ckeditor5/issues/10723).\n\t *\n\t * @observable\n\t */\n\tdeclare public readonly isSelecting: boolean;\n\n\t/**\n\t * True if composition is in progress inside the document.\n\t *\n\t * This property is bound to the {@link module:engine/view/document~Document#isComposing `Document#isComposing`} property.\n\t *\n\t * @observable\n\t */\n\tdeclare public readonly isComposing: boolean;\n\n\t/**\n\t * The text node in which the inline filler was rendered.\n\t */\n\tprivate _inlineFiller: DomText | null = null;\n\n\t/**\n\t * DOM element containing fake selection.\n\t */\n\tprivate _fakeSelectionContainer: DomElement | null = null;\n\n\t/**\n\t * Creates a renderer instance.\n\t *\n\t * @param domConverter Converter instance.\n\t * @param selection View selection.\n\t */\n\tconstructor( domConverter: DomConverter, selection: DocumentSelection ) {\n\t\tsuper();\n\n\t\tthis.domConverter = domConverter;\n\t\tthis.selection = selection;\n\n\t\tthis.set( 'isFocused', false );\n\t\tthis.set( 'isSelecting', false );\n\n\t\t// Rendering the selection and inline filler manipulation should be postponed in (non-Android) Blink until the user finishes\n\t\t// creating the selection in DOM to avoid accidental selection collapsing\n\t\t// (https://github.com/ckeditor/ckeditor5/issues/10562, https://github.com/ckeditor/ckeditor5/issues/10723).\n\t\t// When the user stops selecting, all pending changes should be rendered ASAP, though.\n\t\tif ( env.isBlink && !env.isAndroid ) {\n\t\t\tthis.on<ObservableChangeEvent>( 'change:isSelecting', () => {\n\t\t\t\tif ( !this.isSelecting ) {\n\t\t\t\t\tthis.render();\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\n\t\tthis.set( 'isComposing', false );\n\n\t\tthis.on<ObservableChangeEvent>( 'change:isComposing', () => {\n\t\t\tif ( !this.isComposing ) {\n\t\t\t\tthis.render();\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Marks a view node to be updated in the DOM by {@link #render `render()`}.\n\t *\n\t * Note that only view nodes whose parents have corresponding DOM elements need to be marked to be synchronized.\n\t *\n\t * @see #markedAttributes\n\t * @see #markedChildren\n\t * @see #markedTexts\n\t *\n\t * @param type Type of the change.\n\t * @param node ViewNode to be marked.\n\t */\n\tpublic markToSync( type: ChangeType, node: ViewNode ): void {\n\t\tif ( type === 'text' ) {\n\t\t\tif ( this.domConverter.mapViewToDom( node.parent! ) ) {\n\t\t\t\tthis.markedTexts.add( node );\n\t\t\t}\n\t\t} else {\n\t\t\t// If the node has no DOM element it is not rendered yet,\n\t\t\t// its children/attributes do not need to be marked to be sync.\n\t\t\tif ( !this.domConverter.mapViewToDom( node as ViewElement ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( type === 'attributes' ) {\n\t\t\t\tthis.markedAttributes.add( node as ViewElement );\n\t\t\t} else if ( type === 'children' ) {\n\t\t\t\tthis.markedChildren.add( node as ViewElement );\n\t\t\t} else {\n\t\t\t\t// eslint-disable-next-line @typescript-eslint/no-unused-vars\n\t\t\t\tconst unreachable: never = type;\n\n\t\t\t\t/**\n\t\t\t\t * Unknown type passed to Renderer.markToSync.\n\t\t\t\t *\n\t\t\t\t * @error view-renderer-unknown-type\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'view-renderer-unknown-type', this );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Renders all buffered changes ({@link #markedAttributes}, {@link #markedChildren} and {@link #markedTexts}) and\n\t * the current view selection (if needed) to the DOM by applying a minimal set of changes to it.\n\t *\n\t * Renderer tries not to break the text composition (e.g. IME) and x-index of the selection,\n\t * so it does as little as it is needed to update the DOM.\n\t *\n\t * Renderer also handles {@link module:engine/view/filler fillers}. Especially, it checks if the inline filler is needed\n\t * at the selection position and adds or removes it. To prevent breaking text composition inline filler will not be\n\t * removed as long as the selection is in the text node which needed it at first.\n\t */\n\tpublic render(): void {\n\t\t// Ignore rendering while in the composition mode. Composition events are not cancellable and browser will modify the DOM tree.\n\t\t// All marked elements, attributes, etc. will wait until next render after the composition ends.\n\t\t// On Android composition events are immediately applied to the model, so we don't need to skip rendering,\n\t\t// and we should not do it because the difference between view and DOM could lead to position mapping problems.\n\t\tif ( this.isComposing && !env.isAndroid ) {\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[Renderer]%c Rendering aborted while isComposing',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', ''\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\treturn;\n\t\t}\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.group( '%c[Renderer]%c Rendering',\n\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', ''\n\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t// @if CK_DEBUG_TYPING // }\n\n\t\tlet inlineFillerPosition: ViewPosition | null = null;\n\t\tconst isInlineFillerRenderingPossible = env.isBlink && !env.isAndroid ? !this.isSelecting : true;\n\n\t\t// Refresh mappings.\n\t\tfor ( const element of this.markedChildren ) {\n\t\t\tthis._updateChildrenMappings( element );\n\t\t}\n\n\t\t// Don't manipulate inline fillers while the selection is being made in (non-Android) Blink to prevent accidental\n\t\t// DOM selection collapsing\n\t\t// (https://github.com/ckeditor/ckeditor5/issues/10562, https://github.com/ckeditor/ckeditor5/issues/10723).\n\t\tif ( isInlineFillerRenderingPossible ) {\n\t\t\t// There was inline filler rendered in the DOM but it's not\n\t\t\t// at the selection position any more, so we can remove it\n\t\t\t// (cause even if it's needed, it must be placed in another location).\n\t\t\tif ( this._inlineFiller && !this._isSelectionInInlineFiller() ) {\n\t\t\t\tthis._removeInlineFiller();\n\t\t\t}\n\n\t\t\t// If we've got the filler, let's try to guess its position in the view.\n\t\t\tif ( this._inlineFiller ) {\n\t\t\t\tinlineFillerPosition = this._getInlineFillerPosition();\n\t\t\t}\n\t\t\t// Otherwise, if it's needed, create it at the selection position.\n\t\t\telse if ( this._needsInlineFillerAtSelection() ) {\n\t\t\t\tinlineFillerPosition = this.selection.getFirstPosition()!;\n\n\t\t\t\t// Do not use `markToSync` so it will be added even if the parent is already added.\n\t\t\t\tthis.markedChildren.add( inlineFillerPosition.parent as ViewElement );\n\t\t\t}\n\t\t}\n\t\t// Make sure the inline filler has any parent, so it can be mapped to view position by DomConverter.\n\t\telse if ( this._inlineFiller && this._inlineFiller.parentNode ) {\n\t\t\t// While the user is making selection, preserve the inline filler at its original position.\n\t\t\tinlineFillerPosition = this.domConverter.domPositionToView( this._inlineFiller )!;\n\n\t\t\t// While down-casting the document selection attributes, all existing empty\n\t\t\t// attribute elements (for selection position) are removed from the view and DOM,\n\t\t\t// so make sure that we were able to map filler position.\n\t\t\t// https://github.com/ckeditor/ckeditor5/issues/12026\n\t\t\tif ( inlineFillerPosition && inlineFillerPosition.parent.is( '$text' ) ) {\n\t\t\t\t// The inline filler position is expected to be before the text node.\n\t\t\t\tinlineFillerPosition = ViewPosition._createBefore( inlineFillerPosition.parent );\n\t\t\t}\n\t\t}\n\n\t\tfor ( const element of this.markedAttributes ) {\n\t\t\tthis._updateAttrs( element );\n\t\t}\n\n\t\tfor ( const element of this.markedChildren ) {\n\t\t\tthis._updateChildren( element, { inlineFillerPosition } );\n\t\t}\n\n\t\tfor ( const node of this.markedTexts ) {\n\t\t\tif ( !this.markedChildren.has( node.parent as ViewElement ) && this.domConverter.mapViewToDom( node.parent as ViewElement ) ) {\n\t\t\t\tthis._updateText( node as ViewText, { inlineFillerPosition } );\n\t\t\t}\n\t\t}\n\n\t\t// * Check whether the inline filler is required and where it really is in the DOM.\n\t\t// At this point in most cases it will be in the DOM, but there are exceptions.\n\t\t// For example, if the inline filler was deep in the created DOM structure, it will not be created.\n\t\t// Similarly, if it was removed at the beginning of this function and then neither text nor children were updated,\n\t\t// it will not be present. Fix those and similar scenarios.\n\t\t// * Don't manipulate inline fillers while the selection is being made in (non-Android) Blink to prevent accidental\n\t\t// DOM selection collapsing\n\t\t// (https://github.com/ckeditor/ckeditor5/issues/10562, https://github.com/ckeditor/ckeditor5/issues/10723).\n\t\tif ( isInlineFillerRenderingPossible ) {\n\t\t\tif ( inlineFillerPosition ) {\n\t\t\t\tconst fillerDomPosition = this.domConverter.viewPositionToDom( inlineFillerPosition )!;\n\t\t\t\tconst domDocument = fillerDomPosition.parent.ownerDocument!;\n\n\t\t\t\tif ( !startsWithFiller( fillerDomPosition.parent ) ) {\n\t\t\t\t\t// Filler has not been created at filler position. Create it now.\n\t\t\t\t\tthis._inlineFiller = addInlineFiller( domDocument, fillerDomPosition.parent, fillerDomPosition.offset );\n\t\t\t\t} else {\n\t\t\t\t\t// Filler has been found, save it.\n\t\t\t\t\tthis._inlineFiller = fillerDomPosition.parent as DomText;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t// There is no filler needed.\n\t\t\t\tthis._inlineFiller = null;\n\t\t\t}\n\t\t}\n\n\t\t// First focus the new editing host, then update the selection.\n\t\t// Otherwise, FF may throw an error (https://github.com/ckeditor/ckeditor5/issues/721).\n\t\tthis._updateFocus();\n\t\tthis._updateSelection();\n\n\t\tthis.markedTexts.clear();\n\t\tthis.markedAttributes.clear();\n\t\tthis.markedChildren.clear();\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t// @if CK_DEBUG_TYPING // }\n\t}\n\n\t/**\n\t * Updates mappings of view element's children.\n\t *\n\t * Children that were replaced in the view structure by similar elements (same tag name) are treated as 'replaced'.\n\t * This means that their mappings can be updated so the new view elements are mapped to the existing DOM elements.\n\t * Thanks to that these elements do not need to be re-rendered completely.\n\t *\n\t * @param viewElement The view element whose children mappings will be updated.\n\t */\n\tprivate _updateChildrenMappings( viewElement: ViewElement ): void {\n\t\tconst domElement = this.domConverter.mapViewToDom( viewElement );\n\n\t\tif ( !domElement ) {\n\t\t\t// If there is no `domElement` it means that it was already removed from DOM and there is no need to process it.\n\t\t\treturn;\n\t\t}\n\n\t\t// Removing nodes from the DOM as we iterate can cause `actualDomChildren`\n\t\t// (which is a live-updating `NodeList`) to get out of sync with the\n\t\t// indices that we compute as we iterate over `actions`.\n\t\t// This would produce incorrect element mappings.\n\t\t//\n\t\t// Converting live list to an array to make the list static.\n\t\tconst actualDomChildren = Array.from(\n\t\t\tthis.domConverter.mapViewToDom( viewElement )!.childNodes\n\t\t);\n\t\tconst expectedDomChildren = Array.from(\n\t\t\tthis.domConverter.viewChildrenToDom( viewElement, { withChildren: false } )\n\t\t);\n\t\tconst diff = this._diffNodeLists( actualDomChildren, expectedDomChildren );\n\t\tconst actions = this._findUpdateActions( diff, actualDomChildren, expectedDomChildren, areSimilarElements );\n\n\t\tif ( actions.indexOf( 'update' ) !== -1 ) {\n\t\t\tconst counter = { equal: 0, insert: 0, delete: 0 };\n\n\t\t\tfor ( const action of actions ) {\n\t\t\t\tif ( action === 'update' ) {\n\t\t\t\t\tconst insertIndex = counter.equal + counter.insert;\n\t\t\t\t\tconst deleteIndex = counter.equal + counter.delete;\n\t\t\t\t\tconst viewChild = viewElement.getChild( insertIndex );\n\n\t\t\t\t\t// UIElement and RawElement are special cases. Their children are not stored in a view (#799)\n\t\t\t\t\t// so we cannot use them with replacing flow (since they use view children during rendering\n\t\t\t\t\t// which will always result in rendering empty elements).\n\t\t\t\t\tif ( viewChild && !( viewChild.is( 'uiElement' ) || viewChild.is( 'rawElement' ) ) ) {\n\t\t\t\t\t\tthis._updateElementMappings( viewChild as ViewElement, actualDomChildren[ deleteIndex ] as DomElement );\n\t\t\t\t\t}\n\n\t\t\t\t\tremove( expectedDomChildren[ insertIndex ] );\n\t\t\t\t\tcounter.equal++;\n\t\t\t\t} else {\n\t\t\t\t\tcounter[ action ]++;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Updates mappings of a given view element.\n\t *\n\t * @param viewElement The view element whose mappings will be updated.\n\t * @param domElement The DOM element representing the given view element.\n\t */\n\tprivate _updateElementMappings( viewElement: ViewElement, domElement: DomElement ): void {\n\t\t// Remap 'DomConverter' bindings.\n\t\tthis.domConverter.unbindDomElement( domElement );\n\t\tthis.domConverter.bindElements( domElement, viewElement );\n\n\t\t// View element may have children which needs to be updated, but are not marked, mark them to update.\n\t\tthis.markedChildren.add( viewElement );\n\n\t\t// Because we replace new view element mapping with the existing one, the corresponding DOM element\n\t\t// will not be rerendered. The new view element may have different attributes than the previous one.\n\t\t// Since its corresponding DOM element will not be rerendered, new attributes will not be added\n\t\t// to the DOM, so we need to mark it here to make sure its attributes gets updated. See #1427 for more\n\t\t// detailed case study.\n\t\t// Also there are cases where replaced element is removed from the view structure and then has\n\t\t// its attributes changed or removed. In such cases the element will not be present in `markedAttributes`\n\t\t// and also may be the same (`element.isSimilar()`) as the reused element not having its attributes updated.\n\t\t// To prevent such situations we always mark reused element to have its attributes rerenderd (#1560).\n\t\tthis.markedAttributes.add( viewElement );\n\t}\n\n\t/**\n\t * Gets the position of the inline filler based on the current selection.\n\t * Here, we assume that we know that the filler is needed and\n\t * {@link #_isSelectionInInlineFiller is at the selection position}, and, since it is needed,\n\t * it is somewhere at the selection position.\n\t *\n\t * Note: The filler position cannot be restored based on the filler's DOM text node, because\n\t * when this method is called (before rendering), the bindings will often be broken. View-to-DOM\n\t * bindings are only dependable after rendering.\n\t */\n\tprivate _getInlineFillerPosition(): ViewPosition {\n\t\tconst firstPos = this.selection.getFirstPosition()!;\n\n\t\tif ( firstPos.parent.is( '$text' ) ) {\n\t\t\treturn ViewPosition._createBefore( firstPos.parent );\n\t\t} else {\n\t\t\treturn firstPos;\n\t\t}\n\t}\n\n\t/**\n\t * Returns `true` if the selection has not left the inline filler's text node.\n\t * If it is `true`, it means that the filler had been added for a reason and the selection did not\n\t * leave the filler's text node. For example, the user can be in the middle of a composition so it should not be touched.\n\t *\n\t * @returns `true` if the inline filler and selection are in the same place.\n\t */\n\tprivate _isSelectionInInlineFiller(): boolean {\n\t\tif ( this.selection.rangeCount != 1 || !this.selection.isCollapsed ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Note, we can't check if selection's position equals position of the\n\t\t// this._inlineFiller node, because of #663. We may not be able to calculate\n\t\t// the filler's position in the view at this stage.\n\t\t// Instead, we check it the other way whether selection is anchored in\n\t\t// that text node or next to it.\n\n\t\t// Possible options are:\n\t\t// \"FILLER{}\"\n\t\t// \"FILLERadded-text{}\"\n\t\tconst selectionPosition = this.selection.getFirstPosition()!;\n\t\tconst position = this.domConverter.viewPositionToDom( selectionPosition );\n\n\t\tif ( position && isText( position.parent ) && startsWithFiller( position.parent ) ) {\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Removes the inline filler.\n\t */\n\tprivate _removeInlineFiller(): void {\n\t\tconst domFillerNode = this._inlineFiller!;\n\n\t\t// Something weird happened and the stored node doesn't contain the filler's text.\n\t\tif ( !startsWithFiller( domFillerNode ) ) {\n\t\t\t/**\n\t\t\t * The inline filler node was lost. Most likely, something overwrote the filler text node\n\t\t\t * in the DOM.\n\t\t\t *\n\t\t\t * @error view-renderer-filler-was-lost\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-renderer-filler-was-lost', this );\n\t\t}\n\n\t\tif ( isInlineFiller( domFillerNode ) ) {\n\t\t\tdomFillerNode.remove();\n\t\t} else {\n\t\t\tdomFillerNode.data = domFillerNode.data.substr( INLINE_FILLER_LENGTH );\n\t\t}\n\n\t\tthis._inlineFiller = null;\n\t}\n\n\t/**\n\t * Checks if the inline {@link module:engine/view/filler filler} should be added.\n\t *\n\t * @returns `true` if the inline filler should be added.\n\t */\n\tprivate _needsInlineFillerAtSelection(): boolean {\n\t\tif ( this.selection.rangeCount != 1 || !this.selection.isCollapsed ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst selectionPosition = this.selection.getFirstPosition()!;\n\t\tconst selectionParent = selectionPosition.parent;\n\t\tconst selectionOffset = selectionPosition.offset;\n\n\t\t// If there is no DOM root we do not care about fillers.\n\t\tif ( !this.domConverter.mapViewToDom( selectionParent.root ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( !( selectionParent.is( 'element' ) ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Prevent adding inline filler inside elements with contenteditable=false.\n\t\t// https://github.com/ckeditor/ckeditor5-engine/issues/1170\n\t\tif ( !isEditable( selectionParent ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// We have block filler, we do not need inline one.\n\t\tif ( selectionOffset === selectionParent.getFillerOffset!() ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst nodeBefore = selectionPosition.nodeBefore;\n\t\tconst nodeAfter = selectionPosition.nodeAfter;\n\n\t\tif ( nodeBefore instanceof ViewText || nodeAfter instanceof ViewText ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Do not use inline filler while typing outside inline elements on Android.\n\t\t// The deleteContentBackward would remove part of the inline filler instead of removing last letter in a link.\n\t\tif ( env.isAndroid && ( nodeBefore || nodeAfter ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Checks if text needs to be updated and possibly updates it.\n\t *\n\t * @param viewText View text to update.\n\t * @param options.inlineFillerPosition The position where the inline filler should be rendered.\n\t */\n\tprivate _updateText( viewText: ViewText, options: { inlineFillerPosition?: ViewPosition | null } ) {\n\t\tconst domText = this.domConverter.findCorrespondingDomText( viewText )!;\n\t\tconst newDomText = this.domConverter.viewToDom( viewText ) as DomText;\n\n\t\tlet expectedText = newDomText.data;\n\t\tconst filler = options.inlineFillerPosition;\n\n\t\tif ( filler && filler.parent == viewText.parent && filler.offset == viewText.index ) {\n\t\t\texpectedText = INLINE_FILLER + expectedText;\n\t\t}\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.group( '%c[Renderer]%c Update text',\n\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', ''\n\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t// @if CK_DEBUG_TYPING // }\n\n\t\tupdateTextNode( domText, expectedText );\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t// @if CK_DEBUG_TYPING // }\n\t}\n\n\t/**\n\t * Checks if attribute list needs to be updated and possibly updates it.\n\t *\n\t * @param viewElement The view element to update.\n\t */\n\tprivate _updateAttrs( viewElement: ViewElement ): void {\n\t\tconst domElement = this.domConverter.mapViewToDom( viewElement );\n\n\t\tif ( !domElement ) {\n\t\t\t// If there is no `domElement` it means that 'viewElement' is outdated as its mapping was updated\n\t\t\t// in 'this._updateChildrenMappings()'. There is no need to process it as new view element which\n\t\t\t// replaced old 'viewElement' mapping was also added to 'this.markedAttributes'\n\t\t\t// in 'this._updateChildrenMappings()' so it will be processed separately.\n\t\t\treturn;\n\t\t}\n\n\t\tconst domAttrKeys = Array.from( ( domElement as DomElement ).attributes ).map( attr => attr.name );\n\t\tconst viewAttrKeys = viewElement.getAttributeKeys();\n\n\t\t// Add or overwrite attributes.\n\t\tfor ( const key of viewAttrKeys ) {\n\t\t\tthis.domConverter.setDomElementAttribute( domElement as DomElement, key, viewElement.getAttribute( key )!, viewElement );\n\t\t}\n\n\t\t// Remove from DOM attributes which do not exists in the view.\n\t\tfor ( const key of domAttrKeys ) {\n\t\t\t// All other attributes not present in the DOM should be removed.\n\t\t\tif ( !viewElement.hasAttribute( key ) ) {\n\t\t\t\tthis.domConverter.removeDomElementAttribute( domElement as DomElement, key );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Checks if elements child list needs to be updated and possibly updates it.\n\t *\n\t * Note that on Android, to reduce the risk of composition breaks, it tries to update data of an existing\n\t * child text nodes instead of replacing them completely.\n\t *\n\t * @param viewElement View element to update.\n\t * @param options.inlineFillerPosition The position where the inline filler should be rendered.\n\t */\n\tprivate _updateChildren( viewElement: ViewElement, options: { inlineFillerPosition: ViewPosition | null } ) {\n\t\tconst domElement = this.domConverter.mapViewToDom( viewElement );\n\n\t\tif ( !domElement ) {\n\t\t\t// If there is no `domElement` it means that it was already removed from DOM.\n\t\t\t// There is no need to process it. It will be processed when re-inserted.\n\t\t\treturn;\n\t\t}\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.group( '%c[Renderer]%c Update children',\n\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', ''\n\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t// IME on Android inserts a new text node while typing after a link\n\t\t// instead of updating an existing text node that follows the link.\n\t\t// We must normalize those text nodes so the diff won't get confused.\n\t\t// https://github.com/ckeditor/ckeditor5/issues/12574.\n\t\tif ( env.isAndroid ) {\n\t\t\tlet previousDomNode = null;\n\n\t\t\tfor ( const domNode of Array.from( domElement.childNodes ) ) {\n\t\t\t\tif ( previousDomNode && isText( previousDomNode ) && isText( domNode ) ) {\n\t\t\t\t\tdomElement.normalize();\n\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tpreviousDomNode = domNode;\n\t\t\t}\n\t\t}\n\n\t\tconst inlineFillerPosition = options.inlineFillerPosition;\n\t\tconst actualDomChildren = domElement.childNodes;\n\t\tconst expectedDomChildren = Array.from(\n\t\t\tthis.domConverter.viewChildrenToDom( viewElement, { bind: true } )\n\t\t);\n\n\t\t// Inline filler element has to be created as it is present in the DOM, but not in the view. It is required\n\t\t// during diffing so text nodes could be compared correctly and also during rendering to maintain\n\t\t// proper order and indexes while updating the DOM.\n\t\tif ( inlineFillerPosition && inlineFillerPosition.parent === viewElement ) {\n\t\t\taddInlineFiller( ( domElement as DomElement ).ownerDocument, expectedDomChildren, inlineFillerPosition.offset );\n\t\t}\n\n\t\tconst diff = this._diffNodeLists( actualDomChildren, expectedDomChildren );\n\n\t\t// We need to make sure that we update the existing text node and not replace it with another one.\n\t\t// The composition and different \"language\" browser extensions are fragile to text node being completely replaced.\n\t\tconst actions = this._findUpdateActions( diff, actualDomChildren, expectedDomChildren, areTextNodes );\n\n\t\tlet i = 0;\n\t\tconst nodesToUnbind: Set<DomNode> = new Set();\n\n\t\t// Handle deletions first.\n\t\t// This is to prevent a situation where an element that already exists in `actualDomChildren` is inserted at a different\n\t\t// index in `actualDomChildren`. Since `actualDomChildren` is a `NodeList`, this works like move, not like an insert,\n\t\t// and it disrupts the whole algorithm. See https://github.com/ckeditor/ckeditor5/issues/6367.\n\t\t//\n\t\t// It doesn't matter in what order we remove or add nodes, as long as we remove and add correct nodes at correct indexes.\n\t\tfor ( const action of actions ) {\n\t\t\tif ( action === 'delete' ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[Renderer]%c Remove node',\n\t\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', '', actualDomChildren[ i ]\n\t\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t\t\tnodesToUnbind.add( actualDomChildren[ i ] as DomElement );\n\t\t\t\tremove( actualDomChildren[ i ] );\n\t\t\t} else if ( action === 'equal' || action === 'update' ) {\n\t\t\t\ti++;\n\t\t\t}\n\t\t}\n\n\t\ti = 0;\n\n\t\tfor ( const action of actions ) {\n\t\t\tif ( action === 'insert' ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[Renderer]%c Insert node',\n\t\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', '', expectedDomChildren[ i ]\n\t\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t\tinsertAt( domElement as DomElement, i, expectedDomChildren[ i ] );\n\t\t\t\ti++;\n\t\t\t}\n\t\t\t// Update the existing text node data. Note that replace action is generated only for Android for now.\n\t\t\telse if ( action === 'update' ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconsole.group( '%c[Renderer]%c Update text node',\n\t\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', ''\n\t\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t\tupdateTextNode( actualDomChildren[ i ] as DomText, ( expectedDomChildren[ i ] as DomText ).data );\n\t\t\t\ti++;\n\n\t\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t\t} else if ( action === 'equal' ) {\n\t\t\t\t// Force updating text nodes inside elements which did not change and do not need to be re-rendered (#1125).\n\t\t\t\t// Do it here (not in the loop above) because only after insertions the `i` index is correct.\n\t\t\t\tthis._markDescendantTextToSync( this.domConverter.domToView( expectedDomChildren[ i ] ) as any );\n\t\t\t\ti++;\n\t\t\t}\n\t\t}\n\n\t\t// Unbind removed nodes. When node does not have a parent it means that it was removed from DOM tree during\n\t\t// comparison with the expected DOM. We don't need to check child nodes, because if child node was reinserted,\n\t\t// it was moved to DOM tree out of the removed node.\n\t\tfor ( const node of nodesToUnbind ) {\n\t\t\tif ( !node.parentNode ) {\n\t\t\t\tthis.domConverter.unbindDomElement( node as DomElement );\n\t\t\t}\n\t\t}\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t// @if CK_DEBUG_TYPING // }\n\t}\n\n\t/**\n\t * Shorthand for diffing two arrays or node lists of DOM nodes.\n\t *\n\t * @param actualDomChildren Actual DOM children\n\t * @param expectedDomChildren Expected DOM children.\n\t * @returns The list of actions based on the {@link module:utils/diff~diff} function.\n\t */\n\tprivate _diffNodeLists( actualDomChildren: Array<DomNode> | NodeList, expectedDomChildren: Array<DomNode> | NodeList ) {\n\t\tactualDomChildren = filterOutFakeSelectionContainer( actualDomChildren, this._fakeSelectionContainer );\n\n\t\treturn diff( actualDomChildren, expectedDomChildren, sameNodes.bind( null, this.domConverter ) );\n\t}\n\n\t/**\n\t * Finds DOM nodes that were replaced with the similar nodes (same tag name) in the view. All nodes are compared\n\t * within one `insert`/`delete` action group, for example:\n\t *\n\t * ```\n\t * Actual DOM:\t\t<p><b>Foo</b>Bar<i>Baz</i><b>Bax</b></p>\n\t * Expected DOM:\t<p>Bar<b>123</b><i>Baz</i><b>456</b></p>\n\t * Input actions:\t[ insert, insert, delete, delete, equal, insert, delete ]\n\t * Output actions:\t[ insert, replace, delete, equal, replace ]\n\t * ```\n\t *\n\t * @param actions Actions array which is a result of the {@link module:utils/diff~diff} function.\n\t * @param actualDom Actual DOM children\n\t * @param expectedDom Expected DOM children.\n\t * @param comparator A comparator function that should return `true` if the given node should be reused\n\t * (either by the update of a text node data or an element children list for similar elements).\n\t * @returns Actions array modified with the `update` actions.\n\t */\n\tprivate _findUpdateActions(\n\t\tactions: Array<DiffResult>,\n\t\tactualDom: Array<DomNode> | NodeList,\n\t\texpectedDom: Array<DomNode>,\n\t\tcomparator: ( a: DomNode, b: DomNode ) => boolean\n\t): Array<DiffResult | 'update'> {\n\t\t// If there is no both 'insert' and 'delete' actions, no need to check for replaced elements.\n\t\tif ( actions.indexOf( 'insert' ) === -1 || actions.indexOf( 'delete' ) === -1 ) {\n\t\t\treturn actions;\n\t\t}\n\n\t\tlet newActions: Array<DiffResult | 'update'> = [];\n\t\tlet actualSlice = [];\n\t\tlet expectedSlice = [];\n\n\t\tconst counter = { equal: 0, insert: 0, delete: 0 };\n\n\t\tfor ( const action of actions ) {\n\t\t\tif ( action === 'insert' ) {\n\t\t\t\texpectedSlice.push( expectedDom[ counter.equal + counter.insert ] );\n\t\t\t} else if ( action === 'delete' ) {\n\t\t\t\tactualSlice.push( actualDom[ counter.equal + counter.delete ] );\n\t\t\t} else { // equal\n\t\t\t\tnewActions = newActions.concat(\n\t\t\t\t\tdiff( actualSlice, expectedSlice, comparator )\n\t\t\t\t\t\t.map( action => action === 'equal' ? 'update' : action )\n\t\t\t\t);\n\n\t\t\t\tnewActions.push( 'equal' );\n\n\t\t\t\t// Reset stored elements on 'equal'.\n\t\t\t\tactualSlice = [];\n\t\t\t\texpectedSlice = [];\n\t\t\t}\n\t\t\tcounter[ action ]++;\n\t\t}\n\n\t\treturn newActions.concat(\n\t\t\tdiff( actualSlice, expectedSlice, comparator )\n\t\t\t\t.map( action => action === 'equal' ? 'update' : action )\n\t\t);\n\t}\n\n\t/**\n\t * Marks text nodes to be synchronized.\n\t *\n\t * If a text node is passed, it will be marked. If an element is passed, all descendant text nodes inside it will be marked.\n\t *\n\t * @param viewNode View node to sync.\n\t */\n\tprivate _markDescendantTextToSync( viewNode: ViewNode | undefined ): void {\n\t\tif ( !viewNode ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( viewNode.is( '$text' ) ) {\n\t\t\tthis.markedTexts.add( viewNode );\n\t\t} else if ( viewNode.is( 'element' ) ) {\n\t\t\tfor ( const child of viewNode.getChildren() ) {\n\t\t\t\tthis._markDescendantTextToSync( child );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Checks if the selection needs to be updated and possibly updates it.\n\t */\n\tprivate _updateSelection(): void {\n\t\t// Block updating DOM selection in (non-Android) Blink while the user is selecting to prevent accidental selection collapsing.\n\t\t// Note: Structural changes in DOM must trigger selection rendering, though. Nodes the selection was anchored\n\t\t// to, may disappear in DOM which would break the selection (e.g. in real-time collaboration scenarios).\n\t\t// https://github.com/ckeditor/ckeditor5/issues/10562, https://github.com/ckeditor/ckeditor5/issues/10723\n\t\tif ( env.isBlink && !env.isAndroid && this.isSelecting && !this.markedChildren.size ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// If there is no selection - remove DOM and fake selections.\n\t\tif ( this.selection.rangeCount === 0 ) {\n\t\t\tthis._removeDomSelection();\n\t\t\tthis._removeFakeSelection();\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst domRoot = this.domConverter.mapViewToDom( this.selection.editableElement! );\n\n\t\t// Do nothing if there is no focus, or there is no DOM element corresponding to selection's editable element.\n\t\tif ( !this.isFocused || !domRoot ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Render fake selection - create the fake selection container (if needed) and move DOM selection to it.\n\t\tif ( this.selection.isFake ) {\n\t\t\tthis._updateFakeSelection( domRoot );\n\t\t}\n\t\t// There was a fake selection so remove it and update the DOM selection.\n\t\t// This is especially important on Android because otherwise IME will try to compose over the fake selection container.\n\t\telse if ( this._fakeSelectionContainer && this._fakeSelectionContainer.isConnected ) {\n\t\t\tthis._removeFakeSelection();\n\t\t\tthis._updateDomSelection( domRoot );\n\t\t}\n\t\t// Update the DOM selection in case of a plain selection change (no fake selection is involved).\n\t\t// On non-Android the whole rendering is disabled in composition mode (including DOM selection update),\n\t\t// but updating DOM selection should be also disabled on Android if in the middle of the composition\n\t\t// (to not interrupt it).\n\t\telse if ( !( this.isComposing && env.isAndroid ) ) {\n\t\t\tthis._updateDomSelection( domRoot );\n\t\t}\n\t}\n\n\t/**\n\t * Updates the fake selection.\n\t *\n\t * @param domRoot A valid DOM root where the fake selection container should be added.\n\t */\n\tprivate _updateFakeSelection( domRoot: DomElement ): void {\n\t\tconst domDocument = domRoot.ownerDocument;\n\n\t\tif ( !this._fakeSelectionContainer ) {\n\t\t\tthis._fakeSelectionContainer = createFakeSelectionContainer( domDocument );\n\t\t}\n\n\t\tconst container = this._fakeSelectionContainer;\n\n\t\t// Bind fake selection container with the current selection *position*.\n\t\tthis.domConverter.bindFakeSelection( container, this.selection );\n\n\t\tif ( !this._fakeSelectionNeedsUpdate( domRoot ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !container.parentElement || container.parentElement != domRoot ) {\n\t\t\tdomRoot.appendChild( container );\n\t\t}\n\n\t\tcontainer.textContent = this.selection.fakeSelectionLabel || '\\u00A0';\n\n\t\tconst domSelection = domDocument.getSelection()!;\n\t\tconst domRange = domDocument.createRange();\n\n\t\tdomSelection.removeAllRanges();\n\t\tdomRange.selectNodeContents( container );\n\t\tdomSelection.addRange( domRange );\n\t}\n\n\t/**\n\t * Updates the DOM selection.\n\t *\n\t * @param domRoot A valid DOM root where the DOM selection should be rendered.\n\t */\n\tprivate _updateDomSelection( domRoot: DomElement ) {\n\t\tconst domSelection = domRoot.ownerDocument.defaultView!.getSelection()!;\n\n\t\t// Let's check whether DOM selection needs updating at all.\n\t\tif ( !this._domSelectionNeedsUpdate( domSelection ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Multi-range selection is not available in most browsers, and, at least in Chrome, trying to\n\t\t// set such selection, that is not continuous, throws an error. Because of that, we will just use anchor\n\t\t// and focus of view selection.\n\t\t// Since we are not supporting multi-range selection, we also do not need to check if proper editable is\n\t\t// selected. If there is any editable selected, it is okay (editable is taken from selection anchor).\n\t\tconst anchor = this.domConverter.viewPositionToDom( this.selection.anchor! )!;\n\t\tconst focus = this.domConverter.viewPositionToDom( this.selection.focus! )!;\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[Renderer]%c Update DOM selection:',\n\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', '', anchor, focus\n\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t// @if CK_DEBUG_TYPING // }\n\n\t\tdomSelection.setBaseAndExtent( anchor.parent, anchor.offset, focus.parent, focus.offset );\n\n\t\t// Firefoxspecific hack (https://github.com/ckeditor/ckeditor5-engine/issues/1439).\n\t\tif ( env.isGecko ) {\n\t\t\tfixGeckoSelectionAfterBr( focus, domSelection );\n\t\t}\n\t}\n\n\t/**\n\t * Checks whether a given DOM selection needs to be updated.\n\t *\n\t * @param domSelection The DOM selection to check.\n\t */\n\tprivate _domSelectionNeedsUpdate( domSelection: Selection ): boolean {\n\t\tif ( !this.domConverter.isDomSelectionCorrect( domSelection ) ) {\n\t\t\t// Current DOM selection is in incorrect position. We need to update it.\n\t\t\treturn true;\n\t\t}\n\n\t\tconst oldViewSelection = domSelection && this.domConverter.domSelectionToView( domSelection );\n\n\t\tif ( oldViewSelection && this.selection.isEqual( oldViewSelection ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// If selection is not collapsed, it does not need to be updated if it is similar.\n\t\tif ( !this.selection.isCollapsed && this.selection.isSimilar( oldViewSelection ) ) {\n\t\t\t// Selection did not changed and is correct, do not update.\n\t\t\treturn false;\n\t\t}\n\n\t\t// Selections are not similar.\n\t\treturn true;\n\t}\n\n\t/**\n\t * Checks whether the fake selection needs to be updated.\n\t *\n\t * @param domRoot A valid DOM root where a new fake selection container should be added.\n\t */\n\tprivate _fakeSelectionNeedsUpdate( domRoot: DomElement ): boolean {\n\t\tconst container = this._fakeSelectionContainer;\n\t\tconst domSelection = domRoot.ownerDocument.getSelection()!;\n\n\t\t// Fake selection needs to be updated if there's no fake selection container, or the container currently sits\n\t\t// in a different root.\n\t\tif ( !container || container.parentElement !== domRoot ) {\n\t\t\treturn true;\n\t\t}\n\n\t\t// Make sure that the selection actually is within the fake selection.\n\t\tif ( domSelection.anchorNode !== container && !container.contains( domSelection.anchorNode ) ) {\n\t\t\treturn true;\n\t\t}\n\n\t\treturn container.textContent !== this.selection.fakeSelectionLabel;\n\t}\n\n\t/**\n\t * Removes the DOM selection.\n\t */\n\tprivate _removeDomSelection(): void {\n\t\tfor ( const doc of this.domDocuments ) {\n\t\t\tconst domSelection = doc.getSelection()!;\n\n\t\t\tif ( domSelection.rangeCount ) {\n\t\t\t\tconst activeDomElement = doc.activeElement!;\n\t\t\t\tconst viewElement = this.domConverter.mapDomToView( activeDomElement as DomElement );\n\n\t\t\t\tif ( activeDomElement && viewElement ) {\n\t\t\t\t\tdomSelection.removeAllRanges();\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Removes the fake selection.\n\t */\n\tprivate _removeFakeSelection(): void {\n\t\tconst container = this._fakeSelectionContainer;\n\n\t\tif ( container ) {\n\t\t\tcontainer.remove();\n\t\t}\n\t}\n\n\t/**\n\t * Checks if focus needs to be updated and possibly updates it.\n\t */\n\tprivate _updateFocus(): void {\n\t\tif ( this.isFocused ) {\n\t\t\tconst editable = this.selection.editableElement;\n\n\t\t\tif ( editable ) {\n\t\t\t\tthis.domConverter.focus( editable );\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Checks if provided element is editable.\n */\nfunction isEditable( element: ViewElement ): boolean {\n\tif ( element.getAttribute( 'contenteditable' ) == 'false' ) {\n\t\treturn false;\n\t}\n\n\tconst parent = element.findAncestor( element => element.hasAttribute( 'contenteditable' ) );\n\n\treturn !parent || parent.getAttribute( 'contenteditable' ) == 'true';\n}\n\n/**\n * Adds inline filler at a given position.\n *\n * The position can be given as an array of DOM nodes and an offset in that array,\n * or a DOM parent element and an offset in that element.\n *\n * @returns The DOM text node that contains an inline filler.\n */\nfunction addInlineFiller( domDocument: DomDocument, domParentOrArray: DomNode | Array<DomNode>, offset: number ): DomText {\n\tconst childNodes = domParentOrArray instanceof Array ? domParentOrArray : domParentOrArray.childNodes;\n\tconst nodeAfterFiller = childNodes[ offset ];\n\n\tif ( isText( nodeAfterFiller ) ) {\n\t\tnodeAfterFiller.data = INLINE_FILLER + nodeAfterFiller.data;\n\n\t\treturn nodeAfterFiller;\n\t} else {\n\t\tconst fillerNode = domDocument.createTextNode( INLINE_FILLER );\n\n\t\tif ( Array.isArray( domParentOrArray ) ) {\n\t\t\t( childNodes as Array<DomNode> ).splice( offset, 0, fillerNode );\n\t\t} else {\n\t\t\tinsertAt( domParentOrArray as DomElement, offset, fillerNode );\n\t\t}\n\n\t\treturn fillerNode;\n\t}\n}\n\n/**\n * Whether two DOM nodes should be considered as similar.\n * Nodes are considered similar if they have the same tag name.\n */\nfunction areSimilarElements( node1: DomNode, node2: DomNode ): boolean {\n\treturn isNode( node1 ) && isNode( node2 ) &&\n\t\t!isText( node1 ) && !isText( node2 ) &&\n\t\t!isComment( node1 ) && !isComment( node2 ) &&\n\t\t( node1 as DomElement ).tagName.toLowerCase() === ( node2 as DomElement ).tagName.toLowerCase();\n}\n\n/**\n * Whether two DOM nodes are text nodes.\n */\nfunction areTextNodes( node1: DomNode, node2: DomNode ): boolean {\n\treturn isNode( node1 ) && isNode( node2 ) &&\n\t\tisText( node1 ) && isText( node2 );\n}\n\n/**\n * Whether two dom nodes should be considered as the same.\n * Two nodes which are considered the same are:\n *\n * * Text nodes with the same text.\n * * Element nodes represented by the same object.\n * * Two block filler elements.\n *\n * @param blockFillerMode Block filler mode, see {@link module:engine/view/domconverter~DomConverter#blockFillerMode}.\n */\nfunction sameNodes( domConverter: DomConverter, actualDomChild: DomNode, expectedDomChild: DomNode ): boolean {\n\t// Elements.\n\tif ( actualDomChild === expectedDomChild ) {\n\t\treturn true;\n\t}\n\t// Texts.\n\telse if ( isText( actualDomChild ) && isText( expectedDomChild ) ) {\n\t\treturn actualDomChild.data === expectedDomChild.data;\n\t}\n\t// Block fillers.\n\telse if ( domConverter.isBlockFiller( actualDomChild ) &&\n\t\tdomConverter.isBlockFiller( expectedDomChild ) ) {\n\t\treturn true;\n\t}\n\n\t// Not matching types.\n\treturn false;\n}\n\n/**\n * The following is a Firefoxspecific hack (https://github.com/ckeditor/ckeditor5-engine/issues/1439).\n * When the native DOM selection is at the end of the block and preceded by <br /> e.g.\n *\n * ```html\n * <p>foo<br/>[]</p>\n * ```\n *\n * which happens a lot when using the soft line break, the browser fails to (visually) move the\n * caret to the new line. A quick fix is as simple as forcerefreshing the selection with the same range.\n */\nfunction fixGeckoSelectionAfterBr( focus: ReturnType<DomConverter[ 'viewPositionToDom' ]>, domSelection: DomSelection ) {\n\tconst parent = focus!.parent;\n\n\t// This fix works only when the focus point is at the very end of an element.\n\t// There is no point in running it in cases unrelated to the browser bug.\n\tif ( parent.nodeType != Node.ELEMENT_NODE || focus!.offset != parent.childNodes.length - 1 ) {\n\t\treturn;\n\t}\n\n\tconst childAtOffset = parent.childNodes[ focus!.offset ];\n\n\t// To stay on the safe side, the fix being as specific as possible, it targets only the\n\t// selection which is at the very end of the element and preceded by <br />.\n\tif ( childAtOffset && ( childAtOffset as DomElement ).tagName == 'BR' ) {\n\t\tdomSelection.addRange( domSelection.getRangeAt( 0 ) );\n\t}\n}\n\nfunction filterOutFakeSelectionContainer( domChildList: Array<DomNode> | NodeList, fakeSelectionContainer: DomElement | null ) {\n\tconst childList = Array.from( domChildList );\n\n\tif ( childList.length == 0 || !fakeSelectionContainer ) {\n\t\treturn childList;\n\t}\n\n\tconst last = childList[ childList.length - 1 ];\n\n\tif ( last == fakeSelectionContainer ) {\n\t\tchildList.pop();\n\t}\n\n\treturn childList;\n}\n\n/**\n * Creates a fake selection container for a given document.\n */\nfunction createFakeSelectionContainer( domDocument: DomDocument ): DomElement {\n\tconst container = domDocument.createElement( 'div' );\n\n\tcontainer.className = 'ck-fake-selection-container';\n\n\tObject.assign( container.style, {\n\t\tposition: 'fixed',\n\t\ttop: 0,\n\t\tleft: '-9999px',\n\t\t// See https://github.com/ckeditor/ckeditor5/issues/752.\n\t\twidth: '42px'\n\t} );\n\n\t// Fill it with a text node so we can update it later.\n\tcontainer.textContent = '\\u00A0';\n\n\treturn container;\n}\n\n/**\n * Checks if text needs to be updated and possibly updates it by removing and inserting only parts\n * of the data from the existing text node to reduce impact on the IME composition.\n *\n * @param domText DOM text node to update.\n * @param expectedText The expected data of a text node.\n */\nfunction updateTextNode( domText: DomText, expectedText: string ) {\n\tconst actualText = domText.data;\n\n\tif ( actualText == expectedText ) {\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[Renderer]%c Text node does not need update:',\n\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', '',\n\t\t// @if CK_DEBUG_TYPING // \t\t`\"${ domText.data }\" (${ domText.data.length })`\n\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t// @if CK_DEBUG_TYPING // }\n\n\t\treturn;\n\t}\n\n\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[Renderer]%c Update text node:',\n\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', '',\n\t// @if CK_DEBUG_TYPING // \t\t`\"${ domText.data }\" (${ domText.data.length }) -> \"${ expectedText }\" (${ expectedText.length })`\n\t// @if CK_DEBUG_TYPING // \t);\n\t// @if CK_DEBUG_TYPING // }\n\n\tconst actions = fastDiff( actualText, expectedText );\n\n\tfor ( const action of actions ) {\n\t\tif ( action.type === 'insert' ) {\n\t\t\tdomText.insertData( action.index, action.values.join( '' ) );\n\t\t} else { // 'delete'\n\t\t\tdomText.deleteData( action.index, action.howMany );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/domconverter\n */\n\n/* globals Node, NodeFilter, DOMParser, Text */\n\nimport ViewText from './text';\nimport ViewElement from './element';\nimport ViewUIElement from './uielement';\nimport ViewPosition from './position';\nimport ViewRange from './range';\nimport ViewSelection from './selection';\nimport ViewDocumentFragment from './documentfragment';\nimport ViewTreeWalker from './treewalker';\nimport { default as Matcher, type MatcherPattern } from './matcher';\nimport {\n\tBR_FILLER, INLINE_FILLER_LENGTH, NBSP_FILLER, MARKED_NBSP_FILLER,\n\tgetDataWithoutFiller, isInlineFiller, startsWithFiller\n} from './filler';\n\nimport {\n\tglobal,\n\tlogWarning,\n\tindexOf,\n\tgetAncestors,\n\tisText,\n\tisComment,\n\tisValidAttributeName,\n\tfirst\n} from '@ckeditor/ckeditor5-utils';\n\nimport type ViewNode from './node';\nimport type Document from './document';\nimport type DocumentSelection from './documentselection';\nimport type EditableElement from './editableelement';\nimport type ViewTextProxy from './textproxy';\nimport type ViewRawElement from './rawelement';\n\ntype DomNode = globalThis.Node;\ntype DomElement = globalThis.HTMLElement;\ntype DomDocument = globalThis.Document;\ntype DomDocumentFragment = globalThis.DocumentFragment;\ntype DomComment = globalThis.Comment;\ntype DomRange = globalThis.Range;\ntype DomText = globalThis.Text;\ntype DomSelection = globalThis.Selection;\n\nconst BR_FILLER_REF = BR_FILLER( global.document ); // eslint-disable-line new-cap\nconst NBSP_FILLER_REF = NBSP_FILLER( global.document ); // eslint-disable-line new-cap\nconst MARKED_NBSP_FILLER_REF = MARKED_NBSP_FILLER( global.document ); // eslint-disable-line new-cap\nconst UNSAFE_ATTRIBUTE_NAME_PREFIX = 'data-ck-unsafe-attribute-';\nconst UNSAFE_ELEMENT_REPLACEMENT_ATTRIBUTE = 'data-ck-unsafe-element';\n\n/**\n * `DomConverter` is a set of tools to do transformations between DOM nodes and view nodes. It also handles\n * {@link module:engine/view/domconverter~DomConverter#bindElements bindings} between these nodes.\n *\n * An instance of the DOM converter is available under\n * {@link module:engine/view/view~View#domConverter `editor.editing.view.domConverter`}.\n *\n * The DOM converter does not check which nodes should be rendered (use {@link module:engine/view/renderer~Renderer}), does not keep the\n * state of a tree nor keeps the synchronization between the tree view and the DOM tree (use {@link module:engine/view/document~Document}).\n *\n * The DOM converter keeps DOM elements to view element bindings, so when the converter gets destroyed, the bindings are lost.\n * Two converters will keep separate binding maps, so one tree view can be bound with two DOM trees.\n */\nexport default class DomConverter {\n\tpublic readonly document: Document;\n\n\t/**\n\t * Whether to leave the View-to-DOM conversion result unchanged or improve editing experience by filtering out interactive data.\n\t */\n\tpublic readonly renderingMode: 'data' | 'editing';\n\n\t/**\n\t * The mode of a block filler used by the DOM converter.\n\t */\n\tpublic blockFillerMode: BlockFillerMode;\n\n\t/**\n\t * Elements which are considered pre-formatted elements.\n\t */\n\tpublic readonly preElements: Array<string>;\n\n\t/**\n\t * Elements which are considered block elements (and hence should be filled with a\n\t * {@link #isBlockFiller block filler}).\n\t *\n\t * Whether an element is considered a block element also affects handling of trailing whitespaces.\n\t *\n\t * You can extend this array if you introduce support for block elements which are not yet recognized here.\n\t */\n\tpublic readonly blockElements: Array<string>;\n\n\t/**\n\t * A list of elements that exist inline (in text) but their inner structure cannot be edited because\n\t * of the way they are rendered by the browser. They are mostly HTML form elements but there are other\n\t * elements such as `<img>` or `<iframe>` that also have non-editable children or no children whatsoever.\n\t *\n\t * Whether an element is considered an inline object has an impact on white space rendering (trimming)\n\t * around (and inside of it). In short, white spaces in text nodes next to inline objects are not trimmed.\n\t *\n\t * You can extend this array if you introduce support for inline object elements which are not yet recognized here.\n\t */\n\tpublic readonly inlineObjectElements: Array<string>;\n\n\t/**\n\t * A list of elements which may affect the editing experience. To avoid this, those elements are replaced with\n\t * `<span data-ck-unsafe-element=\"[element name]\"></span>` while rendering in the editing mode.\n\t */\n\tpublic readonly unsafeElements: Array<string>;\n\n\t/**\n\t * The DOM Document used to create DOM nodes.\n\t */\n\tprivate readonly _domDocument: DomDocument;\n\n\t/**\n\t * The DOM-to-view mapping.\n\t */\n\tprivate readonly _domToViewMapping = new WeakMap<DomElement | DomDocumentFragment, ViewElement | ViewDocumentFragment>();\n\n\t/**\n\t * The view-to-DOM mapping.\n\t */\n\tprivate readonly _viewToDomMapping = new WeakMap<ViewElement | ViewDocumentFragment, DomElement | DomDocumentFragment>();\n\n\t/**\n\t * Holds the mapping between fake selection containers and corresponding view selections.\n\t */\n\tprivate readonly _fakeSelectionMapping = new WeakMap<DomElement, ViewSelection>();\n\n\t/**\n\t * Matcher for view elements whose content should be treated as raw data\n\t * and not processed during the conversion from DOM nodes to view elements.\n\t */\n\tprivate readonly _rawContentElementMatcher = new Matcher();\n\n\t/**\n\t * Matcher for inline object view elements. This is an extension of a simple {@link #inlineObjectElements} array of element names.\n\t */\n\tprivate readonly _inlineObjectElementMatcher = new Matcher();\n\n\t/**\n\t * Creates a DOM converter.\n\t *\n\t * @param document The view document instance.\n\t * @param options An object with configuration options.\n\t * @param options.blockFillerMode The type of the block filler to use.\n\t * Default value depends on the options.renderingMode:\n\t * 'nbsp' when options.renderingMode == 'data',\n\t * 'br' when options.renderingMode == 'editing'.\n\t * @param options.renderingMode Whether to leave the View-to-DOM conversion result unchanged\n\t * or improve editing experience by filtering out interactive data.\n\t */\n\tconstructor(\n\t\tdocument: Document,\n\t\t{ blockFillerMode, renderingMode = 'editing' }: {\n\t\t\tblockFillerMode?: BlockFillerMode;\n\t\t\trenderingMode?: 'data' | 'editing';\n\t\t} = {}\n\t) {\n\t\tthis.document = document;\n\t\tthis.renderingMode = renderingMode;\n\t\tthis.blockFillerMode = blockFillerMode || ( renderingMode === 'editing' ? 'br' : 'nbsp' );\n\t\tthis.preElements = [ 'pre' ];\n\t\tthis.blockElements = [\n\t\t\t'address', 'article', 'aside', 'blockquote', 'caption', 'center', 'dd', 'details', 'dir', 'div',\n\t\t\t'dl', 'dt', 'fieldset', 'figcaption', 'figure', 'footer', 'form', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'header',\n\t\t\t'hgroup', 'legend', 'li', 'main', 'menu', 'nav', 'ol', 'p', 'pre', 'section', 'summary', 'table', 'tbody',\n\t\t\t'td', 'tfoot', 'th', 'thead', 'tr', 'ul'\n\t\t];\n\t\tthis.inlineObjectElements = [\n\t\t\t'object', 'iframe', 'input', 'button', 'textarea', 'select', 'option', 'video', 'embed', 'audio', 'img', 'canvas'\n\t\t];\n\t\tthis.unsafeElements = [ 'script', 'style' ];\n\n\t\tthis._domDocument = this.renderingMode === 'editing' ? global.document : global.document.implementation.createHTMLDocument( '' );\n\t}\n\n\t/**\n\t * Binds a given DOM element that represents fake selection to a **position** of a\n\t * {@link module:engine/view/documentselection~DocumentSelection document selection}.\n\t * Document selection copy is stored and can be retrieved by the\n\t * {@link module:engine/view/domconverter~DomConverter#fakeSelectionToView} method.\n\t */\n\tpublic bindFakeSelection( domElement: DomElement, viewDocumentSelection: DocumentSelection ): void {\n\t\tthis._fakeSelectionMapping.set( domElement, new ViewSelection( viewDocumentSelection ) );\n\t}\n\n\t/**\n\t * Returns a {@link module:engine/view/selection~Selection view selection} instance corresponding to a given\n\t * DOM element that represents fake selection. Returns `undefined` if binding to the given DOM element does not exist.\n\t */\n\tpublic fakeSelectionToView( domElement: DomElement ): ViewSelection | undefined {\n\t\treturn this._fakeSelectionMapping.get( domElement );\n\t}\n\n\t/**\n\t * Binds DOM and view elements, so it will be possible to get corresponding elements using\n\t * {@link module:engine/view/domconverter~DomConverter#mapDomToView} and\n\t * {@link module:engine/view/domconverter~DomConverter#mapViewToDom}.\n\t *\n\t * @param domElement The DOM element to bind.\n\t * @param viewElement The view element to bind.\n\t */\n\tpublic bindElements( domElement: DomElement, viewElement: ViewElement ): void {\n\t\tthis._domToViewMapping.set( domElement, viewElement );\n\t\tthis._viewToDomMapping.set( viewElement, domElement );\n\t}\n\n\t/**\n\t * Unbinds a given DOM element from the view element it was bound to. Unbinding is deep, meaning that all children of\n\t * the DOM element will be unbound too.\n\t *\n\t * @param domElement The DOM element to unbind.\n\t */\n\tpublic unbindDomElement( domElement: DomElement ): void {\n\t\tconst viewElement = this._domToViewMapping.get( domElement );\n\n\t\tif ( viewElement ) {\n\t\t\tthis._domToViewMapping.delete( domElement );\n\t\t\tthis._viewToDomMapping.delete( viewElement );\n\n\t\t\tfor ( const child of Array.from( domElement.children ) ) {\n\t\t\t\tthis.unbindDomElement( child as DomElement );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Binds DOM and view document fragments, so it will be possible to get corresponding document fragments using\n\t * {@link module:engine/view/domconverter~DomConverter#mapDomToView} and\n\t * {@link module:engine/view/domconverter~DomConverter#mapViewToDom}.\n\t *\n\t * @param domFragment The DOM document fragment to bind.\n\t * @param viewFragment The view document fragment to bind.\n\t */\n\tpublic bindDocumentFragments( domFragment: DomDocumentFragment, viewFragment: ViewDocumentFragment ): void {\n\t\tthis._domToViewMapping.set( domFragment, viewFragment );\n\t\tthis._viewToDomMapping.set( viewFragment, domFragment );\n\t}\n\n\t/**\n\t * Decides whether a given pair of attribute key and value should be passed further down the pipeline.\n\t *\n\t * @param elementName Element name in lower case.\n\t */\n\tpublic shouldRenderAttribute( attributeKey: string, attributeValue: string, elementName: string ): boolean {\n\t\tif ( this.renderingMode === 'data' ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tattributeKey = attributeKey.toLowerCase();\n\n\t\tif ( attributeKey.startsWith( 'on' ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif (\n\t\t\tattributeKey === 'srcdoc' &&\n\t\t\tattributeValue.match( /\\bon\\S+\\s*=|javascript:|<\\s*\\/*script/i )\n\t\t) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif (\n\t\t\telementName === 'img' &&\n\t\t\t( attributeKey === 'src' || attributeKey === 'srcset' )\n\t\t) {\n\t\t\treturn true;\n\t\t}\n\n\t\tif ( elementName === 'source' && attributeKey === 'srcset' ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tif ( attributeValue.match( /^\\s*(javascript:|data:(image\\/svg|text\\/x?html))/i ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Set `domElement`'s content using provided `html` argument. Apply necessary filtering for the editing pipeline.\n\t *\n\t * @param domElement DOM element that should have `html` set as its content.\n\t * @param html Textual representation of the HTML that will be set on `domElement`.\n\t */\n\tpublic setContentOf( domElement: DomElement, html: string ): void {\n\t\t// For data pipeline we pass the HTML as-is.\n\t\tif ( this.renderingMode === 'data' ) {\n\t\t\tdomElement.innerHTML = html;\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst document = new DOMParser().parseFromString( html, 'text/html' );\n\t\tconst fragment = document.createDocumentFragment();\n\t\tconst bodyChildNodes = document.body.childNodes;\n\n\t\twhile ( bodyChildNodes.length > 0 ) {\n\t\t\tfragment.appendChild( bodyChildNodes[ 0 ] );\n\t\t}\n\n\t\tconst treeWalker = document.createTreeWalker( fragment, NodeFilter.SHOW_ELEMENT );\n\t\tconst nodes: Array<DomElement> = [];\n\n\t\tlet currentNode;\n\n\t\t// eslint-disable-next-line no-cond-assign\n\t\twhile ( currentNode = treeWalker.nextNode() ) {\n\t\t\tnodes.push( currentNode as DomElement );\n\t\t}\n\n\t\tfor ( const currentNode of nodes ) {\n\t\t\t// Go through nodes to remove those that are prohibited in editing pipeline.\n\t\t\tfor ( const attributeName of currentNode.getAttributeNames() ) {\n\t\t\t\tthis.setDomElementAttribute( currentNode, attributeName, currentNode.getAttribute( attributeName )! );\n\t\t\t}\n\n\t\t\tconst elementName = currentNode.tagName.toLowerCase();\n\n\t\t\t// There are certain nodes, that should be renamed to <span> in editing pipeline.\n\t\t\tif ( this._shouldRenameElement( elementName ) ) {\n\t\t\t\t_logUnsafeElement( elementName );\n\n\t\t\t\tcurrentNode.replaceWith( this._createReplacementDomElement( elementName, currentNode ) );\n\t\t\t}\n\t\t}\n\n\t\t// Empty the target element.\n\t\twhile ( domElement.firstChild ) {\n\t\t\tdomElement.firstChild.remove();\n\t\t}\n\n\t\tdomElement.append( fragment );\n\t}\n\n\t/**\n\t * Converts the view to the DOM. For all text nodes, not bound elements and document fragments new items will\n\t * be created. For bound elements and document fragments the method will return corresponding items.\n\t *\n\t * @param viewNode View node or document fragment to transform.\n\t * @param options Conversion options.\n\t * @param options.bind Determines whether new elements will be bound.\n\t * @param options.withChildren If `false`, node's and document fragment's children will not be converted.\n\t * @returns Converted node or DocumentFragment.\n\t */\n\tpublic viewToDom(\n\t\tviewNode: ViewNode | ViewDocumentFragment,\n\t\toptions: { bind?: boolean; withChildren?: boolean } = {}\n\t): DomNode | DomDocumentFragment {\n\t\tif ( viewNode.is( '$text' ) ) {\n\t\t\tconst textData = this._processDataFromViewText( viewNode );\n\n\t\t\treturn this._domDocument.createTextNode( textData );\n\t\t} else {\n\t\t\tif ( this.mapViewToDom( viewNode as ViewElement ) ) {\n\t\t\t\treturn this.mapViewToDom( viewNode as ViewElement )!;\n\t\t\t}\n\n\t\t\tlet domElement: DomElement | DomDocumentFragment | DomComment;\n\n\t\t\tif ( viewNode.is( 'documentFragment' ) ) {\n\t\t\t\t// Create DOM document fragment.\n\t\t\t\tdomElement = this._domDocument.createDocumentFragment();\n\n\t\t\t\tif ( options.bind ) {\n\t\t\t\t\tthis.bindDocumentFragments( domElement, viewNode );\n\t\t\t\t}\n\t\t\t} else if ( viewNode.is( 'uiElement' ) ) {\n\t\t\t\tif ( viewNode.name === '$comment' ) {\n\t\t\t\t\tdomElement = this._domDocument.createComment( viewNode.getCustomProperty( '$rawContent' ) as string );\n\t\t\t\t} else {\n\t\t\t\t\t// UIElement has its own render() method (see #799).\n\t\t\t\t\tdomElement = viewNode.render( this._domDocument, this );\n\t\t\t\t}\n\n\t\t\t\tif ( options.bind ) {\n\t\t\t\t\tthis.bindElements( domElement as DomElement, viewNode );\n\t\t\t\t}\n\n\t\t\t\treturn domElement;\n\t\t\t} else {\n\t\t\t\t// Create DOM element.\n\t\t\t\tif ( this._shouldRenameElement( ( viewNode as ViewElement ).name ) ) {\n\t\t\t\t\t_logUnsafeElement( ( viewNode as ViewElement ).name );\n\n\t\t\t\t\tdomElement = this._createReplacementDomElement( ( viewNode as ViewElement ).name );\n\t\t\t\t} else if ( ( viewNode as ViewElement ).hasAttribute( 'xmlns' ) ) {\n\t\t\t\t\tdomElement = this._domDocument.createElementNS(\n\t\t\t\t\t\t( viewNode as ViewElement ).getAttribute( 'xmlns' )!,\n\t\t\t\t\t\t( viewNode as ViewElement ).name\n\t\t\t\t\t) as HTMLElement;\n\t\t\t\t} else {\n\t\t\t\t\tdomElement = this._domDocument.createElement( ( viewNode as ViewElement ).name );\n\t\t\t\t}\n\n\t\t\t\t// RawElement take care of their children in RawElement#render() method which can be customized\n\t\t\t\t// (see https://github.com/ckeditor/ckeditor5/issues/4469).\n\t\t\t\tif ( viewNode.is( 'rawElement' ) ) {\n\t\t\t\t\tviewNode.render( domElement, this );\n\t\t\t\t}\n\n\t\t\t\tif ( options.bind ) {\n\t\t\t\t\tthis.bindElements( domElement, ( viewNode as ViewElement ) );\n\t\t\t\t}\n\n\t\t\t\t// Copy element's attributes.\n\t\t\t\tfor ( const key of ( viewNode as ViewElement ).getAttributeKeys() ) {\n\t\t\t\t\tthis.setDomElementAttribute(\n\t\t\t\t\t\tdomElement,\n\t\t\t\t\t\tkey,\n\t\t\t\t\t\t( viewNode as ViewElement ).getAttribute( key )!,\n\t\t\t\t\t\t( viewNode as ViewElement )\n\t\t\t\t\t);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( options.withChildren !== false ) {\n\t\t\t\tfor ( const child of this.viewChildrenToDom( viewNode as ViewElement, options ) ) {\n\t\t\t\t\tdomElement!.appendChild( child );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\treturn domElement!;\n\t\t}\n\t}\n\n\t/**\n\t * Sets the attribute on a DOM element.\n\t *\n\t * **Note**: To remove the attribute, use {@link #removeDomElementAttribute}.\n\t *\n\t * @param domElement The DOM element the attribute should be set on.\n\t * @param key The name of the attribute.\n\t * @param value The value of the attribute.\n\t * @param relatedViewElement The view element related to the `domElement` (if there is any).\n\t * It helps decide whether the attribute set is unsafe. For instance, view elements created via the\n\t * {@link module:engine/view/downcastwriter~DowncastWriter} methods can allow certain attributes that would normally be filtered out.\n\t */\n\tpublic setDomElementAttribute( domElement: DomElement, key: string, value: string, relatedViewElement?: ViewElement ): void {\n\t\tconst shouldRenderAttribute = this.shouldRenderAttribute( key, value, domElement.tagName.toLowerCase() ) ||\n\t\t\trelatedViewElement && relatedViewElement.shouldRenderUnsafeAttribute( key );\n\n\t\tif ( !shouldRenderAttribute ) {\n\t\t\tlogWarning( 'domconverter-unsafe-attribute-detected', { domElement, key, value } );\n\t\t}\n\n\t\tif ( !isValidAttributeName( key ) ) {\n\t\t\t/**\n\t\t\t * Invalid attribute name was ignored during rendering.\n\t\t\t *\n\t\t\t * @error domconverter-invalid-attribute-detected\n\t\t\t */\n\t\t\tlogWarning( 'domconverter-invalid-attribute-detected', { domElement, key, value } );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// The old value was safe but the new value is unsafe.\n\t\tif ( domElement.hasAttribute( key ) && !shouldRenderAttribute ) {\n\t\t\tdomElement.removeAttribute( key );\n\t\t}\n\t\t// The old value was unsafe (but prefixed) but the new value will be safe (will be unprefixed).\n\t\telse if ( domElement.hasAttribute( UNSAFE_ATTRIBUTE_NAME_PREFIX + key ) && shouldRenderAttribute ) {\n\t\t\tdomElement.removeAttribute( UNSAFE_ATTRIBUTE_NAME_PREFIX + key );\n\t\t}\n\n\t\t// If the attribute should not be rendered, rename it (instead of removing) to give developers some idea of what\n\t\t// is going on (https://github.com/ckeditor/ckeditor5/issues/10801).\n\t\tdomElement.setAttribute( shouldRenderAttribute ? key : UNSAFE_ATTRIBUTE_NAME_PREFIX + key, value );\n\t}\n\n\t/**\n\t * Removes an attribute from a DOM element.\n\t *\n\t * **Note**: To set the attribute, use {@link #setDomElementAttribute}.\n\t *\n\t * @param domElement The DOM element the attribute should be removed from.\n\t * @param key The name of the attribute.\n\t */\n\tpublic removeDomElementAttribute( domElement: DomElement, key: string ): void {\n\t\t// See #_createReplacementDomElement() to learn what this is.\n\t\tif ( key == UNSAFE_ELEMENT_REPLACEMENT_ATTRIBUTE ) {\n\t\t\treturn;\n\t\t}\n\n\t\tdomElement.removeAttribute( key );\n\n\t\t// See setDomElementAttribute() to learn what this is.\n\t\tdomElement.removeAttribute( UNSAFE_ATTRIBUTE_NAME_PREFIX + key );\n\t}\n\n\t/**\n\t * Converts children of the view element to DOM using the\n\t * {@link module:engine/view/domconverter~DomConverter#viewToDom} method.\n\t * Additionally, this method adds block {@link module:engine/view/filler filler} to the list of children, if needed.\n\t *\n\t * @param viewElement Parent view element.\n\t * @param options See {@link module:engine/view/domconverter~DomConverter#viewToDom} options parameter.\n\t * @returns DOM nodes.\n\t */\n\tpublic* viewChildrenToDom(\n\t\tviewElement: ViewElement,\n\t\toptions: { bind?: boolean; withChildren?: boolean } = {}\n\t): IterableIterator<Node> {\n\t\tconst fillerPositionOffset = viewElement.getFillerOffset && viewElement.getFillerOffset();\n\t\tlet offset = 0;\n\n\t\tfor ( const childView of viewElement.getChildren() ) {\n\t\t\tif ( fillerPositionOffset === offset ) {\n\t\t\t\tyield this._getBlockFiller();\n\t\t\t}\n\n\t\t\tconst transparentRendering = childView.is( 'element' ) &&\n\t\t\t\t!!childView.getCustomProperty( 'dataPipeline:transparentRendering' ) &&\n\t\t\t\t!first( childView.getAttributes() );\n\n\t\t\tif ( transparentRendering && this.renderingMode == 'data' ) {\n\t\t\t\tyield* this.viewChildrenToDom( childView, options );\n\t\t\t} else {\n\t\t\t\tif ( transparentRendering ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * The `dataPipeline:transparentRendering` flag is supported only in the data pipeline.\n\t\t\t\t\t *\n\t\t\t\t\t * @error domconverter-transparent-rendering-unsupported-in-editing-pipeline\n\t\t\t\t\t */\n\t\t\t\t\tlogWarning( 'domconverter-transparent-rendering-unsupported-in-editing-pipeline', { viewElement: childView } );\n\t\t\t\t}\n\n\t\t\t\tyield this.viewToDom( childView, options );\n\t\t\t}\n\n\t\t\toffset++;\n\t\t}\n\n\t\tif ( fillerPositionOffset === offset ) {\n\t\t\tyield this._getBlockFiller();\n\t\t}\n\t}\n\n\t/**\n\t * Converts view {@link module:engine/view/range~Range} to DOM range.\n\t * Inline and block {@link module:engine/view/filler fillers} are handled during the conversion.\n\t *\n\t * @param viewRange View range.\n\t * @returns DOM range.\n\t */\n\tpublic viewRangeToDom( viewRange: ViewRange ): DomRange {\n\t\tconst domStart = this.viewPositionToDom( viewRange.start )!;\n\t\tconst domEnd = this.viewPositionToDom( viewRange.end )!;\n\n\t\tconst domRange = this._domDocument.createRange();\n\t\tdomRange.setStart( domStart.parent, domStart.offset );\n\t\tdomRange.setEnd( domEnd.parent, domEnd.offset );\n\n\t\treturn domRange;\n\t}\n\n\t/**\n\t * Converts view {@link module:engine/view/position~Position} to DOM parent and offset.\n\t *\n\t * Inline and block {@link module:engine/view/filler fillers} are handled during the conversion.\n\t * If the converted position is directly before inline filler it is moved inside the filler.\n\t *\n\t * @param viewPosition View position.\n\t * @returns DOM position or `null` if view position could not be converted to DOM.\n\t * DOM position has two properties:\n\t * * `parent` - DOM position parent.\n\t * * `offset` - DOM position offset.\n\t */\n\tpublic viewPositionToDom( viewPosition: ViewPosition ): { parent: DomNode; offset: number } | null {\n\t\tconst viewParent = viewPosition.parent;\n\n\t\tif ( viewParent.is( '$text' ) ) {\n\t\t\tconst domParent = this.findCorrespondingDomText( viewParent );\n\n\t\t\tif ( !domParent ) {\n\t\t\t\t// Position is in a view text node that has not been rendered to DOM yet.\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tlet offset = viewPosition.offset;\n\n\t\t\tif ( startsWithFiller( domParent ) ) {\n\t\t\t\toffset += INLINE_FILLER_LENGTH;\n\t\t\t}\n\n\t\t\treturn { parent: domParent, offset };\n\t\t} else {\n\t\t\t// viewParent is instance of ViewElement.\n\t\t\tlet domParent, domBefore, domAfter;\n\n\t\t\tif ( viewPosition.offset === 0 ) {\n\t\t\t\tdomParent = this.mapViewToDom( viewParent as ViewElement );\n\n\t\t\t\tif ( !domParent ) {\n\t\t\t\t\t// Position is in a view element that has not been rendered to DOM yet.\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\tdomAfter = domParent.childNodes[ 0 ];\n\t\t\t} else {\n\t\t\t\tconst nodeBefore = viewPosition.nodeBefore!;\n\n\t\t\t\tdomBefore = nodeBefore.is( '$text' ) ?\n\t\t\t\t\tthis.findCorrespondingDomText( nodeBefore ) :\n\t\t\t\t\tthis.mapViewToDom( nodeBefore as ViewElement );\n\n\t\t\t\tif ( !domBefore ) {\n\t\t\t\t\t// Position is after a view element that has not been rendered to DOM yet.\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\tdomParent = domBefore.parentNode;\n\t\t\t\tdomAfter = domBefore.nextSibling;\n\t\t\t}\n\n\t\t\t// If there is an inline filler at position return position inside the filler. We should never return\n\t\t\t// the position before the inline filler.\n\t\t\tif ( isText( domAfter ) && startsWithFiller( domAfter ) ) {\n\t\t\t\treturn { parent: domAfter, offset: INLINE_FILLER_LENGTH };\n\t\t\t}\n\n\t\t\tconst offset = domBefore ? indexOf( domBefore ) + 1 : 0;\n\n\t\t\treturn { parent: domParent!, offset };\n\t\t}\n\t}\n\n\t/**\n\t * Converts DOM to view. For all text nodes, not bound elements and document fragments new items will\n\t * be created. For bound elements and document fragments function will return corresponding items. For\n\t * {@link module:engine/view/filler fillers} `null` will be returned.\n\t * For all DOM elements rendered by {@link module:engine/view/uielement~UIElement} that UIElement will be returned.\n\t *\n\t * @param domNode DOM node or document fragment to transform.\n\t * @param options Conversion options.\n\t * @param options.bind Determines whether new elements will be bound. False by default.\n\t * @param options.withChildren If `true`, node's and document fragment's children will be converted too. True by default.\n\t * @param options.keepOriginalCase If `false`, node's tag name will be converted to lower case. False by default.\n\t * @param options.skipComments If `false`, comment nodes will be converted to `$comment`\n\t * {@link module:engine/view/uielement~UIElement view UI elements}. False by default.\n\t * @returns Converted node or document fragment or `null` if DOM node is a {@link module:engine/view/filler filler}\n\t * or the given node is an empty text node.\n\t */\n\tpublic domToView(\n\t\tdomNode: DomNode,\n\t\toptions: {\n\t\t\tbind?: boolean;\n\t\t\twithChildren?: boolean;\n\t\t\tkeepOriginalCase?: boolean;\n\t\t\tskipComments?: boolean;\n\t\t} = {}\n\t): ViewNode | ViewDocumentFragment | null {\n\t\tconst inlineNodes: Array<ViewNode> = [];\n\t\tconst generator = this._domToView( domNode, options, inlineNodes );\n\n\t\t// Get the first yielded value or a returned value.\n\t\tconst node = generator.next().value;\n\n\t\tif ( !node ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// Trigger children handling.\n\t\tgenerator.next();\n\n\t\t// Whitespace cleaning.\n\t\tthis._processDomInlineNodes( null, inlineNodes, options );\n\n\t\t// Text not got trimmed to an empty string so there is no result node.\n\t\tif ( node.is( '$text' ) && node.data.length == 0 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn node;\n\t}\n\n\t/**\n\t * Converts children of the DOM element to view nodes using\n\t * the {@link module:engine/view/domconverter~DomConverter#domToView} method.\n\t * Additionally this method omits block {@link module:engine/view/filler filler}, if it exists in the DOM parent.\n\t *\n\t * @param domElement Parent DOM element.\n\t * @param options See {@link module:engine/view/domconverter~DomConverter#domToView} options parameter.\n\t * @param inlineNodes An array that will be populated with inline nodes. It's used internally for whitespace processing.\n\t * @returns View nodes.\n\t */\n\tpublic* domChildrenToView(\n\t\tdomElement: DomElement,\n\t\toptions: Parameters<DomConverter[ 'domToView' ]>[ 1 ] = {},\n\t\tinlineNodes: Array<ViewNode> = []\n\t): IterableIterator<ViewNode> {\n\t\tfor ( let i = 0; i < domElement.childNodes.length; i++ ) {\n\t\t\tconst domChild = domElement.childNodes[ i ];\n\t\t\tconst generator = this._domToView( domChild, options, inlineNodes );\n\n\t\t\t// Get the first yielded value or a returned value.\n\t\t\tconst viewChild = generator.next().value as ViewNode | null;\n\n\t\t\tif ( viewChild !== null ) {\n\t\t\t\t// Whitespace cleaning before entering a block element (between block elements).\n\t\t\t\tif ( this._isBlockViewElement( viewChild ) ) {\n\t\t\t\t\tthis._processDomInlineNodes( domElement, inlineNodes, options );\n\t\t\t\t}\n\n\t\t\t\tyield viewChild;\n\n\t\t\t\t// Trigger children handling.\n\t\t\t\tgenerator.next();\n\t\t\t}\n\t\t}\n\n\t\t// Whitespace cleaning before leaving a block element (content of block element).\n\t\tthis._processDomInlineNodes( domElement, inlineNodes, options );\n\t}\n\n\t/**\n\t * Converts DOM selection to view {@link module:engine/view/selection~Selection}.\n\t * Ranges which cannot be converted will be omitted.\n\t *\n\t * @param domSelection DOM selection.\n\t * @returns View selection.\n\t */\n\tpublic domSelectionToView( domSelection: DomSelection ): ViewSelection {\n\t\t// DOM selection might be placed in fake selection container.\n\t\t// If container contains fake selection - return corresponding view selection.\n\t\tif ( domSelection.rangeCount === 1 ) {\n\t\t\tlet container: Node | null = domSelection.getRangeAt( 0 ).startContainer;\n\n\t\t\t// The DOM selection might be moved to the text node inside the fake selection container.\n\t\t\tif ( isText( container ) ) {\n\t\t\t\tcontainer = container.parentNode;\n\t\t\t}\n\n\t\t\tconst viewSelection = this.fakeSelectionToView( container as DomElement );\n\n\t\t\tif ( viewSelection ) {\n\t\t\t\treturn viewSelection;\n\t\t\t}\n\t\t}\n\n\t\tconst isBackward = this.isDomSelectionBackward( domSelection );\n\n\t\tconst viewRanges: Array<ViewRange> = [];\n\n\t\tfor ( let i = 0; i < domSelection.rangeCount; i++ ) {\n\t\t\t// DOM Range have correct start and end, no matter what is the DOM Selection direction. So we don't have to fix anything.\n\t\t\tconst domRange = domSelection.getRangeAt( i );\n\t\t\tconst viewRange = this.domRangeToView( domRange );\n\n\t\t\tif ( viewRange ) {\n\t\t\t\tviewRanges.push( viewRange );\n\t\t\t}\n\t\t}\n\n\t\treturn new ViewSelection( viewRanges, { backward: isBackward } );\n\t}\n\n\t/**\n\t * Converts DOM Range to view {@link module:engine/view/range~Range}.\n\t * If the start or end position can not be converted `null` is returned.\n\t *\n\t * @param domRange DOM range.\n\t * @returns View range.\n\t */\n\tpublic domRangeToView( domRange: DomRange | StaticRange ): ViewRange | null {\n\t\tconst viewStart = this.domPositionToView( domRange.startContainer, domRange.startOffset );\n\t\tconst viewEnd = this.domPositionToView( domRange.endContainer, domRange.endOffset );\n\n\t\tif ( viewStart && viewEnd ) {\n\t\t\treturn new ViewRange( viewStart, viewEnd );\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Converts DOM parent and offset to view {@link module:engine/view/position~Position}.\n\t *\n\t * If the position is inside a {@link module:engine/view/filler filler} which has no corresponding view node,\n\t * position of the filler will be converted and returned.\n\t *\n\t * If the position is inside DOM element rendered by {@link module:engine/view/uielement~UIElement}\n\t * that position will be converted to view position before that UIElement.\n\t *\n\t * If structures are too different and it is not possible to find corresponding position then `null` will be returned.\n\t *\n\t * @param domParent DOM position parent.\n\t * @param domOffset DOM position offset. You can skip it when converting the inline filler node.\n\t * @returns View position.\n\t */\n\tpublic domPositionToView( domParent: DomNode, domOffset: number = 0 ): ViewPosition | null {\n\t\tif ( this.isBlockFiller( domParent ) ) {\n\t\t\treturn this.domPositionToView( domParent.parentNode!, indexOf( domParent ) );\n\t\t}\n\n\t\t// If position is somewhere inside UIElement or a RawElement - return position before that element.\n\t\tconst viewElement = this.mapDomToView( domParent as DomElement );\n\n\t\tif ( viewElement && ( viewElement.is( 'uiElement' ) || viewElement.is( 'rawElement' ) ) ) {\n\t\t\treturn ViewPosition._createBefore( viewElement );\n\t\t}\n\n\t\tif ( isText( domParent ) ) {\n\t\t\tif ( isInlineFiller( domParent ) ) {\n\t\t\t\treturn this.domPositionToView( domParent.parentNode!, indexOf( domParent ) );\n\t\t\t}\n\n\t\t\tconst viewParent = this.findCorrespondingViewText( domParent );\n\t\t\tlet offset = domOffset;\n\n\t\t\tif ( !viewParent ) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tif ( startsWithFiller( domParent ) ) {\n\t\t\t\toffset -= INLINE_FILLER_LENGTH;\n\t\t\t\toffset = offset < 0 ? 0 : offset;\n\t\t\t}\n\n\t\t\treturn new ViewPosition( viewParent, offset );\n\t\t}\n\t\t// domParent instanceof HTMLElement.\n\t\telse {\n\t\t\tif ( domOffset === 0 ) {\n\t\t\t\tconst viewParent = this.mapDomToView( domParent as DomElement );\n\n\t\t\t\tif ( viewParent ) {\n\t\t\t\t\treturn new ViewPosition( viewParent, 0 );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconst domBefore = domParent.childNodes[ domOffset - 1 ];\n\n\t\t\t\t// Jump over an inline filler (and also on Firefox jump over a block filler while pressing backspace in an empty paragraph).\n\t\t\t\tif ( isText( domBefore ) && isInlineFiller( domBefore ) || domBefore && this.isBlockFiller( domBefore ) ) {\n\t\t\t\t\treturn this.domPositionToView( domBefore.parentNode!, indexOf( domBefore ) );\n\t\t\t\t}\n\n\t\t\t\tconst viewBefore = isText( domBefore ) ?\n\t\t\t\t\tthis.findCorrespondingViewText( domBefore ) :\n\t\t\t\t\tthis.mapDomToView( domBefore as DomElement );\n\n\t\t\t\t// TODO #663\n\t\t\t\tif ( viewBefore && viewBefore.parent ) {\n\t\t\t\t\treturn new ViewPosition( viewBefore.parent, viewBefore.index! + 1 );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\treturn null;\n\t\t}\n\t}\n\n\t/**\n\t * Returns corresponding view {@link module:engine/view/element~Element Element} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment} for provided DOM element or\n\t * document fragment. If there is no view item {@link module:engine/view/domconverter~DomConverter#bindElements bound}\n\t * to the given DOM - `undefined` is returned.\n\t *\n\t * For all DOM elements rendered by a {@link module:engine/view/uielement~UIElement} or\n\t * a {@link module:engine/view/rawelement~RawElement}, the parent `UIElement` or `RawElement` will be returned.\n\t *\n\t * @param domElementOrDocumentFragment DOM element or document fragment.\n\t * @returns Corresponding view element, document fragment or `undefined` if no element was bound.\n\t */\n\tpublic mapDomToView( domElementOrDocumentFragment: DomElement | DomDocumentFragment ): ViewElement | ViewDocumentFragment | undefined {\n\t\tconst hostElement = this.getHostViewElement( domElementOrDocumentFragment );\n\n\t\treturn hostElement || this._domToViewMapping.get( domElementOrDocumentFragment );\n\t}\n\n\t/**\n\t * Finds corresponding text node. Text nodes are not {@link module:engine/view/domconverter~DomConverter#bindElements bound},\n\t * corresponding text node is returned based on the sibling or parent.\n\t *\n\t * If the directly previous sibling is a {@link module:engine/view/domconverter~DomConverter#bindElements bound} element, it is used\n\t * to find the corresponding text node.\n\t *\n\t * If this is a first child in the parent and the parent is a {@link module:engine/view/domconverter~DomConverter#bindElements bound}\n\t * element, it is used to find the corresponding text node.\n\t *\n\t * For all text nodes rendered by a {@link module:engine/view/uielement~UIElement} or\n\t * a {@link module:engine/view/rawelement~RawElement}, the parent `UIElement` or `RawElement` will be returned.\n\t *\n\t * Otherwise `null` is returned.\n\t *\n\t * Note that for the block or inline {@link module:engine/view/filler filler} this method returns `null`.\n\t *\n\t * @param domText DOM text node.\n\t * @returns Corresponding view text node or `null`, if it was not possible to find a corresponding node.\n\t */\n\tpublic findCorrespondingViewText( domText: DomText ): ViewText | ViewUIElement | ViewRawElement | null {\n\t\tif ( isInlineFiller( domText ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// If DOM text was rendered by a UIElement or a RawElement - return this parent element.\n\t\tconst hostElement = this.getHostViewElement( domText );\n\n\t\tif ( hostElement ) {\n\t\t\treturn hostElement;\n\t\t}\n\n\t\tconst previousSibling = domText.previousSibling;\n\n\t\t// Try to use previous sibling to find the corresponding text node.\n\t\tif ( previousSibling ) {\n\t\t\tif ( !( this.isElement( previousSibling ) ) ) {\n\t\t\t\t// The previous is text or comment.\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tconst viewElement = this.mapDomToView( previousSibling );\n\n\t\t\tif ( viewElement ) {\n\t\t\t\tconst nextSibling = ( viewElement as ViewElement ).nextSibling;\n\n\t\t\t\t// It might be filler which has no corresponding view node.\n\t\t\t\tif ( nextSibling instanceof ViewText ) {\n\t\t\t\t\treturn nextSibling;\n\t\t\t\t} else {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t\t// Try to use parent to find the corresponding text node.\n\t\telse {\n\t\t\tconst viewElement = this.mapDomToView( domText.parentNode as ( DomElement | DomDocumentFragment ) );\n\n\t\t\tif ( viewElement ) {\n\t\t\t\tconst firstChild = ( viewElement as ViewElement ).getChild( 0 );\n\n\t\t\t\t// It might be filler which has no corresponding view node.\n\t\t\t\tif ( firstChild instanceof ViewText ) {\n\t\t\t\t\treturn firstChild;\n\t\t\t\t} else {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns corresponding DOM item for provided {@link module:engine/view/element~Element Element} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment DocumentFragment}.\n\t * To find a corresponding text for {@link module:engine/view/text~Text view Text instance}\n\t * use {@link #findCorrespondingDomText}.\n\t *\n\t * @label ELEMENT\n\t * @param element View element or document fragment.\n\t * @returns Corresponding DOM node or document fragment.\n\t */\n\tpublic mapViewToDom( element: ViewElement ): DomElement | undefined;\n\n\t/**\n\t * Returns corresponding DOM item for provided {@link module:engine/view/element~Element Element} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment DocumentFragment}.\n\t * To find a corresponding text for {@link module:engine/view/text~Text view Text instance}\n\t * use {@link #findCorrespondingDomText}.\n\t *\n\t * @label DOCUMENT_FRAGMENT\n\t * @param documentFragment View element or document fragment.\n\t * @returns Corresponding DOM node or document fragment.\n\t */\n\tpublic mapViewToDom( documentFragment: ViewDocumentFragment ): DomDocumentFragment | undefined;\n\n\t/**\n\t * Returns corresponding DOM item for provided {@link module:engine/view/element~Element Element} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment DocumentFragment}.\n\t * To find a corresponding text for {@link module:engine/view/text~Text view Text instance}\n\t * use {@link #findCorrespondingDomText}.\n\t *\n\t * @label DOCUMENT_FRAGMENT_OR_ELEMENT\n\t * @param documentFragmentOrElement View element or document fragment.\n\t * @returns Corresponding DOM node or document fragment.\n\t */\n\tpublic mapViewToDom( documentFragmentOrElement: ViewElement | ViewDocumentFragment ): DomElement | DomDocumentFragment | undefined;\n\n\tpublic mapViewToDom( documentFragmentOrElement: ViewElement | ViewDocumentFragment ): DomElement | DomDocumentFragment | undefined {\n\t\treturn this._viewToDomMapping.get( documentFragmentOrElement );\n\t}\n\n\t/**\n\t * Finds corresponding text node. Text nodes are not {@link module:engine/view/domconverter~DomConverter#bindElements bound},\n\t * corresponding text node is returned based on the sibling or parent.\n\t *\n\t * If the directly previous sibling is a {@link module:engine/view/domconverter~DomConverter#bindElements bound} element, it is used\n\t * to find the corresponding text node.\n\t *\n\t * If this is a first child in the parent and the parent is a {@link module:engine/view/domconverter~DomConverter#bindElements bound}\n\t * element, it is used to find the corresponding text node.\n\t *\n\t * Otherwise `null` is returned.\n\t *\n\t * @param viewText View text node.\n\t * @returns Corresponding DOM text node or `null`, if it was not possible to find a corresponding node.\n\t */\n\tpublic findCorrespondingDomText( viewText: ViewText ): DomText | null {\n\t\tconst previousSibling = viewText.previousSibling;\n\n\t\t// Try to use previous sibling to find the corresponding text node.\n\t\tif ( previousSibling && this.mapViewToDom( previousSibling as ViewElement ) ) {\n\t\t\treturn this.mapViewToDom( previousSibling as ViewElement )!.nextSibling as DomText;\n\t\t}\n\n\t\t// If this is a first node, try to use parent to find the corresponding text node.\n\t\tif ( !previousSibling && viewText.parent && this.mapViewToDom( viewText.parent ) ) {\n\t\t\treturn this.mapViewToDom( viewText.parent )!.childNodes[ 0 ] as DomText;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Focuses DOM editable that is corresponding to provided {@link module:engine/view/editableelement~EditableElement}.\n\t */\n\tpublic focus( viewEditable: EditableElement ): void {\n\t\tconst domEditable = this.mapViewToDom( viewEditable );\n\n\t\tif ( domEditable && domEditable.ownerDocument.activeElement !== domEditable ) {\n\t\t\t// Save the scrollX and scrollY positions before the focus.\n\t\t\tconst { scrollX, scrollY } = global.window;\n\t\t\tconst scrollPositions: Array<[ number, number ]> = [];\n\n\t\t\t// Save all scrollLeft and scrollTop values starting from domEditable up to\n\t\t\t// document#documentElement.\n\t\t\tforEachDomElementAncestor( domEditable, node => {\n\t\t\t\tconst { scrollLeft, scrollTop } = ( node as DomElement );\n\n\t\t\t\tscrollPositions.push( [ scrollLeft, scrollTop ] );\n\t\t\t} );\n\n\t\t\tdomEditable.focus();\n\n\t\t\t// Restore scrollLeft and scrollTop values starting from domEditable up to\n\t\t\t// document#documentElement.\n\t\t\t// https://github.com/ckeditor/ckeditor5-engine/issues/951\n\t\t\t// https://github.com/ckeditor/ckeditor5-engine/issues/957\n\t\t\tforEachDomElementAncestor( domEditable, node => {\n\t\t\t\tconst [ scrollLeft, scrollTop ] = scrollPositions.shift() as [ number, number ];\n\n\t\t\t\tnode.scrollLeft = scrollLeft;\n\t\t\t\tnode.scrollTop = scrollTop;\n\t\t\t} );\n\n\t\t\t// Restore the scrollX and scrollY positions after the focus.\n\t\t\t// https://github.com/ckeditor/ckeditor5-engine/issues/951\n\t\t\tglobal.window.scrollTo( scrollX, scrollY );\n\t\t}\n\t}\n\n\t/**\n\t * Remove DOM selection from blurred editable, so it won't interfere with clicking on dropdowns (especially on iOS).\n\t *\n\t * @internal\n\t */\n\tpublic _clearDomSelection(): void {\n\t\tconst domEditable = this.mapViewToDom( this.document.selection.editableElement! );\n\n\t\tif ( !domEditable ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Check if DOM selection is inside editor editable element.\n\t\tconst domSelection = domEditable.ownerDocument.defaultView!.getSelection()!;\n\t\tconst newViewSelection = this.domSelectionToView( domSelection );\n\t\tconst selectionInEditable = newViewSelection && newViewSelection.rangeCount > 0;\n\n\t\tif ( selectionInEditable ) {\n\t\t\tdomSelection.removeAllRanges();\n\t\t}\n\t}\n\n\t/**\n\t * Returns `true` when `node.nodeType` equals `Node.ELEMENT_NODE`.\n\t *\n\t * @param node Node to check.\n\t */\n\tpublic isElement( node: DomNode ): node is DomElement {\n\t\treturn node && node.nodeType == Node.ELEMENT_NODE;\n\t}\n\n\t/**\n\t * Returns `true` when `node.nodeType` equals `Node.DOCUMENT_FRAGMENT_NODE`.\n\t *\n\t * @param node Node to check.\n\t */\n\tpublic isDocumentFragment( node: DomNode ): node is DomDocumentFragment {\n\t\treturn node && node.nodeType == Node.DOCUMENT_FRAGMENT_NODE;\n\t}\n\n\t/**\n\t * Checks if the node is an instance of the block filler for this DOM converter.\n\t *\n\t * ```ts\n\t * const converter = new DomConverter( viewDocument, { blockFillerMode: 'br' } );\n\t *\n\t * converter.isBlockFiller( BR_FILLER( document ) ); // true\n\t * converter.isBlockFiller( NBSP_FILLER( document ) ); // false\n\t * ```\n\t *\n\t * **Note:**: For the `'nbsp'` mode the method also checks context of a node so it cannot be a detached node.\n\t *\n\t * **Note:** A special case in the `'nbsp'` mode exists where the `<br>` in `<p><br></p>` is treated as a block filler.\n\t *\n\t * @param domNode DOM node to check.\n\t * @returns True if a node is considered a block filler for given mode.\n\t */\n\tpublic isBlockFiller( domNode: DomNode ): boolean {\n\t\tif ( this.blockFillerMode == 'br' ) {\n\t\t\treturn domNode.isEqualNode( BR_FILLER_REF );\n\t\t}\n\n\t\t// Special case for <p><br></p> in which <br> should be treated as filler even when we are not in the 'br' mode. See ckeditor5#5564.\n\t\tif (\n\t\t\t( domNode as DomElement ).tagName === 'BR' &&\n\t\t\thasBlockParent( domNode, this.blockElements ) &&\n\t\t\t( domNode as DomElement ).parentNode!.childNodes.length === 1\n\t\t) {\n\t\t\treturn true;\n\t\t}\n\n\t\t// If not in 'br' mode, try recognizing both marked and regular nbsp block fillers.\n\t\treturn domNode.isEqualNode( MARKED_NBSP_FILLER_REF ) || isNbspBlockFiller( domNode, this.blockElements );\n\t}\n\n\t/**\n\t * Returns `true` if given selection is a backward selection, that is, if it's `focus` is before `anchor`.\n\t *\n\t * @param DOM Selection instance to check.\n\t */\n\tpublic isDomSelectionBackward( selection: DomSelection ): boolean {\n\t\tif ( selection.isCollapsed ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Since it takes multiple lines of code to check whether a \"DOM Position\" is before/after another \"DOM Position\",\n\t\t// we will use the fact that range will collapse if it's end is before it's start.\n\t\tconst range = this._domDocument.createRange();\n\n\t\ttry {\n\t\t\trange.setStart( selection.anchorNode!, selection.anchorOffset );\n\t\t\trange.setEnd( selection.focusNode!, selection.focusOffset );\n\t\t} catch ( e ) {\n\t\t\t// Safari sometimes gives us a selection that makes Range.set{Start,End} throw.\n\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/12375.\n\t\t\treturn false;\n\t\t}\n\n\t\tconst backward = range.collapsed;\n\n\t\trange.detach();\n\n\t\treturn backward;\n\t}\n\n\t/**\n\t * Returns a parent {@link module:engine/view/uielement~UIElement} or {@link module:engine/view/rawelement~RawElement}\n\t * that hosts the provided DOM node. Returns `null` if there is no such parent.\n\t */\n\tpublic getHostViewElement( domNode: DomNode ): ViewUIElement | ViewRawElement | null {\n\t\tconst ancestors = getAncestors( domNode );\n\n\t\t// Remove domNode from the list.\n\t\tancestors.pop();\n\n\t\twhile ( ancestors.length ) {\n\t\t\tconst domNode = ancestors.pop();\n\t\t\tconst viewNode = this._domToViewMapping.get( domNode as DomElement );\n\n\t\t\tif ( viewNode && ( viewNode.is( 'uiElement' ) || viewNode.is( 'rawElement' ) ) ) {\n\t\t\t\treturn viewNode;\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Checks if the given selection's boundaries are at correct places.\n\t *\n\t * The following places are considered as incorrect for selection boundaries:\n\t *\n\t * * before or in the middle of an inline filler sequence,\n\t * * inside a DOM element which represents {@link module:engine/view/uielement~UIElement a view UI element},\n\t * * inside a DOM element which represents {@link module:engine/view/rawelement~RawElement a view raw element}.\n\t *\n\t * @param domSelection The DOM selection object to be checked.\n\t * @returns `true` if the given selection is at a correct place, `false` otherwise.\n\t */\n\tpublic isDomSelectionCorrect( domSelection: DomSelection ): boolean {\n\t\treturn this._isDomSelectionPositionCorrect( domSelection.anchorNode!, domSelection.anchorOffset ) &&\n\t\t\tthis._isDomSelectionPositionCorrect( domSelection.focusNode!, domSelection.focusOffset );\n\t}\n\n\t/**\n\t * Registers a {@link module:engine/view/matcher~MatcherPattern} for view elements whose content should be treated as raw data\n\t * and not processed during the conversion from DOM nodes to view elements.\n\t *\n\t * This is affecting how {@link module:engine/view/domconverter~DomConverter#domToView} and\n\t * {@link module:engine/view/domconverter~DomConverter#domChildrenToView} process DOM nodes.\n\t *\n\t * The raw data can be later accessed by a\n\t * {@link module:engine/view/element~Element#getCustomProperty custom property of a view element} called `\"$rawContent\"`.\n\t *\n\t * @param pattern Pattern matching a view element whose content should\n\t * be treated as raw data.\n\t */\n\tpublic registerRawContentMatcher( pattern: MatcherPattern ): void {\n\t\tthis._rawContentElementMatcher.add( pattern );\n\t}\n\n\t/**\n\t * Registers a {@link module:engine/view/matcher~MatcherPattern} for inline object view elements.\n\t *\n\t * This is affecting how {@link module:engine/view/domconverter~DomConverter#domToView} and\n\t * {@link module:engine/view/domconverter~DomConverter#domChildrenToView} process DOM nodes.\n\t *\n\t * This is an extension of a simple {@link #inlineObjectElements} array of element names.\n\t *\n\t * @param pattern Pattern matching a view element which should be treated as an inline object.\n\t */\n\tpublic registerInlineObjectMatcher( pattern: MatcherPattern ): void {\n\t\tthis._inlineObjectElementMatcher.add( pattern );\n\t}\n\n\t/**\n\t * Returns the block {@link module:engine/view/filler filler} node based on the current {@link #blockFillerMode} setting.\n\t */\n\tprivate _getBlockFiller(): DomNode {\n\t\tswitch ( this.blockFillerMode ) {\n\t\t\tcase 'nbsp':\n\t\t\t\treturn NBSP_FILLER( this._domDocument ); // eslint-disable-line new-cap\n\t\t\tcase 'markedNbsp':\n\t\t\t\treturn MARKED_NBSP_FILLER( this._domDocument ); // eslint-disable-line new-cap\n\t\t\tcase 'br':\n\t\t\t\treturn BR_FILLER( this._domDocument ); // eslint-disable-line new-cap\n\t\t}\n\t}\n\n\t/**\n\t * Checks if the given DOM position is a correct place for selection boundary. See {@link #isDomSelectionCorrect}.\n\t *\n\t * @param domParent Position parent.\n\t * @param offset Position offset.\n\t * @returns `true` if given position is at a correct place for selection boundary, `false` otherwise.\n\t */\n\tprivate _isDomSelectionPositionCorrect( domParent: DomNode, offset: number ): boolean {\n\t\t// If selection is before or in the middle of inline filler string, it is incorrect.\n\t\tif ( isText( domParent ) && startsWithFiller( domParent ) && offset < INLINE_FILLER_LENGTH ) {\n\t\t\t// Selection in a text node, at wrong position (before or in the middle of filler).\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( this.isElement( domParent ) && startsWithFiller( domParent.childNodes[ offset ] ) ) {\n\t\t\t// Selection in an element node, before filler text node.\n\t\t\treturn false;\n\t\t}\n\n\t\tconst viewParent = this.mapDomToView( domParent as DomElement );\n\n\t\t// The position is incorrect when anchored inside a UIElement or a RawElement.\n\t\t// Note: In case of UIElement and RawElement, mapDomToView() returns a parent element for any DOM child\n\t\t// so there's no need to perform any additional checks.\n\t\tif ( viewParent && ( viewParent.is( 'uiElement' ) || viewParent.is( 'rawElement' ) ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Internal generator for {@link #domToView}. Also used by {@link #domChildrenToView}.\n\t * Separates DOM nodes conversion from whitespaces processing.\n\t *\n\t * @param domNode DOM node or document fragment to transform.\n\t * @param inlineNodes An array of recently encountered inline nodes truncated to the block element boundaries.\n\t * Used later to process whitespaces.\n\t */\n\tprivate* _domToView(\n\t\tdomNode: DomNode,\n\t\toptions: {\n\t\t\tbind?: boolean;\n\t\t\twithChildren?: boolean;\n\t\t\tkeepOriginalCase?: boolean;\n\t\t\tskipComments?: boolean;\n\t\t},\n\t\tinlineNodes: Array<ViewNode>\n\t): IterableIterator<ViewNode | ViewDocumentFragment | null> {\n\t\tif ( this.isBlockFiller( domNode ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// When node is inside a UIElement or a RawElement return that parent as it's view representation.\n\t\tconst hostElement = this.getHostViewElement( domNode );\n\n\t\tif ( hostElement ) {\n\t\t\treturn hostElement;\n\t\t}\n\n\t\tif ( isComment( domNode ) && options.skipComments ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( isText( domNode ) ) {\n\t\t\tif ( isInlineFiller( domNode ) ) {\n\t\t\t\treturn null;\n\t\t\t} else {\n\t\t\t\tconst textData = domNode.data;\n\n\t\t\t\tif ( textData === '' ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\tconst textNode = new ViewText( this.document, textData );\n\n\t\t\t\tinlineNodes.push( textNode );\n\n\t\t\t\treturn textNode;\n\t\t\t}\n\t\t} else {\n\t\t\tlet viewElement = this.mapDomToView( domNode as ( DomElement | DomDocumentFragment ) );\n\n\t\t\tif ( viewElement ) {\n\t\t\t\tif ( this._isInlineObjectElement( viewElement ) ) {\n\t\t\t\t\tinlineNodes.push( viewElement );\n\t\t\t\t}\n\n\t\t\t\treturn viewElement;\n\t\t\t}\n\n\t\t\tif ( this.isDocumentFragment( domNode ) ) {\n\t\t\t\t// Create view document fragment.\n\t\t\t\tviewElement = new ViewDocumentFragment( this.document );\n\n\t\t\t\tif ( options.bind ) {\n\t\t\t\t\tthis.bindDocumentFragments( domNode, viewElement );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t// Create view element.\n\t\t\t\tviewElement = this._createViewElement( domNode, options );\n\n\t\t\t\tif ( options.bind ) {\n\t\t\t\t\tthis.bindElements( domNode as DomElement, viewElement );\n\t\t\t\t}\n\n\t\t\t\t// Copy element's attributes.\n\t\t\t\tconst attrs = ( domNode as DomElement ).attributes;\n\n\t\t\t\tif ( attrs ) {\n\t\t\t\t\tfor ( let l = attrs.length, i = 0; i < l; i++ ) {\n\t\t\t\t\t\tviewElement._setAttribute( attrs[ i ].name, attrs[ i ].value );\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Treat this element's content as a raw data if it was registered as such.\n\t\t\t\tif ( this._isViewElementWithRawContent( viewElement, options ) ) {\n\t\t\t\t\tviewElement._setCustomProperty( '$rawContent', ( domNode as DomElement ).innerHTML );\n\n\t\t\t\t\tif ( !this._isBlockViewElement( viewElement ) ) {\n\t\t\t\t\t\tinlineNodes.push( viewElement );\n\t\t\t\t\t}\n\n\t\t\t\t\treturn viewElement;\n\t\t\t\t}\n\n\t\t\t\t// Comment node is also treated as an element with raw data.\n\t\t\t\tif ( isComment( domNode ) ) {\n\t\t\t\t\tviewElement._setCustomProperty( '$rawContent', domNode.data );\n\n\t\t\t\t\treturn viewElement;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Yield the element first so the flow of nested inline nodes is not reversed inside elements.\n\t\t\tyield viewElement;\n\n\t\t\tconst nestedInlineNodes: Array<ViewNode> = [];\n\n\t\t\tif ( options.withChildren !== false ) {\n\t\t\t\tfor ( const child of this.domChildrenToView( domNode as DomElement, options, nestedInlineNodes ) ) {\n\t\t\t\t\tviewElement._appendChild( child );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Check if this is an inline object after processing child nodes so matcher\n\t\t\t// for inline objects can verify if the element is empty.\n\t\t\tif ( this._isInlineObjectElement( viewElement ) ) {\n\t\t\t\tinlineNodes.push( viewElement );\n\t\t\t} else {\n\t\t\t\t// It's an inline element that is not an object (like <b>, <i>) or a block element.\n\t\t\t\tfor ( const inlineNode of nestedInlineNodes ) {\n\t\t\t\t\tinlineNodes.push( inlineNode );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Internal helper that walks the list of inline view nodes already generated from DOM nodes\n\t * and handles whitespaces and NBSPs.\n\t *\n\t * @param domParent The DOM parent of the given inline nodes. This should be a document fragment or\n\t * a block element to whitespace processing start cleaning.\n\t * @param inlineNodes An array of recently encountered inline nodes truncated to the block element boundaries.\n\t */\n\tprivate _processDomInlineNodes(\n\t\tdomParent: DomElement | null,\n\t\tinlineNodes: Array<ViewNode>,\n\t\toptions: { withChildren?: boolean }\n\t): void {\n\t\tif ( !inlineNodes.length ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Process text nodes only after reaching a block or document fragment,\n\t\t// do not alter whitespaces while processing an inline element like <b> or <i>.\n\t\tif ( domParent && !this.isDocumentFragment( domParent ) && !this._isBlockDomElement( domParent ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tlet prevNodeEndsWithSpace = false;\n\n\t\tfor ( let i = 0; i < inlineNodes.length; i++ ) {\n\t\t\tconst node = inlineNodes[ i ];\n\n\t\t\tif ( !node.is( '$text' ) ) {\n\t\t\t\tprevNodeEndsWithSpace = false;\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tlet data: string;\n\t\t\tlet nodeEndsWithSpace: boolean = false;\n\n\t\t\tif ( _hasViewParentOfType( node, this.preElements ) ) {\n\t\t\t\tdata = getDataWithoutFiller( node.data );\n\t\t\t} else {\n\t\t\t\t// Change all consecutive whitespace characters (from the [ \\n\\t\\r] set \n\t\t\t\t// see https://github.com/ckeditor/ckeditor5-engine/issues/822#issuecomment-311670249) to a single space character.\n\t\t\t\t// That's how multiple whitespaces are treated when rendered, so we normalize those whitespaces.\n\t\t\t\t// We're replacing 1+ (and not 2+) to also normalize singular \\n\\t\\r characters (#822).\n\t\t\t\tdata = node.data.replace( /[ \\n\\t\\r]{1,}/g, ' ' );\n\t\t\t\tnodeEndsWithSpace = /[^\\S\\u00A0]/.test( data.charAt( data.length - 1 ) );\n\n\t\t\t\tconst prevNode = i > 0 ? inlineNodes[ i - 1 ] : null;\n\t\t\t\tconst nextNode = i + 1 < inlineNodes.length ? inlineNodes[ i + 1 ] : null;\n\n\t\t\t\tconst shouldLeftTrim = !prevNode || prevNode.is( 'element' ) && prevNode.name == 'br' || prevNodeEndsWithSpace;\n\t\t\t\tconst shouldRightTrim = nextNode ? false : !startsWithFiller( node.data );\n\n\t\t\t\t// Do not try to clear whitespaces if this is flat mapping for the purpose of mutation observer and differ in rendering.\n\t\t\t\tif ( options.withChildren !== false ) {\n\t\t\t\t\t// If the previous dom text node does not exist or it ends by whitespace character, remove space character from the\n\t\t\t\t\t// beginning of this text node. Such space character is treated as a whitespace.\n\t\t\t\t\tif ( shouldLeftTrim ) {\n\t\t\t\t\t\tdata = data.replace( /^ /, '' );\n\t\t\t\t\t}\n\n\t\t\t\t\t// If the next text node does not exist remove space character from the end of this text node.\n\t\t\t\t\tif ( shouldRightTrim ) {\n\t\t\t\t\t\tdata = data.replace( / $/, '' );\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// At the beginning and end of a block element, Firefox inserts normal space + <br> instead of non-breaking space.\n\t\t\t\t// This means that the text node starts/end with normal space instead of non-breaking space.\n\t\t\t\t// This causes a problem because the normal space would be removed in `.replace` calls above. To prevent that,\n\t\t\t\t// the inline filler is removed only after the data is initially processed (by the `.replace` above). See ckeditor5#692.\n\t\t\t\tdata = getDataWithoutFiller( data );\n\n\t\t\t\t// At this point we should have removed all whitespaces from DOM text data.\n\t\t\t\t//\n\t\t\t\t// Now, We will reverse the process that happens in `_processDataFromViewText`.\n\t\t\t\t//\n\t\t\t\t// We have to change &nbsp; chars, that were in DOM text data because of rendering reasons, to spaces.\n\t\t\t\t// First, change all ` \\u00A0` pairs (space + &nbsp;) to two spaces. DOM converter changes two spaces from model/view to\n\t\t\t\t// ` \\u00A0` to ensure proper rendering. Since here we convert back, we recognize those pairs and change them back to ` `.\n\t\t\t\tdata = data.replace( / \\u00A0/g, ' ' );\n\n\t\t\t\tconst isNextNodeInlineObjectElement = nextNode && nextNode.is( 'element' ) && nextNode.name != 'br';\n\t\t\t\tconst isNextNodeStartingWithSpace = nextNode && nextNode.is( '$text' ) && nextNode.data.charAt( 0 ) == ' ';\n\n\t\t\t\t// Then, let's change the last nbsp to a space.\n\t\t\t\tif ( /[ \\u00A0]\\u00A0$/.test( data ) || !nextNode || isNextNodeInlineObjectElement || isNextNodeStartingWithSpace ) {\n\t\t\t\t\tdata = data.replace( /\\u00A0$/, ' ' );\n\t\t\t\t}\n\n\t\t\t\t// Then, change &nbsp; character that is at the beginning of the text node to space character.\n\t\t\t\t// We do that replacement only if this is the first node or the previous node ends on whitespace character.\n\t\t\t\tif ( shouldLeftTrim || prevNode && prevNode.is( 'element' ) && prevNode.name != 'br' ) {\n\t\t\t\t\tdata = data.replace( /^\\u00A0/, ' ' );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// At this point, all whitespaces should be removed and all &nbsp; created for rendering reasons should be\n\t\t\t// changed to normal space. All left &nbsp; are &nbsp; inserted intentionally.\n\n\t\t\tif ( data.length == 0 && node.parent ) {\n\t\t\t\tnode._remove();\n\t\t\t\tinlineNodes.splice( i, 1 );\n\t\t\t\ti--;\n\t\t\t} else {\n\t\t\t\tnode._data = data;\n\t\t\t\tprevNodeEndsWithSpace = nodeEndsWithSpace;\n\t\t\t}\n\t\t}\n\n\t\tinlineNodes.length = 0;\n\t}\n\n\t/**\n\t * Takes text data from a given {@link module:engine/view/text~Text#data} and processes it so\n\t * it is correctly displayed in the DOM.\n\t *\n\t * Following changes are done:\n\t *\n\t * * a space at the beginning is changed to `&nbsp;` if this is the first text node in its container\n\t * element or if a previous text node ends with a space character,\n\t * * space at the end of the text node is changed to `&nbsp;` if there are two spaces at the end of a node or if next node\n\t * starts with a space or if it is the last text node in its container,\n\t * * remaining spaces are replaced to a chain of spaces and `&nbsp;` (e.g. `'x x'` becomes `'x &nbsp; x'`).\n\t *\n\t * Content of {@link #preElements} is not processed.\n\t *\n\t * @param node View text node to process.\n\t * @returns Processed text data.\n\t */\n\tprivate _processDataFromViewText( node: ViewText | ViewTextProxy ): string {\n\t\tlet data = node.data;\n\n\t\t// If any of node ancestors has a name which is in `preElements` array, then currently processed\n\t\t// view text node is (will be) in preformatted element. We should not change whitespaces then.\n\t\tif ( node.getAncestors().some( parent => this.preElements.includes( ( parent as ViewElement ).name ) ) ) {\n\t\t\treturn data;\n\t\t}\n\n\t\t// 1. Replace the first space with a nbsp if the previous node ends with a space or there is no previous node\n\t\t// (container element boundary).\n\t\tif ( data.charAt( 0 ) == ' ' ) {\n\t\t\tconst prevNode = this._getTouchingInlineViewNode( node as ViewText, false );\n\t\t\tconst prevEndsWithSpace = prevNode && prevNode.is( '$textProxy' ) && this._nodeEndsWithSpace( prevNode as ViewTextProxy );\n\n\t\t\tif ( prevEndsWithSpace || !prevNode ) {\n\t\t\t\tdata = '\\u00A0' + data.substr( 1 );\n\t\t\t}\n\t\t}\n\n\t\t// 2. Replace the last space with nbsp if there are two spaces at the end or if the next node starts with space or there is no\n\t\t// next node (container element boundary).\n\t\t//\n\t\t// Keep in mind that Firefox prefers $nbsp; before tag, not inside it:\n\t\t//\n\t\t// Foo <span>&nbsp;bar</span> <-- bad.\n\t\t// Foo&nbsp;<span> bar</span> <-- good.\n\t\t//\n\t\t// More here: https://github.com/ckeditor/ckeditor5-engine/issues/1747.\n\t\tif ( data.charAt( data.length - 1 ) == ' ' ) {\n\t\t\tconst nextNode = this._getTouchingInlineViewNode( node as ViewText, true );\n\t\t\tconst nextStartsWithSpace = nextNode && nextNode.is( '$textProxy' ) && nextNode.data.charAt( 0 ) == ' ';\n\n\t\t\tif ( data.charAt( data.length - 2 ) == ' ' || !nextNode || nextStartsWithSpace ) {\n\t\t\t\tdata = data.substr( 0, data.length - 1 ) + '\\u00A0';\n\t\t\t}\n\t\t}\n\n\t\t// 3. Create space+nbsp pairs.\n\t\treturn data.replace( / {2}/g, ' \\u00A0' );\n\t}\n\n\t/**\n\t * Checks whether given node ends with a space character after changing appropriate space characters to `&nbsp;`s.\n\t *\n\t * @param node Node to check.\n\t * @returns `true` if given `node` ends with space, `false` otherwise.\n\t */\n\tprivate _nodeEndsWithSpace( node: ViewTextProxy ): boolean {\n\t\tif ( node.getAncestors().some( parent => this.preElements.includes( ( parent as ViewElement ).name ) ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst data = this._processDataFromViewText( node );\n\n\t\treturn data.charAt( data.length - 1 ) == ' ';\n\t}\n\n\t/**\n\t * Helper function. For given {@link module:engine/view/text~Text view text node}, it finds previous or next sibling\n\t * that is contained in the same container element. If there is no such sibling, `null` is returned.\n\t *\n\t * @param node Reference node.\n\t * @returns Touching text node, an inline object\n\t * or `null` if there is no next or previous touching text node.\n\t */\n\tprivate _getTouchingInlineViewNode( node: ViewText, getNext: boolean ): ViewElement | ViewTextProxy | null {\n\t\tconst treeWalker = new ViewTreeWalker( {\n\t\t\tstartPosition: getNext ? ViewPosition._createAfter( node ) : ViewPosition._createBefore( node ),\n\t\t\tdirection: getNext ? 'forward' : 'backward'\n\t\t} );\n\n\t\tfor ( const value of treeWalker ) {\n\t\t\t// <br> found it works like a block boundary, so do not scan further.\n\t\t\tif ( value.item.is( 'element', 'br' ) ) {\n\t\t\t\treturn null;\n\t\t\t}\n\t\t\t// Found an inline object (for example an image).\n\t\t\telse if ( this._isInlineObjectElement( value.item ) ) {\n\t\t\t\treturn value.item;\n\t\t\t}\n\t\t\t// ViewContainerElement is found on a way to next ViewText node, so given `node` was first/last\n\t\t\t// text node in its container element.\n\t\t\telse if ( value.item.is( 'containerElement' ) ) {\n\t\t\t\treturn null;\n\t\t\t}\n\t\t\t// Found a text node in the same container element.\n\t\t\telse if ( value.item.is( '$textProxy' ) ) {\n\t\t\t\treturn value.item;\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns `true` if a DOM node belongs to {@link #blockElements}. `false` otherwise.\n\t */\n\tprivate _isBlockDomElement( node: DomNode ): boolean {\n\t\treturn this.isElement( node ) && this.blockElements.includes( node.tagName.toLowerCase() );\n\t}\n\n\t/**\n\t * Returns `true` if a view node belongs to {@link #blockElements}. `false` otherwise.\n\t */\n\tprivate _isBlockViewElement( node: ViewNode ): boolean {\n\t\treturn node.is( 'element' ) && this.blockElements.includes( node.name );\n\t}\n\n\t/**\n\t * Returns `true` if a DOM node belongs to {@link #inlineObjectElements}. `false` otherwise.\n\t */\n\tprivate _isInlineObjectElement( node: ViewNode | ViewTextProxy | ViewDocumentFragment ): node is ViewElement {\n\t\tif ( !node.is( 'element' ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn node.name == 'br' ||\n\t\t\tthis.inlineObjectElements.includes( node.name ) ||\n\t\t\t!!this._inlineObjectElementMatcher.match( node );\n\t}\n\n\t/**\n\t * Creates view element basing on the node type.\n\t *\n\t * @param node DOM node to check.\n\t * @param options Conversion options. See {@link module:engine/view/domconverter~DomConverter#domToView} options parameter.\n\t */\n\tprivate _createViewElement( node: DomNode, options: { keepOriginalCase?: boolean } ): ViewElement {\n\t\tif ( isComment( node ) ) {\n\t\t\treturn new ViewUIElement( this.document, '$comment' );\n\t\t}\n\n\t\tconst viewName = options.keepOriginalCase ? ( node as DomElement ).tagName : ( node as DomElement ).tagName.toLowerCase();\n\n\t\treturn new ViewElement( this.document, viewName );\n\t}\n\n\t/**\n\t * Checks if view element's content should be treated as a raw data.\n\t *\n\t * @param viewElement View element to check.\n\t * @param options Conversion options. See {@link module:engine/view/domconverter~DomConverter#domToView} options parameter.\n\t */\n\tprivate _isViewElementWithRawContent( viewElement: ViewElement | ViewDocumentFragment, options: { withChildren?: boolean } ): boolean {\n\t\treturn options.withChildren !== false && viewElement.is( 'element' ) && !!this._rawContentElementMatcher.match( viewElement );\n\t}\n\n\t/**\n\t * Checks whether a given element name should be renamed in a current rendering mode.\n\t *\n\t * @param elementName The name of view element.\n\t */\n\tprivate _shouldRenameElement( elementName: string ): boolean {\n\t\tconst name = elementName.toLowerCase();\n\n\t\treturn this.renderingMode === 'editing' && this.unsafeElements.includes( name );\n\t}\n\n\t/**\n\t * Return a <span> element with a special attribute holding the name of the original element.\n\t * Optionally, copy all the attributes of the original element if that element is provided.\n\t *\n\t * @param elementName The name of view element.\n\t * @param originalDomElement The original DOM element to copy attributes and content from.\n\t */\n\tprivate _createReplacementDomElement( elementName: string, originalDomElement?: DomElement ): DomElement {\n\t\tconst newDomElement = this._domDocument.createElement( 'span' );\n\n\t\t// Mark the span replacing a script as hidden.\n\t\tnewDomElement.setAttribute( UNSAFE_ELEMENT_REPLACEMENT_ATTRIBUTE, elementName );\n\n\t\tif ( originalDomElement ) {\n\t\t\twhile ( originalDomElement.firstChild ) {\n\t\t\t\tnewDomElement.appendChild( originalDomElement.firstChild );\n\t\t\t}\n\n\t\t\tfor ( const attributeName of originalDomElement.getAttributeNames() ) {\n\t\t\t\tnewDomElement.setAttribute( attributeName, originalDomElement.getAttribute( attributeName )! );\n\t\t\t}\n\t\t}\n\n\t\treturn newDomElement;\n\t}\n}\n\n/**\n * Helper function.\n * Used to check if given native `Element` or `Text` node has parent with tag name from `types` array.\n *\n * @returns`true` if such parent exists or `false` if it does not.\n */\nfunction _hasViewParentOfType( node: ViewNode, types: ReadonlyArray<string> ) {\n\treturn node.getAncestors().some( parent => parent.is( 'element' ) && types.includes( parent.name ) );\n}\n\n/**\n * A helper that executes given callback for each DOM node's ancestor, starting from the given node\n * and ending in document#documentElement.\n *\n * @param callback A callback to be executed for each ancestor.\n */\nfunction forEachDomElementAncestor( element: DomElement, callback: ( node: DomElement ) => void ) {\n\tlet node: DomElement | null = element;\n\n\twhile ( node ) {\n\t\tcallback( node as DomElement );\n\t\tnode = node.parentElement;\n\t}\n}\n\n/**\n * Checks if given node is a nbsp block filler.\n *\n * A &nbsp; is a block filler only if it is a single child of a block element.\n *\n * @param domNode DOM node.\n */\nfunction isNbspBlockFiller( domNode: DomNode, blockElements: ReadonlyArray<string> ): boolean {\n\tconst isNBSP = domNode.isEqualNode( NBSP_FILLER_REF );\n\n\treturn isNBSP && hasBlockParent( domNode, blockElements ) && ( domNode as DomElement ).parentNode!.childNodes.length === 1;\n}\n\n/**\n * Checks if domNode has block parent.\n *\n * @param domNode DOM node.\n */\nfunction hasBlockParent( domNode: DomNode, blockElements: ReadonlyArray<string> ): boolean {\n\tconst parent = domNode.parentNode;\n\n\treturn !!parent && !!( parent as DomElement ).tagName && blockElements.includes( ( parent as DomElement ).tagName.toLowerCase() );\n}\n\n/**\n * Log to console the information about element that was replaced.\n * Check UNSAFE_ELEMENTS for all recognized unsafe elements.\n *\n * @param elementName The name of the view element.\n */\nfunction _logUnsafeElement( elementName: string ): void {\n\tif ( elementName === 'script' ) {\n\t\tlogWarning( 'domconverter-unsafe-script-element-detected' );\n\t}\n\n\tif ( elementName === 'style' ) {\n\t\tlogWarning( 'domconverter-unsafe-style-element-detected' );\n\t}\n}\n\n/**\n * Enum representing the type of the block filler.\n *\n * Possible values:\n *\n * * `br` &ndash; For the `<br data-cke-filler=\"true\">` block filler used in the editing view.\n * * `nbsp` &ndash; For the `&nbsp;` block fillers used in the data.\n * * `markedNbsp` &ndash; For the `&nbsp;` block fillers wrapped in `<span>` elements: `<span data-cke-filler=\"true\">&nbsp;</span>`\n * used in the data.\n */\ntype BlockFillerMode = 'br' | 'nbsp' | 'markedNbsp';\n\n/**\n * While rendering the editor content, the {@link module:engine/view/domconverter~DomConverter} detected a `<script>` element that may\n * disrupt the editing experience. To avoid this, the `<script>` element was replaced with `<span data-ck-unsafe-element=\"script\"></span>`.\n *\n * @error domconverter-unsafe-script-element-detected\n */\n\n/**\n * While rendering the editor content, the {@link module:engine/view/domconverter~DomConverter} detected a `<style>` element that may affect\n * the editing experience. To avoid this, the `<style>` element was replaced with `<span data-ck-unsafe-element=\"style\"></span>`.\n *\n * @error domconverter-unsafe-style-element-detected\n */\n\n/**\n * The {@link module:engine/view/domconverter~DomConverter} detected an interactive attribute in the\n * {@glink framework/architecture/editing-engine#editing-pipeline editing pipeline}. For the best\n * editing experience, the attribute was renamed to `data-ck-unsafe-attribute-[original attribute name]`.\n *\n * If you are the author of the plugin that generated this attribute and you want it to be preserved\n * in the editing pipeline, you can configure this when creating the element\n * using {@link module:engine/view/downcastwriter~DowncastWriter} during the\n * {@glink framework/architecture/editing-engine#conversion modelview conversion}. Methods such as\n * {@link module:engine/view/downcastwriter~DowncastWriter#createContainerElement},\n * {@link module:engine/view/downcastwriter~DowncastWriter#createAttributeElement}, or\n * {@link module:engine/view/downcastwriter~DowncastWriter#createEmptyElement}\n * accept an option that will disable filtering of specific attributes:\n *\n * ```ts\n * const paragraph = writer.createContainerElement( 'p',\n * \t{\n * \t\tclass: 'clickable-paragraph',\n * \t\tonclick: 'alert( \"Paragraph clicked!\" )'\n * \t},\n * \t{\n * \t\t// Make sure the \"onclick\" attribute will pass through.\n * \t\trenderUnsafeAttributes: [ 'onclick' ]\n * \t}\n * );\n * ```\n *\n * @error domconverter-unsafe-attribute-detected\n * @param domElement The DOM element the attribute was set on.\n * @param key The original name of the attribute\n * @param value The value of the original attribute\n */\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/isvalidattributename\n */\n\nimport global from './global';\n\n/**\n * Checks if the given attribute name is valid in terms of HTML.\n *\n * @param name Attribute name.\n */\nexport default function isValidAttributeName( name: string ): boolean {\n\ttry {\n\t\tglobal.document.createAttribute( name );\n\t} catch ( error ) {\n\t\treturn false;\n\t}\n\n\treturn true;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals Node */\n\n/**\n * @module utils/dom/getancestors\n */\n\n/**\n * Returns all ancestors of given DOM node, starting from the top-most (root). Includes the given node itself. If the\n * node is a part of `DocumentFragment` that `DocumentFragment` will be returned. In contrary, if the node is\n * appended to a `Document`, that `Document` will not be returned (algorithms operating on DOM tree care for `Document#documentElement`\n * at most, which will be returned).\n *\n * @param node DOM node.\n * @returns Array of given `node` parents.\n */\nexport default function getAncestors( node: Node ): Array<Node> {\n\tconst nodes: Array<Node> = [];\n\tlet currentNode: Node | null = node;\n\n\t// We are interested in `Node`s `DocumentFragment`s only.\n\twhile ( currentNode && currentNode.nodeType != Node.DOCUMENT_NODE ) {\n\t\tnodes.unshift( currentNode );\n\t\tcurrentNode = currentNode.parentNode;\n\t}\n\n\treturn nodes;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/observer\n */\n\nimport { DomEmitterMixin } from '@ckeditor/ckeditor5-utils';\n\nimport type Document from '../document';\nimport type View from '../view';\n\n/**\n * Abstract base observer class. Observers are classes which listen to DOM events, do the preliminary\n * processing and fire events on the {@link module:engine/view/document~Document} objects.\n * Observers can also add features to the view, for instance by updating its status or marking elements\n * which need a refresh on DOM events.\n */\nexport default abstract class Observer extends DomEmitterMixin() {\n\t/**\n\t * An instance of the view controller.\n\t */\n\tpublic readonly view: View;\n\n\t/**\n\t * A reference to the {@link module:engine/view/document~Document} object.\n\t */\n\tpublic readonly document: Document;\n\n\t/**\n\t * The state of the observer. If it is disabled, no events will be fired.\n\t */\n\tprivate _isEnabled: boolean = false;\n\n\t/**\n\t * Creates an instance of the observer.\n\t */\n\tconstructor( view: View ) {\n\t\tsuper();\n\n\t\tthis.view = view;\n\t\tthis.document = view.document;\n\t}\n\n\t/**\n\t * The state of the observer. If it is disabled, no events will be fired.\n\t */\n\tpublic get isEnabled(): boolean {\n\t\treturn this._isEnabled;\n\t}\n\n\t/**\n\t * Enables the observer. This method is called when the observer is registered to the\n\t * {@link module:engine/view/view~View} and after {@link module:engine/view/view~View#forceRender rendering}\n\t * (all observers are {@link #disable disabled} before rendering).\n\t *\n\t * A typical use case for disabling observers is that mutation observers need to be disabled for the rendering.\n\t * However, a child class may not need to be disabled, so it can implement an empty method.\n\t *\n\t * @see module:engine/view/observer/observer~Observer#disable\n\t */\n\tpublic enable(): void {\n\t\tthis._isEnabled = true;\n\t}\n\n\t/**\n\t * Disables the observer. This method is called before\n\t * {@link module:engine/view/view~View#forceRender rendering} to prevent firing events during rendering.\n\t *\n\t * @see module:engine/view/observer/observer~Observer#enable\n\t */\n\tpublic disable(): void {\n\t\tthis._isEnabled = false;\n\t}\n\n\t/**\n\t * Disables and destroys the observer, among others removes event listeners created by the observer.\n\t */\n\tpublic destroy(): void {\n\t\tthis.disable();\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Checks whether a given DOM event should be ignored (should not be turned into a synthetic view document event).\n\t *\n\t * Currently, an event will be ignored only if its target or any of its ancestors has the `data-cke-ignore-events` attribute.\n\t * This attribute can be used inside the structures generated by\n\t * {@link module:engine/view/downcastwriter~DowncastWriter#createUIElement `DowncastWriter#createUIElement()`} to ignore events\n\t * fired within a UI that should be excluded from CKEditor 5's realms.\n\t *\n\t * @param domTarget The DOM event target to check (usually an element, sometimes a text node and\n\t * potentially sometimes a document, too).\n\t * @returns Whether this event should be ignored by the observer.\n\t */\n\tpublic checkShouldIgnoreEventFromTarget( domTarget: Node | null ): boolean {\n\t\tif ( domTarget && domTarget.nodeType === 3 ) {\n\t\t\tdomTarget = domTarget.parentNode as any;\n\t\t}\n\n\t\tif ( !domTarget || domTarget.nodeType !== 1 ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn ( domTarget as any ).matches( '[data-cke-ignore-events], [data-cke-ignore-events] *' );\n\t}\n\n\t/**\n\t * Starts observing given DOM element.\n\t *\n\t * @param domElement DOM element to observe.\n\t * @param name The name of the related root element.\n\t */\n\tpublic abstract observe( domElement: HTMLElement, name: string ): void;\n\n\t/**\n\t * Stops observing given DOM element.\n\t */\n\tpublic abstract stopObserving( domElement: HTMLElement ): void;\n}\n\n/**\n * The constructor of {@link ~Observer} subclass.\n */\nexport type ObserverConstructor = new ( view: View ) => Observer;\n","import copyObject from './_copyObject.js';\nimport createAssigner from './_createAssigner.js';\nimport keysIn from './keysIn.js';\n\n/**\n * This method is like `_.assign` except that it iterates over own and\n * inherited source properties.\n *\n * **Note:** This method mutates `object`.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @alias extend\n * @category Object\n * @param {Object} object The destination object.\n * @param {...Object} [sources] The source objects.\n * @returns {Object} Returns `object`.\n * @see _.assign\n * @example\n *\n * function Foo() {\n * this.a = 1;\n * }\n *\n * function Bar() {\n * this.c = 3;\n * }\n *\n * Foo.prototype.b = 2;\n * Bar.prototype.d = 4;\n *\n * _.assignIn({ 'a': 0 }, new Foo, new Bar);\n * // => { 'a': 1, 'b': 2, 'c': 3, 'd': 4 }\n */\nvar assignIn = createAssigner(function(object, source) {\n copyObject(source, keysIn(source), object);\n});\n\nexport default assignIn;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/domeventdata\n */\n\nimport { extend } from 'lodash-es';\n\nimport type Document from '../document';\nimport type Element from '../element';\nimport type View from '../view';\n\n/**\n * Information about a DOM event in context of the {@link module:engine/view/document~Document}.\n * It wraps the native event, which usually should not be used as the wrapper contains\n * additional data (like key code for keyboard events).\n *\n * @typeParam TEvent The type of DOM Event that this class represents.\n */\nexport default class DomEventData<TEvent extends Event = Event> {\n\t/**\n\t * Instance of the view controller.\n\t */\n\tpublic readonly view: View;\n\n\t/**\n\t * The instance of the document.\n\t */\n\tpublic readonly document: Document;\n\n\t/**\n\t * The DOM event.\n\t */\n\tpublic readonly domEvent: TEvent;\n\n\t/**\n\t * The DOM target.\n\t */\n\tpublic readonly domTarget: HTMLElement;\n\n\t/**\n\t * @param view The instance of the view controller.\n\t * @param domEvent The DOM event.\n\t * @param additionalData Additional properties that the instance should contain.\n\t */\n\tconstructor( view: View, domEvent: TEvent, additionalData?: object ) {\n\t\tthis.view = view;\n\t\tthis.document = view.document;\n\t\tthis.domEvent = domEvent;\n\t\tthis.domTarget = domEvent.target as any;\n\n\t\textend( this, additionalData );\n\t}\n\n\t/**\n\t * The tree view element representing the target.\n\t */\n\tpublic get target(): Element {\n\t\treturn this.view.domConverter.mapDomToView( this.domTarget ) as Element;\n\t}\n\n\t/**\n\t * Prevents the native's event default action.\n\t */\n\tpublic preventDefault(): void {\n\t\tthis.domEvent.preventDefault();\n\t}\n\n\t/**\n\t * Stops native event propagation.\n\t */\n\tpublic stopPropagation(): void {\n\t\tthis.domEvent.stopPropagation();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/domeventobserver\n */\n\nimport Observer from './observer';\nimport DomEventData from './domeventdata';\n\nimport type { EventInfo } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Base class for DOM event observers. This class handles\n * {@link module:engine/view/observer/observer~Observer#observe adding} listeners to DOM elements,\n * {@link module:engine/view/observer/observer~Observer#disable disabling} and\n * {@link module:engine/view/observer/observer~Observer#enable re-enabling} events.\n * Child class needs to define\n * {@link module:engine/view/observer/domeventobserver~DomEventObserver#domEventType DOM event type} and\n * {@link module:engine/view/observer/domeventobserver~DomEventObserver#onDomEvent callback}.\n *\n * For instance:\n *\n * ```ts\n * class ClickObserver extends DomEventObserver<'click'> {\n * \t// It can also be defined as a normal property in the constructor.\n * \tget domEventType(): 'click' {\n * \t\treturn 'click';\n * \t}\n *\n * \tonDomEvent( domEvent: MouseEvent ): void {\n * \t\tthis.fire( 'click', domEvent );\n * \t}\n * }\n * ```\n *\n * @typeParam EventType DOM Event type name or an union of those.\n * @typeParam AdditionalData Additional data passed along with the event.\n */\n\nexport default abstract class DomEventObserver<\n\tEventType extends keyof HTMLElementEventMap,\n\tAdditionalData extends object = object\n> extends Observer {\n\t/**\n\t * Type of the DOM event the observer should listen to. Array of types can be defined\n\t * if the observer should listen to multiple DOM events.\n\t */\n\tpublic abstract get domEventType(): EventType | ReadonlyArray<EventType>;\n\n\t/**\n\t * If set to `true` DOM events will be listened on the capturing phase.\n\t * Default value is `false`.\n\t */\n\tpublic useCapture: boolean = false;\n\n\t/**\n\t * Callback which should be called when the DOM event occurred. Note that the callback will not be called if\n\t * observer {@link #isEnabled is not enabled}.\n\t *\n\t * @see #domEventType\n\t */\n\tpublic abstract onDomEvent( event: HTMLElementEventMap[ EventType ] ): void;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override observe( domElement: HTMLElement ): void {\n\t\tconst types = typeof this.domEventType == 'string' ? [ this.domEventType ] : this.domEventType;\n\n\t\ttypes.forEach( type => {\n\t\t\tthis.listenTo( domElement, type, ( eventInfo, domEvent ) => {\n\t\t\t\tif ( this.isEnabled && !this.checkShouldIgnoreEventFromTarget( domEvent.target as any ) ) {\n\t\t\t\t\tthis.onDomEvent( domEvent );\n\t\t\t\t}\n\t\t\t}, { useCapture: this.useCapture } );\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override stopObserving( domElement: HTMLElement ): void {\n\t\tthis.stopListening( domElement );\n\t}\n\n\t/**\n\t * Calls `Document#fire()` if observer {@link #isEnabled is enabled}.\n\t *\n\t * @see module:utils/emittermixin~Emitter#fire\n\t * @param eventType The event type (name).\n\t * @param domEvent The DOM event.\n\t * @param additionalData The additional data which should extend the\n\t * {@link module:engine/view/observer/domeventdata~DomEventData event data} object.\n\t */\n\tpublic override fire( eventType: string | EventInfo, domEvent: Event, additionalData?: AdditionalData ): void {\n\t\tif ( this.isEnabled ) {\n\t\t\tthis.document.fire( eventType, new DomEventData( this.view, domEvent, additionalData ) );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/keyobserver\n */\n\nimport DomEventObserver from './domeventobserver';\nimport type DomEventData from './domeventdata';\nimport { getCode, type KeystrokeInfo } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Observer for events connected with pressing keyboard keys.\n *\n * Note that this observer is attached by the {@link module:engine/view/view~View} and is available by default.\n */\nexport default class KeyObserver extends DomEventObserver<'keydown' | 'keyup', KeystrokeInfo & { keystroke: number }> {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic readonly domEventType = [ 'keydown', 'keyup' ] as const;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic onDomEvent( domEvt: KeyboardEvent ): void {\n\t\tconst data = {\n\t\t\tkeyCode: domEvt.keyCode,\n\n\t\t\taltKey: domEvt.altKey,\n\t\t\tctrlKey: domEvt.ctrlKey,\n\t\t\tshiftKey: domEvt.shiftKey,\n\t\t\tmetaKey: domEvt.metaKey,\n\n\t\t\tget keystroke() {\n\t\t\t\treturn getCode( this );\n\t\t\t}\n\t\t};\n\n\t\tthis.fire( domEvt.type, domEvt, data );\n\t}\n}\n\n/**\n * Fired when a key has been pressed.\n *\n * Introduced by {@link module:engine/view/observer/keyobserver~KeyObserver}.\n *\n * Note that because {@link module:engine/view/observer/keyobserver~KeyObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/keyobserver~KeyObserver\n * @eventName module:engine/view/document~Document#keydown\n */\nexport type ViewDocumentKeyDownEvent = {\n\tname: 'keydown';\n\targs: [ data: KeyEventData ];\n};\n\n/**\n * Fired when a key has been released.\n *\n * Introduced by {@link module:engine/view/observer/keyobserver~KeyObserver}.\n *\n * Note that because {@link module:engine/view/observer/keyobserver~KeyObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/keyobserver~KeyObserver\n * @eventName module:engine/view/document~Document#keyup\n */\nexport type ViewDocumentKeyUpEvent = {\n\tname: 'keyup';\n\targs: [ data: KeyEventData ];\n};\n\n/**\n * The value of both events - {@link ~ViewDocumentKeyDownEvent} and {@link ~ViewDocumentKeyUpEvent}.\n */\nexport interface KeyEventData extends DomEventData<KeyboardEvent>, KeystrokeInfo {\n\n\t/**\n\t * Code of the whole keystroke. See {@link module:utils/keyboard~getCode}.\n\t */\n\tkeystroke: number;\n}\n","import root from './_root.js';\n\n/**\n * Gets the timestamp of the number of milliseconds that have elapsed since\n * the Unix epoch (1 January 1970 00:00:00 UTC).\n *\n * @static\n * @memberOf _\n * @since 2.4.0\n * @category Date\n * @returns {number} Returns the timestamp.\n * @example\n *\n * _.defer(function(stamp) {\n * console.log(_.now() - stamp);\n * }, _.now());\n * // => Logs the number of milliseconds it took for the deferred invocation.\n */\nvar now = function() {\n return root.Date.now();\n};\n\nexport default now;\n","/** Used to match a single whitespace character. */\nvar reWhitespace = /\\s/;\n\n/**\n * Used by `_.trim` and `_.trimEnd` to get the index of the last non-whitespace\n * character of `string`.\n *\n * @private\n * @param {string} string The string to inspect.\n * @returns {number} Returns the index of the last non-whitespace character.\n */\nfunction trimmedEndIndex(string) {\n var index = string.length;\n\n while (index-- && reWhitespace.test(string.charAt(index))) {}\n return index;\n}\n\nexport default trimmedEndIndex;\n","import trimmedEndIndex from './_trimmedEndIndex.js';\n\n/** Used to match leading whitespace. */\nvar reTrimStart = /^\\s+/;\n\n/**\n * The base implementation of `_.trim`.\n *\n * @private\n * @param {string} string The string to trim.\n * @returns {string} Returns the trimmed string.\n */\nfunction baseTrim(string) {\n return string\n ? string.slice(0, trimmedEndIndex(string) + 1).replace(reTrimStart, '')\n : string;\n}\n\nexport default baseTrim;\n","import baseTrim from './_baseTrim.js';\nimport isObject from './isObject.js';\nimport isSymbol from './isSymbol.js';\n\n/** Used as references for various `Number` constants. */\nvar NAN = 0 / 0;\n\n/** Used to detect bad signed hexadecimal string values. */\nvar reIsBadHex = /^[-+]0x[0-9a-f]+$/i;\n\n/** Used to detect binary string values. */\nvar reIsBinary = /^0b[01]+$/i;\n\n/** Used to detect octal string values. */\nvar reIsOctal = /^0o[0-7]+$/i;\n\n/** Built-in method references without a dependency on `root`. */\nvar freeParseInt = parseInt;\n\n/**\n * Converts `value` to a number.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to process.\n * @returns {number} Returns the number.\n * @example\n *\n * _.toNumber(3.2);\n * // => 3.2\n *\n * _.toNumber(Number.MIN_VALUE);\n * // => 5e-324\n *\n * _.toNumber(Infinity);\n * // => Infinity\n *\n * _.toNumber('3.2');\n * // => 3.2\n */\nfunction toNumber(value) {\n if (typeof value == 'number') {\n return value;\n }\n if (isSymbol(value)) {\n return NAN;\n }\n if (isObject(value)) {\n var other = typeof value.valueOf == 'function' ? value.valueOf() : value;\n value = isObject(other) ? (other + '') : other;\n }\n if (typeof value != 'string') {\n return value === 0 ? value : +value;\n }\n value = baseTrim(value);\n var isBinary = reIsBinary.test(value);\n return (isBinary || reIsOctal.test(value))\n ? freeParseInt(value.slice(2), isBinary ? 2 : 8)\n : (reIsBadHex.test(value) ? NAN : +value);\n}\n\nexport default toNumber;\n","import isObject from './isObject.js';\nimport now from './now.js';\nimport toNumber from './toNumber.js';\n\n/** Error message constants. */\nvar FUNC_ERROR_TEXT = 'Expected a function';\n\n/* Built-in method references for those with the same name as other `lodash` methods. */\nvar nativeMax = Math.max,\n nativeMin = Math.min;\n\n/**\n * Creates a debounced function that delays invoking `func` until after `wait`\n * milliseconds have elapsed since the last time the debounced function was\n * invoked. The debounced function comes with a `cancel` method to cancel\n * delayed `func` invocations and a `flush` method to immediately invoke them.\n * Provide `options` to indicate whether `func` should be invoked on the\n * leading and/or trailing edge of the `wait` timeout. The `func` is invoked\n * with the last arguments provided to the debounced function. Subsequent\n * calls to the debounced function return the result of the last `func`\n * invocation.\n *\n * **Note:** If `leading` and `trailing` options are `true`, `func` is\n * invoked on the trailing edge of the timeout only if the debounced function\n * is invoked more than once during the `wait` timeout.\n *\n * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred\n * until to the next tick, similar to `setTimeout` with a timeout of `0`.\n *\n * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)\n * for details over the differences between `_.debounce` and `_.throttle`.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Function\n * @param {Function} func The function to debounce.\n * @param {number} [wait=0] The number of milliseconds to delay.\n * @param {Object} [options={}] The options object.\n * @param {boolean} [options.leading=false]\n * Specify invoking on the leading edge of the timeout.\n * @param {number} [options.maxWait]\n * The maximum time `func` is allowed to be delayed before it's invoked.\n * @param {boolean} [options.trailing=true]\n * Specify invoking on the trailing edge of the timeout.\n * @returns {Function} Returns the new debounced function.\n * @example\n *\n * // Avoid costly calculations while the window size is in flux.\n * jQuery(window).on('resize', _.debounce(calculateLayout, 150));\n *\n * // Invoke `sendMail` when clicked, debouncing subsequent calls.\n * jQuery(element).on('click', _.debounce(sendMail, 300, {\n * 'leading': true,\n * 'trailing': false\n * }));\n *\n * // Ensure `batchLog` is invoked once after 1 second of debounced calls.\n * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 });\n * var source = new EventSource('/stream');\n * jQuery(source).on('message', debounced);\n *\n * // Cancel the trailing debounced invocation.\n * jQuery(window).on('popstate', debounced.cancel);\n */\nfunction debounce(func, wait, options) {\n var lastArgs,\n lastThis,\n maxWait,\n result,\n timerId,\n lastCallTime,\n lastInvokeTime = 0,\n leading = false,\n maxing = false,\n trailing = true;\n\n if (typeof func != 'function') {\n throw new TypeError(FUNC_ERROR_TEXT);\n }\n wait = toNumber(wait) || 0;\n if (isObject(options)) {\n leading = !!options.leading;\n maxing = 'maxWait' in options;\n maxWait = maxing ? nativeMax(toNumber(options.maxWait) || 0, wait) : maxWait;\n trailing = 'trailing' in options ? !!options.trailing : trailing;\n }\n\n function invokeFunc(time) {\n var args = lastArgs,\n thisArg = lastThis;\n\n lastArgs = lastThis = undefined;\n lastInvokeTime = time;\n result = func.apply(thisArg, args);\n return result;\n }\n\n function leadingEdge(time) {\n // Reset any `maxWait` timer.\n lastInvokeTime = time;\n // Start the timer for the trailing edge.\n timerId = setTimeout(timerExpired, wait);\n // Invoke the leading edge.\n return leading ? invokeFunc(time) : result;\n }\n\n function remainingWait(time) {\n var timeSinceLastCall = time - lastCallTime,\n timeSinceLastInvoke = time - lastInvokeTime,\n timeWaiting = wait - timeSinceLastCall;\n\n return maxing\n ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke)\n : timeWaiting;\n }\n\n function shouldInvoke(time) {\n var timeSinceLastCall = time - lastCallTime,\n timeSinceLastInvoke = time - lastInvokeTime;\n\n // Either this is the first call, activity has stopped and we're at the\n // trailing edge, the system time has gone backwards and we're treating\n // it as the trailing edge, or we've hit the `maxWait` limit.\n return (lastCallTime === undefined || (timeSinceLastCall >= wait) ||\n (timeSinceLastCall < 0) || (maxing && timeSinceLastInvoke >= maxWait));\n }\n\n function timerExpired() {\n var time = now();\n if (shouldInvoke(time)) {\n return trailingEdge(time);\n }\n // Restart the timer.\n timerId = setTimeout(timerExpired, remainingWait(time));\n }\n\n function trailingEdge(time) {\n timerId = undefined;\n\n // Only invoke if we have `lastArgs` which means `func` has been\n // debounced at least once.\n if (trailing && lastArgs) {\n return invokeFunc(time);\n }\n lastArgs = lastThis = undefined;\n return result;\n }\n\n function cancel() {\n if (timerId !== undefined) {\n clearTimeout(timerId);\n }\n lastInvokeTime = 0;\n lastArgs = lastCallTime = lastThis = timerId = undefined;\n }\n\n function flush() {\n return timerId === undefined ? result : trailingEdge(now());\n }\n\n function debounced() {\n var time = now(),\n isInvoking = shouldInvoke(time);\n\n lastArgs = arguments;\n lastThis = this;\n lastCallTime = time;\n\n if (isInvoking) {\n if (timerId === undefined) {\n return leadingEdge(lastCallTime);\n }\n if (maxing) {\n // Handle invocations in a tight loop.\n clearTimeout(timerId);\n timerId = setTimeout(timerExpired, wait);\n return invokeFunc(lastCallTime);\n }\n }\n if (timerId === undefined) {\n timerId = setTimeout(timerExpired, wait);\n }\n return result;\n }\n debounced.cancel = cancel;\n debounced.flush = flush;\n return debounced;\n}\n\nexport default debounce;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/fakeselectionobserver\n */\n\nimport Observer from './observer';\nimport type { ViewDocumentArrowKeyEvent } from './arrowkeysobserver';\nimport ViewSelection from '../selection';\nimport type View from '../view';\nimport type {\n\tViewDocumentSelectionChangeEvent,\n\tViewDocumentSelectionChangeDoneEvent,\n\tViewDocumentSelectionEventData\n} from './selectionobserver';\nimport { keyCodes } from '@ckeditor/ckeditor5-utils';\nimport { debounce, type DebouncedFunc } from 'lodash-es';\n\n/**\n * Fake selection observer class. If view selection is fake it is placed in dummy DOM container. This observer listens\n * on {@link module:engine/view/document~Document#event:keydown keydown} events and handles moving fake view selection to the correct place\n * if arrow keys are pressed.\n * Fires {@link module:engine/view/document~Document#event:selectionChange selectionChange event} simulating natural behaviour of\n * {@link module:engine/view/observer/selectionobserver~SelectionObserver SelectionObserver}.\n */\nexport default class FakeSelectionObserver extends Observer {\n\t/**\n\t * Fires debounced event `selectionChangeDone`. It uses `lodash#debounce` method to delay function call.\n\t */\n\tprivate readonly _fireSelectionChangeDoneDebounced: DebouncedFunc<( data: ViewDocumentSelectionEventData ) => void>;\n\n\t/**\n\t * Creates new FakeSelectionObserver instance.\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tthis._fireSelectionChangeDoneDebounced = debounce( data => {\n\t\t\tthis.document.fire<ViewDocumentSelectionChangeDoneEvent>( 'selectionChangeDone', data );\n\t\t}, 200 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override observe(): void {\n\t\tconst document = this.document;\n\n\t\tdocument.on<ViewDocumentArrowKeyEvent>( 'arrowKey', ( eventInfo, data ) => {\n\t\t\tconst selection = document.selection;\n\n\t\t\tif ( selection.isFake && this.isEnabled ) {\n\t\t\t\t// Prevents default key down handling - no selection change will occur.\n\t\t\t\tdata.preventDefault();\n\t\t\t}\n\t\t}, { context: '$capture' } );\n\n\t\tdocument.on<ViewDocumentArrowKeyEvent>( 'arrowKey', ( eventInfo, data ) => {\n\t\t\tconst selection = document.selection;\n\n\t\t\tif ( selection.isFake && this.isEnabled ) {\n\t\t\t\tthis._handleSelectionMove( data.keyCode );\n\t\t\t}\n\t\t}, { priority: 'lowest' } );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override stopObserving(): void {}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis._fireSelectionChangeDoneDebounced.cancel();\n\t}\n\n\t/**\n\t * Handles collapsing view selection according to given key code. If left or up key is provided - new selection will be\n\t * collapsed to left. If right or down key is pressed - new selection will be collapsed to right.\n\t *\n\t * This method fires {@link module:engine/view/document~Document#event:selectionChange} and\n\t * {@link module:engine/view/document~Document#event:selectionChangeDone} events imitating behaviour of\n\t * {@link module:engine/view/observer/selectionobserver~SelectionObserver}.\n\t */\n\tprivate _handleSelectionMove( keyCode: number ): void {\n\t\tconst selection = this.document.selection;\n\t\tconst newSelection = new ViewSelection( selection.getRanges(), { backward: selection.isBackward, fake: false } );\n\n\t\t// Left or up arrow pressed - move selection to start.\n\t\tif ( keyCode == keyCodes.arrowleft || keyCode == keyCodes.arrowup ) {\n\t\t\tnewSelection.setTo( newSelection.getFirstPosition() );\n\t\t}\n\n\t\t// Right or down arrow pressed - move selection to end.\n\t\tif ( keyCode == keyCodes.arrowright || keyCode == keyCodes.arrowdown ) {\n\t\t\tnewSelection.setTo( newSelection.getLastPosition() );\n\t\t}\n\n\t\tconst data = {\n\t\t\toldSelection: selection,\n\t\t\tnewSelection,\n\t\t\tdomSelection: null\n\t\t};\n\n\t\t// Fire dummy selection change event.\n\t\tthis.document.fire<ViewDocumentSelectionChangeEvent>( 'selectionChange', data );\n\n\t\t// Call` #_fireSelectionChangeDoneDebounced` every time when `selectionChange` event is fired.\n\t\t// This function is debounced what means that `selectionChangeDone` event will be fired only when\n\t\t// defined int the function time will elapse since the last time the function was called.\n\t\t// So `selectionChangeDone` will be fired when selection will stop changing.\n\t\tthis._fireSelectionChangeDoneDebounced( data );\n\t}\n}\n","/** Used to stand-in for `undefined` hash values. */\nvar HASH_UNDEFINED = '__lodash_hash_undefined__';\n\n/**\n * Adds `value` to the array cache.\n *\n * @private\n * @name add\n * @memberOf SetCache\n * @alias push\n * @param {*} value The value to cache.\n * @returns {Object} Returns the cache instance.\n */\nfunction setCacheAdd(value) {\n this.__data__.set(value, HASH_UNDEFINED);\n return this;\n}\n\nexport default setCacheAdd;\n","/**\n * Checks if `value` is in the array cache.\n *\n * @private\n * @name has\n * @memberOf SetCache\n * @param {*} value The value to search for.\n * @returns {number} Returns `true` if `value` is found, else `false`.\n */\nfunction setCacheHas(value) {\n return this.__data__.has(value);\n}\n\nexport default setCacheHas;\n","import MapCache from './_MapCache.js';\nimport setCacheAdd from './_setCacheAdd.js';\nimport setCacheHas from './_setCacheHas.js';\n\n/**\n *\n * Creates an array cache object to store unique values.\n *\n * @private\n * @constructor\n * @param {Array} [values] The values to cache.\n */\nfunction SetCache(values) {\n var index = -1,\n length = values == null ? 0 : values.length;\n\n this.__data__ = new MapCache;\n while (++index < length) {\n this.add(values[index]);\n }\n}\n\n// Add methods to `SetCache`.\nSetCache.prototype.add = SetCache.prototype.push = setCacheAdd;\nSetCache.prototype.has = setCacheHas;\n\nexport default SetCache;\n","/**\n * A specialized version of `_.some` for arrays without support for iteratee\n * shorthands.\n *\n * @private\n * @param {Array} [array] The array to iterate over.\n * @param {Function} predicate The function invoked per iteration.\n * @returns {boolean} Returns `true` if any element passes the predicate check,\n * else `false`.\n */\nfunction arraySome(array, predicate) {\n var index = -1,\n length = array == null ? 0 : array.length;\n\n while (++index < length) {\n if (predicate(array[index], index, array)) {\n return true;\n }\n }\n return false;\n}\n\nexport default arraySome;\n","/**\n * Checks if a `cache` value for `key` exists.\n *\n * @private\n * @param {Object} cache The cache to query.\n * @param {string} key The key of the entry to check.\n * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`.\n */\nfunction cacheHas(cache, key) {\n return cache.has(key);\n}\n\nexport default cacheHas;\n","import SetCache from './_SetCache.js';\nimport arraySome from './_arraySome.js';\nimport cacheHas from './_cacheHas.js';\n\n/** Used to compose bitmasks for value comparisons. */\nvar COMPARE_PARTIAL_FLAG = 1,\n COMPARE_UNORDERED_FLAG = 2;\n\n/**\n * A specialized version of `baseIsEqualDeep` for arrays with support for\n * partial deep comparisons.\n *\n * @private\n * @param {Array} array The array to compare.\n * @param {Array} other The other array to compare.\n * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.\n * @param {Function} customizer The function to customize comparisons.\n * @param {Function} equalFunc The function to determine equivalents of values.\n * @param {Object} stack Tracks traversed `array` and `other` objects.\n * @returns {boolean} Returns `true` if the arrays are equivalent, else `false`.\n */\nfunction equalArrays(array, other, bitmask, customizer, equalFunc, stack) {\n var isPartial = bitmask & COMPARE_PARTIAL_FLAG,\n arrLength = array.length,\n othLength = other.length;\n\n if (arrLength != othLength && !(isPartial && othLength > arrLength)) {\n return false;\n }\n // Check that cyclic values are equal.\n var arrStacked = stack.get(array);\n var othStacked = stack.get(other);\n if (arrStacked && othStacked) {\n return arrStacked == other && othStacked == array;\n }\n var index = -1,\n result = true,\n seen = (bitmask & COMPARE_UNORDERED_FLAG) ? new SetCache : undefined;\n\n stack.set(array, other);\n stack.set(other, array);\n\n // Ignore non-index properties.\n while (++index < arrLength) {\n var arrValue = array[index],\n othValue = other[index];\n\n if (customizer) {\n var compared = isPartial\n ? customizer(othValue, arrValue, index, other, array, stack)\n : customizer(arrValue, othValue, index, array, other, stack);\n }\n if (compared !== undefined) {\n if (compared) {\n continue;\n }\n result = false;\n break;\n }\n // Recursively compare arrays (susceptible to call stack limits).\n if (seen) {\n if (!arraySome(other, function(othValue, othIndex) {\n if (!cacheHas(seen, othIndex) &&\n (arrValue === othValue || equalFunc(arrValue, othValue, bitmask, customizer, stack))) {\n return seen.push(othIndex);\n }\n })) {\n result = false;\n break;\n }\n } else if (!(\n arrValue === othValue ||\n equalFunc(arrValue, othValue, bitmask, customizer, stack)\n )) {\n result = false;\n break;\n }\n }\n stack['delete'](array);\n stack['delete'](other);\n return result;\n}\n\nexport default equalArrays;\n","/**\n * Converts `map` to its key-value pairs.\n *\n * @private\n * @param {Object} map The map to convert.\n * @returns {Array} Returns the key-value pairs.\n */\nfunction mapToArray(map) {\n var index = -1,\n result = Array(map.size);\n\n map.forEach(function(value, key) {\n result[++index] = [key, value];\n });\n return result;\n}\n\nexport default mapToArray;\n","/**\n * Converts `set` to an array of its values.\n *\n * @private\n * @param {Object} set The set to convert.\n * @returns {Array} Returns the values.\n */\nfunction setToArray(set) {\n var index = -1,\n result = Array(set.size);\n\n set.forEach(function(value) {\n result[++index] = value;\n });\n return result;\n}\n\nexport default setToArray;\n","import Symbol from './_Symbol.js';\nimport Uint8Array from './_Uint8Array.js';\nimport eq from './eq.js';\nimport equalArrays from './_equalArrays.js';\nimport mapToArray from './_mapToArray.js';\nimport setToArray from './_setToArray.js';\n\n/** Used to compose bitmasks for value comparisons. */\nvar COMPARE_PARTIAL_FLAG = 1,\n COMPARE_UNORDERED_FLAG = 2;\n\n/** `Object#toString` result references. */\nvar boolTag = '[object Boolean]',\n dateTag = '[object Date]',\n errorTag = '[object Error]',\n mapTag = '[object Map]',\n numberTag = '[object Number]',\n regexpTag = '[object RegExp]',\n setTag = '[object Set]',\n stringTag = '[object String]',\n symbolTag = '[object Symbol]';\n\nvar arrayBufferTag = '[object ArrayBuffer]',\n dataViewTag = '[object DataView]';\n\n/** Used to convert symbols to primitives and strings. */\nvar symbolProto = Symbol ? Symbol.prototype : undefined,\n symbolValueOf = symbolProto ? symbolProto.valueOf : undefined;\n\n/**\n * A specialized version of `baseIsEqualDeep` for comparing objects of\n * the same `toStringTag`.\n *\n * **Note:** This function only supports comparing values with tags of\n * `Boolean`, `Date`, `Error`, `Number`, `RegExp`, or `String`.\n *\n * @private\n * @param {Object} object The object to compare.\n * @param {Object} other The other object to compare.\n * @param {string} tag The `toStringTag` of the objects to compare.\n * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.\n * @param {Function} customizer The function to customize comparisons.\n * @param {Function} equalFunc The function to determine equivalents of values.\n * @param {Object} stack Tracks traversed `object` and `other` objects.\n * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.\n */\nfunction equalByTag(object, other, tag, bitmask, customizer, equalFunc, stack) {\n switch (tag) {\n case dataViewTag:\n if ((object.byteLength != other.byteLength) ||\n (object.byteOffset != other.byteOffset)) {\n return false;\n }\n object = object.buffer;\n other = other.buffer;\n\n case arrayBufferTag:\n if ((object.byteLength != other.byteLength) ||\n !equalFunc(new Uint8Array(object), new Uint8Array(other))) {\n return false;\n }\n return true;\n\n case boolTag:\n case dateTag:\n case numberTag:\n // Coerce booleans to `1` or `0` and dates to milliseconds.\n // Invalid dates are coerced to `NaN`.\n return eq(+object, +other);\n\n case errorTag:\n return object.name == other.name && object.message == other.message;\n\n case regexpTag:\n case stringTag:\n // Coerce regexes to strings and treat strings, primitives and objects,\n // as equal. See http://www.ecma-international.org/ecma-262/7.0/#sec-regexp.prototype.tostring\n // for more details.\n return object == (other + '');\n\n case mapTag:\n var convert = mapToArray;\n\n case setTag:\n var isPartial = bitmask & COMPARE_PARTIAL_FLAG;\n convert || (convert = setToArray);\n\n if (object.size != other.size && !isPartial) {\n return false;\n }\n // Assume cyclic values are equal.\n var stacked = stack.get(object);\n if (stacked) {\n return stacked == other;\n }\n bitmask |= COMPARE_UNORDERED_FLAG;\n\n // Recursively compare objects (susceptible to call stack limits).\n stack.set(object, other);\n var result = equalArrays(convert(object), convert(other), bitmask, customizer, equalFunc, stack);\n stack['delete'](object);\n return result;\n\n case symbolTag:\n if (symbolValueOf) {\n return symbolValueOf.call(object) == symbolValueOf.call(other);\n }\n }\n return false;\n}\n\nexport default equalByTag;\n","import getAllKeys from './_getAllKeys.js';\n\n/** Used to compose bitmasks for value comparisons. */\nvar COMPARE_PARTIAL_FLAG = 1;\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * A specialized version of `baseIsEqualDeep` for objects with support for\n * partial deep comparisons.\n *\n * @private\n * @param {Object} object The object to compare.\n * @param {Object} other The other object to compare.\n * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.\n * @param {Function} customizer The function to customize comparisons.\n * @param {Function} equalFunc The function to determine equivalents of values.\n * @param {Object} stack Tracks traversed `object` and `other` objects.\n * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.\n */\nfunction equalObjects(object, other, bitmask, customizer, equalFunc, stack) {\n var isPartial = bitmask & COMPARE_PARTIAL_FLAG,\n objProps = getAllKeys(object),\n objLength = objProps.length,\n othProps = getAllKeys(other),\n othLength = othProps.length;\n\n if (objLength != othLength && !isPartial) {\n return false;\n }\n var index = objLength;\n while (index--) {\n var key = objProps[index];\n if (!(isPartial ? key in other : hasOwnProperty.call(other, key))) {\n return false;\n }\n }\n // Check that cyclic values are equal.\n var objStacked = stack.get(object);\n var othStacked = stack.get(other);\n if (objStacked && othStacked) {\n return objStacked == other && othStacked == object;\n }\n var result = true;\n stack.set(object, other);\n stack.set(other, object);\n\n var skipCtor = isPartial;\n while (++index < objLength) {\n key = objProps[index];\n var objValue = object[key],\n othValue = other[key];\n\n if (customizer) {\n var compared = isPartial\n ? customizer(othValue, objValue, key, other, object, stack)\n : customizer(objValue, othValue, key, object, other, stack);\n }\n // Recursively compare objects (susceptible to call stack limits).\n if (!(compared === undefined\n ? (objValue === othValue || equalFunc(objValue, othValue, bitmask, customizer, stack))\n : compared\n )) {\n result = false;\n break;\n }\n skipCtor || (skipCtor = key == 'constructor');\n }\n if (result && !skipCtor) {\n var objCtor = object.constructor,\n othCtor = other.constructor;\n\n // Non `Object` object instances with different constructors are not equal.\n if (objCtor != othCtor &&\n ('constructor' in object && 'constructor' in other) &&\n !(typeof objCtor == 'function' && objCtor instanceof objCtor &&\n typeof othCtor == 'function' && othCtor instanceof othCtor)) {\n result = false;\n }\n }\n stack['delete'](object);\n stack['delete'](other);\n return result;\n}\n\nexport default equalObjects;\n","import Stack from './_Stack.js';\nimport equalArrays from './_equalArrays.js';\nimport equalByTag from './_equalByTag.js';\nimport equalObjects from './_equalObjects.js';\nimport getTag from './_getTag.js';\nimport isArray from './isArray.js';\nimport isBuffer from './isBuffer.js';\nimport isTypedArray from './isTypedArray.js';\n\n/** Used to compose bitmasks for value comparisons. */\nvar COMPARE_PARTIAL_FLAG = 1;\n\n/** `Object#toString` result references. */\nvar argsTag = '[object Arguments]',\n arrayTag = '[object Array]',\n objectTag = '[object Object]';\n\n/** Used for built-in method references. */\nvar objectProto = Object.prototype;\n\n/** Used to check objects for own properties. */\nvar hasOwnProperty = objectProto.hasOwnProperty;\n\n/**\n * A specialized version of `baseIsEqual` for arrays and objects which performs\n * deep comparisons and tracks traversed objects enabling objects with circular\n * references to be compared.\n *\n * @private\n * @param {Object} object The object to compare.\n * @param {Object} other The other object to compare.\n * @param {number} bitmask The bitmask flags. See `baseIsEqual` for more details.\n * @param {Function} customizer The function to customize comparisons.\n * @param {Function} equalFunc The function to determine equivalents of values.\n * @param {Object} [stack] Tracks traversed `object` and `other` objects.\n * @returns {boolean} Returns `true` if the objects are equivalent, else `false`.\n */\nfunction baseIsEqualDeep(object, other, bitmask, customizer, equalFunc, stack) {\n var objIsArr = isArray(object),\n othIsArr = isArray(other),\n objTag = objIsArr ? arrayTag : getTag(object),\n othTag = othIsArr ? arrayTag : getTag(other);\n\n objTag = objTag == argsTag ? objectTag : objTag;\n othTag = othTag == argsTag ? objectTag : othTag;\n\n var objIsObj = objTag == objectTag,\n othIsObj = othTag == objectTag,\n isSameTag = objTag == othTag;\n\n if (isSameTag && isBuffer(object)) {\n if (!isBuffer(other)) {\n return false;\n }\n objIsArr = true;\n objIsObj = false;\n }\n if (isSameTag && !objIsObj) {\n stack || (stack = new Stack);\n return (objIsArr || isTypedArray(object))\n ? equalArrays(object, other, bitmask, customizer, equalFunc, stack)\n : equalByTag(object, other, objTag, bitmask, customizer, equalFunc, stack);\n }\n if (!(bitmask & COMPARE_PARTIAL_FLAG)) {\n var objIsWrapped = objIsObj && hasOwnProperty.call(object, '__wrapped__'),\n othIsWrapped = othIsObj && hasOwnProperty.call(other, '__wrapped__');\n\n if (objIsWrapped || othIsWrapped) {\n var objUnwrapped = objIsWrapped ? object.value() : object,\n othUnwrapped = othIsWrapped ? other.value() : other;\n\n stack || (stack = new Stack);\n return equalFunc(objUnwrapped, othUnwrapped, bitmask, customizer, stack);\n }\n }\n if (!isSameTag) {\n return false;\n }\n stack || (stack = new Stack);\n return equalObjects(object, other, bitmask, customizer, equalFunc, stack);\n}\n\nexport default baseIsEqualDeep;\n","import baseIsEqualDeep from './_baseIsEqualDeep.js';\nimport isObjectLike from './isObjectLike.js';\n\n/**\n * The base implementation of `_.isEqual` which supports partial comparisons\n * and tracks traversed objects.\n *\n * @private\n * @param {*} value The value to compare.\n * @param {*} other The other value to compare.\n * @param {boolean} bitmask The bitmask flags.\n * 1 - Unordered comparison\n * 2 - Partial comparison\n * @param {Function} [customizer] The function to customize comparisons.\n * @param {Object} [stack] Tracks traversed `value` and `other` objects.\n * @returns {boolean} Returns `true` if the values are equivalent, else `false`.\n */\nfunction baseIsEqual(value, other, bitmask, customizer, stack) {\n if (value === other) {\n return true;\n }\n if (value == null || other == null || (!isObjectLike(value) && !isObjectLike(other))) {\n return value !== value && other !== other;\n }\n return baseIsEqualDeep(value, other, bitmask, customizer, baseIsEqual, stack);\n}\n\nexport default baseIsEqual;\n","import baseIsEqual from './_baseIsEqual.js';\n\n/**\n * This method is like `_.isEqual` except that it accepts `customizer` which\n * is invoked to compare values. If `customizer` returns `undefined`, comparisons\n * are handled by the method instead. The `customizer` is invoked with up to\n * six arguments: (objValue, othValue [, index|key, object, other, stack]).\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category Lang\n * @param {*} value The value to compare.\n * @param {*} other The other value to compare.\n * @param {Function} [customizer] The function to customize comparisons.\n * @returns {boolean} Returns `true` if the values are equivalent, else `false`.\n * @example\n *\n * function isGreeting(value) {\n * return /^h(?:i|ello)$/.test(value);\n * }\n *\n * function customizer(objValue, othValue) {\n * if (isGreeting(objValue) && isGreeting(othValue)) {\n * return true;\n * }\n * }\n *\n * var array = ['hello', 'goodbye'];\n * var other = ['hi', 'goodbye'];\n *\n * _.isEqualWith(array, other, customizer);\n * // => true\n */\nfunction isEqualWith(value, other, customizer) {\n customizer = typeof customizer == 'function' ? customizer : undefined;\n var result = customizer ? customizer(value, other) : undefined;\n return result === undefined ? baseIsEqual(value, other, undefined, customizer) : !!result;\n}\n\nexport default isEqualWith;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/mutationobserver\n */\n\n/* globals window */\n\nimport Observer from './observer';\nimport { startsWithFiller } from '../filler';\nimport { isEqualWith } from 'lodash-es';\n\nimport type DomConverter from '../domconverter';\nimport type Renderer from '../renderer';\nimport type View from '../view';\nimport type ViewElement from '../element';\nimport type ViewNode from '../node';\nimport type ViewText from '../text';\n\n/**\n * Mutation observer's role is to watch for any DOM changes inside the editor that weren't\n * done by the editor's {@link module:engine/view/renderer~Renderer} itself and reverting these changes.\n *\n * It does this by observing all mutations in the DOM, marking related view elements as changed and calling\n * {@link module:engine/view/renderer~Renderer#render}. Because all mutated nodes are marked as\n * \"to be rendered\" and the {@link module:engine/view/renderer~Renderer#render `render()`} method is called,\n * all changes are reverted in the DOM (the DOM is synced with the editor's view structure).\n *\n * Note that this observer is attached by the {@link module:engine/view/view~View} and is available by default.\n */\nexport default class MutationObserver extends Observer {\n\t/**\n\t * Reference to the {@link module:engine/view/view~View#domConverter}.\n\t */\n\tpublic readonly domConverter: DomConverter;\n\n\t/**\n\t * Reference to the {@link module:engine/view/view~View#_renderer}.\n\t */\n\tpublic readonly renderer: Renderer;\n\n\t/**\n\t * Native mutation observer config.\n\t */\n\tprivate readonly _config: MutationObserverInit;\n\n\t/**\n\t * Observed DOM elements.\n\t */\n\tprivate readonly _domElements: Set<HTMLElement>;\n\n\t/**\n\t * Native mutation observer.\n\t */\n\tprivate _mutationObserver: InstanceType<typeof global.MutationObserver>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tthis._config = {\n\t\t\tchildList: true,\n\t\t\tcharacterData: true,\n\t\t\tsubtree: true\n\t\t};\n\n\t\tthis.domConverter = view.domConverter;\n\t\tthis.renderer = view._renderer;\n\n\t\tthis._domElements = new Set();\n\t\tthis._mutationObserver = new window.MutationObserver( this._onMutations.bind( this ) );\n\t}\n\n\t/**\n\t * Synchronously handles mutations and empties the queue.\n\t */\n\tpublic flush(): void {\n\t\tthis._onMutations( this._mutationObserver.takeRecords() );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic observe( domElement: HTMLElement ): void {\n\t\tthis._domElements.add( domElement );\n\n\t\tif ( this.isEnabled ) {\n\t\t\tthis._mutationObserver.observe( domElement, this._config );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override stopObserving( domElement: HTMLElement ): void {\n\t\tthis._domElements.delete( domElement );\n\n\t\tif ( this.isEnabled ) {\n\t\t\t// Unfortunately, it is not possible to stop observing particular DOM element.\n\t\t\t// In order to stop observing one of multiple DOM elements, we need to re-connect the mutation observer.\n\t\t\tthis._mutationObserver.disconnect();\n\n\t\t\tfor ( const domElement of this._domElements ) {\n\t\t\t\tthis._mutationObserver.observe( domElement, this._config );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override enable(): void {\n\t\tsuper.enable();\n\n\t\tfor ( const domElement of this._domElements ) {\n\t\t\tthis._mutationObserver.observe( domElement, this._config );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override disable(): void {\n\t\tsuper.disable();\n\n\t\tthis._mutationObserver.disconnect();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis._mutationObserver.disconnect();\n\t}\n\n\t/**\n\t * Handles mutations. Mark view elements to sync and call render.\n\t *\n\t * @param domMutations Array of native mutations.\n\t */\n\tprivate _onMutations( domMutations: Array<MutationRecord> ) {\n\t\t// As a result of this.flush() we can have an empty collection.\n\t\tif ( domMutations.length === 0 ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst domConverter = this.domConverter;\n\n\t\t// Use map and set for deduplication.\n\t\tconst mutatedTextNodes = new Set<ViewText>();\n\t\tconst elementsWithMutatedChildren = new Set<ViewElement>();\n\n\t\t// Handle `childList` mutations first, so we will be able to check if the `characterData` mutation is in the\n\t\t// element with changed structure anyway.\n\t\tfor ( const mutation of domMutations ) {\n\t\t\tconst element = domConverter.mapDomToView( mutation.target as HTMLElement );\n\n\t\t\tif ( !element ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// Do not collect mutations from UIElements and RawElements.\n\t\t\tif ( element.is( 'uiElement' ) || element.is( 'rawElement' ) ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tif ( mutation.type === 'childList' && !this._isBogusBrMutation( mutation ) ) {\n\t\t\t\telementsWithMutatedChildren.add( element as ViewElement );\n\t\t\t}\n\t\t}\n\n\t\t// Handle `characterData` mutations later, when we have the full list of nodes which changed structure.\n\t\tfor ( const mutation of domMutations ) {\n\t\t\tconst element = domConverter.mapDomToView( mutation.target as HTMLElement );\n\n\t\t\t// Do not collect mutations from UIElements and RawElements.\n\t\t\tif ( element && ( element.is( 'uiElement' ) || element.is( 'rawElement' ) ) ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tif ( mutation.type === 'characterData' ) {\n\t\t\t\tconst text = domConverter.findCorrespondingViewText( mutation.target as Text ) as ViewText;\n\n\t\t\t\tif ( text && !elementsWithMutatedChildren.has( text.parent as ViewElement ) ) {\n\t\t\t\t\tmutatedTextNodes.add( text );\n\t\t\t\t}\n\t\t\t\t// When we added first letter to the text node which had only inline filler, for the DOM it is mutation\n\t\t\t\t// on text, but for the view, where filler text node did not exist, new text node was created, so we\n\t\t\t\t// need to handle it as a 'children' mutation instead of 'text'.\n\t\t\t\telse if ( !text && startsWithFiller( mutation.target ) ) {\n\t\t\t\t\telementsWithMutatedChildren.add(\n\t\t\t\t\t\tdomConverter.mapDomToView( mutation.target.parentNode as HTMLElement ) as ViewElement\n\t\t\t\t\t);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// Now we build the list of mutations to mark elements. We did not do it earlier to avoid marking the\n\t\t// same node multiple times in case of duplication.\n\n\t\tlet hasMutations = false;\n\n\t\tfor ( const textNode of mutatedTextNodes ) {\n\t\t\thasMutations = true;\n\t\t\tthis.renderer.markToSync( 'text', textNode );\n\t\t}\n\n\t\tfor ( const viewElement of elementsWithMutatedChildren ) {\n\t\t\tconst domElement = domConverter.mapViewToDom( viewElement )!;\n\t\t\tconst viewChildren = Array.from( viewElement.getChildren() );\n\t\t\tconst newViewChildren = Array.from( domConverter.domChildrenToView( domElement, { withChildren: false } ) );\n\n\t\t\t// It may happen that as a result of many changes (sth was inserted and then removed),\n\t\t\t// both elements haven't really changed. #1031\n\t\t\tif ( !isEqualWith( viewChildren, newViewChildren, sameNodes ) ) {\n\t\t\t\thasMutations = true;\n\t\t\t\tthis.renderer.markToSync( 'children', viewElement );\n\t\t\t}\n\t\t}\n\n\t\t// In case only non-relevant mutations were recorded it skips the event and force render (#5600).\n\t\tif ( hasMutations ) {\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.group( '%c[MutationObserver]%c Mutations detected',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight:bold;color:green', ''\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t// At this point we have \"dirty DOM\" (changed) and de-synched view (which has not been changed).\n\t\t\t// In order to \"reset DOM\" we render the view again.\n\t\t\tthis.view.forceRender();\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t}\n\t}\n\n\t/**\n\t * Checks if mutation was generated by the browser inserting bogus br on the end of the block element.\n\t * Such mutations are generated while pressing space or performing native spellchecker correction\n\t * on the end of the block element in Firefox browser.\n\t *\n\t * @param mutation Native mutation object.\n\t */\n\tprivate _isBogusBrMutation( mutation: MutationRecord ) {\n\t\tlet addedNode = null;\n\n\t\t// Check if mutation added only one node on the end of its parent.\n\t\tif ( mutation.nextSibling === null && mutation.removedNodes.length === 0 && mutation.addedNodes.length == 1 ) {\n\t\t\taddedNode = this.domConverter.domToView( mutation.addedNodes[ 0 ], {\n\t\t\t\twithChildren: false\n\t\t\t} );\n\t\t}\n\n\t\treturn addedNode && addedNode.is( 'element', 'br' );\n\t}\n}\n\nfunction sameNodes( child1: ViewNode, child2: ViewNode ) {\n\t// First level of comparison (array of children vs array of children) use the Lodash's default behavior.\n\tif ( Array.isArray( child1 ) ) {\n\t\treturn;\n\t}\n\n\t// Elements.\n\tif ( child1 === child2 ) {\n\t\treturn true;\n\t}\n\t// Texts.\n\telse if ( child1.is( '$text' ) && child2.is( '$text' ) ) {\n\t\treturn child1.data === child2.data;\n\t}\n\n\t// Not matching types.\n\treturn false;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/focusobserver\n */\n\n/* globals setTimeout, clearTimeout */\n\nimport DomEventObserver from './domeventobserver';\nimport type DomEventData from './domeventdata';\nimport type View from '../view';\n\n/**\n * {@link module:engine/view/document~Document#event:focus Focus}\n * and {@link module:engine/view/document~Document#event:blur blur} events observer.\n * Focus observer handle also {@link module:engine/view/rooteditableelement~RootEditableElement#isFocused isFocused} property of the\n * {@link module:engine/view/rooteditableelement~RootEditableElement root elements}.\n *\n * Note that this observer is attached by the {@link module:engine/view/view~View} and is available by default.\n */\nexport default class FocusObserver extends DomEventObserver<'focus' | 'blur'> {\n\t/**\n\t * Identifier of the timeout currently used by focus listener to delay rendering execution.\n\t */\n\tprivate _renderTimeoutId!: ReturnType<typeof setTimeout>;\n\n\t/**\n\t * Set to `true` if the document is in the process of setting the focus.\n\t *\n\t * The flag is used to indicate that setting the focus is in progress.\n\t */\n\tprivate _isFocusChanging: boolean = false;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic readonly domEventType = [ 'focus', 'blur' ] as const;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tthis.useCapture = true;\n\t\tconst document = this.document;\n\n\t\tdocument.on<ViewDocumentFocusEvent>( 'focus', () => {\n\t\t\tthis._isFocusChanging = true;\n\n\t\t\t// Unfortunately native `selectionchange` event is fired asynchronously.\n\t\t\t// We need to wait until `SelectionObserver` handle the event and then render. Otherwise rendering will\n\t\t\t// overwrite new DOM selection with selection from the view.\n\t\t\t// See https://github.com/ckeditor/ckeditor5-engine/issues/795 for more details.\n\t\t\t// Long timeout is needed to solve #676 and https://github.com/ckeditor/ckeditor5-engine/issues/1157 issues.\n\t\t\t//\n\t\t\t// Using `view.change()` instead of `view.forceRender()` to prevent double rendering\n\t\t\t// in a situation where `selectionchange` already caused selection change.\n\t\t\tthis._renderTimeoutId = setTimeout( () => {\n\t\t\t\tthis.flush();\n\t\t\t\tview.change( () => {} );\n\t\t\t}, 50 );\n\t\t} );\n\n\t\tdocument.on<ViewDocumentBlurEvent>( 'blur', ( evt, data ) => {\n\t\t\tconst selectedEditable = document.selection.editableElement;\n\n\t\t\tif ( selectedEditable === null || selectedEditable === data.target ) {\n\t\t\t\tdocument.isFocused = false;\n\t\t\t\tthis._isFocusChanging = false;\n\n\t\t\t\t// Re-render the document to update view elements\n\t\t\t\t// (changing document.isFocused already marked view as changed since last rendering).\n\t\t\t\tview.change( () => {} );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Finishes setting the document focus state.\n\t */\n\tpublic flush(): void {\n\t\tif ( this._isFocusChanging ) {\n\t\t\tthis._isFocusChanging = false;\n\t\t\tthis.document.isFocused = true;\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic onDomEvent( domEvent: FocusEvent ): void {\n\t\tthis.fire( domEvent.type, domEvent );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tif ( this._renderTimeoutId ) {\n\t\t\tclearTimeout( this._renderTimeoutId );\n\t\t}\n\n\t\tsuper.destroy();\n\t}\n}\n\n/**\n * Fired when one of the editables gets focus.\n *\n * Introduced by {@link module:engine/view/observer/focusobserver~FocusObserver}.\n *\n * Note that because {@link module:engine/view/observer/focusobserver~FocusObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/focusobserver~FocusObserver\n * @eventName module:engine/view/document~Document#focus\n * @param data Event data.\n */\nexport type ViewDocumentFocusEvent = {\n\tname: 'focus';\n\targs: [ data: DomEventData<FocusEvent> ];\n};\n\n/**\n * Fired when one of the editables loses focus.\n *\n * Introduced by {@link module:engine/view/observer/focusobserver~FocusObserver}.\n *\n * Note that because {@link module:engine/view/observer/focusobserver~FocusObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/focusobserver~FocusObserver\n * @eventName module:engine/view/document~Document#blur\n * @param data Event data.\n */\nexport type ViewDocumentBlurEvent = {\n\tname: 'blur';\n\targs: [ data: DomEventData<FocusEvent> ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/selectionobserver\n */\n\n/* global setInterval, clearInterval */\n\nimport Observer from './observer';\nimport MutationObserver from './mutationobserver';\nimport { env } from '@ckeditor/ckeditor5-utils';\nimport { debounce, type DebouncedFunc } from 'lodash-es';\n\nimport type View from '../view';\nimport type DocumentSelection from '../documentselection';\nimport type DomConverter from '../domconverter';\nimport type Selection from '../selection';\nimport FocusObserver from './focusobserver';\n\ntype DomSelection = globalThis.Selection;\n\n/**\n * Selection observer class observes selection changes in the document. If a selection changes on the document this\n * observer checks if the DOM selection is different from the {@link module:engine/view/document~Document#selection view selection}.\n * The selection observer fires {@link module:engine/view/document~Document#event:selectionChange} event only if\n * a selection change was the only change in the document and the DOM selection is different from the view selection.\n *\n * This observer also manages the {@link module:engine/view/document~Document#isSelecting} property of the view document.\n *\n * Note that this observer is attached by the {@link module:engine/view/view~View} and is available by default.\n */\nexport default class SelectionObserver extends Observer {\n\t/**\n\t * Instance of the mutation observer. Selection observer calls\n\t * {@link module:engine/view/observer/mutationobserver~MutationObserver#flush} to ensure that the mutations will be handled\n\t * before the {@link module:engine/view/document~Document#event:selectionChange} event is fired.\n\t */\n\tpublic readonly mutationObserver: MutationObserver;\n\n\t/**\n\t * Instance of the focus observer. Focus observer calls\n\t * {@link module:engine/view/observer/focusobserver~FocusObserver#flush} to mark the latest focus change as complete.\n\t */\n\tpublic readonly focusObserver: FocusObserver;\n\n\t/**\n\t * Reference to the view {@link module:engine/view/documentselection~DocumentSelection} object used to compare\n\t * new selection with it.\n\t */\n\tpublic readonly selection: DocumentSelection;\n\n\t/**\n\t * Reference to the {@link module:engine/view/view~View#domConverter}.\n\t */\n\tpublic readonly domConverter: DomConverter;\n\n\t/**\n\t * A set of documents which have added `selectionchange` listener to avoid adding a listener twice to the same\n\t * document.\n\t */\n\tprivate readonly _documents: WeakSet<Document>;\n\n\t/**\n\t * Fires debounced event `selectionChangeDone`. It uses `lodash#debounce` method to delay function call.\n\t */\n\tprivate readonly _fireSelectionChangeDoneDebounced: DebouncedFunc<( data: ViewDocumentSelectionEventData ) => void>;\n\n\t/**\n\t * When called, starts clearing the {@link #_loopbackCounter} counter in time intervals. When the number of selection\n\t * changes exceeds a certain limit within the interval of time, the observer will not fire `selectionChange` but warn about\n\t * possible infinite selection loop.\n\t */\n\tprivate readonly _clearInfiniteLoopInterval: ReturnType<typeof setInterval>;\n\n\t/**\n\t * Unlocks the `isSelecting` state of the view document in case the selection observer did not record this fact\n\t * correctly (for whatever reason). It is a safeguard (paranoid check), that returns document to the normal state\n\t * after a certain period of time (debounced, postponed by each selectionchange event).\n\t */\n\tprivate readonly _documentIsSelectingInactivityTimeoutDebounced: DebouncedFunc<() => void>;\n\n\t/**\n\t * Private property to check if the code does not enter infinite loop.\n\t */\n\tprivate _loopbackCounter: number;\n\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tthis.mutationObserver = view.getObserver( MutationObserver );\n\t\tthis.focusObserver = view.getObserver( FocusObserver );\n\t\tthis.selection = this.document.selection;\n\t\tthis.domConverter = view.domConverter;\n\n\t\tthis._documents = new WeakSet();\n\n\t\tthis._fireSelectionChangeDoneDebounced = debounce( data => {\n\t\t\tthis.document.fire<ViewDocumentSelectionChangeDoneEvent>( 'selectionChangeDone', data );\n\t\t}, 200 );\n\n\t\tthis._clearInfiniteLoopInterval = setInterval( () => this._clearInfiniteLoop(), 1000 );\n\n\t\tthis._documentIsSelectingInactivityTimeoutDebounced = debounce( () => ( this.document.isSelecting = false ), 5000 );\n\n\t\tthis._loopbackCounter = 0;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override observe( domElement: HTMLElement ): void {\n\t\tconst domDocument = domElement.ownerDocument;\n\n\t\tconst startDocumentIsSelecting = () => {\n\t\t\tthis.document.isSelecting = true;\n\n\t\t\t// Let's activate the safety timeout each time the document enters the \"is selecting\" state.\n\t\t\tthis._documentIsSelectingInactivityTimeoutDebounced();\n\t\t};\n\n\t\tconst endDocumentIsSelecting = () => {\n\t\t\tif ( !this.document.isSelecting ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Make sure that model selection is up-to-date at the end of selecting process.\n\t\t\t// Sometimes `selectionchange` events could arrive after the `mouseup` event and that selection could be already outdated.\n\t\t\tthis._handleSelectionChange( null, domDocument );\n\n\t\t\tthis.document.isSelecting = false;\n\n\t\t\t// The safety timeout can be canceled when the document leaves the \"is selecting\" state.\n\t\t\tthis._documentIsSelectingInactivityTimeoutDebounced.cancel();\n\t\t};\n\n\t\t// The document has the \"is selecting\" state while the user keeps making (extending) the selection\n\t\t// (e.g. by holding the mouse button and moving the cursor). The state resets when they either released\n\t\t// the mouse button or interrupted the process by pressing or releasing any key.\n\t\tthis.listenTo( domElement, 'selectstart', startDocumentIsSelecting, { priority: 'highest' } );\n\n\t\tthis.listenTo( domElement, 'keydown', endDocumentIsSelecting, { priority: 'highest', useCapture: true } );\n\t\tthis.listenTo( domElement, 'keyup', endDocumentIsSelecting, { priority: 'highest', useCapture: true } );\n\n\t\t// Add document-wide listeners only once. This method could be called for multiple editing roots.\n\t\tif ( this._documents.has( domDocument ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// This listener is using capture mode to make sure that selection is upcasted before any other\n\t\t// handler would like to check it and update (for example table multi cell selection).\n\t\tthis.listenTo( domDocument, 'mouseup', endDocumentIsSelecting, { priority: 'highest', useCapture: true } );\n\n\t\tthis.listenTo( domDocument, 'selectionchange', ( evt, domEvent ) => {\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconst domSelection = domDocument.defaultView!.getSelection();\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.group( '%c[SelectionObserver]%c selectionchange', 'color:green', ''\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[SelectionObserver]%c DOM Selection:', 'font-weight:bold;color:green', '',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t{ node: domSelection!.anchorNode, offset: domSelection!.anchorOffset },\n\t\t\t// @if CK_DEBUG_TYPING // \t\t{ node: domSelection!.focusNode, offset: domSelection!.focusOffset }\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t// The Renderer is disabled while composing on non-android browsers, so we can't update the view selection\n\t\t\t// because the DOM and view tree drifted apart. Position mapping could fail because of it.\n\t\t\tif ( this.document.isComposing && !env.isAndroid ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[SelectionObserver]%c Selection change ignored (isComposing)',\n\t\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight:bold;color:green', ''\n\t\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._handleSelectionChange( domEvent, domDocument );\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t// Defer the safety timeout when the selection changes (e.g. the user keeps extending the selection\n\t\t\t// using their mouse).\n\t\t\tthis._documentIsSelectingInactivityTimeoutDebounced();\n\t\t} );\n\n\t\tthis._documents.add( domDocument );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override stopObserving( domElement: HTMLElement ): void {\n\t\tthis.stopListening( domElement );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tclearInterval( this._clearInfiniteLoopInterval );\n\t\tthis._fireSelectionChangeDoneDebounced.cancel();\n\t\tthis._documentIsSelectingInactivityTimeoutDebounced.cancel();\n\t}\n\n\t/* istanbul ignore next -- @preserve */\n\tprivate _reportInfiniteLoop() {\n\t// @if CK_DEBUG //\t\tthrow new Error(\n\t// @if CK_DEBUG //\t\t\t'Selection change observer detected an infinite rendering loop.\\n\\n' +\n\t// @if CK_DEBUG //\t \t\t'⚠️⚠️ Report this error on https://github.com/ckeditor/ckeditor5/issues/11658.'\n\t// @if CK_DEBUG //\t\t);\n\t}\n\n\t/**\n\t * Selection change listener. {@link module:engine/view/observer/mutationobserver~MutationObserver#flush Flush} mutations, check if\n\t * a selection changes and fires {@link module:engine/view/document~Document#event:selectionChange} event on every change\n\t * and {@link module:engine/view/document~Document#event:selectionChangeDone} when a selection stop changing.\n\t *\n\t * @param domEvent DOM event.\n\t * @param domDocument DOM document.\n\t */\n\tprivate _handleSelectionChange( domEvent: unknown, domDocument: Document ) {\n\t\tif ( !this.isEnabled ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst domSelection = domDocument.defaultView!.getSelection()!;\n\n\t\tif ( this.checkShouldIgnoreEventFromTarget( domSelection.anchorNode! ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Ensure the mutation event will be before selection event on all browsers.\n\t\tthis.mutationObserver.flush();\n\n\t\tconst newViewSelection = this.domConverter.domSelectionToView( domSelection );\n\n\t\t// Do not convert selection change if the new view selection has no ranges in it.\n\t\t//\n\t\t// It means that the DOM selection is in some way incorrect. Ranges that were in the DOM selection could not be\n\t\t// converted to the view. This happens when the DOM selection was moved outside of the editable element.\n\t\tif ( newViewSelection.rangeCount == 0 ) {\n\t\t\tthis.view.hasDomSelection = false;\n\n\t\t\treturn;\n\t\t}\n\n\t\tthis.view.hasDomSelection = true;\n\n\t\t// Mark the latest focus change as complete (we got new selection after the focus so the selection is in the focused element).\n\t\tthis.focusObserver.flush();\n\n\t\tif ( this.selection.isEqual( newViewSelection ) && this.domConverter.isDomSelectionCorrect( domSelection ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Ensure we are not in the infinite loop (#400).\n\t\t// This counter is reset each second. 60 selection changes in 1 second is enough high number\n\t\t// to be very difficult (impossible) to achieve using just keyboard keys (during normal editor use).\n\t\tif ( ++this._loopbackCounter > 60 ) {\n\t\t\t// Selection change observer detected an infinite rendering loop.\n\t\t\t// Most probably you try to put the selection in the position which is not allowed\n\t\t\t// by the browser and browser fixes it automatically what causes `selectionchange` event on\n\t\t\t// which a loopback through a model tries to re-render the wrong selection and again.\n\t\t\tthis._reportInfiniteLoop();\n\n\t\t\treturn;\n\t\t}\n\n\t\tif ( this.selection.isSimilar( newViewSelection ) ) {\n\t\t\t// If selection was equal and we are at this point of algorithm, it means that it was incorrect.\n\t\t\t// Just re-render it, no need to fire any events, etc.\n\t\t\tthis.view.forceRender();\n\t\t} else {\n\t\t\tconst data: ViewDocumentSelectionEventData = {\n\t\t\t\toldSelection: this.selection,\n\t\t\t\tnewSelection: newViewSelection,\n\t\t\t\tdomSelection\n\t\t\t};\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[SelectionObserver]%c Fire selection change:',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight:bold;color:green', '',\n\t\t\t// @if CK_DEBUG_TYPING // \t\tnewViewSelection.getFirstRange()\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t// Prepare data for new selection and fire appropriate events.\n\t\t\tthis.document.fire<ViewDocumentSelectionChangeEvent>( 'selectionChange', data );\n\n\t\t\t// Call `#_fireSelectionChangeDoneDebounced` every time when `selectionChange` event is fired.\n\t\t\t// This function is debounced what means that `selectionChangeDone` event will be fired only when\n\t\t\t// defined int the function time will elapse since the last time the function was called.\n\t\t\t// So `selectionChangeDone` will be fired when selection will stop changing.\n\t\t\tthis._fireSelectionChangeDoneDebounced( data );\n\t\t}\n\t}\n\n\t/**\n\t * Clears `SelectionObserver` internal properties connected with preventing infinite loop.\n\t */\n\tprivate _clearInfiniteLoop(): void {\n\t\tthis._loopbackCounter = 0;\n\t}\n}\n\n/**\n * The value of {@link ~ViewDocumentSelectionChangeEvent} and {@link ~ViewDocumentSelectionChangeDoneEvent} events.\n */\nexport type ViewDocumentSelectionEventData = {\n\n\t/**\n\t * Old View selection which is {@link module:engine/view/document~Document#selection}.\n\t */\n\toldSelection: DocumentSelection;\n\n\t/**\n\t * New View selection which is converted DOM selection.\n\t */\n\tnewSelection: Selection;\n\n\t/**\n\t * Native DOM selection.\n\t */\n\tdomSelection: DomSelection | null;\n};\n\n/**\n * Fired when a selection has changed. This event is fired only when the selection change was the only change that happened\n * in the document, and the old selection is different then the new selection.\n *\n * Introduced by {@link module:engine/view/observer/selectionobserver~SelectionObserver}.\n *\n * Note that because {@link module:engine/view/observer/selectionobserver~SelectionObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/selectionobserver~SelectionObserver\n * @eventName module:engine/view/document~Document#selectionChange\n */\nexport type ViewDocumentSelectionChangeEvent = {\n\tname: 'selectionChange';\n\targs: [ ViewDocumentSelectionEventData ];\n};\n\n/**\n * Fired when selection stops changing.\n *\n * Introduced by {@link module:engine/view/observer/selectionobserver~SelectionObserver}.\n *\n * Note that because {@link module:engine/view/observer/selectionobserver~SelectionObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/selectionobserver~SelectionObserver\n * @eventName module:engine/view/document~Document#selectionChangeDone\n */\nexport type ViewDocumentSelectionChangeDoneEvent = {\n\tname: 'selectionChangeDone';\n\targs: [ ViewDocumentSelectionEventData ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/compositionobserver\n */\n\nimport DomEventObserver from './domeventobserver';\nimport type View from '../view';\nimport type DomEventData from './domeventdata';\n\n/**\n * {@link module:engine/view/document~Document#event:compositionstart Compositionstart},\n * {@link module:engine/view/document~Document#event:compositionupdate compositionupdate} and\n * {@link module:engine/view/document~Document#event:compositionend compositionend} events observer.\n *\n * Note that this observer is attached by the {@link module:engine/view/view~View} and is available by default.\n */\nexport default class CompositionObserver extends DomEventObserver<'compositionstart' | 'compositionupdate' | 'compositionend'> {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic readonly domEventType = [ 'compositionstart', 'compositionupdate', 'compositionend' ] as const;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tconst document = this.document;\n\n\t\tdocument.on<ViewDocumentCompositionStartEvent>( 'compositionstart', () => {\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.log( '%c[CompositionObserver] ' +\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'┌───────────────────────────── isComposing = true ─────────────────────────────┐',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight: bold; color: green'\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t\tdocument.isComposing = true;\n\t\t}, { priority: 'low' } );\n\n\t\tdocument.on<ViewDocumentCompositionEndEvent>( 'compositionend', () => {\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.log( '%c[CompositionObserver] ' +\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'└───────────────────────────── isComposing = false ─────────────────────────────┘',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight: bold; color: green'\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t\tdocument.isComposing = false;\n\t\t}, { priority: 'low' } );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic onDomEvent( domEvent: CompositionEvent ): void {\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.group( `%c[CompositionObserver]%c ${ domEvent.type }`, 'color: green', '' );\n\t\t// @if CK_DEBUG_TYPING // }\n\n\t\tthis.fire( domEvent.type, domEvent, {\n\t\t\tdata: domEvent.data\n\t\t} );\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t// @if CK_DEBUG_TYPING // }\n\t}\n}\n\nexport interface CompositionEventData extends DomEventData<CompositionEvent> {\n\tdata: string | null;\n}\n\n/**\n * Fired when composition starts inside one of the editables.\n *\n * Introduced by {@link module:engine/view/observer/compositionobserver~CompositionObserver}.\n *\n * Note that because {@link module:engine/view/observer/compositionobserver~CompositionObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/compositionobserver~CompositionObserver\n * @eventName module:engine/view/document~Document#compositionstart\n * @param data Event data.\n */\nexport type ViewDocumentCompositionStartEvent = {\n\tname: 'compositionstart';\n\targs: [ data: CompositionEventData ];\n};\n\n/**\n * Fired when composition is updated inside one of the editables.\n *\n * Introduced by {@link module:engine/view/observer/compositionobserver~CompositionObserver}.\n *\n * Note that because {@link module:engine/view/observer/compositionobserver~CompositionObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/compositionobserver~CompositionObserver\n * @eventName module:engine/view/document~Document#compositionupdate\n * @param data Event data.\n */\nexport type ViewDocumentCompositionUpdateEvent = {\n\tname: 'compositionupdate';\n\targs: [ data: CompositionEventData ];\n};\n\n/**\n * Fired when composition ends inside one of the editables.\n *\n * Introduced by {@link module:engine/view/observer/compositionobserver~CompositionObserver}.\n *\n * Note that because {@link module:engine/view/observer/compositionobserver~CompositionObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @see module:engine/view/observer/compositionobserver~CompositionObserver\n * @eventName module:engine/view/document~Document#compositionend\n * @param data Event data.\n */\nexport type ViewDocumentCompositionEndEvent = {\n\tname: 'compositionend';\n\targs: [ data: CompositionEventData ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/datatransfer\n */\n\ntype DomDataTransfer = globalThis.DataTransfer;\n\n/**\n * A facade over the native [`DataTransfer`](https://developer.mozilla.org/en-US/docs/Web/API/DataTransfer) object.\n */\nexport default class DataTransfer {\n\t/**\n\t * The array of files created from the native `DataTransfer#files` or `DataTransfer#items`.\n\t */\n\tprivate _files: Array<File> | null;\n\n\t/**\n\t * The native DataTransfer object.\n\t */\n\tprivate _native: DomDataTransfer;\n\n\t/**\n\t * @param nativeDataTransfer The native [`DataTransfer`](https://developer.mozilla.org/en-US/docs/Web/API/DataTransfer) object.\n\t * @param options.cacheFiles Whether `files` list should be initialized in the constructor.\n\t */\n\tconstructor( nativeDataTransfer: DomDataTransfer, options: { cacheFiles?: boolean } = {} ) {\n\t\t// We should store references to the File instances in case someone would like to process this files\n\t\t// outside the event handler. Files are stored only for `drop` and `paste` events because they are not usable\n\t\t// in other events and are generating a huge delay on Firefox while dragging.\n\t\t// See https://github.com/ckeditor/ckeditor5/issues/13366.\n\t\tthis._files = options.cacheFiles ? getFiles( nativeDataTransfer ) : null;\n\n\t\tthis._native = nativeDataTransfer;\n\t}\n\n\t/**\n\t * The array of files created from the native `DataTransfer#files` or `DataTransfer#items`.\n\t */\n\tpublic get files(): Array<File> {\n\t\tif ( !this._files ) {\n\t\t\tthis._files = getFiles( this._native );\n\t\t}\n\n\t\treturn this._files;\n\t}\n\n\t/**\n\t * Returns an array of available native content types.\n\t */\n\tpublic get types(): ReadonlyArray<string> {\n\t\treturn this._native.types;\n\t}\n\n\t/**\n\t * Gets the data from the data transfer by its MIME type.\n\t *\n\t * ```ts\n\t * dataTransfer.getData( 'text/plain' );\n\t * ```\n\t *\n\t * @param type The MIME type. E.g. `text/html` or `text/plain`.\n\t */\n\tpublic getData( type: string ): string {\n\t\treturn this._native.getData( type );\n\t}\n\n\t/**\n\t * Sets the data in the data transfer.\n\t *\n\t * @param type The MIME type. E.g. `text/html` or `text/plain`.\n\t */\n\tpublic setData( type: string, data: string ): void {\n\t\tthis._native.setData( type, data );\n\t}\n\n\t/**\n\t * The effect that is allowed for a drag operation.\n\t */\n\tpublic set effectAllowed( value: EffectAllowed ) {\n\t\tthis._native.effectAllowed = value;\n\t}\n\n\tpublic get effectAllowed(): EffectAllowed {\n\t\treturn this._native.effectAllowed;\n\t}\n\n\t/**\n\t * The actual drop effect.\n\t */\n\tpublic set dropEffect( value: DropEffect ) {\n\t\tthis._native.dropEffect = value;\n\t}\n\n\tpublic get dropEffect(): DropEffect {\n\t\treturn this._native.dropEffect;\n\t}\n\n\t/**\n\t * Set a preview image of the dragged content.\n\t */\n\tpublic setDragImage( image: Element, x: number, y: number ): void {\n\t\tthis._native.setDragImage( image, x, y );\n\t}\n\n\t/**\n\t * Whether the dragging operation was canceled.\n\t */\n\tpublic get isCanceled(): boolean {\n\t\treturn this._native.dropEffect == 'none' || !!( this._native as any ).mozUserCancelled;\n\t}\n}\n\n/**\n * The effect that is allowed for a drag operation.\n */\nexport type EffectAllowed = DomDataTransfer[ 'effectAllowed' ];\n\n/**\n * The actual drop effect.\n */\nexport type DropEffect = DomDataTransfer[ 'dropEffect' ];\n\nfunction getFiles( nativeDataTransfer: DomDataTransfer ): Array<File> {\n\t// DataTransfer.files and items are array-like and might not have an iterable interface.\n\tconst files = Array.from( nativeDataTransfer.files || [] );\n\tconst items = Array.from( nativeDataTransfer.items || [] );\n\n\tif ( files.length ) {\n\t\treturn files;\n\t}\n\n\t// Chrome has empty DataTransfer.files, but allows getting files through the items interface.\n\treturn items\n\t\t.filter( item => item.kind === 'file' )\n\t\t.map( item => item.getAsFile()! );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/inputobserver\n */\n\nimport DomEventObserver from './domeventobserver';\nimport type DomEventData from './domeventdata';\nimport type ViewRange from '../range';\nimport DataTransfer from '../datatransfer';\nimport { env } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Observer for events connected with data input.\n *\n * **Note**: This observer is attached by {@link module:engine/view/view~View} and available by default in all\n * editor instances.\n */\nexport default class InputObserver extends DomEventObserver<'beforeinput'> {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic readonly domEventType = 'beforeinput' as const;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic onDomEvent( domEvent: InputEvent ): void {\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.group( `%c[InputObserver]%c ${ domEvent.type }: ${ domEvent.inputType }`,\n\t\t// @if CK_DEBUG_TYPING // \t\t'color: green', 'color: default'\n\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t// @if CK_DEBUG_TYPING // }\n\n\t\tconst domTargetRanges = domEvent.getTargetRanges();\n\t\tconst view = this.view;\n\t\tconst viewDocument = view.document;\n\n\t\tlet dataTransfer: DataTransfer | null = null;\n\t\tlet data: string | null = null;\n\t\tlet targetRanges: Array<ViewRange> = [];\n\n\t\tif ( domEvent.dataTransfer ) {\n\t\t\tdataTransfer = new DataTransfer( domEvent.dataTransfer );\n\t\t}\n\n\t\tif ( domEvent.data !== null ) {\n\t\t\tdata = domEvent.data;\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( `%c[InputObserver]%c event data: %c${ JSON.stringify( data ) }`,\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', 'font-weight:bold', 'color: blue;'\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t} else if ( dataTransfer ) {\n\t\t\tdata = dataTransfer.getData( 'text/plain' );\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( `%c[InputObserver]%c event data transfer: %c${ JSON.stringify( data ) }`,\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', 'font-weight:bold', 'color: blue;'\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t}\n\n\t\t// If the editor selection is fake (an object is selected), the DOM range does not make sense because it is anchored\n\t\t// in the fake selection container.\n\t\tif ( viewDocument.selection.isFake ) {\n\t\t\t// Future-proof: in case of multi-range fake selections being possible.\n\t\t\ttargetRanges = Array.from( viewDocument.selection.getRanges() );\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[InputObserver]%c using fake selection:',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', 'font-weight:bold', targetRanges,\n\t\t\t// @if CK_DEBUG_TYPING // \t\tviewDocument.selection.isFake ? 'fake view selection' : 'fake DOM parent'\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t} else if ( domTargetRanges.length ) {\n\t\t\ttargetRanges = domTargetRanges.map( domRange => {\n\t\t\t\t// Sometimes browser provides range that starts before editable node.\n\t\t\t\t// We try to fall back to collapsed range at the valid end position.\n\t\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/14411.\n\t\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/14050.\n\t\t\t\tconst viewStart = view.domConverter.domPositionToView( domRange.startContainer, domRange.startOffset );\n\t\t\t\tconst viewEnd = view.domConverter.domPositionToView( domRange.endContainer, domRange.endOffset );\n\n\t\t\t\tif ( viewStart ) {\n\t\t\t\t\treturn view.createRange( viewStart, viewEnd );\n\t\t\t\t} else if ( viewEnd ) {\n\t\t\t\t\treturn view.createRange( viewEnd );\n\t\t\t\t}\n\t\t\t} ).filter( ( range ): range is ViewRange => !!range );\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[InputObserver]%c using target ranges:',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', 'font-weight:bold', targetRanges\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t}\n\t\t// For Android devices we use a fallback to the current DOM selection, Android modifies it according\n\t\t// to the expected target ranges of input event.\n\t\telse if ( env.isAndroid ) {\n\t\t\tconst domSelection = ( domEvent.target as HTMLElement ).ownerDocument.defaultView!.getSelection()!;\n\n\t\t\ttargetRanges = Array.from( view.domConverter.domSelectionToView( domSelection ).getRanges() );\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[InputObserver]%c using selection ranges:',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'color: green;font-weight: bold', 'font-weight:bold', targetRanges\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\t\t}\n\n\t\t// Android sometimes fires insertCompositionText with a new-line character at the end of the data\n\t\t// instead of firing insertParagraph beforeInput event.\n\t\t// Fire the correct type of beforeInput event and ignore the replaced fragment of text because\n\t\t// it wants to replace \"test\" with \"test\\n\".\n\t\t// https://github.com/ckeditor/ckeditor5/issues/12368.\n\t\tif ( env.isAndroid && domEvent.inputType == 'insertCompositionText' && data && data.endsWith( '\\n' ) ) {\n\t\t\tthis.fire( domEvent.type, domEvent, {\n\t\t\t\tinputType: 'insertParagraph',\n\t\t\t\ttargetRanges: [ view.createRange( targetRanges[ 0 ].end ) ]\n\t\t\t} );\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Normalize the insertText data that includes new-line characters.\n\t\t// https://github.com/ckeditor/ckeditor5/issues/2045.\n\t\tif ( domEvent.inputType == 'insertText' && data && data.includes( '\\n' ) ) {\n\t\t\t// There might be a single new-line or double for new paragraph, but we translate\n\t\t\t// it to paragraphs as it is our default action for enter handling.\n\t\t\tconst parts = data.split( /\\n{1,2}/g );\n\n\t\t\tlet partTargetRanges = targetRanges;\n\n\t\t\tfor ( let i = 0; i < parts.length; i++ ) {\n\t\t\t\tconst dataPart = parts[ i ];\n\n\t\t\t\tif ( dataPart != '' ) {\n\t\t\t\t\tthis.fire( domEvent.type, domEvent, {\n\t\t\t\t\t\tdata: dataPart,\n\t\t\t\t\t\tdataTransfer,\n\t\t\t\t\t\ttargetRanges: partTargetRanges,\n\t\t\t\t\t\tinputType: domEvent.inputType,\n\t\t\t\t\t\tisComposing: domEvent.isComposing\n\t\t\t\t\t} );\n\n\t\t\t\t\t// Use the result view selection so following events will be added one after another.\n\t\t\t\t\tpartTargetRanges = [ viewDocument.selection.getFirstRange()! ];\n\t\t\t\t}\n\n\t\t\t\tif ( i + 1 < parts.length ) {\n\t\t\t\t\tthis.fire( domEvent.type, domEvent, {\n\t\t\t\t\t\tinputType: 'insertParagraph',\n\t\t\t\t\t\ttargetRanges: partTargetRanges\n\t\t\t\t\t} );\n\n\t\t\t\t\t// Use the result view selection so following events will be added one after another.\n\t\t\t\t\tpartTargetRanges = [ viewDocument.selection.getFirstRange()! ];\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Fire the normalized beforeInput event.\n\t\tthis.fire( domEvent.type, domEvent, {\n\t\t\tdata,\n\t\t\tdataTransfer,\n\t\t\ttargetRanges,\n\t\t\tinputType: domEvent.inputType,\n\t\t\tisComposing: domEvent.isComposing\n\t\t} );\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.groupEnd();\n\t\t// @if CK_DEBUG_TYPING // }\n\t}\n}\n\n/**\n * Fired before the web browser inputs, deletes, or formats some data.\n *\n * This event is introduced by {@link module:engine/view/observer/inputobserver~InputObserver} and available\n * by default in all editor instances (attached by {@link module:engine/view/view~View}).\n *\n * @see module:engine/view/observer/inputobserver~InputObserver\n * @eventName module:engine/view/document~Document#beforeinput\n * @param data Event data containing detailed information about the event.\n */\nexport type ViewDocumentInputEvent = {\n\tname: 'beforeinput';\n\targs: [ data: InputEventData ];\n};\n\n/**\n * The value of the {@link ~ViewDocumentInputEvent} event.\n */\nexport interface InputEventData extends DomEventData<InputEvent> {\n\n\t/**\n\t * The type of the input event (e.g. \"insertText\" or \"deleteWordBackward\"). Corresponds to native `InputEvent#inputType`.\n\t */\n\treadonly inputType: string;\n\n\t/**\n\t * A unified text data passed along with the input event. Depending on:\n\t *\n\t * * the web browser and input events implementation (for instance [Level 1](https://www.w3.org/TR/input-events-1/) or\n\t * [Level 2](https://www.w3.org/TR/input-events-2/)),\n\t * * {@link module:engine/view/observer/inputobserver~InputEventData#inputType input type}\n\t *\n\t * text data is sometimes passed in the `data` and sometimes in the `dataTransfer` property.\n\t *\n\t * * If `InputEvent#data` was set, this property reflects its value.\n\t * * If `InputEvent#data` is unavailable, this property contains the `'text/plain'` data from\n\t * {@link module:engine/view/observer/inputobserver~InputEventData#dataTransfer}.\n\t * * If the event ({@link module:engine/view/observer/inputobserver~InputEventData#inputType input type})\n\t * provides no data whatsoever, this property is `null`.\n\t */\n\treadonly data: string | null;\n\n\t/**\n\t * The data transfer instance of the input event. Corresponds to native `InputEvent#dataTransfer`.\n\t *\n\t * The value is `null` when no `dataTransfer` was passed along with the input event.\n\t */\n\treadonly dataTransfer: DataTransfer;\n\n\t/**\n\t * A flag indicating that the `beforeinput` event was fired during composition.\n\t *\n\t * Corresponds to the\n\t * {@link module:engine/view/document~Document#event:compositionstart},\n\t * {@link module:engine/view/document~Document#event:compositionupdate},\n\t * and {@link module:engine/view/document~Document#event:compositionend } trio.\n\t */\n\treadonly isComposing: boolean;\n\n\t/**\n\t * Editing {@link module:engine/view/range~Range view ranges} corresponding to DOM ranges provided by the web browser\n\t * (as returned by `InputEvent#getTargetRanges()`).\n\t */\n\treadonly targetRanges: Array<ViewRange>;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/arrowkeysobserver\n */\n\nimport Observer from './observer';\nimport BubblingEventInfo from './bubblingeventinfo';\nimport type View from '../view';\nimport type { KeyEventData, ViewDocumentKeyDownEvent } from './keyobserver';\nimport type { BubblingEvent } from './bubblingemittermixin';\n\nimport { isArrowKeyCode } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Arrow keys observer introduces the {@link module:engine/view/document~Document#event:arrowKey `Document#arrowKey`} event.\n *\n * Note that this observer is attached by the {@link module:engine/view/view~View} and is available by default.\n */\nexport default class ArrowKeysObserver extends Observer {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tthis.document.on<ViewDocumentKeyDownEvent>( 'keydown', ( event, data ) => {\n\t\t\tif ( this.isEnabled && isArrowKeyCode( data.keyCode ) ) {\n\t\t\t\tconst eventInfo = new BubblingEventInfo( this.document, 'arrowKey', this.document.selection.getFirstRange()! );\n\n\t\t\t\tthis.document.fire<ViewDocumentArrowKeyEvent>( eventInfo, data );\n\n\t\t\t\tif ( eventInfo.stop.called ) {\n\t\t\t\t\tevent.stop();\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override observe(): void {}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override stopObserving(): void {}\n}\n\n/**\n * Event fired when the user presses an arrow keys.\n *\n * Introduced by {@link module:engine/view/observer/arrowkeysobserver~ArrowKeysObserver}.\n *\n * Note that because {@link module:engine/view/observer/arrowkeysobserver~ArrowKeysObserver} is attached by the\n * {@link module:engine/view/view~View} this event is available by default.\n *\n * @eventName module:engine/view/document~Document#arrowKey\n * @param data\n */\n\nexport type ViewDocumentArrowKeyEvent = BubblingEvent<{\n\tname: 'arrowKey';\n\targs: [ data: KeyEventData ];\n}>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/tabobserver\n */\n\nimport type View from '../view';\nimport Observer from './observer';\nimport BubblingEventInfo from './bubblingeventinfo';\nimport type { KeyEventData, ViewDocumentKeyDownEvent } from './keyobserver';\nimport type { BubblingEvent } from './bubblingemittermixin';\n\nimport { keyCodes } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Tab observer introduces the {@link module:engine/view/document~Document#event:tab `Document#tab`} event.\n *\n * Note that because {@link module:engine/view/observer/tabobserver~TabObserver} is attached by the\n * {@link module:engine/view/view~View}, this event is available by default.\n */\nexport default class TabObserver extends Observer {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tconst doc = this.document;\n\n\t\tdoc.on<ViewDocumentKeyDownEvent>( 'keydown', ( evt, data ) => {\n\t\t\tif (\n\t\t\t\t!this.isEnabled ||\n\t\t\t\tdata.keyCode != keyCodes.tab ||\n\t\t\t\tdata.ctrlKey\n\t\t\t) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst event = new BubblingEventInfo( doc, 'tab', doc.selection.getFirstRange()! );\n\n\t\t\tdoc.fire<ViewDocumentTabEvent>( event, data );\n\n\t\t\tif ( event.stop.called ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override observe(): void {}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override stopObserving(): void {}\n}\n\n/**\n * Event fired when the user presses a tab key.\n *\n * Introduced by {@link module:engine/view/observer/tabobserver~TabObserver}.\n *\n * Note that because {@link module:engine/view/observer/tabobserver~TabObserver} is attached by the\n * {@link module:engine/view/view~View}, this event is available by default.\n *\n * @eventName module:engine/view/document~Document#tab\n * @param data\n */\nexport type ViewDocumentTabEvent = BubblingEvent<{\n\tname: 'tab';\n\targs: [ data: KeyEventData ];\n}>;\n","import baseClone from './_baseClone.js';\n\n/** Used to compose bitmasks for cloning. */\nvar CLONE_DEEP_FLAG = 1,\n CLONE_SYMBOLS_FLAG = 4;\n\n/**\n * This method is like `_.clone` except that it recursively clones `value`.\n *\n * @static\n * @memberOf _\n * @since 1.0.0\n * @category Lang\n * @param {*} value The value to recursively clone.\n * @returns {*} Returns the deep cloned value.\n * @see _.clone\n * @example\n *\n * var objects = [{ 'a': 1 }, { 'b': 2 }];\n *\n * var deep = _.cloneDeep(objects);\n * console.log(deep[0] === objects[0]);\n * // => false\n */\nfunction cloneDeep(value) {\n return baseClone(value, CLONE_DEEP_FLAG | CLONE_SYMBOLS_FLAG);\n}\n\nexport default cloneDeep;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/view\n */\n\nimport Document, { type ViewDocumentLayoutChangedEvent } from './document';\nimport DowncastWriter from './downcastwriter';\nimport Renderer from './renderer';\nimport DomConverter from './domconverter';\nimport Position, { type PositionOffset } from './position';\nimport Range from './range';\nimport Selection, {\n\ttype PlaceOrOffset,\n\ttype Selectable,\n\ttype SelectionOptions\n} from './selection';\n\nimport type { default as Observer, ObserverConstructor } from './observer/observer';\nimport type { ViewDocumentSelectionChangeEvent } from './documentselection';\nimport type { StylesProcessor } from './stylesmap';\nimport type Element from './element';\nimport type Node from './node';\nimport type Item from './item';\n\nimport KeyObserver from './observer/keyobserver';\nimport FakeSelectionObserver from './observer/fakeselectionobserver';\nimport MutationObserver from './observer/mutationobserver';\nimport SelectionObserver from './observer/selectionobserver';\nimport FocusObserver, { type ViewDocumentBlurEvent } from './observer/focusobserver';\nimport CompositionObserver from './observer/compositionobserver';\nimport InputObserver from './observer/inputobserver';\nimport ArrowKeysObserver from './observer/arrowkeysobserver';\nimport TabObserver from './observer/tabobserver';\n\nimport {\n\tCKEditorError,\n\tenv,\n\tObservableMixin,\n\tscrollViewportToShowTarget,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\nimport { injectUiElementHandling } from './uielement';\nimport { injectQuirksHandling } from './filler';\n\nimport { cloneDeep } from 'lodash-es';\n\ntype IfTrue<T> = T extends true ? true : never;\ntype DomRange = globalThis.Range;\n\n/**\n * Editor's view controller class. Its main responsibility is DOM - View management for editing purposes, to provide\n * abstraction over the DOM structure and events and hide all browsers quirks.\n *\n * View controller renders view document to DOM whenever view structure changes. To determine when view can be rendered,\n * all changes need to be done using the {@link module:engine/view/view~View#change} method, using\n * {@link module:engine/view/downcastwriter~DowncastWriter}:\n *\n * ```ts\n * view.change( writer => {\n * \twriter.insert( position, writer.createText( 'foo' ) );\n * } );\n * ```\n *\n * View controller also register {@link module:engine/view/observer/observer~Observer observers} which observes changes\n * on DOM and fire events on the {@link module:engine/view/document~Document Document}.\n * Note that the following observers are added by the class constructor and are always available:\n *\n * * {@link module:engine/view/observer/selectionobserver~SelectionObserver},\n * * {@link module:engine/view/observer/focusobserver~FocusObserver},\n * * {@link module:engine/view/observer/keyobserver~KeyObserver},\n * * {@link module:engine/view/observer/fakeselectionobserver~FakeSelectionObserver}.\n * * {@link module:engine/view/observer/compositionobserver~CompositionObserver}.\n * * {@link module:engine/view/observer/inputobserver~InputObserver}.\n * * {@link module:engine/view/observer/arrowkeysobserver~ArrowKeysObserver}.\n * * {@link module:engine/view/observer/tabobserver~TabObserver}.\n *\n * This class also {@link module:engine/view/view~View#attachDomRoot binds the DOM and the view elements}.\n *\n * If you do not need full a DOM - view management, and only want to transform a tree of view elements to a tree of DOM\n * elements you do not need this controller. You can use the {@link module:engine/view/domconverter~DomConverter DomConverter} instead.\n */\nexport default class View extends ObservableMixin() {\n\t/**\n\t * Instance of the {@link module:engine/view/document~Document} associated with this view controller.\n\t */\n\tpublic readonly document: Document;\n\n\t/**\n\t * Instance of the {@link module:engine/view/domconverter~DomConverter domConverter} used by\n\t * {@link module:engine/view/view~View#_renderer renderer}\n\t * and {@link module:engine/view/observer/observer~Observer observers}.\n\t */\n\tpublic readonly domConverter: DomConverter;\n\n\t/**\n\t * Roots of the DOM tree. Map on the `HTMLElement`s with roots names as keys.\n\t */\n\tpublic readonly domRoots: Map<string, HTMLElement> = new Map();\n\n\t/**\n\t * Used to prevent calling {@link #forceRender} and {@link #change} during rendering view to the DOM.\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public isRenderingInProgress: boolean;\n\n\t/**\n\t * Informs whether the DOM selection is inside any of the DOM roots managed by the view.\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public hasDomSelection: boolean;\n\n\t/**\n\t * Instance of the {@link module:engine/view/renderer~Renderer renderer}.\n\t *\n\t * @internal\n\t */\n\tpublic readonly _renderer: Renderer;\n\n\t/**\n\t * A DOM root attributes cache. It saves the initial values of DOM root attributes before the DOM element\n\t * is {@link module:engine/view/view~View#attachDomRoot attached} to the view so later on, when\n\t * the view is destroyed ({@link module:engine/view/view~View#detachDomRoot}), they can be easily restored.\n\t * This way, the DOM element can go back to the (clean) state as if the editing view never used it.\n\t */\n\tprivate readonly _initialDomRootAttributes: WeakMap<HTMLElement, Record<string, string>> = new WeakMap();\n\n\t/**\n\t * Map of registered {@link module:engine/view/observer/observer~Observer observers}.\n\t */\n\tprivate readonly _observers: Map<ObserverConstructor, Observer> = new Map();\n\n\t/**\n\t * DowncastWriter instance used in {@link #change change method} callbacks.\n\t */\n\tprivate readonly _writer: DowncastWriter;\n\n\t/**\n\t * Is set to `true` when {@link #change view changes} are currently in progress.\n\t */\n\tprivate _ongoingChange: boolean = false;\n\n\t/**\n\t * Used to prevent calling {@link #forceRender} and {@link #change} during rendering view to the DOM.\n\t */\n\tprivate _postFixersInProgress: boolean = false;\n\n\t/**\n\t * Internal flag to temporary disable rendering. See the usage in the {@link #_disableRendering}.\n\t */\n\tprivate _renderingDisabled: boolean = false;\n\n\t/**\n\t * Internal flag that disables rendering when there are no changes since the last rendering.\n\t * It stores information about changed selection and changed elements from attached document roots.\n\t */\n\tprivate _hasChangedSinceTheLastRendering: boolean = false;\n\n\t/**\n\t * @param stylesProcessor The styles processor instance.\n\t */\n\tconstructor( stylesProcessor: StylesProcessor ) {\n\t\tsuper();\n\n\t\tthis.document = new Document( stylesProcessor );\n\t\tthis.domConverter = new DomConverter( this.document );\n\n\t\tthis.set( 'isRenderingInProgress', false );\n\t\tthis.set( 'hasDomSelection', false );\n\n\t\tthis._renderer = new Renderer( this.domConverter, this.document.selection );\n\t\tthis._renderer.bind( 'isFocused', 'isSelecting', 'isComposing' )\n\t\t\t.to( this.document, 'isFocused', 'isSelecting', 'isComposing' );\n\n\t\tthis._writer = new DowncastWriter( this.document );\n\n\t\t// Add default observers.\n\t\t// Make sure that this list matches AlwaysRegisteredObservers type.\n\t\tthis.addObserver( MutationObserver );\n\t\tthis.addObserver( FocusObserver );\n\t\tthis.addObserver( SelectionObserver );\n\t\tthis.addObserver( KeyObserver );\n\t\tthis.addObserver( FakeSelectionObserver );\n\t\tthis.addObserver( CompositionObserver );\n\t\tthis.addObserver( ArrowKeysObserver );\n\t\tthis.addObserver( InputObserver );\n\t\tthis.addObserver( TabObserver );\n\n\t\t// Inject quirks handlers.\n\t\tinjectQuirksHandling( this );\n\t\tinjectUiElementHandling( this );\n\n\t\t// Use 'normal' priority so that rendering is performed as first when using that priority.\n\t\tthis.on<ViewRenderEvent>( 'render', () => {\n\t\t\tthis._render();\n\n\t\t\t// Informs that layout has changed after render.\n\t\t\tthis.document.fire<ViewDocumentLayoutChangedEvent>( 'layoutChanged' );\n\n\t\t\t// Reset the `_hasChangedSinceTheLastRendering` flag after rendering.\n\t\t\tthis._hasChangedSinceTheLastRendering = false;\n\t\t} );\n\n\t\t// Listen to the document selection changes directly.\n\t\tthis.listenTo<ViewDocumentSelectionChangeEvent>( this.document.selection, 'change', () => {\n\t\t\tthis._hasChangedSinceTheLastRendering = true;\n\t\t} );\n\n\t\t// Trigger re-render if only the focus changed.\n\t\tthis.listenTo<ObservableChangeEvent>( this.document, 'change:isFocused', () => {\n\t\t\tthis._hasChangedSinceTheLastRendering = true;\n\t\t} );\n\n\t\t// Remove ranges from DOM selection if editor is blurred.\n\t\t// See https://github.com/ckeditor/ckeditor5/issues/5753.\n\t\tif ( env.isiOS ) {\n\t\t\tthis.listenTo<ViewDocumentBlurEvent>( this.document, 'blur', ( evt, data ) => {\n\t\t\t\tconst relatedViewElement = this.domConverter.mapDomToView( data.domEvent.relatedTarget as HTMLElement );\n\n\t\t\t\t// Do not modify DOM selection if focus is moved to other editable of the same editor.\n\t\t\t\tif ( !relatedViewElement ) {\n\t\t\t\t\tthis.domConverter._clearDomSelection();\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Attaches a DOM root element to the view element and enable all observers on that element.\n\t * Also {@link module:engine/view/renderer~Renderer#markToSync mark element} to be synchronized\n\t * with the view what means that all child nodes will be removed and replaced with content of the view root.\n\t *\n\t * This method also will change view element name as the same as tag name of given dom root.\n\t * Name is always transformed to lower case.\n\t *\n\t * **Note:** Use {@link #detachDomRoot `detachDomRoot()`} to revert this action.\n\t *\n\t * @param domRoot DOM root element.\n\t * @param name Name of the root.\n\t */\n\tpublic attachDomRoot( domRoot: HTMLElement, name: string = 'main' ): void {\n\t\tconst viewRoot = this.document.getRoot( name )!;\n\n\t\t// Set view root name the same as DOM root tag name.\n\t\tviewRoot._name = domRoot.tagName.toLowerCase();\n\n\t\tconst initialDomRootAttributes: Record<string, string> = {};\n\n\t\t// 1. Copy and cache the attributes to remember the state of the element before attaching.\n\t\t// The cached attributes will be restored in detachDomRoot() so the element goes to the\n\t\t// clean state as if the editing view never used it.\n\t\t// 2. Apply the attributes using the view writer, so they all go under the control of the engine.\n\t\t// The editing view takes over the attribute management completely because various\n\t\t// features (e.g. addPlaceholder()) require dynamic changes of those attributes and they\n\t\t// cannot be managed by the engine and the UI library at the same time.\n\t\tfor ( const { name, value } of Array.from( domRoot.attributes ) ) {\n\t\t\tinitialDomRootAttributes[ name ] = value;\n\n\t\t\t// Do not use writer.setAttribute() for the class attribute. The EditableUIView class\n\t\t\t// and its descendants could have already set some using the writer.addClass() on the view\n\t\t\t// document root. They haven't been rendered yet so they are not present in the DOM root.\n\t\t\t// Using writer.setAttribute( 'class', ... ) would override them completely.\n\t\t\tif ( name === 'class' ) {\n\t\t\t\tthis._writer.addClass( value.split( ' ' ), viewRoot );\n\t\t\t} else {\n\t\t\t\tthis._writer.setAttribute( name, value, viewRoot );\n\t\t\t}\n\t\t}\n\n\t\tthis._initialDomRootAttributes.set( domRoot, initialDomRootAttributes );\n\n\t\tconst updateContenteditableAttribute = () => {\n\t\t\tthis._writer.setAttribute( 'contenteditable', ( !viewRoot.isReadOnly ).toString(), viewRoot );\n\n\t\t\tif ( viewRoot.isReadOnly ) {\n\t\t\t\tthis._writer.addClass( 'ck-read-only', viewRoot );\n\t\t\t} else {\n\t\t\t\tthis._writer.removeClass( 'ck-read-only', viewRoot );\n\t\t\t}\n\t\t};\n\n\t\t// Set initial value.\n\t\tupdateContenteditableAttribute();\n\n\t\tthis.domRoots.set( name, domRoot );\n\t\tthis.domConverter.bindElements( domRoot, viewRoot );\n\t\tthis._renderer.markToSync( 'children', viewRoot );\n\t\tthis._renderer.markToSync( 'attributes', viewRoot );\n\t\tthis._renderer.domDocuments.add( domRoot.ownerDocument );\n\n\t\tviewRoot.on( 'change:children', ( evt, node ) => this._renderer.markToSync( 'children', node ) );\n\t\tviewRoot.on( 'change:attributes', ( evt, node ) => this._renderer.markToSync( 'attributes', node ) );\n\t\tviewRoot.on( 'change:text', ( evt, node ) => this._renderer.markToSync( 'text', node ) );\n\t\tviewRoot.on( 'change:isReadOnly', () => this.change( updateContenteditableAttribute ) );\n\n\t\tviewRoot.on( 'change', () => {\n\t\t\tthis._hasChangedSinceTheLastRendering = true;\n\t\t} );\n\n\t\tfor ( const observer of this._observers.values() ) {\n\t\t\tobserver.observe( domRoot, name );\n\t\t}\n\t}\n\n\t/**\n\t * Detaches a DOM root element from the view element and restores its attributes to the state before\n\t * {@link #attachDomRoot `attachDomRoot()`}.\n\t *\n\t * @param name Name of the root to detach.\n\t */\n\tpublic detachDomRoot( name: string ): void {\n\t\tconst domRoot = this.domRoots.get( name )!;\n\n\t\t// Remove all root attributes so the DOM element is \"bare\".\n\t\tArray.from( domRoot.attributes ).forEach( ( { name } ) => domRoot.removeAttribute( name ) );\n\n\t\tconst initialDomRootAttributes = this._initialDomRootAttributes.get( domRoot );\n\n\t\t// Revert all view root attributes back to the state before attachDomRoot was called.\n\t\tfor ( const attribute in initialDomRootAttributes ) {\n\t\t\tdomRoot.setAttribute( attribute, initialDomRootAttributes[ attribute ] );\n\t\t}\n\n\t\tthis.domRoots.delete( name );\n\t\tthis.domConverter.unbindDomElement( domRoot );\n\n\t\tfor ( const observer of this._observers.values() ) {\n\t\t\tobserver.stopObserving( domRoot );\n\t\t}\n\t}\n\n\t/**\n\t * Gets DOM root element.\n\t *\n\t * @param name Name of the root.\n\t * @returns DOM root element instance.\n\t */\n\tpublic getDomRoot( name: string = 'main' ): HTMLElement | undefined {\n\t\treturn this.domRoots.get( name );\n\t}\n\n\t/**\n\t * Creates observer of the given type if not yet created, {@link module:engine/view/observer/observer~Observer#enable enables} it\n\t * and {@link module:engine/view/observer/observer~Observer#observe attaches} to all existing and future\n\t * {@link #domRoots DOM roots}.\n\t *\n\t * Note: Observers are recognized by their constructor (classes). A single observer will be instantiated and used only\n\t * when registered for the first time. This means that features and other components can register a single observer\n\t * multiple times without caring whether it has been already added or not.\n\t *\n\t * @param ObserverConstructor The constructor of an observer to add.\n\t * Should create an instance inheriting from {@link module:engine/view/observer/observer~Observer}.\n\t * @returns Added observer instance.\n\t */\n\tpublic addObserver( ObserverConstructor: ObserverConstructor ): Observer {\n\t\tlet observer = this._observers.get( ObserverConstructor );\n\n\t\tif ( observer ) {\n\t\t\treturn observer;\n\t\t}\n\n\t\tobserver = new ObserverConstructor( this );\n\n\t\tthis._observers.set( ObserverConstructor, observer );\n\n\t\tfor ( const [ name, domElement ] of this.domRoots ) {\n\t\t\tobserver.observe( domElement, name );\n\t\t}\n\n\t\tobserver.enable();\n\n\t\treturn observer;\n\t}\n\n\tpublic getObserver<T extends ObserverConstructor>( ObserverConstructor: T ):\n\t\tT extends AlwaysRegisteredObservers ? InstanceType<T> : InstanceType<T> | undefined;\n\n\t/**\n\t * Returns observer of the given type or `undefined` if such observer has not been added yet.\n\t *\n\t * @param ObserverConstructor The constructor of an observer to get.\n\t * @returns Observer instance or undefined.\n\t */\n\tpublic getObserver<T extends ObserverConstructor>( ObserverConstructor: T ): InstanceType<T> | undefined {\n\t\treturn this._observers.get( ObserverConstructor ) as ( InstanceType<T> | undefined );\n\t}\n\n\t/**\n\t * Disables all added observers.\n\t */\n\tpublic disableObservers(): void {\n\t\tfor ( const observer of this._observers.values() ) {\n\t\t\tobserver.disable();\n\t\t}\n\t}\n\n\t/**\n\t * Enables all added observers.\n\t */\n\tpublic enableObservers(): void {\n\t\tfor ( const observer of this._observers.values() ) {\n\t\t\tobserver.enable();\n\t\t}\n\t}\n\n\t/**\n\t * Scrolls the page viewport and {@link #domRoots} with their ancestors to reveal the\n\t * caret, **if not already visible to the user**.\n\t *\n\t * **Note**: Calling this method fires the {@link module:engine/view/view~ViewScrollToTheSelectionEvent} event that\n\t * allows custom behaviors.\n\t *\n\t * @param options Additional configuration of the scrolling behavior.\n\t * @param options.viewportOffset A distance between the DOM selection and the viewport boundary to be maintained\n\t * while scrolling to the selection (default is 20px). Setting this value to `0` will reveal the selection precisely at\n\t * the viewport boundary.\n\t * @param options.ancestorOffset A distance between the DOM selection and scrollable DOM root ancestor(s) to be maintained\n\t * while scrolling to the selection (default is 20px). Setting this value to `0` will reveal the selection precisely at\n\t * the scrollable ancestor(s) boundary.\n\t * @param options.alignToTop When set `true`, the DOM selection will be aligned to the top of the viewport if not already visible\n\t * (see `forceScroll` to learn more).\n\t * @param options.forceScroll When set `true`, the DOM selection will be aligned to the top of the viewport and scrollable ancestors\n\t * whether it is already visible or not. This option will only work when `alignToTop` is `true`.\n\t */\n\tpublic scrollToTheSelection<T extends boolean, U extends IfTrue<T>>( {\n\t\talignToTop,\n\t\tforceScroll,\n\t\tviewportOffset = 20,\n\t\tancestorOffset = 20\n\t}: {\n\t\treadonly viewportOffset?: number | { top: number; bottom: number; left: number; right: number };\n\t\treadonly ancestorOffset?: number;\n\t\treadonly alignToTop?: T;\n\t\treadonly forceScroll?: U;\n\t} = {} ): void {\n\t\tconst range = this.document.selection.getFirstRange();\n\n\t\tif ( !range ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Clone to make sure properties like `viewportOffset` are not mutated in the event listeners.\n\t\tconst originalArgs = cloneDeep( { alignToTop, forceScroll, viewportOffset, ancestorOffset } );\n\n\t\tif ( typeof viewportOffset === 'number' ) {\n\t\t\tviewportOffset = {\n\t\t\t\ttop: viewportOffset,\n\t\t\t\tbottom: viewportOffset,\n\t\t\t\tleft: viewportOffset,\n\t\t\t\tright: viewportOffset\n\t\t\t};\n\t\t}\n\n\t\tconst options = {\n\t\t\ttarget: this.domConverter.viewRangeToDom( range ),\n\t\t\tviewportOffset,\n\t\t\tancestorOffset,\n\t\t\talignToTop,\n\t\t\tforceScroll\n\t\t};\n\n\t\tthis.fire<ViewScrollToTheSelectionEvent>( 'scrollToTheSelection', options, originalArgs );\n\n\t\tscrollViewportToShowTarget( options );\n\t}\n\n\t/**\n\t * It will focus DOM element representing {@link module:engine/view/editableelement~EditableElement EditableElement}\n\t * that is currently having selection inside.\n\t */\n\tpublic focus(): void {\n\t\tif ( !this.document.isFocused ) {\n\t\t\tconst editable = this.document.selection.editableElement;\n\n\t\t\tif ( editable ) {\n\t\t\t\tthis.domConverter.focus( editable );\n\t\t\t\tthis.forceRender();\n\t\t\t} else {\n\t\t\t\t// Before focusing view document, selection should be placed inside one of the view's editables.\n\t\t\t\t// Normally its selection will be converted from model document (which have default selection), but\n\t\t\t\t// when using view document on its own, we need to manually place selection before focusing it.\n\t\t\t\t//\n\t\t\t\t// @if CK_DEBUG // console.warn( 'There is no selection in any editable to focus.' );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * The `change()` method is the primary way of changing the view. You should use it to modify any node in the view tree.\n\t * It makes sure that after all changes are made the view is rendered to the DOM (assuming that the view will be changed\n\t * inside the callback). It prevents situations when the DOM is updated when the view state is not yet correct. It allows\n\t * to nest calls one inside another and still performs a single rendering after all those changes are made.\n\t * It also returns the return value of its callback.\n\t *\n\t * ```ts\n\t * const text = view.change( writer => {\n\t * \tconst newText = writer.createText( 'foo' );\n\t * \twriter.insert( position1, newText );\n\t *\n\t * \tview.change( writer => {\n\t * \t\twriter.insert( position2, writer.createText( 'bar' ) );\n\t * \t} );\n\t *\n\t * \twriter.remove( range );\n\t *\n\t * \treturn newText;\n\t * } );\n\t * ```\n\t *\n\t * When the outermost change block is done and rendering to the DOM is over the\n\t * {@link module:engine/view/view~View#event:render `View#render`} event is fired.\n\t *\n\t * This method throws a `applying-view-changes-on-rendering` error when\n\t * the change block is used after rendering to the DOM has started.\n\t *\n\t * @param callback Callback function which may modify the view.\n\t * @returns Value returned by the callback.\n\t */\n\tpublic change<TReturn>( callback: ( writer: DowncastWriter ) => TReturn ): TReturn {\n\t\tif ( this.isRenderingInProgress || this._postFixersInProgress ) {\n\t\t\t/**\n\t\t\t * Thrown when there is an attempt to make changes to the view tree when it is in incorrect state. This may\n\t\t\t * cause some unexpected behaviour and inconsistency between the DOM and the view.\n\t\t\t * This may be caused by:\n\t\t\t *\n\t\t\t * * calling {@link module:engine/view/view~View#change} or {@link module:engine/view/view~View#forceRender} during rendering\n\t\t\t * process,\n\t\t\t * * calling {@link module:engine/view/view~View#change} or {@link module:engine/view/view~View#forceRender} inside of\n\t\t\t * {@link module:engine/view/document~Document#registerPostFixer post-fixer function}.\n\t\t\t *\n\t\t\t * @error cannot-change-view-tree\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'cannot-change-view-tree',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\ttry {\n\t\t\t// Recursive call to view.change() method - execute listener immediately.\n\t\t\tif ( this._ongoingChange ) {\n\t\t\t\treturn callback( this._writer );\n\t\t\t}\n\n\t\t\t// This lock will assure that all recursive calls to view.change() will end up in same block - one \"render\"\n\t\t\t// event for all nested calls.\n\t\t\tthis._ongoingChange = true;\n\t\t\tconst callbackResult = callback( this._writer );\n\t\t\tthis._ongoingChange = false;\n\n\t\t\t// This lock is used by editing controller to render changes from outer most model.change() once. As plugins might call\n\t\t\t// view.change() inside model.change() block - this will ensures that postfixers and rendering are called once after all\n\t\t\t// changes. Also, we don't need to render anything if there're no changes since last rendering.\n\t\t\tif ( !this._renderingDisabled && this._hasChangedSinceTheLastRendering ) {\n\t\t\t\tthis._postFixersInProgress = true;\n\t\t\t\tthis.document._callPostFixers( this._writer );\n\t\t\t\tthis._postFixersInProgress = false;\n\n\t\t\t\tthis.fire<ViewRenderEvent>( 'render' );\n\t\t\t}\n\n\t\t\treturn callbackResult;\n\t\t} catch ( err: any ) {\n\t\t\t// @if CK_DEBUG // throw err;\n\t\t\t/* istanbul ignore next -- @preserve */\n\t\t\tCKEditorError.rethrowUnexpectedError( err, this );\n\t\t}\n\t}\n\n\t/**\n\t * Forces rendering {@link module:engine/view/document~Document view document} to DOM. If any view changes are\n\t * currently in progress, rendering will start after all {@link #change change blocks} are processed.\n\t *\n\t * Note that this method is dedicated for special cases. All view changes should be wrapped in the {@link #change}\n\t * block and the view will automatically check whether it needs to render DOM or not.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `applying-view-changes-on-rendering` when\n\t * trying to re-render when rendering to DOM has already started.\n\t */\n\tpublic forceRender(): void {\n\t\tthis._hasChangedSinceTheLastRendering = true;\n\t\tthis.getObserver( FocusObserver ).flush();\n\t\tthis.change( () => {} );\n\t}\n\n\t/**\n\t * Destroys this instance. Makes sure that all observers are destroyed and listeners removed.\n\t */\n\tpublic destroy(): void {\n\t\tfor ( const observer of this._observers.values() ) {\n\t\t\tobserver.destroy();\n\t\t}\n\n\t\tthis.document.destroy();\n\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Creates position at the given location. The location can be specified as:\n\t *\n\t * * a {@link module:engine/view/position~Position position},\n\t * * parent element and offset (offset defaults to `0`),\n\t * * parent element and `'end'` (sets position at the end of that element),\n\t * * {@link module:engine/view/item~Item view item} and `'before'` or `'after'` (sets position before or after given view item).\n\t *\n\t * This method is a shortcut to other constructors such as:\n\t *\n\t * * {@link #createPositionBefore},\n\t * * {@link #createPositionAfter},\n\t *\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/view/item~Item view item}.\n\t */\n\tpublic createPositionAt( itemOrPosition: Item | Position, offset?: PositionOffset ): Position {\n\t\treturn Position._createAt( itemOrPosition, offset );\n\t}\n\n\t/**\n\t * Creates a new position after given view item.\n\t *\n\t * @param item View item after which the position should be located.\n\t */\n\tpublic createPositionAfter( item: Item ): Position {\n\t\treturn Position._createAfter( item );\n\t}\n\n\t/**\n\t * Creates a new position before given view item.\n\t *\n\t * @param item View item before which the position should be located.\n\t */\n\tpublic createPositionBefore( item: Item ): Position {\n\t\treturn Position._createBefore( item );\n\t}\n\n\t/**\n\t * Creates a range spanning from `start` position to `end` position.\n\t *\n\t * **Note:** This factory method creates it's own {@link module:engine/view/position~Position} instances basing on passed values.\n\t *\n\t * @param start Start position.\n\t * @param end End position. If not set, range will be collapsed at `start` position.\n\t */\n\tpublic createRange( start: Position, end?: Position | null ): Range {\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Creates a range that starts before given {@link module:engine/view/item~Item view item} and ends after it.\n\t */\n\tpublic createRangeOn( item: Item ): Range {\n\t\treturn Range._createOn( item );\n\t}\n\n\t/**\n\t * Creates a range inside an {@link module:engine/view/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * @param element Element which is a parent for the range.\n\t */\n\tpublic createRangeIn( element: Element ): Range {\n\t\treturn Range._createIn( element );\n\t}\n\n\t/**\n\t * Creates new {@link module:engine/view/selection~Selection} instance.\n\t *\n\t * ```ts\n\t * // Creates collapsed selection at the position of given item and offset.\n\t * const paragraph = view.createContainerElement( 'paragraph' );\n\t * const selection = view.createSelection( paragraph, offset );\n\t *\n\t * // Creates a range inside an {@link module:engine/view/element~Element element} which starts before the\n\t * // first child of that element and ends after the last child of that element.\n\t * const selection = view.createSelection( paragraph, 'in' );\n\t *\n\t * // Creates a range on an {@link module:engine/view/item~Item item} which starts before the item and ends\n\t * // just after the item.\n\t * const selection = view.createSelection( paragraph, 'on' );\n\t * ```\n\t *\n\t * `Selection`'s factory method allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = view.createSelection( paragraph, 'in', { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = view.createSelection( element, 'in', { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #createSelection:SELECTABLE `createSelection( selectable, options )`}.\n\t *\n\t * @label NODE_OFFSET\n\t */\n\tpublic createSelection( selectable: Node, placeOrOffset: PlaceOrOffset, options?: SelectionOptions ): Selection;\n\n\t/**\n\t * Creates new {@link module:engine/view/selection~Selection} instance.\n\t *\n\t * ```ts\n\t * // Creates empty selection without ranges.\n\t * const selection = view.createSelection();\n\t *\n\t * // Creates selection at the given range.\n\t * const range = view.createRange( start, end );\n\t * const selection = view.createSelection( range );\n\t *\n\t * // Creates selection at the given ranges\n\t * const ranges = [ view.createRange( start1, end2 ), view.createRange( star2, end2 ) ];\n\t * const selection = view.createSelection( ranges );\n\t *\n\t * // Creates selection from the other selection.\n\t * const otherSelection = view.createSelection();\n\t * const selection = view.createSelection( otherSelection );\n\t *\n\t * // Creates selection from the document selection.\n\t * const selection = view.createSelection( editor.editing.view.document.selection );\n\t *\n\t * // Creates selection at the given position.\n\t * const position = view.createPositionFromPath( root, path );\n\t * const selection = view.createSelection( position );\n\t * ```\n\t *\n\t * `Selection`'s factory method allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = view.createSelection( range, { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = view.createSelection( range, { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #createSelection:NODE_OFFSET `createSelection( node, placeOrOffset, options )`}.\n\t *\n\t * @label SELECTABLE\n\t */\n\tpublic createSelection( selectable?: Exclude<Selectable, Node>, options?: SelectionOptions ): Selection;\n\n\tpublic createSelection( ...args: ConstructorParameters<typeof Selection> ): Selection {\n\t\treturn new Selection( ...args );\n\t}\n\n\t/**\n\t * Disables or enables rendering. If the flag is set to `true` then the rendering will be disabled.\n\t * If the flag is set to `false` and if there was some change in the meantime, then the rendering action will be performed.\n\t *\n\t * @internal\n\t * @param flag A flag indicates whether the rendering should be disabled.\n\t */\n\tpublic _disableRendering( flag: boolean ): void {\n\t\tthis._renderingDisabled = flag;\n\n\t\tif ( flag == false ) {\n\t\t\t// Render when you stop blocking rendering.\n\t\t\tthis.change( () => {} );\n\t\t}\n\t}\n\n\t/**\n\t * Renders all changes. In order to avoid triggering the observers (e.g. selection) all observers are disabled\n\t * before rendering and re-enabled after that.\n\t */\n\tprivate _render(): void {\n\t\tthis.isRenderingInProgress = true;\n\t\tthis.disableObservers();\n\t\tthis._renderer.render();\n\t\tthis.enableObservers();\n\t\tthis.isRenderingInProgress = false;\n\t}\n}\n\n/**\n * Fired after a topmost {@link module:engine/view/view~View#change change block} and all\n * {@link module:engine/view/document~Document#registerPostFixer post-fixers} are executed.\n *\n * Actual rendering is performed as a first listener on 'normal' priority.\n *\n * ```ts\n * view.on( 'render', () => {\n * \t// Rendering to the DOM is complete.\n * } );\n * ```\n *\n * This event is useful when you want to update interface elements after the rendering, e.g. position of the\n * balloon panel. If you wants to change view structure use\n * {@link module:engine/view/document~Document#registerPostFixer post-fixers}.\n *\n * @eventName ~View#render\n */\nexport type ViewRenderEvent = {\n\tname: 'render';\n\targs: [];\n};\n\n/**\n * An event fired at the moment of {@link module:engine/view/view~View#scrollToTheSelection} being called. It\n * carries two objects in its payload (`args`):\n *\n * * The first argument is the {@link module:engine/view/view~ViewScrollToTheSelectionEventData object containing data} that gets\n * passed down to the {@link module:utils/dom/scroll~scrollViewportToShowTarget} helper. If some event listener modifies it, it can\n * adjust the behavior of the scrolling (e.g. include additional `viewportOffset`).\n * * The second argument corresponds to the original arguments passed to {@link module:utils/dom/scroll~scrollViewportToShowTarget}.\n * It allows listeners to re-execute the `scrollViewportToShowTarget()` method with its original arguments if there is such a need,\n * for instance, if the integration requires rescrolling after certain interaction.\n *\n * @eventName ~View#scrollToTheSelection\n */\nexport type ViewScrollToTheSelectionEvent = {\n\tname: 'scrollToTheSelection';\n\targs: [\n\t\tViewScrollToTheSelectionEventData,\n\t\tParameters<View[ 'scrollToTheSelection' ]>[ 0 ]\n\t];\n};\n\n/**\n * An object passed down to the {@link module:utils/dom/scroll~scrollViewportToShowTarget} helper while calling\n * {@link module:engine/view/view~View#scrollToTheSelection}.\n */\nexport type ViewScrollToTheSelectionEventData = {\n\ttarget: DomRange;\n\tviewportOffset: { top: number; bottom: number; left: number; right: number };\n\tancestorOffset: number;\n\talignToTop?: boolean;\n\tforceScroll?: boolean;\n};\n\n/**\n * Observers that are always registered.\n */\nexport type AlwaysRegisteredObservers =\n\t| typeof MutationObserver\n\t| typeof FocusObserver\n\t| typeof SelectionObserver\n\t| typeof KeyObserver\n\t| typeof FakeSelectionObserver\n\t| typeof CompositionObserver\n\t| typeof ArrowKeysObserver\n\t| typeof InputObserver\n\t| typeof TabObserver;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/typecheckable\n */\n\nimport type { Marker } from './markercollection';\nimport type DocumentFragment from './documentfragment';\nimport type DocumentSelection from './documentselection';\nimport type Element from './element';\nimport type LivePosition from './liveposition';\nimport type LiveRange from './liverange';\nimport type Node from './node';\nimport type Position from './position';\nimport type Range from './range';\nimport type RootElement from './rootelement';\nimport type Selection from './selection';\nimport type Text from './text';\nimport type TextProxy from './textproxy';\n\nexport default abstract class TypeCheckable {\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/node~Node} or its subclass.\n\t *\n\t * This method is useful when processing model objects that are of unknown type. For example, a function\n\t * may return a {@link module:engine/model/documentfragment~DocumentFragment} or a {@link module:engine/model/node~Node}\n\t * that can be either a text node or an element. This method can be used to check what kind of object is returned.\n\t *\n\t * ```ts\n\t * someObject.is( 'element' ); // -> true if this is an element\n\t * someObject.is( 'node' ); // -> true if this is a node (a text node or an element)\n\t * someObject.is( 'documentFragment' ); // -> true if this is a document fragment\n\t * ```\n\t *\n\t * Since this method is also available on a range of view objects, you can prefix the type of the object with\n\t * `model:` or `view:` to check, for example, if this is the model's or view's element:\n\t *\n\t * ```ts\n\t * modelElement.is( 'model:element' ); // -> true\n\t * modelElement.is( 'view:element' ); // -> false\n\t * ```\n\t *\n\t * By using this method it is also possible to check a name of an element:\n\t *\n\t * ```ts\n\t * imageElement.is( 'element', 'imageBlock' ); // -> true\n\t * imageElement.is( 'element', 'imageBlock' ); // -> same as above\n\t * imageElement.is( 'model:element', 'imageBlock' ); // -> same as above, but more precise\n\t * ```\n\t *\n\t * @label NODE\n\t */\n\tpublic is( type: 'node' | 'model:node' ): this is Node | Element | Text | RootElement;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/element~Element} or its subclass.\n\t *\n\t * ```ts\n\t * element.is( 'element' ); // -> true\n\t * element.is( 'node' ); // -> true\n\t * element.is( 'model:element' ); // -> true\n\t * element.is( 'model:node' ); // -> true\n\t *\n\t * element.is( 'view:element' ); // -> false\n\t * element.is( 'documentSelection' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is an element, you can also check its\n\t * {@link module:engine/model/element~Element#name name}:\n\t *\n\t * ```ts\n\t * element.is( 'element', 'imageBlock' ); // -> true if this is an <imageBlock> element\n\t * text.is( 'element', 'imageBlock' ); -> false\n\t * ```\n\t *\n\t * @label ELEMENT\n\t */\n\tpublic is( type: 'element' | 'model:element' ): this is Element | RootElement;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/rootelement~RootElement}.\n\t *\n\t * ```ts\n\t * rootElement.is( 'rootElement' ); // -> true\n\t * rootElement.is( 'element' ); // -> true\n\t * rootElement.is( 'node' ); // -> true\n\t * rootElement.is( 'model:rootElement' ); // -> true\n\t * rootElement.is( 'model:element' ); // -> true\n\t * rootElement.is( 'model:node' ); // -> true\n\t *\n\t * rootElement.is( 'view:element' ); // -> false\n\t * rootElement.is( 'documentFragment' ); // -> false\n\t * ```\n\t *\n\t * Assuming that the object being checked is an element, you can also check its\n\t * {@link module:engine/model/element~Element#name name}:\n\t *\n\t * ```ts\n\t * rootElement.is( 'rootElement', '$root' ); // -> same as above\n\t * ```\n\t *\n\t * @label ROOT_ELEMENT\n\t */\n\tpublic is( type: 'rootElement' | 'model:rootElement' ): this is RootElement;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/text~Text}.\n\t *\n\t * ```ts\n\t * text.is( '$text' ); // -> true\n\t * text.is( 'node' ); // -> true\n\t * text.is( 'model:$text' ); // -> true\n\t * text.is( 'model:node' ); // -> true\n\t *\n\t * text.is( 'view:$text' ); // -> false\n\t * text.is( 'documentSelection' ); // -> false\n\t * ```\n\t *\n\t * **Note:** Until version 20.0.0 this method wasn't accepting `'$text'` type. The legacy `'text'` type is still\n\t * accepted for backward compatibility.\n\t *\n\t * @label TEXT\n\t */\n\tpublic is( type: '$text' | 'model:$text' ): this is Text;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/position~Position} or its subclass.\n\t *\n\t * ```ts\n\t * position.is( 'position' ); // -> true\n\t * position.is( 'model:position' ); // -> true\n\t *\n\t * position.is( 'view:position' ); // -> false\n\t * position.is( 'documentSelection' ); // -> false\n\t * ```\n\t *\n\t * @label POSITION\n\t */\n\tpublic is( type: 'position' | 'model:position' ): this is Position | LivePosition;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/liveposition~LivePosition}.\n\t *\n\t * ```ts\n\t * livePosition.is( 'position' ); // -> true\n\t * livePosition.is( 'model:position' ); // -> true\n\t * livePosition.is( 'liveposition' ); // -> true\n\t * livePosition.is( 'model:livePosition' ); // -> true\n\t *\n\t * livePosition.is( 'view:position' ); // -> false\n\t * livePosition.is( 'documentSelection' ); // -> false\n\t * ```\n\t *\n\t * @label LIVE_POSITION\n\t */\n\tpublic is( type: 'livePosition' | 'model:livePosition' ): this is LivePosition;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/range~Range} or its subclass.\n\t *\n\t * ```ts\n\t * range.is( 'range' ); // -> true\n\t * range.is( 'model:range' ); // -> true\n\t *\n\t * range.is( 'view:range' ); // -> false\n\t * range.is( 'documentSelection' ); // -> false\n\t * ```\n\t *\n\t * @label RANGE\n\t */\n\tpublic is( type: 'range' | 'model:range' ): this is Range | LiveRange;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/liverange~LiveRange}.\n\t *\n\t * ```ts\n\t * liveRange.is( 'range' ); // -> true\n\t * liveRange.is( 'model:range' ); // -> true\n\t * liveRange.is( 'liveRange' ); // -> true\n\t * liveRange.is( 'model:liveRange' ); // -> true\n\t *\n\t * liveRange.is( 'view:range' ); // -> false\n\t * liveRange.is( 'documentSelection' ); // -> false\n\t * ```\n\t *\n\t * @label LIVE_RANGE\n\t */\n\tpublic is( type: 'liveRange' | 'model:liveRange' ): this is LiveRange;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/documentfragment~DocumentFragment}.\n\t *\n\t * ```ts\n\t * docFrag.is( 'documentFragment' ); // -> true\n\t * docFrag.is( 'model:documentFragment' ); // -> true\n\t *\n\t * docFrag.is( 'view:documentFragment' ); // -> false\n\t * docFrag.is( 'element' ); // -> false\n\t * docFrag.is( 'node' ); // -> false\n\t * ```\n\t *\n\t * @label DOCUMENT_FRAGMENT\n\t */\n\tpublic is( type: 'documentFragment' | 'model:documentFragment' ): this is DocumentFragment;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/selection~Selection}\n\t * or {@link module:engine/model/documentselection~DocumentSelection}.\n\t *\n\t * ```ts\n\t * selection.is( 'selection' ); // -> true\n\t * selection.is( 'model:selection' ); // -> true\n\t *\n\t * selection.is( 'view:selection' ); // -> false\n\t * selection.is( 'range' ); // -> false\n\t * ```\n\t *\n\t * @label SELECTION\n\t */\n\tpublic is( type: 'selection' | 'model:selection' ): this is Selection | DocumentSelection;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/documentselection~DocumentSelection}.\n\t *\n\t * ```ts\n\t * selection.is( 'selection' ); // -> true\n\t * selection.is( 'documentSelection' ); // -> true\n\t * selection.is( 'model:selection' ); // -> true\n\t * selection.is( 'model:documentSelection' ); // -> true\n\t *\n\t * selection.is( 'view:selection' ); // -> false\n\t * selection.is( 'element' ); // -> false\n\t * selection.is( 'node' ); // -> false\n\t * ```\n\t *\n\t * @label DOCUMENT_SELECTION\n\t */\n\tpublic is( type: 'documentSelection' | 'model:documentSelection' ): this is DocumentSelection;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/markercollection~Marker}.\n\t *\n\t * ```ts\n\t * marker.is( 'marker' ); // -> true\n\t * marker.is( 'model:marker' ); // -> true\n\t *\n\t * marker.is( 'view:element' ); // -> false\n\t * marker.is( 'documentSelection' ); // -> false\n\t * ```\n\t *\n\t * @label MARKER\n\t */\n\tpublic is( type: 'marker' | 'model:marker' ): this is Marker;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/textproxy~TextProxy}.\n\t *\n\t * ```ts\n\t * textProxy.is( '$textProxy' ); // -> true\n\t * textProxy.is( 'model:$textProxy' ); // -> true\n\t *\n\t * textProxy.is( 'view:$textProxy' ); // -> false\n\t * textProxy.is( 'range' ); // -> false\n\t * ```\n\t *\n\t * **Note:** Until version 20.0.0 this method wasn't accepting `'$textProxy'` type. The legacy `'textProxt'` type is still\n\t * accepted for backward compatibility.\n\t *\n\t * @label TEXT_PROXY\n\t */\n\tpublic is( type: '$textProxy' | 'model:$textProxy' ): this is TextProxy;\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/element~Element} or its subclass and has the specified `name`.\n\t *\n\t * ```ts\n\t * element.is( 'element', 'imageBlock' ); // -> true if this is an <imageBlock> element\n\t * text.is( 'element', 'imageBlock' ); -> false\n\t * ```\n\t *\n\t * @label ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'element' | 'model:element', name: N ): this is ( Element | RootElement ) & { name: N };\n\n\t/**\n\t * Checks whether the object is of type {@link module:engine/model/rootelement~RootElement} and has the specified `name`.\n\t *\n\t * ```ts\n\t * rootElement.is( 'rootElement', '$root' );\n\t * ```\n\t *\n\t * @label ROOT_ELEMENT_NAME\n\t */\n\tpublic is<N extends string>( type: 'rootElement' | 'model:rootElement', name: N ): this is RootElement & { name: N };\n\n\t/* istanbul ignore next -- @preserve */\n\tpublic is(): boolean {\n\t\t// There are a lot of overloads above.\n\t\t// Overriding method in derived classes remove them and only `is( type: string ): boolean` is visible which we don't want.\n\t\t// One option would be to copy them all to all classes, but that's ugly.\n\t\t// It's best when TypeScript compiler doesn't see those overloads, except the one in the top base class.\n\t\t// To overload a method, but not let the compiler see it, do after class definition:\n\t\t// `MyClass.prototype.is = function( type: string ) {...}`\n\t\tthrow new Error( 'is() method is abstract' );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* eslint-disable @typescript-eslint/no-unused-vars */\n\n/**\n * @module engine/model/node\n */\n\nimport TypeCheckable from './typecheckable';\n\nimport type Document from './document';\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\n\nimport { CKEditorError, compareArrays, toMap } from '@ckeditor/ckeditor5-utils';\n\n// To check if component is loaded more than once.\nimport '@ckeditor/ckeditor5-utils/src/version';\n\n/**\n * Model node. Most basic structure of model tree.\n *\n * This is an abstract class that is a base for other classes representing different nodes in model.\n *\n * **Note:** If a node is detached from the model tree, you can manipulate it using it's API.\n * However, it is **very important** that nodes already attached to model tree should be only changed through\n * {@link module:engine/model/writer~Writer Writer API}.\n *\n * Changes done by `Node` methods, like {@link module:engine/model/element~Element#_insertChild _insertChild} or\n * {@link module:engine/model/node~Node#_setAttribute _setAttribute}\n * do not generate {@link module:engine/model/operation/operation~Operation operations}\n * which are essential for correct editor work if you modify nodes in {@link module:engine/model/document~Document document} root.\n *\n * The flow of working on `Node` (and classes that inherits from it) is as such:\n * 1. You can create a `Node` instance, modify it using it's API.\n * 2. Add `Node` to the model using `Batch` API.\n * 3. Change `Node` that was already added to the model using `Batch` API.\n *\n * Similarly, you cannot use `Batch` API on a node that has not been added to the model tree, with the exception\n * of {@link module:engine/model/writer~Writer#insert inserting} that node to the model tree.\n *\n * Be aware that using {@link module:engine/model/writer~Writer#remove remove from Batch API} does not allow to use `Node` API because\n * the information about `Node` is still kept in model document.\n *\n * In case of {@link module:engine/model/element~Element element node}, adding and removing children also counts as changing a node and\n * follows same rules.\n */\nexport default abstract class Node extends TypeCheckable {\n\t/**\n\t * Parent of this node. It could be {@link module:engine/model/element~Element}\n\t * or {@link module:engine/model/documentfragment~DocumentFragment}.\n\t * Equals to `null` if the node has no parent.\n\t */\n\tpublic readonly parent: Element | DocumentFragment | null = null;\n\n\t/**\n\t * Unique root name used to identify this root element by {@link module:engine/model/document~Document}.\n\t */\n\tdeclare public readonly rootName: string | undefined;\n\n\t/**\n\t * Attributes set on this node.\n\t */\n\tprivate _attrs: Map<string, unknown>;\n\n\t/**\n\t * Creates a model node.\n\t *\n\t * This is an abstract class, so this constructor should not be used directly.\n\t *\n\t * @param attrs Node's attributes. See {@link module:utils/tomap~toMap} for a list of accepted values.\n\t */\n\tconstructor( attrs?: NodeAttributes ) {\n\t\tsuper();\n\n\t\tthis._attrs = toMap( attrs! );\n\t}\n\n\t/**\n\t * {@link module:engine/model/document~Document Document} that owns this root element.\n\t */\n\tpublic get document(): Document | null {\n\t\treturn null;\n\t}\n\n\t/**\n\t * Index of this node in its parent or `null` if the node has no parent.\n\t *\n\t * Accessing this property throws an error if this node's parent element does not contain it.\n\t * This means that model tree got broken.\n\t */\n\tpublic get index(): number | null {\n\t\tlet pos;\n\n\t\tif ( !this.parent ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( ( pos = this.parent.getChildIndex( this ) ) === null ) {\n\t\t\tthrow new CKEditorError( 'model-node-not-found-in-parent', this );\n\t\t}\n\n\t\treturn pos;\n\t}\n\n\t/**\n\t * Offset at which this node starts in its parent. It is equal to the sum of {@link #offsetSize offsetSize}\n\t * of all its previous siblings. Equals to `null` if node has no parent.\n\t *\n\t * Accessing this property throws an error if this node's parent element does not contain it.\n\t * This means that model tree got broken.\n\t */\n\tpublic get startOffset(): number | null {\n\t\tlet pos;\n\n\t\tif ( !this.parent ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( ( pos = this.parent.getChildStartOffset( this ) ) === null ) {\n\t\t\tthrow new CKEditorError( 'model-node-not-found-in-parent', this );\n\t\t}\n\n\t\treturn pos;\n\t}\n\n\t/**\n\t * Offset size of this node. Represents how much \"offset space\" is occupied by the node in it's parent.\n\t * It is important for {@link module:engine/model/position~Position position}. When node has `offsetSize` greater than `1`, position\n\t * can be placed between that node start and end. `offsetSize` greater than `1` is for nodes that represents more\n\t * than one entity, i.e. {@link module:engine/model/text~Text text node}.\n\t */\n\tpublic get offsetSize(): number {\n\t\treturn 1;\n\t}\n\n\t/**\n\t * Offset at which this node ends in it's parent. It is equal to the sum of this node's\n\t * {@link module:engine/model/node~Node#startOffset start offset} and {@link #offsetSize offset size}.\n\t * Equals to `null` if the node has no parent.\n\t */\n\tpublic get endOffset(): number | null {\n\t\tif ( !this.parent ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn this.startOffset! + this.offsetSize;\n\t}\n\n\t/**\n\t * Node's next sibling or `null` if the node is a last child of it's parent or if the node has no parent.\n\t */\n\tpublic get nextSibling(): Node | null {\n\t\tconst index = this.index;\n\n\t\treturn ( index !== null && this.parent!.getChild( index + 1 ) ) || null;\n\t}\n\n\t/**\n\t * Node's previous sibling or `null` if the node is a first child of it's parent or if the node has no parent.\n\t */\n\tpublic get previousSibling(): Node | null {\n\t\tconst index = this.index;\n\n\t\treturn ( index !== null && this.parent!.getChild( index - 1 ) ) || null;\n\t}\n\n\t/**\n\t * The top-most ancestor of the node. If node has no parent it is the root itself. If the node is a part\n\t * of {@link module:engine/model/documentfragment~DocumentFragment}, it's `root` is equal to that `DocumentFragment`.\n\t */\n\tpublic get root(): Node | DocumentFragment {\n\t\t// eslint-disable-next-line @typescript-eslint/no-this-alias, consistent-this\n\t\tlet root: Node | DocumentFragment = this;\n\n\t\twhile ( root.parent ) {\n\t\t\troot = root.parent;\n\t\t}\n\n\t\treturn root;\n\t}\n\n\t/**\n\t * Returns `true` if the node is inside a document root that is attached to the document.\n\t */\n\tpublic isAttached(): boolean {\n\t\t// If the node has no parent it means that it is a root.\n\t\t// But this is not a `RootElement`, so it means that it is not attached.\n\t\t//\n\t\t// If this is not the root, check if this element's root is attached.\n\t\treturn this.parent === null ? false : this.root.isAttached();\n\t}\n\n\t/**\n\t * Gets path to the node. The path is an array containing starting offsets of consecutive ancestors of this node,\n\t * beginning from {@link module:engine/model/node~Node#root root}, down to this node's starting offset. The path can be used to\n\t * create {@link module:engine/model/position~Position Position} instance.\n\t *\n\t * ```ts\n\t * const abc = new Text( 'abc' );\n\t * const foo = new Text( 'foo' );\n\t * const h1 = new Element( 'h1', null, new Text( 'header' ) );\n\t * const p = new Element( 'p', null, [ abc, foo ] );\n\t * const div = new Element( 'div', null, [ h1, p ] );\n\t * foo.getPath(); // Returns [ 1, 3 ]. `foo` is in `p` which is in `div`. `p` starts at offset 1, while `foo` at 3.\n\t * h1.getPath(); // Returns [ 0 ].\n\t * div.getPath(); // Returns [].\n\t * ```\n\t */\n\tpublic getPath(): Array<number> {\n\t\tconst path = [];\n\t\t// eslint-disable-next-line @typescript-eslint/no-this-alias, consistent-this\n\t\tlet node: Node | DocumentFragment = this;\n\n\t\twhile ( node.parent ) {\n\t\t\tpath.unshift( node.startOffset! );\n\t\t\tnode = node.parent;\n\t\t}\n\n\t\treturn path;\n\t}\n\n\t/**\n\t * Returns ancestors array of this node.\n\t *\n\t * @param options Options object.\n\t * @param options.includeSelf When set to `true` this node will be also included in parent's array.\n\t * @param options.parentFirst When set to `true`, array will be sorted from node's parent to root element,\n\t * otherwise root element will be the first item in the array.\n\t * @returns Array with ancestors.\n\t */\n\tpublic getAncestors( options: { includeSelf?: boolean; parentFirst?: boolean } = {} ): Array<Node | DocumentFragment> {\n\t\tconst ancestors: Array<Node | DocumentFragment> = [];\n\t\tlet parent = options.includeSelf ? this : this.parent;\n\n\t\twhile ( parent ) {\n\t\t\tancestors[ options.parentFirst ? 'push' : 'unshift' ]( parent );\n\t\t\tparent = parent.parent;\n\t\t}\n\n\t\treturn ancestors;\n\t}\n\n\t/**\n\t * Returns a {@link module:engine/model/element~Element} or {@link module:engine/model/documentfragment~DocumentFragment}\n\t * which is a common ancestor of both nodes.\n\t *\n\t * @param node The second node.\n\t * @param options Options object.\n\t * @param options.includeSelf When set to `true` both nodes will be considered \"ancestors\" too.\n\t * Which means that if e.g. node A is inside B, then their common ancestor will be B.\n\t */\n\tpublic getCommonAncestor( node: Node, options: { includeSelf?: boolean } = {} ): Element | DocumentFragment | null {\n\t\tconst ancestorsA = this.getAncestors( options );\n\t\tconst ancestorsB = node.getAncestors( options );\n\n\t\tlet i = 0;\n\n\t\twhile ( ancestorsA[ i ] == ancestorsB[ i ] && ancestorsA[ i ] ) {\n\t\t\ti++;\n\t\t}\n\n\t\treturn i === 0 ? null : ancestorsA[ i - 1 ] as ( Element | DocumentFragment );\n\t}\n\n\t/**\n\t * Returns whether this node is before given node. `false` is returned if nodes are in different trees (for example,\n\t * in different {@link module:engine/model/documentfragment~DocumentFragment}s).\n\t *\n\t * @param node Node to compare with.\n\t */\n\tpublic isBefore( node: Node ): boolean {\n\t\t// Given node is not before this node if they are same.\n\t\tif ( this == node ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Return `false` if it is impossible to compare nodes.\n\t\tif ( this.root !== node.root ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst thisPath = this.getPath();\n\t\tconst nodePath = node.getPath();\n\n\t\tconst result = compareArrays( thisPath, nodePath );\n\n\t\tswitch ( result ) {\n\t\t\tcase 'prefix':\n\t\t\t\treturn true;\n\n\t\t\tcase 'extension':\n\t\t\t\treturn false;\n\n\t\t\tdefault:\n\t\t\t\treturn thisPath[ result as number ] < nodePath[ result as number ];\n\t\t}\n\t}\n\n\t/**\n\t * Returns whether this node is after given node. `false` is returned if nodes are in different trees (for example,\n\t * in different {@link module:engine/model/documentfragment~DocumentFragment}s).\n\t *\n\t * @param node Node to compare with.\n\t */\n\tpublic isAfter( node: Node ): boolean {\n\t\t// Given node is not before this node if they are same.\n\t\tif ( this == node ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Return `false` if it is impossible to compare nodes.\n\t\tif ( this.root !== node.root ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// In other cases, just check if the `node` is before, and return the opposite.\n\t\treturn !this.isBefore( node );\n\t}\n\n\t/**\n\t * Checks if the node has an attribute with given key.\n\t *\n\t * @param key Key of attribute to check.\n\t * @returns `true` if attribute with given key is set on node, `false` otherwise.\n\t */\n\tpublic hasAttribute( key: string ): boolean {\n\t\treturn this._attrs.has( key );\n\t}\n\n\t/**\n\t * Gets an attribute value for given key or `undefined` if that attribute is not set on node.\n\t *\n\t * @param key Key of attribute to look for.\n\t * @returns Attribute value or `undefined`.\n\t */\n\tpublic getAttribute( key: string ): unknown {\n\t\treturn this._attrs.get( key );\n\t}\n\n\t/**\n\t * Returns iterator that iterates over this node's attributes.\n\t *\n\t * Attributes are returned as arrays containing two items. First one is attribute key and second is attribute value.\n\t * This format is accepted by native `Map` object and also can be passed in `Node` constructor.\n\t */\n\tpublic getAttributes(): IterableIterator<[ string, unknown ]> {\n\t\treturn this._attrs.entries();\n\t}\n\n\t/**\n\t * Returns iterator that iterates over this node's attribute keys.\n\t */\n\tpublic getAttributeKeys(): IterableIterator<string> {\n\t\treturn this._attrs.keys();\n\t}\n\n\t/**\n\t * Converts `Node` to plain object and returns it.\n\t *\n\t * @returns `Node` converted to plain object.\n\t */\n\tpublic toJSON(): unknown {\n\t\tconst json: any = {};\n\n\t\t// Serializes attributes to the object.\n\t\t// attributes = { a: 'foo', b: 1, c: true }.\n\t\tif ( this._attrs.size ) {\n\t\t\tjson.attributes = Array.from( this._attrs ).reduce( ( result, attr ) => {\n\t\t\t\tresult[ attr[ 0 ] ] = attr[ 1 ];\n\n\t\t\t\treturn result;\n\t\t\t}, {} as any );\n\t\t}\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * Creates a copy of this node, that is a node with exactly same attributes, and returns it.\n\t *\n\t * @internal\n\t * @returns Node with same attributes as this node.\n\t */\n\tpublic _clone( _deep?: boolean ): Node {\n\t\treturn new ( this.constructor as any )( this._attrs );\n\t}\n\n\t/**\n\t * Removes this node from it's parent.\n\t *\n\t * @internal\n\t * @see module:engine/model/writer~Writer#remove\n\t */\n\tpublic _remove(): void {\n\t\tthis.parent!._removeChildren( this.index! );\n\t}\n\n\t/**\n\t * Sets attribute on the node. If attribute with the same key already is set, it's value is overwritten.\n\t *\n\t * @see module:engine/model/writer~Writer#setAttribute\n\t * @internal\n\t * @param key Key of attribute to set.\n\t * @param value Attribute value.\n\t */\n\tpublic _setAttribute( key: string, value: unknown ): void {\n\t\tthis._attrs.set( key, value );\n\t}\n\n\t/**\n\t * Removes all attributes from the node and sets given attributes.\n\t *\n\t * @see module:engine/model/writer~Writer#setAttributes\n\t * @internal\n\t * @param attrs Attributes to set. See {@link module:utils/tomap~toMap} for a list of accepted values.\n\t */\n\tpublic _setAttributesTo( attrs: NodeAttributes ): void {\n\t\tthis._attrs = toMap( attrs );\n\t}\n\n\t/**\n\t * Removes an attribute with given key from the node.\n\t *\n\t * @see module:engine/model/writer~Writer#removeAttribute\n\t * @internal\n\t * @param key Key of attribute to remove.\n\t * @returns `true` if the attribute was set on the element, `false` otherwise.\n\t */\n\tpublic _removeAttribute( key: string ): boolean {\n\t\treturn this._attrs.delete( key );\n\t}\n\n\t/**\n\t * Removes all attributes from the node.\n\t *\n\t * @see module:engine/model/writer~Writer#clearAttributes\n\t * @internal\n\t */\n\tpublic _clearAttributes(): void {\n\t\tthis._attrs.clear();\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nNode.prototype.is = function( type: string ): boolean {\n\treturn type === 'node' || type === 'model:node';\n};\n\n/**\n * The node's parent does not contain this node.\n *\n * @error model-node-not-found-in-parent\n */\n\n/**\n * Node's attributes. See {@link module:utils/tomap~toMap} for a list of accepted values.\n */\nexport type NodeAttributes = Record<string, unknown> | Iterable<[ string, unknown ]>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/nodelist\n */\n\nimport Node from './node';\n\nimport { CKEditorError, spliceArray } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Provides an interface to operate on a list of {@link module:engine/model/node~Node nodes}. `NodeList` is used internally\n * in classes like {@link module:engine/model/element~Element Element}\n * or {@link module:engine/model/documentfragment~DocumentFragment DocumentFragment}.\n */\nexport default class NodeList implements Iterable<Node> {\n\t/**\n\t * Nodes contained in this node list.\n\t */\n\tprivate _nodes: Array<Node> = [];\n\n\t/**\n\t * Creates an empty node list.\n\t *\n\t * @internal\n\t * @param nodes Nodes contained in this node list.\n\t */\n\tconstructor( nodes?: Iterable<Node> ) {\n\t\tif ( nodes ) {\n\t\t\tthis._insertNodes( 0, nodes );\n\t\t}\n\t}\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * Iterates over all nodes contained inside this node list.\n\t */\n\tpublic [ Symbol.iterator ](): IterableIterator<Node> {\n\t\treturn this._nodes[ Symbol.iterator ]();\n\t}\n\n\t/**\n\t * Number of nodes contained inside this node list.\n\t */\n\tpublic get length(): number {\n\t\treturn this._nodes.length;\n\t}\n\n\t/**\n\t * Sum of {@link module:engine/model/node~Node#offsetSize offset sizes} of all nodes contained inside this node list.\n\t */\n\tpublic get maxOffset(): number {\n\t\treturn this._nodes.reduce( ( sum, node ) => sum + node.offsetSize, 0 );\n\t}\n\n\t/**\n\t * Gets the node at the given index. Returns `null` if incorrect index was passed.\n\t */\n\tpublic getNode( index: number ): Node | null {\n\t\treturn this._nodes[ index ] || null;\n\t}\n\n\t/**\n\t * Returns an index of the given node. Returns `null` if given node is not inside this node list.\n\t */\n\tpublic getNodeIndex( node: Node ): number | null {\n\t\tconst index = this._nodes.indexOf( node );\n\n\t\treturn index == -1 ? null : index;\n\t}\n\n\t/**\n\t * Returns the starting offset of given node. Starting offset is equal to the sum of\n\t * {@link module:engine/model/node~Node#offsetSize offset sizes} of all nodes that are before this node in this node list.\n\t */\n\tpublic getNodeStartOffset( node: Node ): number | null {\n\t\tconst index = this.getNodeIndex( node );\n\n\t\treturn index === null ? null : this._nodes.slice( 0, index ).reduce( ( sum, node ) => sum + node.offsetSize, 0 );\n\t}\n\n\t/**\n\t * Converts index to offset in node list.\n\t *\n\t * Returns starting offset of a node that is at given index. Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError}\n\t * `model-nodelist-index-out-of-bounds` if given index is less than `0` or more than {@link #length}.\n\t */\n\tpublic indexToOffset( index: number ): number {\n\t\tif ( index == this._nodes.length ) {\n\t\t\treturn this.maxOffset;\n\t\t}\n\n\t\tconst node = this._nodes[ index ];\n\n\t\tif ( !node ) {\n\t\t\t/**\n\t\t\t * Given index cannot be found in the node list.\n\t\t\t *\n\t\t\t * @error model-nodelist-index-out-of-bounds\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-nodelist-index-out-of-bounds', this );\n\t\t}\n\n\t\treturn this.getNodeStartOffset( node )!;\n\t}\n\n\t/**\n\t * Converts offset in node list to index.\n\t *\n\t * Returns index of a node that occupies given offset. Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError}\n\t * `model-nodelist-offset-out-of-bounds` if given offset is less than `0` or more than {@link #maxOffset}.\n\t */\n\tpublic offsetToIndex( offset: number ): number {\n\t\tlet totalOffset = 0;\n\n\t\tfor ( const node of this._nodes ) {\n\t\t\tif ( offset >= totalOffset && offset < totalOffset + node.offsetSize ) {\n\t\t\t\treturn this.getNodeIndex( node )!;\n\t\t\t}\n\n\t\t\ttotalOffset += node.offsetSize;\n\t\t}\n\n\t\tif ( totalOffset != offset ) {\n\t\t\t/**\n\t\t\t * Given offset cannot be found in the node list.\n\t\t\t *\n\t\t\t * @error model-nodelist-offset-out-of-bounds\n\t\t\t * @param offset\n\t\t\t * @param nodeList Stringified node list.\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-nodelist-offset-out-of-bounds',\n\t\t\t\tthis,\n\t\t\t\t{\n\t\t\t\t\toffset,\n\t\t\t\t\tnodeList: this\n\t\t\t\t}\n\t\t\t);\n\t\t}\n\n\t\treturn this.length;\n\t}\n\n\t/**\n\t * Inserts given nodes at given index.\n\t *\n\t * @internal\n\t * @param index Index at which nodes should be inserted.\n\t * @param nodes Nodes to be inserted.\n\t */\n\tpublic _insertNodes( index: number, nodes: Iterable<Node> ): void {\n\t\t// Validation.\n\t\tfor ( const node of nodes ) {\n\t\t\tif ( !( node instanceof Node ) ) {\n\t\t\t\t/**\n\t\t\t\t * Trying to insert an object which is not a Node instance.\n\t\t\t\t *\n\t\t\t\t * @error model-nodelist-insertnodes-not-node\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'model-nodelist-insertnodes-not-node', this );\n\t\t\t}\n\t\t}\n\n\t\tthis._nodes = spliceArray<Node>( this._nodes, Array.from( nodes ), index, 0 );\n\t}\n\n\t/**\n\t * Removes one or more nodes starting at the given index.\n\t *\n\t * @internal\n\t * @param indexStart Index of the first node to remove.\n\t * @param howMany Number of nodes to remove.\n\t * @returns Array containing removed nodes.\n\t */\n\tpublic _removeNodes( indexStart: number, howMany: number = 1 ): Array<Node> {\n\t\treturn this._nodes.splice( indexStart, howMany );\n\t}\n\n\t/**\n\t * Converts `NodeList` instance to an array containing nodes that were inserted in the node list. Nodes\n\t * are also converted to their plain object representation.\n\t *\n\t * @returns `NodeList` instance converted to `Array`.\n\t */\n\tpublic toJSON(): unknown {\n\t\treturn this._nodes.map( node => node.toJSON() );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/splicearray\n */\n\nconst BIG_CHUNK_SIZE = 10000;\n\n/**\n * Splices one array into another. To be used instead of `Array.prototype.splice` as the latter may\n * throw \"Maximum call stack size exceeded\" when passed huge number of items to insert.\n *\n * Note: in contrary to Array.splice, this function does not modify the original `target`.\n *\n * ```ts\n * spliceArray( [ 1, 2 ], [ 3, 4 ], 0, 0 );\t// [ 3, 4, 1, 2 ]\n * spliceArray( [ 1, 2 ], [ 3, 4 ], 1, 1 );\t// [ 1, 3, 4 ]\n * spliceArray( [ 1, 2 ], [ 3, 4 ], 1, 0 );\t// [ 1, 3, 4, 2 ]\n * spliceArray( [ 1, 2 ], [ 3, 4 ], 2, 0 );\t// [ 1, 2, 3, 4 ]\n * spliceArray( [ 1, 2 ], [], 0, 1 );\t// [ 2 ]\n * ```\n *\n * @param target Array to be spliced.\n * @param source Array of elements to be inserted to target.\n * @param start Index at which nodes should be inserted/removed.\n * @param count Number of items.\n *\n * @returns New spliced array.\n */\nexport default function spliceArray<T>( target: Array<T>, source: Array<T>, start: number, count: number ): Array<T> {\n\t// In case of performance problems, see: https://github.com/ckeditor/ckeditor5/pull/12429/files#r965850568\n\tif ( Math.max( source.length, target.length ) > BIG_CHUNK_SIZE ) {\n\t\treturn target.slice( 0, start ).concat( source ).concat( target.slice( start + count, target.length ) );\n\t} else {\n\t\tconst newTarget = Array.from( target );\n\t\tnewTarget.splice( start, count, ...source );\n\n\t\treturn newTarget;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/text\n */\n\nimport Node, { type NodeAttributes } from './node';\n\n// @if CK_DEBUG_ENGINE // const { convertMapToStringifiedObject } = require( '../dev-utils/utils' );\n\n/**\n * Model text node. Type of {@link module:engine/model/node~Node node} that contains {@link module:engine/model/text~Text#data text data}.\n *\n * **Important:** see {@link module:engine/model/node~Node} to read about restrictions using `Text` and `Node` API.\n *\n * **Note:** keep in mind that `Text` instances might indirectly got removed from model tree when model is changed.\n * This happens when {@link module:engine/model/writer~Writer model writer} is used to change model and the text node is merged with\n * another text node. Then, both text nodes are removed and a new text node is inserted into the model. Because of\n * this behavior, keeping references to `Text` is not recommended. Instead, consider creating\n * {@link module:engine/model/liveposition~LivePosition live position} placed before the text node.\n */\nexport default class Text extends Node {\n\t/**\n\t * Text data contained in this text node.\n\t *\n\t * @internal\n\t */\n\tpublic _data: string;\n\n\t/**\n\t * Creates a text node.\n\t *\n\t * **Note:** Constructor of this class shouldn't be used directly in the code.\n\t * Use the {@link module:engine/model/writer~Writer#createText} method instead.\n\t *\n\t * @internal\n\t * @param data Node's text.\n\t * @param attrs Node's attributes. See {@link module:utils/tomap~toMap} for a list of accepted values.\n\t */\n\tconstructor( data?: string, attrs?: NodeAttributes ) {\n\t\tsuper( attrs );\n\t\tthis._data = data || '';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override get offsetSize(): number {\n\t\treturn this.data.length;\n\t}\n\n\t/**\n\t * Returns a text data contained in the node.\n\t */\n\tpublic get data(): string {\n\t\treturn this._data;\n\t}\n\n\t/**\n\t * Converts `Text` instance to plain object and returns it.\n\t *\n\t * @returns`Text` instance converted to plain object.\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.data = this.data;\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * Creates a copy of this text node and returns it. Created text node has same text data and attributes as original text node.\n\t *\n\t * @internal\n\t * @returns `Text` instance created using given plain object.\n\t */\n\tpublic override _clone(): Text {\n\t\treturn new Text( this.data, this.getAttributes() );\n\t}\n\n\t/**\n\t * Creates a `Text` instance from given plain object (i.e. parsed JSON string).\n\t *\n\t * @param json Plain object to be converted to `Text`.\n\t * @returns `Text` instance created using given plain object.\n\t */\n\tpublic static fromJSON( json: any ): Text {\n\t\treturn new Text( json.data, json.attributes );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `#${ this.data }`;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logExtended(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( `ModelText: ${ this }, attrs: ${ convertMapToStringifiedObject( this.getAttributes() ) }` );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ModelText: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nText.prototype.is = function( type: string ): boolean {\n\treturn type === '$text' || type === 'model:$text' ||\n\t\t// This are legacy values kept for backward compatibility.\n\t\ttype === 'text' || type === 'model:text' ||\n\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\ttype === 'node' || type === 'model:node';\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/textproxy\n */\n\nimport TypeCheckable from './typecheckable';\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\nimport type Node from './node';\nimport type Text from './text';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\n// @if CK_DEBUG_ENGINE // const { convertMapToStringifiedObject } = require( '../dev-utils/utils' );\n\n/**\n * `TextProxy` represents a part of {@link module:engine/model/text~Text text node}.\n *\n * Since {@link module:engine/model/position~Position positions} can be placed between characters of a text node,\n * {@link module:engine/model/range~Range ranges} may contain only parts of text nodes. When {@link module:engine/model/range~Range#getItems\n * getting items}\n * contained in such range, we need to represent a part of that text node, since returning the whole text node would be incorrect.\n * `TextProxy` solves this issue.\n *\n * `TextProxy` has an API similar to {@link module:engine/model/text~Text Text} and allows to do most of the common tasks performed\n * on model nodes.\n *\n * **Note:** Some `TextProxy` instances may represent whole text node, not just a part of it.\n * See {@link module:engine/model/textproxy~TextProxy#isPartial}.\n *\n * **Note:** `TextProxy` is not an instance of {@link module:engine/model/node~Node node}. Keep this in mind when using it as a\n * parameter of methods.\n *\n * **Note:** `TextProxy` is a readonly interface. If you want to perform changes on model data represented by a `TextProxy`\n * use {@link module:engine/model/writer~Writer model writer API}.\n *\n * **Note:** `TextProxy` instances are created on the fly, basing on the current state of model. Because of this, it is\n * highly unrecommended to store references to `TextProxy` instances. `TextProxy` instances are not refreshed when\n * model changes, so they might get invalidated. Instead, consider creating {@link module:engine/model/liveposition~LivePosition live\n * position}.\n *\n * `TextProxy` instances are created by {@link module:engine/model/treewalker~TreeWalker model tree walker}. You should not need to create\n * an instance of this class by your own.\n */\nexport default class TextProxy extends TypeCheckable {\n\t/**\n\t * Text node which part is represented by this text proxy.\n\t */\n\tpublic readonly textNode: Text;\n\n\t/**\n\t * Text data represented by this text proxy.\n\t */\n\tpublic readonly data: string;\n\n\t/**\n\t * Offset in {@link module:engine/model/textproxy~TextProxy#textNode text node} from which the text proxy starts.\n\t */\n\tpublic readonly offsetInText: number;\n\n\t/**\n\t * Creates a text proxy.\n\t *\n\t * @internal\n\t * @param textNode Text node which part is represented by this text proxy.\n\t * @param offsetInText Offset in {@link module:engine/model/textproxy~TextProxy#textNode text node} from which the text proxy\n\t * starts.\n\t * @param length Text proxy length, that is how many text node's characters, starting from `offsetInText` it represents.\n\t */\n\tconstructor( textNode: Text, offsetInText: number, length: number ) {\n\t\tsuper();\n\n\t\tthis.textNode = textNode;\n\n\t\tif ( offsetInText < 0 || offsetInText > textNode.offsetSize ) {\n\t\t\t/**\n\t\t\t * Given `offsetInText` value is incorrect.\n\t\t\t *\n\t\t\t * @error model-textproxy-wrong-offsetintext\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-textproxy-wrong-offsetintext', this );\n\t\t}\n\n\t\tif ( length < 0 || offsetInText + length > textNode.offsetSize ) {\n\t\t\t/**\n\t\t\t * Given `length` value is incorrect.\n\t\t\t *\n\t\t\t * @error model-textproxy-wrong-length\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-textproxy-wrong-length', this );\n\t\t}\n\t\tthis.data = textNode.data.substring( offsetInText, offsetInText + length );\n\n\t\tthis.offsetInText = offsetInText;\n\t}\n\n\t/**\n\t * Offset at which this text proxy starts in it's parent.\n\t *\n\t * @see module:engine/model/node~Node#startOffset\n\t */\n\tpublic get startOffset(): number | null {\n\t\treturn this.textNode.startOffset !== null ? this.textNode.startOffset + this.offsetInText : null;\n\t}\n\n\t/**\n\t * Offset size of this text proxy. Equal to the number of characters represented by the text proxy.\n\t *\n\t * @see module:engine/model/node~Node#offsetSize\n\t */\n\tpublic get offsetSize(): number {\n\t\treturn this.data.length;\n\t}\n\n\t/**\n\t * Offset at which this text proxy ends in it's parent.\n\t *\n\t * @see module:engine/model/node~Node#endOffset\n\t */\n\tpublic get endOffset(): number | null {\n\t\treturn this.startOffset !== null ? this.startOffset + this.offsetSize : null;\n\t}\n\n\t/**\n\t * Flag indicating whether `TextProxy` instance covers only part of the original {@link module:engine/model/text~Text text node}\n\t * (`true`) or the whole text node (`false`).\n\t *\n\t * This is `false` when text proxy starts at the very beginning of {@link module:engine/model/textproxy~TextProxy#textNode textNode}\n\t * ({@link module:engine/model/textproxy~TextProxy#offsetInText offsetInText} equals `0`) and text proxy sizes is equal to\n\t * text node size.\n\t */\n\tpublic get isPartial(): boolean {\n\t\treturn this.offsetSize !== this.textNode.offsetSize;\n\t}\n\n\t/**\n\t * Parent of this text proxy, which is same as parent of text node represented by this text proxy.\n\t */\n\tpublic get parent(): Element | DocumentFragment | null {\n\t\treturn this.textNode.parent;\n\t}\n\n\t/**\n\t * Root of this text proxy, which is same as root of text node represented by this text proxy.\n\t */\n\tpublic get root(): Node | DocumentFragment {\n\t\treturn this.textNode.root;\n\t}\n\n\t/**\n\t * Gets path to this text proxy.\n\t *\n\t * @see module:engine/model/node~Node#getPath\n\t */\n\tpublic getPath(): Array<number> {\n\t\tconst path = this.textNode.getPath();\n\n\t\tif ( path.length > 0 ) {\n\t\t\tpath[ path.length - 1 ] += this.offsetInText;\n\t\t}\n\n\t\treturn path;\n\t}\n\n\t/**\n\t * Returns ancestors array of this text proxy.\n\t *\n\t * @param options Options object.\n\t * @param options.includeSelf When set to `true` this text proxy will be also included in parent's array.\n\t * @param options.parentFirst When set to `true`, array will be sorted from text proxy parent to root element,\n\t * otherwise root element will be the first item in the array.\n\t * @returns Array with ancestors.\n\t */\n\tpublic getAncestors( options: {\n\t\tincludeSelf?: boolean;\n\t\tparentFirst?: boolean;\n\t} = {} ): Array<TextProxy | Element | DocumentFragment> {\n\t\tconst ancestors: Array<TextProxy | Element | DocumentFragment> = [];\n\t\tlet parent: TextProxy | Element | DocumentFragment | null = options.includeSelf ? this : this.parent;\n\n\t\twhile ( parent ) {\n\t\t\tancestors[ options.parentFirst ? 'push' : 'unshift' ]( parent );\n\t\t\tparent = parent.parent;\n\t\t}\n\n\t\treturn ancestors;\n\t}\n\n\t/**\n\t * Checks if this text proxy has an attribute for given key.\n\t *\n\t * @param key Key of attribute to check.\n\t * @returns `true` if attribute with given key is set on text proxy, `false` otherwise.\n\t */\n\tpublic hasAttribute( key: string ): boolean {\n\t\treturn this.textNode.hasAttribute( key );\n\t}\n\n\t/**\n\t * Gets an attribute value for given key or `undefined` if that attribute is not set on text proxy.\n\t *\n\t * @param key Key of attribute to look for.\n\t * @returns Attribute value or `undefined`.\n\t */\n\tpublic getAttribute( key: string ): unknown {\n\t\treturn this.textNode.getAttribute( key );\n\t}\n\n\t/**\n\t * Returns iterator that iterates over this node's attributes. Attributes are returned as arrays containing two\n\t * items. First one is attribute key and second is attribute value.\n\t *\n\t * This format is accepted by native `Map` object and also can be passed in `Node` constructor.\n\t */\n\tpublic getAttributes(): IterableIterator<[ string, unknown ]> {\n\t\treturn this.textNode.getAttributes();\n\t}\n\n\t/**\n\t * Returns iterator that iterates over this node's attribute keys.\n\t */\n\tpublic getAttributeKeys(): IterableIterator<string> {\n\t\treturn this.textNode.getAttributeKeys();\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `#${ this.data }`;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ModelTextProxy: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logExtended(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( `ModelTextProxy: ${ this }, ` +\n\t// @if CK_DEBUG_ENGINE // \t\t`attrs: ${ convertMapToStringifiedObject( this.getAttributes() ) }` );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nTextProxy.prototype.is = function( type: string ): boolean {\n\treturn type === '$textProxy' || type === 'model:$textProxy' ||\n\t\t// This are legacy values kept for backward compatibility.\n\t\ttype === 'textProxy' || type === 'model:textProxy';\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/element\n */\n\nimport Node, { type NodeAttributes } from './node';\nimport NodeList from './nodelist';\nimport Text from './text';\nimport TextProxy from './textproxy';\n\nimport type Item from './item';\n\nimport { isIterable } from '@ckeditor/ckeditor5-utils';\n\n// @if CK_DEBUG_ENGINE // const { stringifyMap, convertMapToStringifiedObject, convertMapToTags } = require( '../dev-utils/utils' );\n\n/**\n * Model element. Type of {@link module:engine/model/node~Node node} that has a {@link module:engine/model/element~Element#name name} and\n * {@link module:engine/model/element~Element#getChildren child nodes}.\n *\n * **Important**: see {@link module:engine/model/node~Node} to read about restrictions using `Element` and `Node` API.\n */\nexport default class Element extends Node {\n\t/**\n\t * Element name.\n\t */\n\tpublic readonly name: string;\n\n\t/**\n\t * List of children nodes.\n\t */\n\tprivate readonly _children: NodeList = new NodeList();\n\n\t/**\n\t * Creates a model element.\n\t *\n\t * **Note:** Constructor of this class shouldn't be used directly in the code.\n\t * Use the {@link module:engine/model/writer~Writer#createElement} method instead.\n\t *\n\t * @internal\n\t * @param name Element's name.\n\t * @param attrs Element's attributes. See {@link module:utils/tomap~toMap} for a list of accepted values.\n\t * @param children One or more nodes to be inserted as children of created element.\n\t */\n\tconstructor(\n\t\tname: string,\n\t\tattrs?: NodeAttributes,\n\t\tchildren?: string | Item | Iterable<string | Item>\n\t) {\n\t\tsuper( attrs );\n\n\t\tthis.name = name;\n\n\t\tif ( children ) {\n\t\t\tthis._insertChild( 0, children );\n\t\t}\n\t}\n\n\t/**\n\t * Number of this element's children.\n\t */\n\tpublic get childCount(): number {\n\t\treturn this._children.length;\n\t}\n\n\t/**\n\t * Sum of {@link module:engine/model/node~Node#offsetSize offset sizes} of all of this element's children.\n\t */\n\tpublic get maxOffset(): number {\n\t\treturn this._children.maxOffset;\n\t}\n\n\t/**\n\t * Is `true` if there are no nodes inside this element, `false` otherwise.\n\t */\n\tpublic get isEmpty(): boolean {\n\t\treturn this.childCount === 0;\n\t}\n\n\t/**\n\t * Gets the child at the given index.\n\t */\n\tpublic getChild( index: number ): Node | null {\n\t\treturn this._children.getNode( index );\n\t}\n\n\t/**\n\t * Returns an iterator that iterates over all of this element's children.\n\t */\n\tpublic getChildren(): IterableIterator<Node> {\n\t\treturn this._children[ Symbol.iterator ]();\n\t}\n\n\t/**\n\t * Returns an index of the given child node. Returns `null` if given node is not a child of this element.\n\t *\n\t * @param node Child node to look for.\n\t * @returns Child node's index in this element.\n\t */\n\tpublic getChildIndex( node: Node ): number | null {\n\t\treturn this._children.getNodeIndex( node );\n\t}\n\n\t/**\n\t * Returns the starting offset of given child. Starting offset is equal to the sum of\n\t * {@link module:engine/model/node~Node#offsetSize offset sizes} of all node's siblings that are before it. Returns `null` if\n\t * given node is not a child of this element.\n\t *\n\t * @param node Child node to look for.\n\t * @returns Child node's starting offset.\n\t */\n\tpublic getChildStartOffset( node: Node ): number | null {\n\t\treturn this._children.getNodeStartOffset( node );\n\t}\n\n\t/**\n\t * Returns index of a node that occupies given offset. If given offset is too low, returns `0`. If given offset is\n\t * too high, returns {@link module:engine/model/element~Element#getChildIndex index after last child}.\n\t *\n\t * ```ts\n\t * const textNode = new Text( 'foo' );\n\t * const pElement = new Element( 'p' );\n\t * const divElement = new Element( [ textNode, pElement ] );\n\t * divElement.offsetToIndex( -1 ); // Returns 0, because offset is too low.\n\t * divElement.offsetToIndex( 0 ); // Returns 0, because offset 0 is taken by `textNode` which is at index 0.\n\t * divElement.offsetToIndex( 1 ); // Returns 0, because `textNode` has `offsetSize` equal to 3, so it occupies offset 1 too.\n\t * divElement.offsetToIndex( 2 ); // Returns 0.\n\t * divElement.offsetToIndex( 3 ); // Returns 1.\n\t * divElement.offsetToIndex( 4 ); // Returns 2. There are no nodes at offset 4, so last available index is returned.\n\t * ```\n\t */\n\tpublic offsetToIndex( offset: number ): number {\n\t\treturn this._children.offsetToIndex( offset );\n\t}\n\n\t/**\n\t * Returns a descendant node by its path relative to this element.\n\t *\n\t * ```ts\n\t * // <this>a<b>c</b></this>\n\t * this.getNodeByPath( [ 0 ] ); // -> \"a\"\n\t * this.getNodeByPath( [ 1 ] ); // -> <b>\n\t * this.getNodeByPath( [ 1, 0 ] ); // -> \"c\"\n\t * ```\n\t *\n\t * @param relativePath Path of the node to find, relative to this element.\n\t */\n\tpublic getNodeByPath( relativePath: Array<number> ): Node {\n\t\t// eslint-disable-next-line @typescript-eslint/no-this-alias, consistent-this\n\t\tlet node: Node = this;\n\n\t\tfor ( const index of relativePath ) {\n\t\t\tnode = ( node as Element ).getChild( ( node as Element ).offsetToIndex( index ) )!;\n\t\t}\n\n\t\treturn node;\n\t}\n\n\t/**\n\t * Returns the parent element of the given name. Returns null if the element is not inside the desired parent.\n\t *\n\t * @param parentName The name of the parent element to find.\n\t * @param options Options object.\n\t * @param options.includeSelf When set to `true` this node will be also included while searching.\n\t */\n\tpublic findAncestor( parentName: string, options: { includeSelf?: boolean } = {} ): Element | null {\n\t\tlet parent = options.includeSelf ? this : this.parent;\n\n\t\twhile ( parent ) {\n\t\t\tif ( parent.name === parentName ) {\n\t\t\t\treturn parent;\n\t\t\t}\n\n\t\t\tparent = parent.parent;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Converts `Element` instance to plain object and returns it. Takes care of converting all of this element's children.\n\t *\n\t * @returns `Element` instance converted to plain object.\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.name = this.name;\n\n\t\tif ( this._children.length > 0 ) {\n\t\t\tjson.children = [];\n\n\t\t\tfor ( const node of this._children ) {\n\t\t\t\tjson.children.push( node.toJSON() );\n\t\t\t}\n\t\t}\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * Creates a copy of this element and returns it. Created element has the same name and attributes as the original element.\n\t * If clone is deep, the original element's children are also cloned. If not, then empty element is returned.\n\t *\n\t * @internal\n\t * @param deep If set to `true` clones element and all its children recursively. When set to `false`,\n\t * element will be cloned without any child.\n\t */\n\tpublic override _clone( deep = false ): Element {\n\t\tconst children = deep ? Array.from( this._children ).map( node => node._clone( true ) ) : undefined;\n\n\t\treturn new Element( this.name, this.getAttributes(), children );\n\t}\n\n\t/**\n\t * {@link module:engine/model/element~Element#_insertChild Inserts} one or more nodes at the end of this element.\n\t *\n\t * @see module:engine/model/writer~Writer#append\n\t * @internal\n\t * @param nodes Nodes to be inserted.\n\t */\n\tpublic _appendChild( nodes: string | Item | Iterable<string | Item> ): void {\n\t\tthis._insertChild( this.childCount, nodes );\n\t}\n\n\t/**\n\t * Inserts one or more nodes at the given index and sets {@link module:engine/model/node~Node#parent parent} of these nodes\n\t * to this element.\n\t *\n\t * @see module:engine/model/writer~Writer#insert\n\t * @internal\n\t * @param index Index at which nodes should be inserted.\n\t * @param items Items to be inserted.\n\t */\n\tpublic _insertChild( index: number, items: string | Item | Iterable<string | Item> ): void {\n\t\tconst nodes = normalize( items );\n\n\t\tfor ( const node of nodes ) {\n\t\t\t// If node that is being added to this element is already inside another element, first remove it from the old parent.\n\t\t\tif ( node.parent !== null ) {\n\t\t\t\tnode._remove();\n\t\t\t}\n\n\t\t\t( node as any ).parent = this;\n\t\t}\n\n\t\tthis._children._insertNodes( index, nodes );\n\t}\n\n\t/**\n\t * Removes one or more nodes starting at the given index and sets\n\t * {@link module:engine/model/node~Node#parent parent} of these nodes to `null`.\n\t *\n\t * @see module:engine/model/writer~Writer#remove\n\t * @internal\n\t * @param index Index of the first node to remove.\n\t * @param howMany Number of nodes to remove.\n\t * @returns Array containing removed nodes.\n\t */\n\tpublic _removeChildren( index: number, howMany: number = 1 ): Array<Node> {\n\t\tconst nodes = this._children._removeNodes( index, howMany );\n\n\t\tfor ( const node of nodes ) {\n\t\t\t( node as any ).parent = null;\n\t\t}\n\n\t\treturn nodes;\n\t}\n\n\t/**\n\t * Creates an `Element` instance from given plain object (i.e. parsed JSON string).\n\t * Converts `Element` children to proper nodes.\n\t *\n\t * @param json Plain object to be converted to `Element`.\n\t * @returns `Element` instance created using given plain object.\n\t */\n\tpublic static fromJSON( json: any ): Element {\n\t\tlet children: Array<Node> | undefined;\n\n\t\tif ( json.children ) {\n\t\t\tchildren = [];\n\n\t\t\tfor ( const child of json.children ) {\n\t\t\t\tif ( child.name ) {\n\t\t\t\t\t// If child has name property, it is an Element.\n\t\t\t\t\tchildren.push( Element.fromJSON( child ) );\n\t\t\t\t} else {\n\t\t\t\t\t// Otherwise, it is a Text node.\n\t\t\t\t\tchildren.push( Text.fromJSON( child ) );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn new Element( json.name, json.attributes, children );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `<${ this.rootName || this.name }>`;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ModelElement: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logExtended(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( `ModelElement: ${ this }, ${ this.childCount } children,\n\t// @if CK_DEBUG_ENGINE // \tattrs: ${ convertMapToStringifiedObject( this.getAttributes() ) }` );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logAll(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( '--------------------' );\n\n\t// @if CK_DEBUG_ENGINE // \tthis.logExtended();\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'List of children:' );\n\n\t// @if CK_DEBUG_ENGINE // \tfor ( const child of this.getChildren() as any ) {\n\t// @if CK_DEBUG_ENGINE // \t\tchild.log();\n\t// @if CK_DEBUG_ENGINE // \t}\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public printTree( level = 0 ): string {\n\t// @if CK_DEBUG_ENGINE // \tlet string = '';\n\n\t// @if CK_DEBUG_ENGINE // \tstring += '\\t'.repeat( level );\n\t// @if CK_DEBUG_ENGINE // \tstring += `<${ this.rootName || this.name }${ convertMapToTags( this.getAttributes() ) }>`;\n\n\t// @if CK_DEBUG_ENGINE // \tfor ( const child of this.getChildren() as any ) {\n\t// @if CK_DEBUG_ENGINE // \t\tstring += '\\n';\n\n\t// @if CK_DEBUG_ENGINE // \t\tif ( child.is( '$text' ) ) {\n\t// @if CK_DEBUG_ENGINE // \t\t\tconst textAttrs = convertMapToTags( child._attrs );\n\n\t// @if CK_DEBUG_ENGINE // \t\t\tstring += '\\t'.repeat( level + 1 );\n\n\t// @if CK_DEBUG_ENGINE // \t\t\tif ( textAttrs !== '' ) {\n\t// @if CK_DEBUG_ENGINE // \t\t\t\tstring += `<$text${ textAttrs }>` + child.data + '</$text>';\n\t// @if CK_DEBUG_ENGINE // \t\t\t} else {\n\t// @if CK_DEBUG_ENGINE // \t\t\t\tstring += child.data;\n\t// @if CK_DEBUG_ENGINE // \t\t\t}\n\t// @if CK_DEBUG_ENGINE // \t\t} else {\n\t// @if CK_DEBUG_ENGINE // \t\t\tstring += child.printTree( level + 1 );\n\t// @if CK_DEBUG_ENGINE // \t\t}\n\t// @if CK_DEBUG_ENGINE // \t}\n\n\t// @if CK_DEBUG_ENGINE // \tif ( this.childCount ) {\n\t// @if CK_DEBUG_ENGINE // \t\tstring += '\\n' + '\\t'.repeat( level );\n\t// @if CK_DEBUG_ENGINE // \t}\n\n\t// @if CK_DEBUG_ENGINE // \tstring += `</${ this.rootName || this.name }>`;\n\n\t// @if CK_DEBUG_ENGINE // \treturn string;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logTree(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( this.printTree() );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'element' || type === 'model:element' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'node' || type === 'model:node';\n\t}\n\n\treturn name === this.name && ( type === 'element' || type === 'model:element' );\n};\n\n/**\n * Converts strings to Text and non-iterables to arrays.\n */\nfunction normalize( nodes: string | Item | Iterable<string | Item> ): Iterable<Node> {\n\t// Separate condition because string is iterable.\n\tif ( typeof nodes == 'string' ) {\n\t\treturn [ new Text( nodes ) ];\n\t}\n\n\tif ( !isIterable( nodes ) ) {\n\t\tnodes = [ nodes ];\n\t}\n\n\t// Array.from to enable .map() on non-arrays.\n\treturn Array.from( nodes )\n\t\t.map( node => {\n\t\t\tif ( typeof node == 'string' ) {\n\t\t\t\treturn new Text( node );\n\t\t\t}\n\n\t\t\tif ( node instanceof TextProxy ) {\n\t\t\t\treturn new Text( node.data, node.getAttributes() );\n\t\t\t}\n\n\t\t\treturn node;\n\t\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/treewalker\n */\n\nimport Element from './element';\nimport {\n\tdefault as Position,\n\tgetTextNodeAtPosition,\n\tgetNodeAfterPosition,\n\tgetNodeBeforePosition\n} from './position';\nimport Text from './text';\nimport TextProxy from './textproxy';\n\nimport type DocumentFragment from './documentfragment';\nimport type Item from './item';\nimport type Range from './range';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Position iterator class. It allows to iterate forward and backward over the document.\n */\nexport default class TreeWalker implements Iterable<TreeWalkerValue> {\n\t/**\n\t * Walking direction. Defaults `'forward'`.\n\t */\n\tpublic readonly direction: TreeWalkerDirection;\n\n\t/**\n\t * Iterator boundaries.\n\t *\n\t * When the iterator is walking `'forward'` on the end of boundary or is walking `'backward'`\n\t * on the start of boundary, then `{ done: true }` is returned.\n\t *\n\t * If boundaries are not defined they are set before first and after last child of the root node.\n\t */\n\tpublic readonly boundaries: Range | null;\n\n\t/**\n\t * Flag indicating whether all consecutive characters with the same attributes should be\n\t * returned as one {@link module:engine/model/textproxy~TextProxy} (`true`) or one by one (`false`).\n\t */\n\tpublic readonly singleCharacters: boolean;\n\n\t/**\n\t * Flag indicating whether iterator should enter elements or not. If the iterator is shallow child nodes of any\n\t * iterated node will not be returned along with `elementEnd` tag.\n\t */\n\tpublic readonly shallow: boolean;\n\n\t/**\n\t * Flag indicating whether iterator should ignore `elementEnd` tags. If the option is true walker will not\n\t * return a parent node of the start position. If this option is `true` each {@link module:engine/model/element~Element} will\n\t * be returned once, while if the option is `false` they might be returned twice:\n\t * for `'elementStart'` and `'elementEnd'`.\n\t */\n\tpublic readonly ignoreElementEnd: boolean;\n\n\t/**\n\t * Iterator position. This is always static position, even if the initial position was a\n\t * {@link module:engine/model/liveposition~LivePosition live position}. If start position is not defined then position depends\n\t * on {@link #direction}. If direction is `'forward'` position starts form the beginning, when direction\n\t * is `'backward'` position starts from the end.\n\t */\n\tprivate _position: Position;\n\n\t/**\n\t * Start boundary cached for optimization purposes.\n\t */\n\tprivate _boundaryStartParent: Element | DocumentFragment | null;\n\n\t/**\n\t * End boundary cached for optimization purposes.\n\t */\n\tprivate _boundaryEndParent: Element | DocumentFragment | null;\n\n\t/**\n\t * Parent of the most recently visited node. Cached for optimization purposes.\n\t */\n\tprivate _visitedParent: Element | DocumentFragment;\n\n\t/**\n\t * Creates a range iterator. All parameters are optional, but you have to specify either `boundaries` or `startPosition`.\n\t *\n\t * @param options Object with configuration.\n\t */\n\tconstructor( options: TreeWalkerOptions ) {\n\t\tif ( !options || ( !options.boundaries && !options.startPosition ) ) {\n\t\t\t/**\n\t\t\t * Neither boundaries nor starting position of a `TreeWalker` have been defined.\n\t\t\t *\n\t\t\t * @error model-tree-walker-no-start-position\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'model-tree-walker-no-start-position',\n\t\t\t\tnull\n\t\t\t);\n\t\t}\n\n\t\tconst direction = options.direction || 'forward';\n\n\t\tif ( direction != 'forward' && direction != 'backward' ) {\n\t\t\t/**\n\t\t\t * Only `backward` and `forward` direction allowed.\n\t\t\t *\n\t\t\t * @error model-tree-walker-unknown-direction\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-tree-walker-unknown-direction', options, { direction } );\n\t\t}\n\n\t\tthis.direction = direction;\n\t\tthis.boundaries = options.boundaries || null;\n\n\t\tif ( options.startPosition ) {\n\t\t\tthis._position = options.startPosition.clone();\n\t\t} else {\n\t\t\tthis._position = Position._createAt( this.boundaries![ this.direction == 'backward' ? 'end' : 'start' ] );\n\t\t}\n\n\t\t// Reset position stickiness in case it was set to other value, as the stickiness is kept after cloning.\n\t\tthis.position.stickiness = 'toNone';\n\n\t\tthis.singleCharacters = !!options.singleCharacters;\n\t\tthis.shallow = !!options.shallow;\n\t\tthis.ignoreElementEnd = !!options.ignoreElementEnd;\n\n\t\tthis._boundaryStartParent = this.boundaries ? this.boundaries.start.parent : null;\n\t\tthis._boundaryEndParent = this.boundaries ? this.boundaries.end.parent : null;\n\t\tthis._visitedParent = this.position.parent;\n\t}\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * @returns {Iterable.<module:engine/model/treewalker~TreeWalkerValue>}\n\t */\n\tpublic [ Symbol.iterator ](): IterableIterator<TreeWalkerValue> {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Iterator position. This is always static position, even if the initial position was a\n\t * {@link module:engine/model/liveposition~LivePosition live position}. If start position is not defined then position depends\n\t * on {@link #direction}. If direction is `'forward'` position starts form the beginning, when direction\n\t * is `'backward'` position starts from the end.\n\t */\n\tpublic get position(): Position {\n\t\treturn this._position;\n\t}\n\n\t/**\n\t * Moves {@link #position} in the {@link #direction} skipping values as long as the callback function returns `true`.\n\t *\n\t * For example:\n\t *\n\t * ```ts\n\t * walker.skip( value => value.type == 'text' ); // <paragraph>[]foo</paragraph> -> <paragraph>foo[]</paragraph>\n\t * walker.skip( () => true ); // Move the position to the end: <paragraph>[]foo</paragraph> -> <paragraph>foo</paragraph>[]\n\t * walker.skip( () => false ); // Do not move the position.\n\t * ```\n\t *\n\t * @param skip Callback function. Gets {@link module:engine/model/treewalker~TreeWalkerValue} and should\n\t * return `true` if the value should be skipped or `false` if not.\n\t */\n\tpublic skip( skip: ( value: TreeWalkerValue ) => boolean ): void {\n\t\tlet done, value, prevPosition, prevVisitedParent;\n\n\t\tdo {\n\t\t\tprevPosition = this.position;\n\t\t\tprevVisitedParent = this._visitedParent;\n\n\t\t\t( { done, value } = this.next() );\n\t\t} while ( !done && skip( value ) );\n\n\t\tif ( !done ) {\n\t\t\tthis._position = prevPosition;\n\t\t\tthis._visitedParent = prevVisitedParent;\n\t\t}\n\t}\n\n\t/**\n\t * Gets the next tree walker's value.\n\t */\n\tpublic next(): IteratorResult<TreeWalkerValue> {\n\t\tif ( this.direction == 'forward' ) {\n\t\t\treturn this._next();\n\t\t} else {\n\t\t\treturn this._previous();\n\t\t}\n\t}\n\n\t/**\n\t * Makes a step forward in model. Moves the {@link #position} to the next position and returns the encountered value.\n\t */\n\tprivate _next(): IteratorResult<TreeWalkerValue> {\n\t\tconst previousPosition = this.position;\n\t\tconst position = this.position.clone();\n\t\tconst parent = this._visitedParent;\n\n\t\t// We are at the end of the root.\n\t\tif ( parent.parent === null && position.offset === parent.maxOffset ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// We reached the walker boundary.\n\t\tif ( parent === this._boundaryEndParent && position.offset == this.boundaries!.end.offset ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// Get node just after the current position.\n\t\t// Use a highly optimized version instead of checking the text node first and then getting the node after. See #6582.\n\t\tconst textNodeAtPosition = getTextNodeAtPosition( position, parent );\n\t\tconst node = textNodeAtPosition || getNodeAfterPosition( position, parent, textNodeAtPosition );\n\n\t\tif ( node instanceof Element ) {\n\t\t\tif ( !this.shallow ) {\n\t\t\t\t// Manual operations on path internals for optimization purposes. Here and in the rest of the method.\n\t\t\t\t( position.path as Array<number> ).push( 0 );\n\t\t\t\tthis._visitedParent = node;\n\t\t\t} else {\n\t\t\t\t// We are past the walker boundaries.\n\t\t\t\tif ( this.boundaries && this.boundaries.end.isBefore( position ) ) {\n\t\t\t\t\treturn { done: true, value: undefined };\n\t\t\t\t}\n\n\t\t\t\tposition.offset++;\n\t\t\t}\n\n\t\t\tthis._position = position;\n\n\t\t\treturn formatReturnValue( 'elementStart', node, previousPosition, position, 1 );\n\t\t}\n\n\t\tif ( node instanceof Text ) {\n\t\t\tlet charactersCount;\n\n\t\t\tif ( this.singleCharacters ) {\n\t\t\t\tcharactersCount = 1;\n\t\t\t} else {\n\t\t\t\tlet offset = node.endOffset!;\n\n\t\t\t\tif ( this._boundaryEndParent == parent && this.boundaries!.end.offset < offset ) {\n\t\t\t\t\toffset = this.boundaries!.end.offset;\n\t\t\t\t}\n\n\t\t\t\tcharactersCount = offset - position.offset;\n\t\t\t}\n\n\t\t\tconst offsetInTextNode = position.offset - node.startOffset!;\n\t\t\tconst item = new TextProxy( node, offsetInTextNode, charactersCount );\n\n\t\t\tposition.offset += charactersCount;\n\t\t\tthis._position = position;\n\n\t\t\treturn formatReturnValue( 'text', item, previousPosition, position, charactersCount );\n\t\t}\n\n\t\t// `node` is not set, we reached the end of current `parent`.\n\t\t( position.path as Array<number> ).pop();\n\t\tposition.offset++;\n\t\tthis._position = position;\n\t\tthis._visitedParent = parent.parent!;\n\n\t\tif ( this.ignoreElementEnd ) {\n\t\t\treturn this._next();\n\t\t}\n\n\t\treturn formatReturnValue( 'elementEnd', parent as Element, previousPosition, position );\n\t}\n\n\t/**\n\t * Makes a step backward in model. Moves the {@link #position} to the previous position and returns the encountered value.\n\t */\n\tprivate _previous(): IteratorResult<TreeWalkerValue> {\n\t\tconst previousPosition = this.position;\n\t\tconst position = this.position.clone();\n\t\tconst parent = this._visitedParent;\n\n\t\t// We are at the beginning of the root.\n\t\tif ( parent.parent === null && position.offset === 0 ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// We reached the walker boundary.\n\t\tif ( parent == this._boundaryStartParent && position.offset == this.boundaries!.start.offset ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// Get node just before the current position.\n\t\t// Use a highly optimized version instead of checking the text node first and then getting the node before. See #6582.\n\t\tconst positionParent = position.parent;\n\t\tconst textNodeAtPosition = getTextNodeAtPosition( position, positionParent );\n\t\tconst node = textNodeAtPosition || getNodeBeforePosition( position, positionParent, textNodeAtPosition );\n\n\t\tif ( node instanceof Element ) {\n\t\t\tposition.offset--;\n\n\t\t\tif ( this.shallow ) {\n\t\t\t\tthis._position = position;\n\n\t\t\t\treturn formatReturnValue( 'elementStart', node, previousPosition, position, 1 );\n\t\t\t}\n\n\t\t\t( position.path as Array<number> ).push( node.maxOffset );\n\t\t\tthis._position = position;\n\t\t\tthis._visitedParent = node;\n\n\t\t\tif ( this.ignoreElementEnd ) {\n\t\t\t\treturn this._previous();\n\t\t\t}\n\n\t\t\treturn formatReturnValue( 'elementEnd', node, previousPosition, position );\n\t\t}\n\n\t\tif ( node instanceof Text ) {\n\t\t\tlet charactersCount;\n\n\t\t\tif ( this.singleCharacters ) {\n\t\t\t\tcharactersCount = 1;\n\t\t\t} else {\n\t\t\t\tlet offset = node.startOffset!;\n\n\t\t\t\tif ( this._boundaryStartParent == parent && this.boundaries!.start.offset > offset ) {\n\t\t\t\t\toffset = this.boundaries!.start.offset;\n\t\t\t\t}\n\n\t\t\t\tcharactersCount = position.offset - offset;\n\t\t\t}\n\n\t\t\tconst offsetInTextNode = position.offset - node.startOffset!;\n\t\t\tconst item = new TextProxy( node, offsetInTextNode - charactersCount, charactersCount );\n\n\t\t\tposition.offset -= charactersCount;\n\t\t\tthis._position = position;\n\n\t\t\treturn formatReturnValue( 'text', item, previousPosition, position, charactersCount );\n\t\t}\n\n\t\t// `node` is not set, we reached the beginning of current `parent`.\n\t\t( position.path as Array<number> ).pop();\n\t\tthis._position = position;\n\t\tthis._visitedParent = parent.parent!;\n\n\t\treturn formatReturnValue( 'elementStart', parent as Element, previousPosition, position, 1 );\n\t}\n}\n\nfunction formatReturnValue(\n\ttype: TreeWalkerValueType,\n\titem: Item,\n\tpreviousPosition: Position,\n\tnextPosition: Position,\n\tlength?: number\n): IteratorYieldResult<TreeWalkerValue> {\n\treturn {\n\t\tdone: false,\n\t\tvalue: {\n\t\t\ttype,\n\t\t\titem,\n\t\t\tpreviousPosition,\n\t\t\tnextPosition,\n\t\t\tlength\n\t\t}\n\t};\n}\n\n/**\n * Type of the step made by {@link module:engine/model/treewalker~TreeWalker}.\n * Possible values: `'elementStart'` if walker is at the beginning of a node, `'elementEnd'` if walker is at the end of node,\n * or `'text'` if walker traversed over text.\n */\nexport type TreeWalkerValueType = 'elementStart' | 'elementEnd' | 'text';\n\n/**\n * Object returned by {@link module:engine/model/treewalker~TreeWalker} when traversing tree model.\n */\nexport interface TreeWalkerValue {\n\ttype: TreeWalkerValueType;\n\n\t/**\n\t * Item between old and new positions of {@link module:engine/model/treewalker~TreeWalker}.\n\t */\n\titem: Item;\n\n\t/**\n\t * Previous position of the iterator.\n\t * * Forward iteration: For `'elementEnd'` it is the last position inside the element. For all other types it is the\n\t * position before the item.\n\t * * Backward iteration: For `'elementStart'` it is the first position inside the element. For all other types it is\n\t * the position after item.\n\t */\n\tpreviousPosition: Position;\n\n\t/**\n\t * Next position of the iterator.\n\t * * Forward iteration: For `'elementStart'` it is the first position inside the element. For all other types it is\n\t * the position after the item.\n\t * * Backward iteration: For `'elementEnd'` it is last position inside element. For all other types it is the position\n\t * before the item.\n\t */\n\tnextPosition: Position;\n\n\t/**\n\t * Length of the item. For `'elementStart'` it is 1. For `'text'` it is the length of the text. For `'elementEnd'` it is `undefined`.\n\t */\n\tlength?: number;\n}\n\n/**\n * Tree walking direction.\n */\nexport type TreeWalkerDirection = 'forward' | 'backward';\n\n/**\n * The configuration of TreeWalker.\n *\n * All parameters are optional, but you have to specify either `boundaries` or `startPosition`.\n */\nexport interface TreeWalkerOptions {\n\n\t/**\n\t * Walking direction.\n\t *\n\t * @default 'forward'\n\t */\n\tdirection?: TreeWalkerDirection;\n\n\t/**\n\t * Range to define boundaries of the iterator.\n\t */\n\tboundaries?: Range | null;\n\n\t/**\n\t * Starting position.\n\t */\n\tstartPosition?: Position;\n\n\t/**\n\t * Flag indicating whether all consecutive characters with the same attributes\n\t * should be returned one by one as multiple {@link module:engine/model/textproxy~TextProxy} (`true`) objects or as one\n\t * {@link module:engine/model/textproxy~TextProxy} (`false`).\n\t */\n\tsingleCharacters?: boolean;\n\n\t/**\n\t * Flag indicating whether iterator should enter elements or not. If the\n\t * iterator is shallow child nodes of any iterated node will not be returned along with `elementEnd` tag.\n\t */\n\tshallow?: boolean;\n\n\t/**\n\t * Flag indicating whether iterator should ignore `elementEnd` tags.\n\t * If the option is true walker will not return a parent node of start position. If this option is `true`\n\t * each {@link module:engine/model/element~Element} will be returned once, while if the option is `false` they might be returned\n\t * twice: for `'elementStart'` and `'elementEnd'`.\n\t */\n\tignoreElementEnd?: boolean;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/position\n */\n\nimport TypeCheckable from './typecheckable';\nimport TreeWalker, { type TreeWalkerOptions, type TreeWalkerValue } from './treewalker';\n\nimport type Document from './document';\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\nimport type InsertOperation from './operation/insertoperation';\nimport type Item from './item';\nimport type MergeOperation from './operation/mergeoperation';\nimport type MoveOperation from './operation/moveoperation';\nimport type Node from './node';\nimport type Operation from './operation/operation';\nimport type SplitOperation from './operation/splitoperation';\nimport type Text from './text';\n\nimport { CKEditorError, compareArrays } from '@ckeditor/ckeditor5-utils';\n\n// To check if component is loaded more than once.\nimport '@ckeditor/ckeditor5-utils/src/version';\n\n/**\n * Represents a position in the model tree.\n *\n * A position is represented by its {@link module:engine/model/position~Position#root} and\n * a {@link module:engine/model/position~Position#path} in that root.\n *\n * You can create position instances via its constructor or the `createPosition*()` factory methods of\n * {@link module:engine/model/model~Model} and {@link module:engine/model/writer~Writer}.\n *\n * **Note:** Position is based on offsets, not indexes. This means that a position between two text nodes\n * `foo` and `bar` has offset `3`, not `1`. See {@link module:engine/model/position~Position#path} for more information.\n *\n * Since a position in the model is represented by a {@link module:engine/model/position~Position#root position root} and\n * {@link module:engine/model/position~Position#path position path} it is possible to create positions placed in non-existing places.\n * This requirement is important for operational transformation algorithms.\n *\n * Also, {@link module:engine/model/operation/operation~Operation operations}\n * kept in the {@link module:engine/model/document~Document#history document history}\n * are storing positions (and ranges) which were correct when those operations were applied, but may not be correct\n * after the document has changed.\n *\n * When changes are applied to the model, it may also happen that {@link module:engine/model/position~Position#parent position parent}\n * will change even if position path has not changed. Keep in mind, that if a position leads to non-existing element,\n * {@link module:engine/model/position~Position#parent} and some other properties and methods will throw errors.\n *\n * In most cases, position with wrong path is caused by an error in code, but it is sometimes needed, as described above.\n */\nexport default class Position extends TypeCheckable {\n\t/**\n\t * Root of the position path.\n\t */\n\tpublic readonly root: Element | DocumentFragment;\n\n\t/**\n\t * Position of the node in the tree. **Path contains offsets, not indexes.**\n\t *\n\t * Position can be placed before, after or in a {@link module:engine/model/node~Node node} if that node has\n\t * {@link module:engine/model/node~Node#offsetSize} greater than `1`. Items in position path are\n\t * {@link module:engine/model/node~Node#startOffset starting offsets} of position ancestors, starting from direct root children,\n\t * down to the position offset in it's parent.\n\t *\n\t * ```\n\t * ROOT\n\t * |- P before: [ 0 ] after: [ 1 ]\n\t * |- UL before: [ 1 ] after: [ 2 ]\n\t * |- LI before: [ 1, 0 ] after: [ 1, 1 ]\n\t * | |- foo before: [ 1, 0, 0 ] after: [ 1, 0, 3 ]\n\t * |- LI before: [ 1, 1 ] after: [ 1, 2 ]\n\t * |- bar before: [ 1, 1, 0 ] after: [ 1, 1, 3 ]\n\t * ```\n\t *\n\t * `foo` and `bar` are representing {@link module:engine/model/text~Text text nodes}. Since text nodes has offset size\n\t * greater than `1` you can place position offset between their start and end:\n\t *\n\t * ```\n\t * ROOT\n\t * |- P\n\t * |- UL\n\t * |- LI\n\t * | |- f^o|o ^ has path: [ 1, 0, 1 ] | has path: [ 1, 0, 2 ]\n\t * |- LI\n\t * |- b^a|r ^ has path: [ 1, 1, 1 ] | has path: [ 1, 1, 2 ]\n\t * ```\n\t */\n\tpublic readonly path: ReadonlyArray<number>;\n\n\t/**\n\t * Position stickiness. See {@link module:engine/model/position~PositionStickiness}.\n\t */\n\tpublic stickiness: PositionStickiness;\n\n\t/**\n\t * Creates a position.\n\t *\n\t * @param root Root of the position.\n\t * @param path Position path. See {@link module:engine/model/position~Position#path}.\n\t * @param stickiness Position stickiness. See {@link module:engine/model/position~PositionStickiness}.\n\t */\n\tconstructor(\n\t\troot: Element | DocumentFragment,\n\t\tpath: ReadonlyArray<number>,\n\t\tstickiness: PositionStickiness = 'toNone'\n\t) {\n\t\tsuper();\n\n\t\tif ( !root.is( 'element' ) && !root.is( 'documentFragment' ) ) {\n\t\t\t/**\n\t\t\t * Position root is invalid.\n\t\t\t *\n\t\t\t * Positions can only be anchored in elements or document fragments.\n\t\t\t *\n\t\t\t * @error model-position-root-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'model-position-root-invalid',\n\t\t\t\troot\n\t\t\t);\n\t\t}\n\n\t\tif ( !( path instanceof Array ) || path.length === 0 ) {\n\t\t\t/**\n\t\t\t * Position path must be an array with at least one item.\n\t\t\t *\n\t\t\t * @error model-position-path-incorrect-format\n\t\t\t * @param path\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'model-position-path-incorrect-format',\n\t\t\t\troot,\n\t\t\t\t{ path }\n\t\t\t);\n\t\t}\n\n\t\t// Normalize the root and path when element (not root) is passed.\n\t\tif ( root.is( 'rootElement' ) ) {\n\t\t\tpath = path.slice();\n\t\t} else {\n\t\t\tpath = [ ...root.getPath(), ...path ];\n\t\t\troot = root.root as any;\n\t\t}\n\n\t\tthis.root = root;\n\t\tthis.path = path;\n\t\tthis.stickiness = stickiness;\n\t}\n\n\t/**\n\t * Offset at which this position is located in its {@link module:engine/model/position~Position#parent parent}. It is equal\n\t * to the last item in position {@link module:engine/model/position~Position#path path}.\n\t *\n\t * @type {Number}\n\t */\n\tpublic get offset(): number {\n\t\treturn this.path[ this.path.length - 1 ];\n\t}\n\n\tpublic set offset( newOffset: number ) {\n\t\t( this.path as Array<number> )[ this.path.length - 1 ] = newOffset;\n\t}\n\n\t/**\n\t * Parent element of this position.\n\t *\n\t * Keep in mind that `parent` value is calculated when the property is accessed.\n\t * If {@link module:engine/model/position~Position#path position path}\n\t * leads to a non-existing element, `parent` property will throw error.\n\t *\n\t * Also it is a good idea to cache `parent` property if it is used frequently in an algorithm (i.e. in a long loop).\n\t */\n\tpublic get parent(): Element | DocumentFragment {\n\t\tlet parent: any = this.root;\n\n\t\tfor ( let i = 0; i < this.path.length - 1; i++ ) {\n\t\t\tparent = parent.getChild( parent.offsetToIndex( this.path[ i ] ) );\n\n\t\t\tif ( !parent ) {\n\t\t\t\t/**\n\t\t\t\t * The position's path is incorrect. This means that a position does not point to\n\t\t\t\t * a correct place in the tree and hence, some of its methods and getters cannot work correctly.\n\t\t\t\t *\n\t\t\t\t * **Note**: Unlike DOM and view positions, in the model, the\n\t\t\t\t * {@link module:engine/model/position~Position#parent position's parent} is always an element or a document fragment.\n\t\t\t\t * The last offset in the {@link module:engine/model/position~Position#path position's path} is the point in this element\n\t\t\t\t * where this position points.\n\t\t\t\t *\n\t\t\t\t * Read more about model positions and offsets in\n\t\t\t\t * the {@glink framework/architecture/editing-engine#indexes-and-offsets Editing engine architecture} guide.\n\t\t\t\t *\n\t\t\t\t * @error model-position-path-incorrect\n\t\t\t\t * @param position The incorrect position.\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'model-position-path-incorrect', this, { position: this } );\n\t\t\t}\n\t\t}\n\n\t\tif ( parent.is( '$text' ) ) {\n\t\t\tthrow new CKEditorError( 'model-position-path-incorrect', this, { position: this } );\n\t\t}\n\n\t\treturn parent;\n\t}\n\n\t/**\n\t * Position {@link module:engine/model/position~Position#offset offset} converted to an index in position's parent node. It is\n\t * equal to the {@link module:engine/model/node~Node#index index} of a node after this position. If position is placed\n\t * in text node, position index is equal to the index of that text node.\n\t */\n\tpublic get index(): number {\n\t\treturn this.parent.offsetToIndex( this.offset );\n\t}\n\n\t/**\n\t * Returns {@link module:engine/model/text~Text text node} instance in which this position is placed or `null` if this\n\t * position is not in a text node.\n\t */\n\tpublic get textNode(): Text | null {\n\t\treturn getTextNodeAtPosition( this, this.parent );\n\t}\n\n\t/**\n\t * Node directly after this position or `null` if this position is in text node.\n\t */\n\tpublic get nodeAfter(): Node | null {\n\t\t// Cache the parent and reuse for performance reasons. See #6579 and #6582.\n\t\tconst parent = this.parent;\n\n\t\treturn getNodeAfterPosition( this, parent, getTextNodeAtPosition( this, parent ) );\n\t}\n\n\t/**\n\t * Node directly before this position or `null` if this position is in text node.\n\t */\n\tpublic get nodeBefore(): Node | null {\n\t\t// Cache the parent and reuse for performance reasons. See #6579 and #6582.\n\t\tconst parent = this.parent;\n\n\t\treturn getNodeBeforePosition( this, parent, getTextNodeAtPosition( this, parent ) );\n\t}\n\n\t/**\n\t * Is `true` if position is at the beginning of its {@link module:engine/model/position~Position#parent parent}, `false` otherwise.\n\t */\n\tpublic get isAtStart(): boolean {\n\t\treturn this.offset === 0;\n\t}\n\n\t/**\n\t * Is `true` if position is at the end of its {@link module:engine/model/position~Position#parent parent}, `false` otherwise.\n\t */\n\tpublic get isAtEnd(): boolean {\n\t\treturn this.offset == this.parent.maxOffset;\n\t}\n\n\t/**\n\t * Checks whether this position is before or after given position.\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t */\n\tpublic compareWith( otherPosition: Position ): PositionRelation {\n\t\tif ( this.root != otherPosition.root ) {\n\t\t\treturn 'different';\n\t\t}\n\n\t\tconst result = compareArrays( this.path, otherPosition.path );\n\n\t\tswitch ( result ) {\n\t\t\tcase 'same':\n\t\t\t\treturn 'same';\n\n\t\t\tcase 'prefix':\n\t\t\t\treturn 'before';\n\n\t\t\tcase 'extension':\n\t\t\t\treturn 'after';\n\n\t\t\tdefault:\n\t\t\t\treturn this.path[ result ] < otherPosition.path[ result ] ? 'before' : 'after';\n\t\t}\n\t}\n\n\t/**\n\t * Gets the farthest position which matches the callback using\n\t * {@link module:engine/model/treewalker~TreeWalker TreeWalker}.\n\t *\n\t * For example:\n\t *\n\t * ```ts\n\t * getLastMatchingPosition( value => value.type == 'text' );\n\t * // <paragraph>[]foo</paragraph> -> <paragraph>foo[]</paragraph>\n\t *\n\t * getLastMatchingPosition( value => value.type == 'text', { direction: 'backward' } );\n\t * // <paragraph>foo[]</paragraph> -> <paragraph>[]foo</paragraph>\n\t *\n\t * getLastMatchingPosition( value => false );\n\t * // Do not move the position.\n\t * ```\n\t *\n\t * @param skip Callback function. Gets {@link module:engine/model/treewalker~TreeWalkerValue} and should\n\t * return `true` if the value should be skipped or `false` if not.\n\t * @param options Object with configuration options. See {@link module:engine/model/treewalker~TreeWalker}.\n\t *\n\t * @returns The position after the last item which matches the `skip` callback test.\n\t */\n\tpublic getLastMatchingPosition(\n\t\tskip: ( value: TreeWalkerValue ) => boolean,\n\t\toptions: TreeWalkerOptions = {}\n\t): Position {\n\t\toptions.startPosition = this;\n\n\t\tconst treeWalker = new TreeWalker( options );\n\t\ttreeWalker.skip( skip );\n\n\t\treturn treeWalker.position;\n\t}\n\n\t/**\n\t * Returns a path to this position's parent. Parent path is equal to position {@link module:engine/model/position~Position#path path}\n\t * but without the last item.\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t *\n\t * @returns Path to the parent.\n\t */\n\tpublic getParentPath(): Array<number> {\n\t\treturn this.path.slice( 0, -1 );\n\t}\n\n\t/**\n\t * Returns ancestors array of this position, that is this position's parent and its ancestors.\n\t *\n\t * @returns Array with ancestors.\n\t */\n\tpublic getAncestors(): Array<Element | DocumentFragment> {\n\t\tconst parent = this.parent;\n\n\t\tif ( parent.is( 'documentFragment' ) ) {\n\t\t\treturn [ parent ];\n\t\t} else {\n\t\t\treturn parent.getAncestors( { includeSelf: true } ) as any;\n\t\t}\n\t}\n\n\t/**\n\t * Returns the parent element of the given name. Returns null if the position is not inside the desired parent.\n\t *\n\t * @param parentName The name of the parent element to find.\n\t */\n\tpublic findAncestor( parentName: string ): Element | null {\n\t\tconst parent = this.parent;\n\n\t\tif ( parent.is( 'element' ) ) {\n\t\t\treturn parent.findAncestor( parentName, { includeSelf: true } );\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns the slice of two position {@link #path paths} which is identical. The {@link #root roots}\n\t * of these two paths must be identical.\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t *\n\t * @param position The second position.\n\t * @returns The common path.\n\t */\n\tpublic getCommonPath( position: Position ): Array<number> {\n\t\tif ( this.root != position.root ) {\n\t\t\treturn [];\n\t\t}\n\n\t\t// We find on which tree-level start and end have the lowest common ancestor\n\t\tconst cmp = compareArrays( this.path, position.path );\n\t\t// If comparison returned string it means that arrays are same.\n\t\tconst diffAt = ( typeof cmp == 'string' ) ? Math.min( this.path.length, position.path.length ) : cmp;\n\n\t\treturn this.path.slice( 0, diffAt );\n\t}\n\n\t/**\n\t * Returns an {@link module:engine/model/element~Element} or {@link module:engine/model/documentfragment~DocumentFragment}\n\t * which is a common ancestor of both positions. The {@link #root roots} of these two positions must be identical.\n\t *\n\t * @param position The second position.\n\t */\n\tpublic getCommonAncestor( position: Position ): Element | DocumentFragment | null {\n\t\tconst ancestorsA = this.getAncestors();\n\t\tconst ancestorsB = position.getAncestors();\n\n\t\tlet i = 0;\n\n\t\twhile ( ancestorsA[ i ] == ancestorsB[ i ] && ancestorsA[ i ] ) {\n\t\t\ti++;\n\t\t}\n\n\t\treturn i === 0 ? null : ancestorsA[ i - 1 ];\n\t}\n\n\t/**\n\t * Returns a new instance of `Position`, that has same {@link #parent parent} but it's offset\n\t * is shifted by `shift` value (can be a negative value).\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t *\n\t * @param shift Offset shift. Can be a negative value.\n\t * @returns Shifted position.\n\t */\n\tpublic getShiftedBy( shift: number ): Position {\n\t\tconst shifted = this.clone();\n\n\t\tconst offset = shifted.offset + shift;\n\t\tshifted.offset = offset < 0 ? 0 : offset;\n\n\t\treturn shifted;\n\t}\n\n\t/**\n\t * Checks whether this position is after given position.\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t *\n\t * @see module:engine/model/position~Position#isBefore\n\t * @param otherPosition Position to compare with.\n\t * @returns True if this position is after given position.\n\t */\n\tpublic isAfter( otherPosition: Position ): boolean {\n\t\treturn this.compareWith( otherPosition ) == 'after';\n\t}\n\n\t/**\n\t * Checks whether this position is before given position.\n\t *\n\t * **Note:** watch out when using negation of the value returned by this method, because the negation will also\n\t * be `true` if positions are in different roots and you might not expect this. You should probably use\n\t * `a.isAfter( b ) || a.isEqual( b )` or `!a.isBefore( p ) && a.root == b.root` in most scenarios. If your\n\t * condition uses multiple `isAfter` and `isBefore` checks, build them so they do not use negated values, i.e.:\n\t *\n\t * ```ts\n\t * if ( a.isBefore( b ) && c.isAfter( d ) ) {\n\t * \t// do A.\n\t * } else {\n\t * \t// do B.\n\t * }\n\t * ```\n\t *\n\t * or, if you have only one if-branch:\n\t *\n\t * ```ts\n\t * if ( !( a.isBefore( b ) && c.isAfter( d ) ) {\n\t * \t// do B.\n\t * }\n\t * ```\n\t *\n\t * rather than:\n\t *\n\t * ```ts\n\t * if ( !a.isBefore( b ) || && !c.isAfter( d ) ) {\n\t * \t// do B.\n\t * } else {\n\t * \t// do A.\n\t * }\n\t * ```\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t *\n\t * @param otherPosition Position to compare with.\n\t * @returns True if this position is before given position.\n\t */\n\tpublic isBefore( otherPosition: Position ): boolean {\n\t\treturn this.compareWith( otherPosition ) == 'before';\n\t}\n\n\t/**\n\t * Checks whether this position is equal to given position.\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t *\n\t * @param otherPosition Position to compare with.\n\t * @returns True if positions are same.\n\t */\n\tpublic isEqual( otherPosition: Position ): boolean {\n\t\treturn this.compareWith( otherPosition ) == 'same';\n\t}\n\n\t/**\n\t * Checks whether this position is touching given position. Positions touch when there are no text nodes\n\t * or empty nodes in a range between them. Technically, those positions are not equal but in many cases\n\t * they are very similar or even indistinguishable.\n\t *\n\t * @param otherPosition Position to compare with.\n\t * @returns True if positions touch.\n\t */\n\tpublic isTouching( otherPosition: Position ): boolean {\n\t\tif ( this.root !== otherPosition.root ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst commonLevel = Math.min( this.path.length, otherPosition.path.length );\n\n\t\tfor ( let level = 0; level < commonLevel; level++ ) {\n\t\t\tconst diff = this.path[ level ] - otherPosition.path[ level ];\n\n\t\t\t// Positions are spread by a node, so they are not touching.\n\t\t\tif ( diff < -1 || diff > 1 ) {\n\t\t\t\treturn false;\n\t\t\t} else if ( diff === 1 ) {\n\t\t\t\t// `otherPosition` is on the left.\n\t\t\t\t// `this` is on the right.\n\t\t\t\treturn checkTouchingBranch( otherPosition, this, level );\n\t\t\t} else if ( diff === -1 ) {\n\t\t\t\t// `this` is on the left.\n\t\t\t\t// `otherPosition` is on the right.\n\t\t\t\treturn checkTouchingBranch( this, otherPosition, level );\n\t\t\t}\n\t\t\t// `diff === 0`.\n\t\t\t// Positions are inside the same element on this level, compare deeper.\n\t\t}\n\n\t\t// If we ended up here, it means that positions paths have the same beginning.\n\t\t// If the paths have the same length, then it means that they are identical, so the positions are same.\n\t\tif ( this.path.length === otherPosition.path.length ) {\n\t\t\treturn true;\n\t\t}\n\t\t// If positions have different length of paths, then the common part is the same.\n\t\t// In this case, the \"shorter\" position is on the left, the \"longer\" position is on the right.\n\t\t//\n\t\t// If the positions are touching, the \"longer\" position must have only zeroes. For example:\n\t\t// [ 1, 2 ] vs [ 1, 2, 0 ]\n\t\t// [ 1, 2 ] vs [ 1, 2, 0, 0, 0 ]\n\t\telse if ( this.path.length > otherPosition.path.length ) {\n\t\t\treturn checkOnlyZeroes( this.path, commonLevel );\n\t\t} else {\n\t\t\treturn checkOnlyZeroes( otherPosition.path, commonLevel );\n\t\t}\n\t}\n\n\t/**\n\t * Checks if two positions are in the same parent.\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t *\n\t * @param position Position to compare with.\n\t * @returns `true` if positions have the same parent, `false` otherwise.\n\t */\n\tpublic hasSameParentAs( position: Position ): boolean {\n\t\tif ( this.root !== position.root ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst thisParentPath = this.getParentPath();\n\t\tconst posParentPath = position.getParentPath();\n\n\t\treturn compareArrays( thisParentPath, posParentPath ) == 'same';\n\t}\n\n\t/**\n\t * Returns a copy of this position that is transformed by given `operation`.\n\t *\n\t * The new position's parameters are updated accordingly to the effect of the `operation`.\n\t *\n\t * For example, if `n` nodes are inserted before the position, the returned position {@link ~Position#offset} will be\n\t * increased by `n`. If the position was in a merged element, it will be accordingly moved to the new element, etc.\n\t *\n\t * This method is safe to use it on non-existing positions (for example during operational transformation).\n\t *\n\t * @param operation Operation to transform by.\n\t * @returns Transformed position.\n\t */\n\tpublic getTransformedByOperation( operation: Operation ): Position {\n\t\tlet result;\n\n\t\tswitch ( operation.type ) {\n\t\t\tcase 'insert':\n\t\t\t\tresult = this._getTransformedByInsertOperation( operation as InsertOperation );\n\t\t\t\tbreak;\n\t\t\tcase 'move':\n\t\t\tcase 'remove':\n\t\t\tcase 'reinsert':\n\t\t\t\tresult = this._getTransformedByMoveOperation( operation as MoveOperation );\n\t\t\t\tbreak;\n\t\t\tcase 'split':\n\t\t\t\tresult = this._getTransformedBySplitOperation( operation as SplitOperation );\n\t\t\t\tbreak;\n\t\t\tcase 'merge':\n\t\t\t\tresult = this._getTransformedByMergeOperation( operation as MergeOperation );\n\t\t\t\tbreak;\n\t\t\tdefault:\n\t\t\t\tresult = Position._createAt( this );\n\t\t\t\tbreak;\n\t\t}\n\n\t\treturn result;\n\t}\n\n\t/**\n\t * Returns a copy of this position transformed by an insert operation.\n\t *\n\t * @internal\n\t */\n\tpublic _getTransformedByInsertOperation( operation: InsertOperation ): Position {\n\t\treturn this._getTransformedByInsertion( operation.position, operation.howMany );\n\t}\n\n\t/**\n\t * Returns a copy of this position transformed by a move operation.\n\t *\n\t * @internal\n\t */\n\tpublic _getTransformedByMoveOperation( operation: MoveOperation ): Position {\n\t\treturn this._getTransformedByMove( operation.sourcePosition, operation.targetPosition, operation.howMany );\n\t}\n\n\t/**\n\t * Returns a copy of this position transformed by a split operation.\n\t *\n\t * @internal\n\t */\n\tpublic _getTransformedBySplitOperation( operation: SplitOperation ): Position {\n\t\tconst movedRange = operation.movedRange;\n\n\t\tconst isContained = movedRange.containsPosition( this ) ||\n\t\t\t( movedRange.start.isEqual( this ) && this.stickiness == 'toNext' );\n\n\t\tif ( isContained ) {\n\t\t\treturn this._getCombined( operation.splitPosition, operation.moveTargetPosition );\n\t\t} else {\n\t\t\tif ( operation.graveyardPosition ) {\n\t\t\t\treturn this._getTransformedByMove( operation.graveyardPosition, operation.insertionPosition, 1 );\n\t\t\t} else {\n\t\t\t\treturn this._getTransformedByInsertion( operation.insertionPosition, 1 );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Returns a copy of this position transformed by merge operation.\n\t *\n\t * @internal\n\t */\n\tpublic _getTransformedByMergeOperation( operation: MergeOperation ): Position {\n\t\tconst movedRange = operation.movedRange;\n\t\tconst isContained = movedRange.containsPosition( this ) || movedRange.start.isEqual( this );\n\n\t\tlet pos;\n\n\t\tif ( isContained ) {\n\t\t\tpos = this._getCombined( operation.sourcePosition, operation.targetPosition );\n\n\t\t\tif ( operation.sourcePosition.isBefore( operation.targetPosition ) ) {\n\t\t\t\t// Above happens during OT when the merged element is moved before the merged-to element.\n\t\t\t\tpos = pos._getTransformedByDeletion( operation.deletionPosition, 1 )!;\n\t\t\t}\n\t\t} else if ( this.isEqual( operation.deletionPosition ) ) {\n\t\t\tpos = Position._createAt( operation.deletionPosition );\n\t\t} else {\n\t\t\tpos = this._getTransformedByMove( operation.deletionPosition, operation.graveyardPosition, 1 );\n\t\t}\n\n\t\treturn pos;\n\t}\n\n\t/**\n\t * Returns a copy of this position that is updated by removing `howMany` nodes starting from `deletePosition`.\n\t * It may happen that this position is in a removed node. If that is the case, `null` is returned instead.\n\t *\n\t * @internal\n\t * @param deletePosition Position before the first removed node.\n\t * @param howMany How many nodes are removed.\n\t * @returns Transformed position or `null`.\n\t */\n\tpublic _getTransformedByDeletion( deletePosition: Position, howMany: number ): Position | null {\n\t\tconst transformed = Position._createAt( this );\n\n\t\t// This position can't be affected if deletion was in a different root.\n\t\tif ( this.root != deletePosition.root ) {\n\t\t\treturn transformed;\n\t\t}\n\n\t\tif ( compareArrays( deletePosition.getParentPath(), this.getParentPath() ) == 'same' ) {\n\t\t\t// If nodes are removed from the node that is pointed by this position...\n\t\t\tif ( deletePosition.offset < this.offset ) {\n\t\t\t\t// And are removed from before an offset of that position...\n\t\t\t\tif ( deletePosition.offset + howMany > this.offset ) {\n\t\t\t\t\t// Position is in removed range, it's no longer in the tree.\n\t\t\t\t\treturn null;\n\t\t\t\t} else {\n\t\t\t\t\t// Decrement the offset accordingly.\n\t\t\t\t\ttransformed.offset -= howMany;\n\t\t\t\t}\n\t\t\t}\n\t\t} else if ( compareArrays( deletePosition.getParentPath(), this.getParentPath() ) == 'prefix' ) {\n\t\t\t// If nodes are removed from a node that is on a path to this position...\n\t\t\tconst i = deletePosition.path.length - 1;\n\n\t\t\tif ( deletePosition.offset <= this.path[ i ] ) {\n\t\t\t\t// And are removed from before next node of that path...\n\t\t\t\tif ( deletePosition.offset + howMany > this.path[ i ] ) {\n\t\t\t\t\t// If the next node of that path is removed return null\n\t\t\t\t\t// because the node containing this position got removed.\n\t\t\t\t\treturn null;\n\t\t\t\t} else {\n\t\t\t\t\t// Otherwise, decrement index on that path.\n\t\t\t\t\t( transformed.path as Array<number> )[ i ] -= howMany;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn transformed;\n\t}\n\n\t/**\n\t * Returns a copy of this position that is updated by inserting `howMany` nodes at `insertPosition`.\n\t *\n\t * @internal\n\t * @param insertPosition Position where nodes are inserted.\n\t * @param howMany How many nodes are inserted.\n\t * @returns Transformed position.\n\t */\n\tpublic _getTransformedByInsertion( insertPosition: Position, howMany: number ): Position {\n\t\tconst transformed = Position._createAt( this );\n\n\t\t// This position can't be affected if insertion was in a different root.\n\t\tif ( this.root != insertPosition.root ) {\n\t\t\treturn transformed;\n\t\t}\n\n\t\tif ( compareArrays( insertPosition.getParentPath(), this.getParentPath() ) == 'same' ) {\n\t\t\t// If nodes are inserted in the node that is pointed by this position...\n\t\t\tif ( insertPosition.offset < this.offset || ( insertPosition.offset == this.offset && this.stickiness != 'toPrevious' ) ) {\n\t\t\t\t// And are inserted before an offset of that position...\n\t\t\t\t// \"Push\" this positions offset.\n\t\t\t\ttransformed.offset += howMany;\n\t\t\t}\n\t\t} else if ( compareArrays( insertPosition.getParentPath(), this.getParentPath() ) == 'prefix' ) {\n\t\t\t// If nodes are inserted in a node that is on a path to this position...\n\t\t\tconst i = insertPosition.path.length - 1;\n\n\t\t\tif ( insertPosition.offset <= this.path[ i ] ) {\n\t\t\t\t// And are inserted before next node of that path...\n\t\t\t\t// \"Push\" the index on that path.\n\t\t\t\t( transformed.path as Array<number> )[ i ] += howMany;\n\t\t\t}\n\t\t}\n\n\t\treturn transformed;\n\t}\n\n\t/**\n\t * Returns a copy of this position that is updated by moving `howMany` nodes from `sourcePosition` to `targetPosition`.\n\t *\n\t * @internal\n\t * @param sourcePosition Position before the first element to move.\n\t * @param targetPosition Position where moved elements will be inserted.\n\t * @param howMany How many consecutive nodes to move, starting from `sourcePosition`.\n\t * @returns Transformed position.\n\t */\n\tpublic _getTransformedByMove( sourcePosition: Position, targetPosition: Position, howMany: number ): Position {\n\t\t// Update target position, as it could be affected by nodes removal.\n\t\ttargetPosition = targetPosition._getTransformedByDeletion( sourcePosition, howMany )!;\n\n\t\tif ( sourcePosition.isEqual( targetPosition ) ) {\n\t\t\t// If `targetPosition` is equal to `sourcePosition` this isn't really any move. Just return position as it is.\n\t\t\treturn Position._createAt( this );\n\t\t}\n\n\t\t// Moving a range removes nodes from their original position. We acknowledge this by proper transformation.\n\t\tconst transformed = this._getTransformedByDeletion( sourcePosition, howMany );\n\n\t\tconst isMoved = transformed === null ||\n\t\t\t( sourcePosition.isEqual( this ) && this.stickiness == 'toNext' ) ||\n\t\t\t( sourcePosition.getShiftedBy( howMany ).isEqual( this ) && this.stickiness == 'toPrevious' );\n\n\t\tif ( isMoved ) {\n\t\t\t// This position is inside moved range (or sticks to it).\n\t\t\t// In this case, we calculate a combination of this position, move source position and target position.\n\t\t\treturn this._getCombined( sourcePosition, targetPosition );\n\t\t} else {\n\t\t\t// This position is not inside a removed range.\n\t\t\t//\n\t\t\t// In next step, we simply reflect inserting `howMany` nodes, which might further affect the position.\n\t\t\treturn transformed._getTransformedByInsertion( targetPosition, howMany );\n\t\t}\n\t}\n\n\t/**\n\t * Returns a new position that is a combination of this position and given positions.\n\t *\n\t * The combined position is a copy of this position transformed by moving a range starting at `source` position\n\t * to the `target` position. It is expected that this position is inside the moved range.\n\t *\n\t * Example:\n\t *\n\t * ```ts\n\t * let original = model.createPositionFromPath( root, [ 2, 3, 1 ] );\n\t * let source = model.createPositionFromPath( root, [ 2, 2 ] );\n\t * let target = model.createPositionFromPath( otherRoot, [ 1, 1, 3 ] );\n\t * original._getCombined( source, target ); // path is [ 1, 1, 4, 1 ], root is `otherRoot`\n\t * ```\n\t *\n\t * Explanation:\n\t *\n\t * We have a position `[ 2, 3, 1 ]` and move some nodes from `[ 2, 2 ]` to `[ 1, 1, 3 ]`. The original position\n\t * was inside moved nodes and now should point to the new place. The moved nodes will be after\n\t * positions `[ 1, 1, 3 ]`, `[ 1, 1, 4 ]`, `[ 1, 1, 5 ]`. Since our position was in the second moved node,\n\t * the transformed position will be in a sub-tree of a node at `[ 1, 1, 4 ]`. Looking at original path, we\n\t * took care of `[ 2, 3 ]` part of it. Now we have to add the rest of the original path to the transformed path.\n\t * Finally, the transformed position will point to `[ 1, 1, 4, 1 ]`.\n\t *\n\t * @internal\n\t * @param source Beginning of the moved range.\n\t * @param target Position where the range is moved.\n\t * @returns Combined position.\n\t */\n\tpublic _getCombined( source: Position, target: Position ): Position {\n\t\tconst i = source.path.length - 1;\n\n\t\t// The first part of a path to combined position is a path to the place where nodes were moved.\n\t\tconst combined = Position._createAt( target );\n\t\tcombined.stickiness = this.stickiness;\n\n\t\t// Then we have to update the rest of the path.\n\n\t\t// Fix the offset because this position might be after `from` position and we have to reflect that.\n\t\tcombined.offset = combined.offset + this.path[ i ] - source.offset;\n\n\t\t// Then, add the rest of the path.\n\t\t// If this position is at the same level as `from` position nothing will get added.\n\t\t( combined as any ).path = [ ...combined.path, ...this.path.slice( i + 1 ) ];\n\n\t\treturn combined;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic toJSON(): unknown {\n\t\treturn {\n\t\t\troot: this.root.toJSON(),\n\t\t\tpath: Array.from( this.path ),\n\t\t\tstickiness: this.stickiness\n\t\t};\n\t}\n\n\t/**\n\t * Returns a new position that is equal to current position.\n\t */\n\tpublic clone(): this {\n\t\treturn new ( this.constructor as any )( this.root, this.path, this.stickiness );\n\t}\n\n\t/**\n\t * Creates position at the given location. The location can be specified as:\n\t *\n\t * * a {@link module:engine/model/position~Position position},\n\t * * parent element and offset (offset defaults to `0`),\n\t * * parent element and `'end'` (sets position at the end of that element),\n\t * * {@link module:engine/model/item~Item model item} and `'before'` or `'after'` (sets position before or after given model item).\n\t *\n\t * This method is a shortcut to other factory methods such as:\n\t *\n\t * * {@link module:engine/model/position~Position._createBefore},\n\t * * {@link module:engine/model/position~Position._createAfter}.\n\t *\n\t * @internal\n\t * @param offset Offset or one of the flags. Used only when the first parameter is a {@link module:engine/model/item~Item model item}.\n\t * @param stickiness Position stickiness. Used only when the first parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic static _createAt(\n\t\titemOrPosition: Item | Position | DocumentFragment,\n\t\toffset?: PositionOffset,\n\t\tstickiness: PositionStickiness = 'toNone'\n\t): Position {\n\t\tif ( itemOrPosition instanceof Position ) {\n\t\t\treturn new Position( itemOrPosition.root, itemOrPosition.path, itemOrPosition.stickiness );\n\t\t} else {\n\t\t\tconst node = itemOrPosition;\n\n\t\t\tif ( offset == 'end' ) {\n\t\t\t\toffset = ( node as any ).maxOffset;\n\t\t\t} else if ( offset == 'before' ) {\n\t\t\t\treturn this._createBefore( node, stickiness );\n\t\t\t} else if ( offset == 'after' ) {\n\t\t\t\treturn this._createAfter( node, stickiness );\n\t\t\t} else if ( offset !== 0 && !offset ) {\n\t\t\t\t/**\n\t\t\t\t * {@link module:engine/model/model~Model#createPositionAt `Model#createPositionAt()`}\n\t\t\t\t * requires the offset to be specified when the first parameter is a model item.\n\t\t\t\t *\n\t\t\t\t * @error model-createpositionat-offset-required\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'model-createpositionat-offset-required', [ this, itemOrPosition ] );\n\t\t\t}\n\n\t\t\tif ( !node.is( 'element' ) && !node.is( 'documentFragment' ) ) {\n\t\t\t\t/**\n\t\t\t\t * Position parent have to be a model element or model document fragment.\n\t\t\t\t *\n\t\t\t\t * @error model-position-parent-incorrect\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'model-position-parent-incorrect',\n\t\t\t\t\t[ this, itemOrPosition ]\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tconst path = node.getPath();\n\n\t\t\tpath.push( offset as any );\n\n\t\t\treturn new this( node.root as any, path, stickiness );\n\t\t}\n\t}\n\n\t/**\n\t * Creates a new position, after given {@link module:engine/model/item~Item model item}.\n\t *\n\t * @internal\n\t * @param item Item after which the position should be placed.\n\t * @param stickiness Position stickiness.\n\t */\n\tpublic static _createAfter( item: Item | DocumentFragment, stickiness?: PositionStickiness ): Position {\n\t\tif ( !item.parent ) {\n\t\t\t/**\n\t\t\t * You can not make a position after a root element.\n\t\t\t *\n\t\t\t * @error model-position-after-root\n\t\t\t * @param root\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'model-position-after-root',\n\t\t\t\t[ this, item ],\n\t\t\t\t{ root: item }\n\t\t\t);\n\t\t}\n\n\t\treturn this._createAt( item.parent, item.endOffset!, stickiness );\n\t}\n\n\t/**\n\t * Creates a new position, before the given {@link module:engine/model/item~Item model item}.\n\t *\n\t * @internal\n\t * @param item Item before which the position should be placed.\n\t * @param stickiness Position stickiness.\n\t */\n\tpublic static _createBefore( item: Item | DocumentFragment, stickiness?: PositionStickiness ): Position {\n\t\tif ( !item.parent ) {\n\t\t\t/**\n\t\t\t * You can not make a position before a root element.\n\t\t\t *\n\t\t\t * @error model-position-before-root\n\t\t\t * @param root\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'model-position-before-root',\n\t\t\t\titem,\n\t\t\t\t{ root: item }\n\t\t\t);\n\t\t}\n\n\t\treturn this._createAt( item.parent, item.startOffset!, stickiness );\n\t}\n\n\t/**\n\t * Creates a `Position` instance from given plain object (i.e. parsed JSON string).\n\t *\n\t * @param json Plain object to be converted to `Position`.\n\t * @param doc Document object that will be position owner.\n\t * @returns `Position` instance created using given plain object.\n\t */\n\tpublic static fromJSON( json: any, doc: Document ): Position {\n\t\tif ( json.root === '$graveyard' ) {\n\t\t\tconst pos = new Position( doc.graveyard, json.path );\n\t\t\tpos.stickiness = json.stickiness;\n\n\t\t\treturn pos;\n\t\t}\n\n\t\tif ( !doc.getRoot( json.root ) ) {\n\t\t\t/**\n\t\t\t * Cannot create position for document. Root with specified name does not exist.\n\t\t\t *\n\t\t\t * @error model-position-fromjson-no-root\n\t\t\t * @param rootName\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'model-position-fromjson-no-root',\n\t\t\t\tdoc,\n\t\t\t\t{ rootName: json.root }\n\t\t\t);\n\t\t}\n\n\t\treturn new Position( doc.getRoot( json.root )!, json.path, json.stickiness );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `${ this.root } [ ${ this.path.join( ', ' ) } ]`;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ModelPosition: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nPosition.prototype.is = function( type: string ): boolean {\n\treturn type === 'position' || type === 'model:position';\n};\n\n/**\n * A flag indicating whether this position is `'before'` or `'after'` or `'same'` as given position.\n * If positions are in different roots `'different'` flag is returned.\n */\nexport type PositionRelation = 'before' | 'after' | 'same' | 'different';\n\n/**\n * Offset or one of the flags.\n */\nexport type PositionOffset = number | 'before' | 'after' | 'end';\n\n/**\n * Represents how position is \"sticking\" with neighbour nodes. Used to define how position should be transformed (moved)\n * in edge cases. Possible values: `'toNone'`, `'toNext'`, `'toPrevious'`.\n *\n * Examples:\n *\n * ```\n * Insert. Position is at | and nodes are inserted at the same position, marked as ^:\n *\n * - sticks to none: <p>f^|oo</p> -> <p>fbar|oo</p>\n * - sticks to next node: <p>f^|oo</p> -> <p>fbar|oo</p>\n * - sticks to previous node: <p>f|^oo</p> -> <p>f|baroo</p>\n * ```\n *\n *\n * Move. Position is at | and range [oo] is moved to position ^:\n *\n * ```\n * - sticks to none: <p>f|[oo]</p><p>b^ar</p> -> <p>f|</p><p>booar</p>\n * - sticks to none: <p>f[oo]|</p><p>b^ar</p> -> <p>f|</p><p>booar</p>\n *\n * - sticks to next node: <p>f|[oo]</p><p>b^ar</p> -> <p>f</p><p>b|ooar</p>\n * - sticks to next node: <p>f[oo]|</p><p>b^ar</p> -> <p>f|</p><p>booar</p>\n *\n * - sticks to previous node: <p>f|[oo]</p><p>b^ar</p> -> <p>f|</p><p>booar</p>\n * - sticks to previous node: <p>f[oo]|</p><p>b^ar</p> -> <p>f</p><p>boo|ar</p>\n * ```\n */\nexport type PositionStickiness = 'toNone' | 'toNext' | 'toPrevious';\n\n/**\n * Returns a text node at the given position.\n *\n * This is a helper function optimized to reuse the position parent instance for performance reasons.\n *\n * Normally, you should use {@link module:engine/model/position~Position#textNode `Position#textNode`}.\n * If you start hitting performance issues with {@link module:engine/model/position~Position#parent `Position#parent`}\n * check if your algorithm does not access it multiple times (which can happen directly or indirectly via other position properties).\n *\n * See https://github.com/ckeditor/ckeditor5/issues/6579.\n *\n * See also:\n *\n * * {@link module:engine/model/position~getNodeAfterPosition}\n * * {@link module:engine/model/position~getNodeBeforePosition}\n *\n * @param positionParent The parent of the given position.\n */\nexport function getTextNodeAtPosition( position: Position, positionParent: Element | DocumentFragment ): Text | null {\n\tconst node = positionParent.getChild( positionParent.offsetToIndex( position.offset ) );\n\n\tif ( node && node.is( '$text' ) && node.startOffset! < position.offset ) {\n\t\treturn node;\n\t}\n\n\treturn null;\n}\n\n/**\n * Returns the node after the given position.\n *\n * This is a helper function optimized to reuse the position parent instance and the calculation of the text node at the\n * specific position for performance reasons.\n *\n * Normally, you should use {@link module:engine/model/position~Position#nodeAfter `Position#nodeAfter`}.\n * If you start hitting performance issues with {@link module:engine/model/position~Position#parent `Position#parent`} and/or\n * {@link module:engine/model/position~Position#textNode `Position#textNode`}\n * check if your algorithm does not access those properties multiple times\n * (which can happen directly or indirectly via other position properties).\n *\n * See https://github.com/ckeditor/ckeditor5/issues/6579 and https://github.com/ckeditor/ckeditor5/issues/6582.\n *\n * See also:\n *\n * * {@link module:engine/model/position~getTextNodeAtPosition}\n * * {@link module:engine/model/position~getNodeBeforePosition}\n *\n * @param positionParent The parent of the given position.\n * @param textNode Text node at the given position.\n */\nexport function getNodeAfterPosition(\n\tposition: Position,\n\tpositionParent: Element | DocumentFragment,\n\ttextNode: Text | null\n): Node | null {\n\tif ( textNode !== null ) {\n\t\treturn null;\n\t}\n\n\treturn positionParent.getChild( positionParent.offsetToIndex( position.offset ) );\n}\n\n/**\n * Returns the node before the given position.\n *\n * Refer to {@link module:engine/model/position~getNodeBeforePosition} for documentation on when to use this util method.\n *\n * See also:\n *\n * * {@link module:engine/model/position~getTextNodeAtPosition}\n * * {@link module:engine/model/position~getNodeAfterPosition}\n *\n * @param positionParent The parent of the given position.\n * @param textNode Text node at the given position.\n */\nexport function getNodeBeforePosition(\n\tposition: Position,\n\tpositionParent: Element | DocumentFragment,\n\ttextNode: Text | null\n): Node | null {\n\tif ( textNode !== null ) {\n\t\treturn null;\n\t}\n\n\treturn positionParent.getChild( positionParent.offsetToIndex( position.offset ) - 1 );\n}\n\n/**\n * This is a helper function for `Position#isTouching()`.\n *\n * It checks whether to given positions are touching, considering that they have the same root and paths\n * until given level, and at given level they differ by 1 (so they are branching at `level` point).\n *\n * The exact requirements for touching positions are described in `Position#isTouching()` and also\n * in the body of this function.\n *\n * @param left Position \"on the left\" (it is before `right`).\n * @param right Position \"on the right\" (it is after `left`).\n * @param level Level on which the positions are different.\n */\nfunction checkTouchingBranch( left: Position, right: Position, level: number ): boolean {\n\tif ( level + 1 === left.path.length ) {\n\t\t// Left position does not have any more entries after the point where the positions differ.\n\t\t// [ 2 ] vs [ 3 ]\n\t\t// [ 2 ] vs [ 3, 0, 0 ]\n\t\t// The positions are spread by node at [ 2 ].\n\t\treturn false;\n\t}\n\n\tif ( !checkOnlyZeroes( right.path, level + 1 ) ) {\n\t\t// Right position does not have only zeroes, so we have situation like:\n\t\t// [ 2, maxOffset ] vs [ 3, 1 ]\n\t\t// [ 2, maxOffset ] vs [ 3, 1, 0, 0 ]\n\t\t// The positions are spread by node at [ 3, 0 ].\n\t\treturn false;\n\t}\n\n\tif ( !checkOnlyMaxOffset( left, level + 1 ) ) {\n\t\t// Left position does not have only max offsets, so we have situation like:\n\t\t// [ 2, 4 ] vs [ 3 ]\n\t\t// [ 2, 4 ] vs [ 3, 0, 0 ]\n\t\t// The positions are spread by node at [ 2, 5 ].\n\t\treturn false;\n\t}\n\n\t// Left position has only max offsets and right position has only zeroes or nothing.\n\t// [ 2, maxOffset ] vs [ 3 ]\n\t// [ 2, maxOffset, maxOffset ] vs [ 3, 0 ]\n\t// There are not elements between positions. The positions are touching.\n\treturn true;\n}\n\n/**\n * Checks whether for given array, starting from given index until the end of the array, all items are `0`s.\n *\n * This is a helper function for `Position#isTouching()`.\n */\nfunction checkOnlyZeroes( arr: ReadonlyArray<number>, idx: number ): boolean {\n\twhile ( idx < arr.length ) {\n\t\tif ( arr[ idx ] !== 0 ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tidx++;\n\t}\n\n\treturn true;\n}\n\n/**\n * Checks whether for given position, starting from given path level, whether the position is at the end of\n * its parent and whether each element on the path to the position is also at at the end of its parent.\n *\n * This is a helper function for `Position#isTouching()`.\n */\nfunction checkOnlyMaxOffset( pos: Position, level: number ): boolean {\n\tlet parent = pos.parent;\n\tlet idx = pos.path.length - 1;\n\tlet add = 0;\n\n\twhile ( idx >= level ) {\n\t\tif ( pos.path[ idx ] + add !== parent.maxOffset ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// After the first check, we \"go up\", and check whether the position's parent-parent is the last element.\n\t\t// However, we need to add 1 to the value in the path to \"simulate\" moving the path after the parent.\n\t\t// It happens just once.\n\t\tadd = 1;\n\t\tidx--;\n\t\tparent = parent.parent!;\n\t}\n\n\treturn true;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/range\n */\n\nimport TypeCheckable from './typecheckable';\nimport Position from './position';\nimport TreeWalker, { type TreeWalkerOptions, type TreeWalkerValue } from './treewalker';\n\nimport type Document from './document';\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\nimport type InsertOperation from './operation/insertoperation';\nimport type Item from './item';\nimport type MergeOperation from './operation/mergeoperation';\nimport type MoveOperation from './operation/moveoperation';\nimport type Operation from './operation/operation';\nimport type SplitOperation from './operation/splitoperation';\n\nimport { CKEditorError, compareArrays } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Represents a range in the model tree.\n *\n * A range is defined by its {@link module:engine/model/range~Range#start} and {@link module:engine/model/range~Range#end}\n * positions.\n *\n * You can create range instances via its constructor or the `createRange*()` factory methods of\n * {@link module:engine/model/model~Model} and {@link module:engine/model/writer~Writer}.\n */\nexport default class Range extends TypeCheckable implements Iterable<TreeWalkerValue> {\n\t/**\n\t * Start position.\n\t */\n\tpublic readonly start: Position;\n\n\t/**\n\t * End position.\n\t */\n\tpublic readonly end: Position;\n\n\t/**\n\t * Creates a range spanning from `start` position to `end` position.\n\t *\n\t * @param start The start position.\n\t * @param end The end position. If not set, the range will be collapsed at the `start` position.\n\t */\n\tconstructor( start: Position, end?: Position | null ) {\n\t\tsuper();\n\n\t\tthis.start = Position._createAt( start );\n\t\tthis.end = end ? Position._createAt( end ) : Position._createAt( start );\n\n\t\t// If the range is collapsed, treat in a similar way as a position and set its boundaries stickiness to 'toNone'.\n\t\t// In other case, make the boundaries stick to the \"inside\" of the range.\n\t\tthis.start.stickiness = this.isCollapsed ? 'toNone' : 'toNext';\n\t\tthis.end.stickiness = this.isCollapsed ? 'toNone' : 'toPrevious';\n\t}\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * Iterates over all {@link module:engine/model/item~Item items} that are in this range and returns\n\t * them together with additional information like length or {@link module:engine/model/position~Position positions},\n\t * grouped as {@link module:engine/model/treewalker~TreeWalkerValue}.\n\t * It iterates over all {@link module:engine/model/textproxy~TextProxy text contents} that are inside the range\n\t * and all the {@link module:engine/model/element~Element}s that are entered into when iterating over this range.\n\t *\n\t * This iterator uses {@link module:engine/model/treewalker~TreeWalker} with `boundaries` set to this range\n\t * and `ignoreElementEnd` option set to `true`.\n\t */\n\tpublic* [ Symbol.iterator ](): IterableIterator<TreeWalkerValue> {\n\t\tyield* new TreeWalker( { boundaries: this, ignoreElementEnd: true } );\n\t}\n\n\t/**\n\t * Describes whether the range is collapsed, that is if {@link #start} and\n\t * {@link #end} positions are equal.\n\t */\n\tpublic get isCollapsed(): boolean {\n\t\treturn this.start.isEqual( this.end );\n\t}\n\n\t/**\n\t * Describes whether this range is flat, that is if {@link #start} position and\n\t * {@link #end} position are in the same {@link module:engine/model/position~Position#parent}.\n\t */\n\tpublic get isFlat(): boolean {\n\t\tconst startParentPath = this.start.getParentPath();\n\t\tconst endParentPath = this.end.getParentPath();\n\n\t\treturn compareArrays( startParentPath, endParentPath ) == 'same';\n\t}\n\n\t/**\n\t * Range root element.\n\t */\n\tpublic get root(): Element | DocumentFragment {\n\t\treturn this.start.root;\n\t}\n\n\t/**\n\t * Checks whether this range contains given {@link module:engine/model/position~Position position}.\n\t *\n\t * @param position Position to check.\n\t * @returns `true` if given {@link module:engine/model/position~Position position} is contained\n\t * in this range,`false` otherwise.\n\t */\n\tpublic containsPosition( position: Position ): boolean {\n\t\treturn position.isAfter( this.start ) && position.isBefore( this.end );\n\t}\n\n\t/**\n\t * Checks whether this range contains given {@link ~Range range}.\n\t *\n\t * @param otherRange Range to check.\n\t * @param loose Whether the check is loose or strict. If the check is strict (`false`), compared range cannot\n\t * start or end at the same position as this range boundaries. If the check is loose (`true`), compared range can start, end or\n\t * even be equal to this range. Note that collapsed ranges are always compared in strict mode.\n\t * @returns {Boolean} `true` if given {@link ~Range range} boundaries are contained by this range, `false` otherwise.\n\t */\n\tpublic containsRange( otherRange: Range, loose: boolean = false ): boolean {\n\t\tif ( otherRange.isCollapsed ) {\n\t\t\tloose = false;\n\t\t}\n\n\t\tconst containsStart = this.containsPosition( otherRange.start ) || ( loose && this.start.isEqual( otherRange.start ) );\n\t\tconst containsEnd = this.containsPosition( otherRange.end ) || ( loose && this.end.isEqual( otherRange.end ) );\n\n\t\treturn containsStart && containsEnd;\n\t}\n\n\t/**\n\t * Checks whether given {@link module:engine/model/item~Item} is inside this range.\n\t */\n\tpublic containsItem( item: Item ): boolean {\n\t\tconst pos = Position._createBefore( item );\n\n\t\treturn this.containsPosition( pos ) || this.start.isEqual( pos );\n\t}\n\n\t/**\n\t * Two ranges are equal if their {@link #start} and {@link #end} positions are equal.\n\t *\n\t * @param otherRange Range to compare with.\n\t * @returns `true` if ranges are equal, `false` otherwise.\n\t */\n\tpublic isEqual( otherRange: Range ): boolean {\n\t\treturn this.start.isEqual( otherRange.start ) && this.end.isEqual( otherRange.end );\n\t}\n\n\t/**\n\t * Checks and returns whether this range intersects with given range.\n\t *\n\t * @param otherRange Range to compare with.\n\t * @returns `true` if ranges intersect, `false` otherwise.\n\t */\n\tpublic isIntersecting( otherRange: Range ): boolean {\n\t\treturn this.start.isBefore( otherRange.end ) && this.end.isAfter( otherRange.start );\n\t}\n\n\t/**\n\t * Computes which part(s) of this {@link ~Range range} is not a part of given {@link ~Range range}.\n\t * Returned array contains zero, one or two {@link ~Range ranges}.\n\t *\n\t * Examples:\n\t *\n\t * ```ts\n\t * let range = model.createRange(\n\t * \tmodel.createPositionFromPath( root, [ 2, 7 ] ),\n\t * \tmodel.createPositionFromPath( root, [ 4, 0, 1 ] )\n\t * );\n\t * let otherRange = model.createRange( model.createPositionFromPath( root, [ 1 ] ), model.createPositionFromPath( root, [ 5 ] ) );\n\t * let transformed = range.getDifference( otherRange );\n\t * // transformed array has no ranges because `otherRange` contains `range`\n\t *\n\t * otherRange = model.createRange( model.createPositionFromPath( root, [ 1 ] ), model.createPositionFromPath( root, [ 3 ] ) );\n\t * transformed = range.getDifference( otherRange );\n\t * // transformed array has one range: from [ 3 ] to [ 4, 0, 1 ]\n\t *\n\t * otherRange = model.createRange( model.createPositionFromPath( root, [ 3 ] ), model.createPositionFromPath( root, [ 4 ] ) );\n\t * transformed = range.getDifference( otherRange );\n\t * // transformed array has two ranges: from [ 2, 7 ] to [ 3 ] and from [ 4 ] to [ 4, 0, 1 ]\n\t * ```\n\t *\n\t * @param otherRange Range to differentiate against.\n\t * @returns The difference between ranges.\n\t */\n\tpublic getDifference( otherRange: Range ): Array<Range> {\n\t\tconst ranges = [];\n\n\t\tif ( this.isIntersecting( otherRange ) ) {\n\t\t\t// Ranges intersect.\n\n\t\t\tif ( this.containsPosition( otherRange.start ) ) {\n\t\t\t\t// Given range start is inside this range. This means that we have to\n\t\t\t\t// add shrunken range - from the start to the middle of this range.\n\t\t\t\tranges.push( new Range( this.start, otherRange.start ) );\n\t\t\t}\n\n\t\t\tif ( this.containsPosition( otherRange.end ) ) {\n\t\t\t\t// Given range end is inside this range. This means that we have to\n\t\t\t\t// add shrunken range - from the middle of this range to the end.\n\t\t\t\tranges.push( new Range( otherRange.end, this.end ) );\n\t\t\t}\n\t\t} else {\n\t\t\t// Ranges do not intersect, return the original range.\n\t\t\tranges.push( new Range( this.start, this.end ) );\n\t\t}\n\n\t\treturn ranges;\n\t}\n\n\t/**\n\t * Returns an intersection of this {@link ~Range range} and given {@link ~Range range}.\n\t * Intersection is a common part of both of those ranges. If ranges has no common part, returns `null`.\n\t *\n\t * Examples:\n\t *\n\t * ```ts\n\t * let range = model.createRange(\n\t * \tmodel.createPositionFromPath( root, [ 2, 7 ] ),\n\t * \tmodel.createPositionFromPath( root, [ 4, 0, 1 ] )\n\t * );\n\t * let otherRange = model.createRange( model.createPositionFromPath( root, [ 1 ] ), model.createPositionFromPath( root, [ 2 ] ) );\n\t * let transformed = range.getIntersection( otherRange ); // null - ranges have no common part\n\t *\n\t * otherRange = model.createRange( model.createPositionFromPath( root, [ 3 ] ), model.createPositionFromPath( root, [ 5 ] ) );\n\t * transformed = range.getIntersection( otherRange ); // range from [ 3 ] to [ 4, 0, 1 ]\n\t * ```\n\t *\n\t * @param otherRange Range to check for intersection.\n\t * @returns A common part of given ranges or `null` if ranges have no common part.\n\t */\n\tpublic getIntersection( otherRange: Range ): Range | null {\n\t\tif ( this.isIntersecting( otherRange ) ) {\n\t\t\t// Ranges intersect, so a common range will be returned.\n\t\t\t// At most, it will be same as this range.\n\t\t\tlet commonRangeStart = this.start;\n\t\t\tlet commonRangeEnd = this.end;\n\n\t\t\tif ( this.containsPosition( otherRange.start ) ) {\n\t\t\t\t// Given range start is inside this range. This means thaNt we have to\n\t\t\t\t// shrink common range to the given range start.\n\t\t\t\tcommonRangeStart = otherRange.start;\n\t\t\t}\n\n\t\t\tif ( this.containsPosition( otherRange.end ) ) {\n\t\t\t\t// Given range end is inside this range. This means that we have to\n\t\t\t\t// shrink common range to the given range end.\n\t\t\t\tcommonRangeEnd = otherRange.end;\n\t\t\t}\n\n\t\t\treturn new Range( commonRangeStart, commonRangeEnd );\n\t\t}\n\n\t\t// Ranges do not intersect, so they do not have common part.\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns a range created by joining this {@link ~Range range} with the given {@link ~Range range}.\n\t * If ranges have no common part, returns `null`.\n\t *\n\t * Examples:\n\t *\n\t * ```ts\n\t * let range = model.createRange(\n\t * \tmodel.createPositionFromPath( root, [ 2, 7 ] ),\n\t * \tmodel.createPositionFromPath( root, [ 4, 0, 1 ] )\n\t * );\n\t * let otherRange = model.createRange(\n\t * \tmodel.createPositionFromPath( root, [ 1 ] ),\n\t * \tmodel.createPositionFromPath( root, [ 2 ] )\n \t * );\n\t * let transformed = range.getJoined( otherRange ); // null - ranges have no common part\n\t *\n\t * otherRange = model.createRange(\n\t * \tmodel.createPositionFromPath( root, [ 3 ] ),\n\t * \tmodel.createPositionFromPath( root, [ 5 ] )\n\t * );\n\t * transformed = range.getJoined( otherRange ); // range from [ 2, 7 ] to [ 5 ]\n\t * ```\n\t *\n\t * @param otherRange Range to be joined.\n\t * @param loose Whether the intersection check is loose or strict. If the check is strict (`false`),\n\t * ranges are tested for intersection or whether start/end positions are equal. If the check is loose (`true`),\n\t * compared range is also checked if it's {@link module:engine/model/position~Position#isTouching touching} current range.\n\t * @returns A sum of given ranges or `null` if ranges have no common part.\n\t */\n\tpublic getJoined( otherRange: Range, loose: boolean = false ): Range | null {\n\t\tlet shouldJoin = this.isIntersecting( otherRange );\n\n\t\tif ( !shouldJoin ) {\n\t\t\tif ( this.start.isBefore( otherRange.start ) ) {\n\t\t\t\tshouldJoin = loose ? this.end.isTouching( otherRange.start ) : this.end.isEqual( otherRange.start );\n\t\t\t} else {\n\t\t\t\tshouldJoin = loose ? otherRange.end.isTouching( this.start ) : otherRange.end.isEqual( this.start );\n\t\t\t}\n\t\t}\n\n\t\tif ( !shouldJoin ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tlet startPosition = this.start;\n\t\tlet endPosition = this.end;\n\n\t\tif ( otherRange.start.isBefore( startPosition ) ) {\n\t\t\tstartPosition = otherRange.start;\n\t\t}\n\n\t\tif ( otherRange.end.isAfter( endPosition ) ) {\n\t\t\tendPosition = otherRange.end;\n\t\t}\n\n\t\treturn new Range( startPosition, endPosition );\n\t}\n\n\t/**\n\t * Computes and returns the smallest set of {@link #isFlat flat} ranges, that covers this range in whole.\n\t *\n\t * See an example of a model structure (`[` and `]` are range boundaries):\n\t *\n\t * ```\n\t * root root\n\t * |- element DIV DIV P2 P3 DIV\n\t * | |- element H H P1 f o o b a r H P4\n\t * | | |- \"fir[st\" fir[st lorem se]cond ipsum\n\t * | |- element P1\n\t * | | |- \"lorem\" ||\n\t * |- element P2 ||\n\t * | |- \"foo\" VV\n\t * |- element P3\n\t * | |- \"bar\" root\n\t * |- element DIV DIV [P2 P3] DIV\n\t * | |- element H H [P1] f o o b a r H P4\n\t * | | |- \"se]cond\" fir[st] lorem [se]cond ipsum\n\t * | |- element P4\n\t * | | |- \"ipsum\"\n\t * ```\n\t *\n\t * As it can be seen, letters contained in the range are: `stloremfoobarse`, spread across different parents.\n\t * We are looking for minimal set of flat ranges that contains the same nodes.\n\t *\n\t * Minimal flat ranges for above range `( [ 0, 0, 3 ], [ 3, 0, 2 ] )` will be:\n\t *\n\t * ```\n\t * ( [ 0, 0, 3 ], [ 0, 0, 5 ] ) = \"st\"\n\t * ( [ 0, 1 ], [ 0, 2 ] ) = element P1 (\"lorem\")\n\t * ( [ 1 ], [ 3 ] ) = element P2, element P3 (\"foobar\")\n\t * ( [ 3, 0, 0 ], [ 3, 0, 2 ] ) = \"se\"\n\t * ```\n\t *\n\t * **Note:** if an {@link module:engine/model/element~Element element} is not wholly contained in this range, it won't be returned\n\t * in any of the returned flat ranges. See in the example how `H` elements at the beginning and at the end of the range\n\t * were omitted. Only their parts that were wholly in the range were returned.\n\t *\n\t * **Note:** this method is not returning flat ranges that contain no nodes.\n\t *\n\t * @returns Array of flat ranges covering this range.\n\t */\n\tpublic getMinimalFlatRanges(): Array<Range> {\n\t\tconst ranges = [];\n\t\tconst diffAt = this.start.getCommonPath( this.end ).length;\n\n\t\tconst pos = Position._createAt( this.start );\n\t\tlet posParent = pos.parent;\n\n\t\t// Go up.\n\t\twhile ( pos.path.length > diffAt + 1 ) {\n\t\t\tconst howMany = posParent.maxOffset - pos.offset;\n\n\t\t\tif ( howMany !== 0 ) {\n\t\t\t\tranges.push( new Range( pos, pos.getShiftedBy( howMany ) ) );\n\t\t\t}\n\n\t\t\t( pos as any ).path = pos.path.slice( 0, -1 );\n\t\t\tpos.offset++;\n\t\t\tposParent = posParent.parent!;\n\t\t}\n\n\t\t// Go down.\n\t\twhile ( pos.path.length <= this.end.path.length ) {\n\t\t\tconst offset = this.end.path[ pos.path.length - 1 ];\n\t\t\tconst howMany = offset - pos.offset;\n\n\t\t\tif ( howMany !== 0 ) {\n\t\t\t\tranges.push( new Range( pos, pos.getShiftedBy( howMany ) ) );\n\t\t\t}\n\n\t\t\tpos.offset = offset;\n\t\t\t( pos.path as Array<number> ).push( 0 );\n\t\t}\n\n\t\treturn ranges;\n\t}\n\n\t/**\n\t * Creates a {@link module:engine/model/treewalker~TreeWalker TreeWalker} instance with this range as a boundary.\n\t *\n\t * For example, to iterate over all items in the entire document root:\n\t *\n\t * ```ts\n\t * // Create a range spanning over the entire root content:\n\t * const range = editor.model.createRangeIn( editor.model.document.getRoot() );\n\t *\n\t * // Iterate over all items in this range:\n\t * for ( const value of range.getWalker() ) {\n\t * \tconsole.log( value.item );\n\t * }\n\t * ```\n\t *\n\t * @param options Object with configuration options. See {@link module:engine/model/treewalker~TreeWalker}.\n\t */\n\tpublic getWalker( options: TreeWalkerOptions = {} ): TreeWalker {\n\t\toptions.boundaries = this;\n\n\t\treturn new TreeWalker( options );\n\t}\n\n\t/**\n\t * Returns an iterator that iterates over all {@link module:engine/model/item~Item items} that are in this range and returns\n\t * them.\n\t *\n\t * This method uses {@link module:engine/model/treewalker~TreeWalker} with `boundaries` set to this range and `ignoreElementEnd` option\n\t * set to `true`. However it returns only {@link module:engine/model/item~Item model items},\n\t * not {@link module:engine/model/treewalker~TreeWalkerValue}.\n\t *\n\t * You may specify additional options for the tree walker. See {@link module:engine/model/treewalker~TreeWalker} for\n\t * a full list of available options.\n\t *\n\t * @param options Object with configuration options. See {@link module:engine/model/treewalker~TreeWalker}.\n\t */\n\tpublic* getItems( options: TreeWalkerOptions = {} ): IterableIterator<Item> {\n\t\toptions.boundaries = this;\n\t\toptions.ignoreElementEnd = true;\n\n\t\tconst treeWalker = new TreeWalker( options );\n\n\t\tfor ( const value of treeWalker ) {\n\t\t\tyield value.item;\n\t\t}\n\t}\n\n\t/**\n\t * Returns an iterator that iterates over all {@link module:engine/model/position~Position positions} that are boundaries or\n\t * contained in this range.\n\t *\n\t * This method uses {@link module:engine/model/treewalker~TreeWalker} with `boundaries` set to this range. However it returns only\n\t * {@link module:engine/model/position~Position positions}, not {@link module:engine/model/treewalker~TreeWalkerValue}.\n\t *\n\t * You may specify additional options for the tree walker. See {@link module:engine/model/treewalker~TreeWalker} for\n\t * a full list of available options.\n\t *\n\t * @param options Object with configuration options. See {@link module:engine/model/treewalker~TreeWalker}.\n\t */\n\tpublic* getPositions( options: TreeWalkerOptions = {} ): IterableIterator<Position> {\n\t\toptions.boundaries = this;\n\n\t\tconst treeWalker = new TreeWalker( options );\n\n\t\tyield treeWalker.position;\n\n\t\tfor ( const value of treeWalker ) {\n\t\t\tyield value.nextPosition;\n\t\t}\n\t}\n\n\t/**\n\t * Returns a range that is a result of transforming this range by given `operation`.\n\t *\n\t * **Note:** transformation may break one range into multiple ranges (for example, when a part of the range is\n\t * moved to a different part of document tree). For this reason, an array is returned by this method and it\n\t * may contain one or more `Range` instances.\n\t *\n\t * @param operation Operation to transform range by.\n\t * @returns Range which is the result of transformation.\n\t */\n\tpublic getTransformedByOperation( operation: Operation ): Array<Range> {\n\t\tswitch ( operation.type ) {\n\t\t\tcase 'insert':\n\t\t\t\treturn this._getTransformedByInsertOperation( operation as InsertOperation );\n\t\t\tcase 'move':\n\t\t\tcase 'remove':\n\t\t\tcase 'reinsert':\n\t\t\t\treturn this._getTransformedByMoveOperation( operation as MoveOperation );\n\t\t\tcase 'split':\n\t\t\t\treturn [ this._getTransformedBySplitOperation( operation as SplitOperation ) ];\n\t\t\tcase 'merge':\n\t\t\t\treturn [ this._getTransformedByMergeOperation( operation as MergeOperation ) ];\n\t\t}\n\n\t\treturn [ new Range( this.start, this.end ) ];\n\t}\n\n\t/**\n\t * Returns a range that is a result of transforming this range by multiple `operations`.\n\t *\n\t * @see ~Range#getTransformedByOperation\n\t * @param operations Operations to transform the range by.\n\t * @returns Range which is the result of transformation.\n\t */\n\tpublic getTransformedByOperations( operations: Iterable<Operation> ): Array<Range> {\n\t\tconst ranges = [ new Range( this.start, this.end ) ];\n\n\t\tfor ( const operation of operations ) {\n\t\t\tfor ( let i = 0; i < ranges.length; i++ ) {\n\t\t\t\tconst result = ranges[ i ].getTransformedByOperation( operation );\n\n\t\t\t\tranges.splice( i, 1, ...result );\n\t\t\t\ti += result.length - 1;\n\t\t\t}\n\t\t}\n\n\t\t// It may happen that a range is split into two, and then the part of second \"piece\" is moved into first\n\t\t// \"piece\". In this case we will have incorrect third range, which should not be included in the result --\n\t\t// because it is already included in the first \"piece\". In this loop we are looking for all such ranges that\n\t\t// are inside other ranges and we simply remove them.\n\t\tfor ( let i = 0; i < ranges.length; i++ ) {\n\t\t\tconst range = ranges[ i ];\n\n\t\t\tfor ( let j = i + 1; j < ranges.length; j++ ) {\n\t\t\t\tconst next = ranges[ j ];\n\n\t\t\t\tif ( range.containsRange( next ) || next.containsRange( range ) || range.isEqual( next ) ) {\n\t\t\t\t\tranges.splice( j, 1 );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn ranges;\n\t}\n\n\t/**\n\t * Returns an {@link module:engine/model/element~Element} or {@link module:engine/model/documentfragment~DocumentFragment}\n\t * which is a common ancestor of the range's both ends (in which the entire range is contained).\n\t */\n\tpublic getCommonAncestor(): Element | DocumentFragment | null {\n\t\treturn this.start.getCommonAncestor( this.end );\n\t}\n\n\t/**\n\t * Returns an {@link module:engine/model/element~Element Element} contained by the range.\n\t * The element will be returned when it is the **only** node within the range and **fullycontained**\n\t * at the same time.\n\t */\n\tpublic getContainedElement(): Element | null {\n\t\tif ( this.isCollapsed ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst nodeAfterStart = this.start.nodeAfter;\n\t\tconst nodeBeforeEnd = this.end.nodeBefore;\n\n\t\tif ( nodeAfterStart && nodeAfterStart.is( 'element' ) && nodeAfterStart === nodeBeforeEnd ) {\n\t\t\treturn nodeAfterStart;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Converts `Range` to plain object and returns it.\n\t *\n\t * @returns `Node` converted to plain object.\n\t */\n\tpublic toJSON(): unknown {\n\t\treturn {\n\t\t\tstart: this.start.toJSON(),\n\t\t\tend: this.end.toJSON()\n\t\t};\n\t}\n\n\t/**\n\t * Returns a new range that is equal to current range.\n\t */\n\tpublic clone(): this {\n\t\treturn new ( this.constructor as any )( this.start, this.end );\n\t}\n\n\t/**\n\t * Returns a result of transforming a copy of this range by insert operation.\n\t *\n\t * One or more ranges may be returned as a result of this transformation.\n\t *\n\t * @internal\n\t */\n\tpublic _getTransformedByInsertOperation( operation: InsertOperation, spread: boolean = false ): Array<Range> {\n\t\treturn this._getTransformedByInsertion( operation.position, operation.howMany, spread );\n\t}\n\n\t/**\n\t * Returns a result of transforming a copy of this range by move operation.\n\t *\n\t * One or more ranges may be returned as a result of this transformation.\n\t *\n\t * @internal\n\t */\n\tpublic _getTransformedByMoveOperation( operation: MoveOperation, spread: boolean = false ): Array<Range> {\n\t\tconst sourcePosition = operation.sourcePosition;\n\t\tconst howMany = operation.howMany;\n\t\tconst targetPosition = operation.targetPosition;\n\n\t\treturn this._getTransformedByMove( sourcePosition, targetPosition, howMany, spread );\n\t}\n\n\t/**\n\t * Returns a result of transforming a copy of this range by split operation.\n\t *\n\t * Always one range is returned. The transformation is done in a way to not break the range.\n\t *\n\t * @internal\n\t */\n\tpublic _getTransformedBySplitOperation( operation: SplitOperation ): Range {\n\t\tconst start = this.start._getTransformedBySplitOperation( operation );\n\t\tlet end = this.end._getTransformedBySplitOperation( operation );\n\n\t\tif ( this.end.isEqual( operation.insertionPosition ) ) {\n\t\t\tend = this.end.getShiftedBy( 1 );\n\t\t}\n\n\t\t// Below may happen when range contains graveyard element used by split operation.\n\t\tif ( start.root != end.root ) {\n\t\t\t// End position was next to the moved graveyard element and was moved with it.\n\t\t\t// Fix it by using old `end` which has proper `root`.\n\t\t\tend = this.end.getShiftedBy( -1 );\n\t\t}\n\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Returns a result of transforming a copy of this range by merge operation.\n\t *\n\t * Always one range is returned. The transformation is done in a way to not break the range.\n\t *\n\t * @internal\n\t */\n\tpublic _getTransformedByMergeOperation( operation: MergeOperation ): Range {\n\t\t// Special case when the marker is set on \"the closing tag\" of an element. Marker can be set like that during\n\t\t// transformations, especially when a content of a few block elements were removed. For example:\n\t\t//\n\t\t// {} is the transformed range, [] is the removed range.\n\t\t// <p>F[o{o</p><p>B}ar</p><p>Xy]z</p>\n\t\t//\n\t\t// <p>Fo{o</p><p>B}ar</p><p>z</p>\n\t\t// <p>F{</p><p>B}ar</p><p>z</p>\n\t\t// <p>F{</p>}<p>z</p>\n\t\t// <p>F{}z</p>\n\t\t//\n\t\tif ( this.start.isEqual( operation.targetPosition ) && this.end.isEqual( operation.deletionPosition ) ) {\n\t\t\treturn new Range( this.start );\n\t\t}\n\n\t\tlet start = this.start._getTransformedByMergeOperation( operation );\n\t\tlet end = this.end._getTransformedByMergeOperation( operation );\n\n\t\tif ( start.root != end.root ) {\n\t\t\t// This happens when the end position was next to the merged (deleted) element.\n\t\t\t// Then, the end position was moved to the graveyard root. In this case we need to fix\n\t\t\t// the range cause its boundaries would be in different roots.\n\t\t\tend = this.end.getShiftedBy( -1 );\n\t\t}\n\n\t\tif ( start.isAfter( end ) ) {\n\t\t\t// This happens in three following cases:\n\t\t\t//\n\t\t\t// Case 1: Merge operation source position is before the target position (due to some transformations, OT, etc.)\n\t\t\t// This means that start can be moved before the end of the range.\n\t\t\t//\n\t\t\t// Before: <p>a{a</p><p>b}b</p><p>cc</p>\n\t\t\t// Merge: <p>b}b</p><p>cca{a</p>\n\t\t\t// Fix: <p>{b}b</p><p>ccaa</p>\n\t\t\t//\n\t\t\t// Case 2: Range start is before merged node but not directly.\n\t\t\t// Result should include all nodes that were in the original range.\n\t\t\t//\n\t\t\t// Before: <p>aa</p>{<p>cc</p><p>b}b</p>\n\t\t\t// Merge: <p>aab}b</p>{<p>cc</p>\n\t\t\t// Fix: <p>aa{bb</p><p>cc</p>}\n\t\t\t//\n\t\t\t// The range is expanded by an additional `b` letter but it is better than dropping the whole `cc` paragraph.\n\t\t\t//\n\t\t\t// Case 3: Range start is directly before merged node.\n\t\t\t// Resulting range should include only nodes from the merged element:\n\t\t\t//\n\t\t\t// Before: <p>aa</p>{<p>b}b</p><p>cc</p>\n\t\t\t// Merge: <p>aab}b</p>{<p>cc</p>\n\t\t\t// Fix: <p>aa{b}b</p><p>cc</p>\n\t\t\t//\n\n\t\t\tif ( operation.sourcePosition.isBefore( operation.targetPosition ) ) {\n\t\t\t\t// Case 1.\n\t\t\t\tstart = Position._createAt( end );\n\t\t\t\tstart.offset = 0;\n\t\t\t} else {\n\t\t\t\tif ( !operation.deletionPosition.isEqual( start ) ) {\n\t\t\t\t\t// Case 2.\n\t\t\t\t\tend = operation.deletionPosition;\n\t\t\t\t}\n\n\t\t\t\t// In both case 2 and 3 start is at the end of the merge-to element.\n\t\t\t\tstart = operation.targetPosition;\n\t\t\t}\n\n\t\t\treturn new Range( start, end );\n\t\t}\n\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Returns an array containing one or two {@link ~Range ranges} that are a result of transforming this\n\t * {@link ~Range range} by inserting `howMany` nodes at `insertPosition`. Two {@link ~Range ranges} are\n\t * returned if the insertion was inside this {@link ~Range range} and `spread` is set to `true`.\n\t *\n\t * Examples:\n\t *\n\t * ```ts\n\t * let range = model.createRange(\n\t * \tmodel.createPositionFromPath( root, [ 2, 7 ] ),\n\t * \tmodel.createPositionFromPath( root, [ 4, 0, 1 ] )\n\t * );\n\t * let transformed = range._getTransformedByInsertion( model.createPositionFromPath( root, [ 1 ] ), 2 );\n\t * // transformed array has one range from [ 4, 7 ] to [ 6, 0, 1 ]\n\t *\n\t * transformed = range._getTransformedByInsertion( model.createPositionFromPath( root, [ 4, 0, 0 ] ), 4 );\n\t * // transformed array has one range from [ 2, 7 ] to [ 4, 0, 5 ]\n\t *\n\t * transformed = range._getTransformedByInsertion( model.createPositionFromPath( root, [ 3, 2 ] ), 4 );\n\t * // transformed array has one range, which is equal to original range\n\t *\n\t * transformed = range._getTransformedByInsertion( model.createPositionFromPath( root, [ 3, 2 ] ), 4, true );\n\t * // transformed array has two ranges: from [ 2, 7 ] to [ 3, 2 ] and from [ 3, 6 ] to [ 4, 0, 1 ]\n\t * ```\n\t *\n\t * @internal\n\t * @param insertPosition Position where nodes are inserted.\n\t * @param howMany How many nodes are inserted.\n\t * @param spread Flag indicating whether this range should be spread if insertion\n\t * was inside the range. Defaults to `false`.\n\t * @returns Result of the transformation.\n\t */\n\tpublic _getTransformedByInsertion( insertPosition: Position, howMany: number, spread: boolean = false ): Array<Range> {\n\t\tif ( spread && this.containsPosition( insertPosition ) ) {\n\t\t\t// Range has to be spread. The first part is from original start to the spread point.\n\t\t\t// The other part is from spread point to the original end, but transformed by\n\t\t\t// insertion to reflect insertion changes.\n\n\t\t\treturn [\n\t\t\t\tnew Range( this.start, insertPosition ),\n\t\t\t\tnew Range(\n\t\t\t\t\tinsertPosition.getShiftedBy( howMany ),\n\t\t\t\t\tthis.end._getTransformedByInsertion( insertPosition, howMany )\n\t\t\t\t)\n\t\t\t];\n\t\t} else {\n\t\t\tconst range = new Range( this.start, this.end );\n\n\t\t\t( range as any ).start = range.start._getTransformedByInsertion( insertPosition, howMany );\n\t\t\t( range as any ).end = range.end._getTransformedByInsertion( insertPosition, howMany );\n\n\t\t\treturn [ range ];\n\t\t}\n\t}\n\n\t/**\n\t * Returns an array containing {@link ~Range ranges} that are a result of transforming this\n\t * {@link ~Range range} by moving `howMany` nodes from `sourcePosition` to `targetPosition`.\n\t *\n\t * @internal\n\t * @param sourcePosition Position from which nodes are moved.\n\t * @param targetPosition Position to where nodes are moved.\n\t * @param howMany How many nodes are moved.\n\t * @param spread Whether the range should be spread if the move points inside the range.\n\t * @returns Result of the transformation.\n\t */\n\tpublic _getTransformedByMove(\n\t\tsourcePosition: Position,\n\t\ttargetPosition: Position,\n\t\thowMany: number,\n\t\tspread: boolean = false\n\t): Array<Range> {\n\t\t// Special case for transforming a collapsed range. Just transform it like a position.\n\t\tif ( this.isCollapsed ) {\n\t\t\tconst newPos = this.start._getTransformedByMove( sourcePosition, targetPosition, howMany );\n\n\t\t\treturn [ new Range( newPos ) ];\n\t\t}\n\n\t\t// Special case for transformation when a part of the range is moved towards the range.\n\t\t//\n\t\t// Examples:\n\t\t//\n\t\t// <div><p>ab</p><p>c[d</p></div><p>e]f</p> --> <div><p>ab</p></div><p>c[d</p><p>e]f</p>\n\t\t// <p>e[f</p><div><p>a]b</p><p>cd</p></div> --> <p>e[f</p><p>a]b</p><div><p>cd</p></div>\n\t\t//\n\t\t// Without this special condition, the default algorithm leaves an \"artifact\" range from one of `differenceSet` parts:\n\t\t//\n\t\t// <div><p>ab</p><p>c[d</p></div><p>e]f</p> --> <div><p>ab</p>{</div>}<p>c[d</p><p>e]f</p>\n\t\t//\n\t\t// This special case is applied only if the range is to be kept together (not spread).\n\t\tconst moveRange = Range._createFromPositionAndShift( sourcePosition, howMany );\n\t\tconst insertPosition = targetPosition._getTransformedByDeletion( sourcePosition, howMany );\n\n\t\tif ( this.containsPosition( targetPosition ) && !spread ) {\n\t\t\tif ( moveRange.containsPosition( this.start ) || moveRange.containsPosition( this.end ) ) {\n\t\t\t\tconst start = this.start._getTransformedByMove( sourcePosition, targetPosition, howMany );\n\t\t\t\tconst end = this.end._getTransformedByMove( sourcePosition, targetPosition, howMany );\n\n\t\t\t\treturn [ new Range( start, end ) ];\n\t\t\t}\n\t\t}\n\n\t\t// Default algorithm.\n\t\tlet result: Array<Range>;\n\n\t\tconst differenceSet = this.getDifference( moveRange );\n\t\tlet difference = null;\n\n\t\tconst common = this.getIntersection( moveRange );\n\n\t\tif ( differenceSet.length == 1 ) {\n\t\t\t// `moveRange` and this range may intersect but may be separate.\n\t\t\tdifference = new Range(\n\t\t\t\tdifferenceSet[ 0 ].start._getTransformedByDeletion( sourcePosition, howMany )!,\n\t\t\t\tdifferenceSet[ 0 ].end._getTransformedByDeletion( sourcePosition, howMany )!\n\t\t\t);\n\t\t} else if ( differenceSet.length == 2 ) {\n\t\t\t// `moveRange` is inside this range.\n\t\t\tdifference = new Range(\n\t\t\t\tthis.start,\n\t\t\t\tthis.end._getTransformedByDeletion( sourcePosition, howMany )\n\t\t\t);\n\t\t} // else, `moveRange` contains this range.\n\n\t\tif ( difference ) {\n\t\t\tresult = difference._getTransformedByInsertion( insertPosition!, howMany, common !== null || spread );\n\t\t} else {\n\t\t\tresult = [];\n\t\t}\n\n\t\tif ( common ) {\n\t\t\tconst transformedCommon = new Range(\n\t\t\t\tcommon.start._getCombined( moveRange.start, insertPosition! ),\n\t\t\t\tcommon.end._getCombined( moveRange.start, insertPosition! )\n\t\t\t);\n\n\t\t\tif ( result.length == 2 ) {\n\t\t\t\tresult.splice( 1, 0, transformedCommon );\n\t\t\t} else {\n\t\t\t\tresult.push( transformedCommon );\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t}\n\n\t/**\n\t * Returns a copy of this range that is transformed by deletion of `howMany` nodes from `deletePosition`.\n\t *\n\t * If the deleted range is intersecting with the transformed range, the transformed range will be shrank.\n\t *\n\t * If the deleted range contains transformed range, `null` will be returned.\n\t *\n\t * @internal\n\t * @param deletionPosition Position from which nodes are removed.\n\t * @param howMany How many nodes are removed.\n\t * @returns Result of the transformation.\n\t */\n\tpublic _getTransformedByDeletion( deletePosition: Position, howMany: number ): Range | null {\n\t\tlet newStart = this.start._getTransformedByDeletion( deletePosition, howMany );\n\t\tlet newEnd = this.end._getTransformedByDeletion( deletePosition, howMany );\n\n\t\tif ( newStart == null && newEnd == null ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( newStart == null ) {\n\t\t\tnewStart = deletePosition;\n\t\t}\n\n\t\tif ( newEnd == null ) {\n\t\t\tnewEnd = deletePosition;\n\t\t}\n\n\t\treturn new Range( newStart, newEnd );\n\t}\n\n\t/**\n\t * Creates a new range, spreading from specified {@link module:engine/model/position~Position position} to a position moved by\n\t * given `shift`. If `shift` is a negative value, shifted position is treated as the beginning of the range.\n\t *\n\t * @internal\n\t * @param position Beginning of the range.\n\t * @param shift How long the range should be.\n\t */\n\tpublic static _createFromPositionAndShift( position: Position, shift: number ): Range {\n\t\tconst start = position;\n\t\tconst end = position.getShiftedBy( shift );\n\n\t\treturn shift > 0 ? new this( start, end ) : new this( end, start );\n\t}\n\n\t/**\n\t * Creates a range inside an {@link module:engine/model/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * @internal\n\t * @param element Element which is a parent for the range.\n\t */\n\tpublic static _createIn( element: Element | DocumentFragment ): Range {\n\t\treturn new this( Position._createAt( element, 0 ), Position._createAt( element, element.maxOffset ) );\n\t}\n\n\t/**\n\t * Creates a range that starts before given {@link module:engine/model/item~Item model item} and ends after it.\n\t *\n\t * @internal\n\t */\n\tpublic static _createOn( item: Item ): Range {\n\t\treturn this._createFromPositionAndShift( Position._createBefore( item ), item.offsetSize );\n\t}\n\n\t/**\n\t * Combines all ranges from the passed array into a one range. At least one range has to be passed.\n\t * Passed ranges must not have common parts.\n\t *\n\t * The first range from the array is a reference range. If other ranges start or end on the exactly same position where\n\t * the reference range, they get combined into one range.\n\t *\n\t * ```\n\t * [ ][] [ ][ ][ ][ ][] [ ] // Passed ranges, shown sorted\n\t * [ ] // The result of the function if the first range was a reference range.\n\t * [ ] // The result of the function if the third-to-seventh range was a reference range.\n\t * [ ] // The result of the function if the last range was a reference range.\n\t * ```\n\t *\n\t * @internal\n\t * @param ranges Ranges to combine.\n\t * @returns Combined range.\n\t */\n\tpublic static _createFromRanges( ranges: Array<Range> ): Range {\n\t\tif ( ranges.length === 0 ) {\n\t\t\t/**\n\t\t\t * At least one range has to be passed to\n\t\t\t * {@link module:engine/model/range~Range._createFromRanges `Range._createFromRanges()`}.\n\t\t\t *\n\t\t\t * @error range-create-from-ranges-empty-array\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'range-create-from-ranges-empty-array',\n\t\t\t\tnull\n\t\t\t);\n\t\t} else if ( ranges.length == 1 ) {\n\t\t\treturn ranges[ 0 ].clone();\n\t\t}\n\n\t\t// 1. Set the first range in `ranges` array as a reference range.\n\t\t// If we are going to return just a one range, one of the ranges need to be the reference one.\n\t\t// Other ranges will be stuck to that range, if possible.\n\t\tconst ref = ranges[ 0 ];\n\n\t\t// 2. Sort all the ranges so it's easier to process them.\n\t\tranges.sort( ( a, b ) => {\n\t\t\treturn a.start.isAfter( b.start ) ? 1 : -1;\n\t\t} );\n\n\t\t// 3. Check at which index the reference range is now.\n\t\tconst refIndex = ranges.indexOf( ref );\n\n\t\t// 4. At this moment we don't need the original range.\n\t\t// We are going to modify the result and we need to return a new instance of Range.\n\t\t// We have to create a copy of the reference range.\n\t\tconst result = new this( ref.start, ref.end );\n\n\t\t// 5. Ranges should be checked and glued starting from the range that is closest to the reference range.\n\t\t// Since ranges are sorted, start with the range with index that is closest to reference range index.\n\t\tif ( refIndex > 0 ) {\n\t\t\t// eslint-disable-next-line no-constant-condition\n\t\t\tfor ( let i = refIndex - 1; true; i++ ) {\n\t\t\t\tif ( ranges[ i ].end.isEqual( result.start ) ) {\n\t\t\t\t\t( result as any ).start = Position._createAt( ranges[ i ].start );\n\t\t\t\t} else {\n\t\t\t\t\t// If ranges are not starting/ending at the same position there is no point in looking further.\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// 6. Ranges should be checked and glued starting from the range that is closest to the reference range.\n\t\t// Since ranges are sorted, start with the range with index that is closest to reference range index.\n\t\tfor ( let i = refIndex + 1; i < ranges.length; i++ ) {\n\t\t\tif ( ranges[ i ].start.isEqual( result.end ) ) {\n\t\t\t\t( result as any ).end = Position._createAt( ranges[ i ].end );\n\t\t\t} else {\n\t\t\t\t// If ranges are not starting/ending at the same position there is no point in looking further.\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t}\n\n\t/**\n\t * Creates a `Range` instance from given plain object (i.e. parsed JSON string).\n\t *\n\t * @param json Plain object to be converted to `Range`.\n\t * @param doc Document object that will be range owner.\n\t * @returns `Range` instance created using given plain object.\n\t */\n\tpublic static fromJSON( json: any, doc: Document ): Range {\n\t\treturn new this( Position.fromJSON( json.start, doc ), Position.fromJSON( json.end, doc ) );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `${ this.root } [ ${ this.start.path.join( ', ' ) } ] - [ ${ this.end.path.join( ', ' ) } ]`;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ModelPosition: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nRange.prototype.is = function( type: string ): boolean {\n\treturn type === 'range' || type === 'model:range';\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/conversion/mapper\n */\n\nimport ModelPosition from '../model/position';\nimport ModelRange from '../model/range';\n\nimport ViewPosition from '../view/position';\nimport ViewRange from '../view/range';\nimport ViewText from '../view/text';\n\nimport { CKEditorError, EmitterMixin } from '@ckeditor/ckeditor5-utils';\n\nimport type ViewDocumentFragment from '../view/documentfragment';\nimport type ViewElement from '../view/element';\nimport type ModelElement from '../model/element';\nimport type ModelDocumentFragment from '../model/documentfragment';\nimport type ViewNode from '../view/node';\n\n/**\n * Maps elements, positions and markers between the {@link module:engine/view/document~Document view} and\n * the {@link module:engine/model/model model}.\n *\n * The instance of the Mapper used for the editing pipeline is available in\n * {@link module:engine/controller/editingcontroller~EditingController#mapper `editor.editing.mapper`}.\n *\n * Mapper uses bound elements to find corresponding elements and positions, so, to get proper results,\n * all model elements should be {@link module:engine/conversion/mapper~Mapper#bindElements bound}.\n *\n * To map the complex model to/from view relations, you may provide custom callbacks for the\n * {@link module:engine/conversion/mapper~Mapper#event:modelToViewPosition modelToViewPosition event} and\n * {@link module:engine/conversion/mapper~Mapper#event:viewToModelPosition viewToModelPosition event} that are fired whenever\n * a position mapping request occurs.\n * Those events are fired by the {@link module:engine/conversion/mapper~Mapper#toViewPosition toViewPosition}\n * and {@link module:engine/conversion/mapper~Mapper#toModelPosition toModelPosition} methods. `Mapper` adds its own default callbacks\n * with `'lowest'` priority. To override default `Mapper` mapping, add custom callback with higher priority and\n * stop the event.\n */\nexport default class Mapper extends EmitterMixin() {\n\t/**\n\t * Model element to view element mapping.\n\t */\n\tprivate _modelToViewMapping = new WeakMap<ModelElement | ModelDocumentFragment, ViewElement | ViewDocumentFragment>();\n\n\t/**\n\t * View element to model element mapping.\n\t */\n\tprivate _viewToModelMapping = new WeakMap<ViewElement | ViewDocumentFragment, ModelElement | ModelDocumentFragment>();\n\n\t/**\n\t * A map containing callbacks between view element names and functions evaluating length of view elements\n\t * in model.\n\t */\n\tprivate _viewToModelLengthCallbacks = new Map<string, ( element: ViewElement ) => number>();\n\n\t/**\n\t * Model marker name to view elements mapping.\n\t *\n\t * Keys are `String`s while values are `Set`s with {@link module:engine/view/element~Element view elements}.\n\t * One marker (name) can be mapped to multiple elements.\n\t */\n\tprivate _markerNameToElements = new Map<string, Set<ViewElement>>();\n\n\t/**\n\t * View element to model marker names mapping.\n\t *\n\t * This is reverse to {@link ~Mapper#_markerNameToElements} map.\n\t */\n\tprivate _elementToMarkerNames = new Map<ViewElement, Set<string>>();\n\n\t/**\n\t * The map of removed view elements with their current root (used for deferred unbinding).\n\t */\n\tprivate _deferredBindingRemovals = new Map<ViewElement, ViewElement | ViewDocumentFragment>();\n\n\t/**\n\t * Stores marker names of markers which have changed due to unbinding a view element (so it is assumed that the view element\n\t * has been removed, moved or renamed).\n\t */\n\tprivate _unboundMarkerNames = new Set<string>();\n\n\t/**\n\t * Creates an instance of the mapper.\n\t */\n\tconstructor() {\n\t\tsuper();\n\n\t\t// Default mapper algorithm for mapping model position to view position.\n\t\tthis.on<MapperModelToViewPositionEvent>( 'modelToViewPosition', ( evt, data ) => {\n\t\t\tif ( data.viewPosition ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst viewContainer = this._modelToViewMapping.get( data.modelPosition.parent as ModelElement );\n\n\t\t\tif ( !viewContainer ) {\n\t\t\t\t/**\n\t\t\t\t * A model position could not be mapped to the view because the parent of the model position\n\t\t\t\t * does not have a mapped view element (might have not been converted yet or it has no converter).\n\t\t\t\t *\n\t\t\t\t * Make sure that the model element is correctly converted to the view.\n\t\t\t\t *\n\t\t\t\t * @error mapping-model-position-view-parent-not-found\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'mapping-model-position-view-parent-not-found', this, { modelPosition: data.modelPosition } );\n\t\t\t}\n\n\t\t\tdata.viewPosition = this.findPositionIn( viewContainer, data.modelPosition.offset );\n\t\t}, { priority: 'low' } );\n\n\t\t// Default mapper algorithm for mapping view position to model position.\n\t\tthis.on<MapperViewToModelPositionEvent>( 'viewToModelPosition', ( evt, data ) => {\n\t\t\tif ( data.modelPosition ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst viewBlock = this.findMappedViewAncestor( data.viewPosition );\n\t\t\tconst modelParent = this._viewToModelMapping.get( viewBlock );\n\t\t\tconst modelOffset = this._toModelOffset( data.viewPosition.parent as ViewElement, data.viewPosition.offset, viewBlock );\n\n\t\t\tdata.modelPosition = ModelPosition._createAt( modelParent!, modelOffset );\n\t\t}, { priority: 'low' } );\n\t}\n\n\t/**\n\t * Marks model and view elements as corresponding. Corresponding elements can be retrieved by using\n\t * the {@link module:engine/conversion/mapper~Mapper#toModelElement toModelElement} and\n\t * {@link module:engine/conversion/mapper~Mapper#toViewElement toViewElement} methods.\n\t * The information that elements are bound is also used to translate positions.\n\t *\n\t * @param modelElement Model element.\n\t * @param viewElement View element.\n\t */\n\tpublic bindElements(\n\t\tmodelElement: ModelElement | ModelDocumentFragment,\n\t\tviewElement: ViewElement | ViewDocumentFragment\n\t): void {\n\t\tthis._modelToViewMapping.set( modelElement, viewElement );\n\t\tthis._viewToModelMapping.set( viewElement, modelElement );\n\t}\n\n\t/**\n\t * Unbinds the given {@link module:engine/view/element~Element view element} from the map.\n\t *\n\t * **Note:** view-to-model binding will be removed, if it existed. However, corresponding model-to-view binding\n\t * will be removed only if model element is still bound to the passed `viewElement`.\n\t *\n\t * This behavior allows for re-binding model element to another view element without fear of losing the new binding\n\t * when the previously bound view element is unbound.\n\t *\n\t * @param viewElement View element to unbind.\n\t * @param options The options object.\n\t * @param options.defer Controls whether the binding should be removed immediately or deferred until a\n\t * {@link #flushDeferredBindings `flushDeferredBindings()`} call.\n\t */\n\tpublic unbindViewElement(\n\t\tviewElement: ViewElement,\n\t\toptions: { defer?: boolean } = {}\n\t): void {\n\t\tconst modelElement = this.toModelElement( viewElement )!;\n\n\t\tif ( this._elementToMarkerNames.has( viewElement ) ) {\n\t\t\tfor ( const markerName of this._elementToMarkerNames.get( viewElement )! ) {\n\t\t\t\tthis._unboundMarkerNames.add( markerName );\n\t\t\t}\n\t\t}\n\n\t\tif ( options.defer ) {\n\t\t\tthis._deferredBindingRemovals.set( viewElement, viewElement.root );\n\t\t} else {\n\t\t\tthis._viewToModelMapping.delete( viewElement );\n\n\t\t\tif ( this._modelToViewMapping.get( modelElement ) == viewElement ) {\n\t\t\t\tthis._modelToViewMapping.delete( modelElement );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Unbinds the given {@link module:engine/model/element~Element model element} from the map.\n\t *\n\t * **Note:** the model-to-view binding will be removed, if it existed. However, the corresponding view-to-model binding\n\t * will be removed only if the view element is still bound to the passed `modelElement`.\n\t *\n\t * This behavior lets for re-binding view element to another model element without fear of losing the new binding\n\t * when the previously bound model element is unbound.\n\t *\n\t * @param modelElement Model element to unbind.\n\t */\n\tpublic unbindModelElement( modelElement: ModelElement ): void {\n\t\tconst viewElement = this.toViewElement( modelElement )!;\n\n\t\tthis._modelToViewMapping.delete( modelElement );\n\n\t\tif ( this._viewToModelMapping.get( viewElement ) == modelElement ) {\n\t\t\tthis._viewToModelMapping.delete( viewElement );\n\t\t}\n\t}\n\n\t/**\n\t * Binds the given marker name with the given {@link module:engine/view/element~Element view element}. The element\n\t * will be added to the current set of elements bound with the given marker name.\n\t *\n\t * @param element Element to bind.\n\t * @param name Marker name.\n\t */\n\tpublic bindElementToMarker( element: ViewElement, name: string ): void {\n\t\tconst elements = this._markerNameToElements.get( name ) || new Set();\n\t\telements.add( element );\n\n\t\tconst names = this._elementToMarkerNames.get( element ) || new Set();\n\t\tnames.add( name );\n\n\t\tthis._markerNameToElements.set( name, elements );\n\t\tthis._elementToMarkerNames.set( element, names );\n\t}\n\n\t/**\n\t * Unbinds an element from given marker name.\n\t *\n\t * @param element Element to unbind.\n\t * @param name Marker name.\n\t */\n\tpublic unbindElementFromMarkerName( element: ViewElement, name: string ): void {\n\t\tconst nameToElements = this._markerNameToElements.get( name );\n\n\t\tif ( nameToElements ) {\n\t\t\tnameToElements.delete( element );\n\n\t\t\tif ( nameToElements.size == 0 ) {\n\t\t\t\tthis._markerNameToElements.delete( name );\n\t\t\t}\n\t\t}\n\n\t\tconst elementToNames = this._elementToMarkerNames.get( element );\n\n\t\tif ( elementToNames ) {\n\t\t\telementToNames.delete( name );\n\n\t\t\tif ( elementToNames.size == 0 ) {\n\t\t\t\tthis._elementToMarkerNames.delete( element );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Returns all marker names of markers which have changed due to unbinding a view element (so it is assumed that the view element\n\t * has been removed, moved or renamed) since the last flush. After returning, the marker names list is cleared.\n\t */\n\tpublic flushUnboundMarkerNames(): Array<string> {\n\t\tconst markerNames = Array.from( this._unboundMarkerNames );\n\n\t\tthis._unboundMarkerNames.clear();\n\n\t\treturn markerNames;\n\t}\n\n\t/**\n\t * Unbinds all deferred binding removals of view elements that in the meantime were not re-attached to some root or document fragment.\n\t *\n\t * See: {@link #unbindViewElement `unbindViewElement()`}.\n\t */\n\tpublic flushDeferredBindings(): void {\n\t\tfor ( const [ viewElement, root ] of this._deferredBindingRemovals ) {\n\t\t\t// Unbind it only if it wasn't re-attached to some root or document fragment.\n\t\t\tif ( viewElement.root == root ) {\n\t\t\t\tthis.unbindViewElement( viewElement );\n\t\t\t}\n\t\t}\n\n\t\tthis._deferredBindingRemovals = new Map();\n\t}\n\n\t/**\n\t * Removes all model to view and view to model bindings.\n\t */\n\tpublic clearBindings(): void {\n\t\tthis._modelToViewMapping = new WeakMap();\n\t\tthis._viewToModelMapping = new WeakMap();\n\t\tthis._markerNameToElements = new Map();\n\t\tthis._elementToMarkerNames = new Map();\n\t\tthis._unboundMarkerNames = new Set();\n\t\tthis._deferredBindingRemovals = new Map();\n\t}\n\n\t/**\n\t * Gets the corresponding model element.\n\t *\n\t * **Note:** {@link module:engine/view/uielement~UIElement} does not have corresponding element in model.\n\t *\n\t * @label ELEMENT\n\t * @param viewElement View element.\n\t * @returns Corresponding model element or `undefined` if not found.\n\t */\n\tpublic toModelElement( viewElement: ViewElement ): ModelElement | undefined;\n\n\t/**\n\t * Gets the corresponding model document fragment.\n\t *\n\t * @label DOCUMENT_FRAGMENT\n\t * @param viewDocumentFragment View document fragment.\n\t * @returns Corresponding model document fragment or `undefined` if not found.\n\t */\n\tpublic toModelElement( viewDocumentFragment: ViewDocumentFragment ): ModelDocumentFragment | undefined;\n\n\tpublic toModelElement( viewElement: ViewElement | ViewDocumentFragment ): ModelElement | ModelDocumentFragment | undefined {\n\t\treturn this._viewToModelMapping.get( viewElement );\n\t}\n\n\t/**\n\t * Gets the corresponding view element.\n\t *\n\t * @label ELEMENT\n\t * @param modelElement Model element.\n\t * @returns Corresponding view element or `undefined` if not found.\n\t */\n\tpublic toViewElement( modelElement: ModelElement ): ViewElement | undefined;\n\n\t/**\n\t * Gets the corresponding view document fragment.\n\t *\n\t * @label DOCUMENT_FRAGMENT\n\t * @param modelDocumentFragment Model document fragment.\n\t * @returns Corresponding view document fragment or `undefined` if not found.\n\t */\n\tpublic toViewElement( modelDocumentFragment: ModelDocumentFragment ): ViewDocumentFragment | undefined;\n\n\tpublic toViewElement( modelElement: ModelElement | ModelDocumentFragment ): ViewElement | ViewDocumentFragment | undefined {\n\t\treturn this._modelToViewMapping.get( modelElement );\n\t}\n\n\t/**\n\t * Gets the corresponding model range.\n\t *\n\t * @param viewRange View range.\n\t * @returns Corresponding model range.\n\t */\n\tpublic toModelRange( viewRange: ViewRange ): ModelRange {\n\t\treturn new ModelRange( this.toModelPosition( viewRange.start ), this.toModelPosition( viewRange.end ) );\n\t}\n\n\t/**\n\t * Gets the corresponding view range.\n\t *\n\t * @param modelRange Model range.\n\t * @returns Corresponding view range.\n\t */\n\tpublic toViewRange( modelRange: ModelRange ): ViewRange {\n\t\treturn new ViewRange( this.toViewPosition( modelRange.start ), this.toViewPosition( modelRange.end ) );\n\t}\n\n\t/**\n\t * Gets the corresponding model position.\n\t *\n\t * @fires viewToModelPosition\n\t * @param viewPosition View position.\n\t * @returns Corresponding model position.\n\t */\n\tpublic toModelPosition( viewPosition: ViewPosition ): ModelPosition {\n\t\tconst data: MapperViewToModelPositionEvent[ 'args' ][ 0 ] = {\n\t\t\tviewPosition,\n\t\t\tmapper: this\n\t\t};\n\n\t\tthis.fire<MapperViewToModelPositionEvent>( 'viewToModelPosition', data );\n\n\t\treturn data.modelPosition!;\n\t}\n\n\t/**\n\t * Gets the corresponding view position.\n\t *\n\t * @fires modelToViewPosition\n\t * @param modelPosition Model position.\n\t * @param options Additional options for position mapping process.\n\t * @param options.isPhantom Should be set to `true` if the model position to map is pointing to a place\n\t * in model tree which no longer exists. For example, it could be an end of a removed model range.\n\t * @returns Corresponding view position.\n\t */\n\tpublic toViewPosition(\n\t\tmodelPosition: ModelPosition,\n\t\toptions: { isPhantom?: boolean } = {}\n\t): ViewPosition {\n\t\tconst data: MapperModelToViewPositionEvent[ 'args' ][ 0 ] = {\n\t\t\tmodelPosition,\n\t\t\tmapper: this,\n\t\t\tisPhantom: options.isPhantom\n\t\t};\n\n\t\tthis.fire<MapperModelToViewPositionEvent>( 'modelToViewPosition', data );\n\n\t\treturn data.viewPosition!;\n\t}\n\n\t/**\n\t * Gets all view elements bound to the given marker name.\n\t *\n\t * @param name Marker name.\n\t * @returns View elements bound with the given marker name or `null`\n\t * if no elements are bound to the given marker name.\n\t */\n\tpublic markerNameToElements( name: string ): Set<ViewElement> | null {\n\t\tconst boundElements = this._markerNameToElements.get( name );\n\n\t\tif ( !boundElements ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst elements = new Set<ViewElement>();\n\n\t\tfor ( const element of boundElements ) {\n\t\t\tif ( element.is( 'attributeElement' ) ) {\n\t\t\t\tfor ( const clone of element.getElementsWithSameId() ) {\n\t\t\t\t\telements.add( clone );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\telements.add( element );\n\t\t\t}\n\t\t}\n\n\t\treturn elements;\n\t}\n\n\t/**\n\t * Registers a callback that evaluates the length in the model of a view element with the given name.\n\t *\n\t * The callback is fired with one argument, which is a view element instance. The callback is expected to return\n\t * a number representing the length of the view element in the model.\n\t *\n\t * ```ts\n\t * // List item in view may contain nested list, which have other list items. In model though,\n\t * // the lists are represented by flat structure. Because of those differences, length of list view element\n\t * // may be greater than one. In the callback it's checked how many nested list items are in evaluated list item.\n\t *\n\t * function getViewListItemLength( element ) {\n\t * \tlet length = 1;\n\t *\n\t * \tfor ( let child of element.getChildren() ) {\n\t * \t\tif ( child.name == 'ul' || child.name == 'ol' ) {\n\t * \t\t\tfor ( let item of child.getChildren() ) {\n\t * \t\t\t\tlength += getViewListItemLength( item );\n\t * \t\t\t}\n\t * \t\t}\n\t * \t}\n\t *\n\t * \treturn length;\n\t * }\n\t *\n\t * mapper.registerViewToModelLength( 'li', getViewListItemLength );\n\t * ```\n\t *\n\t * @param viewElementName Name of view element for which callback is registered.\n\t * @param lengthCallback Function return a length of view element instance in model.\n\t */\n\tpublic registerViewToModelLength(\n\t\tviewElementName: string,\n\t\tlengthCallback: ( element: ViewElement ) => number\n\t): void {\n\t\tthis._viewToModelLengthCallbacks.set( viewElementName, lengthCallback );\n\t}\n\n\t/**\n\t * For the given `viewPosition`, finds and returns the closest ancestor of this position that has a mapping to\n\t * the model.\n\t *\n\t * @param viewPosition Position for which a mapped ancestor should be found.\n\t */\n\tpublic findMappedViewAncestor( viewPosition: ViewPosition ): ViewElement {\n\t\tlet parent: any = viewPosition.parent;\n\n\t\twhile ( !this._viewToModelMapping.has( parent ) ) {\n\t\t\tparent = parent.parent;\n\t\t}\n\n\t\treturn parent;\n\t}\n\n\t/**\n\t * Calculates model offset based on the view position and the block element.\n\t *\n\t * Example:\n\t *\n\t * ```html\n\t * <p>foo<b>ba|r</b></p> // _toModelOffset( b, 2, p ) -> 5\n\t * ```\n\t *\n\t * Is a sum of:\n\t *\n\t * ```html\n\t * <p>foo|<b>bar</b></p> // _toModelOffset( p, 3, p ) -> 3\n\t * <p>foo<b>ba|r</b></p> // _toModelOffset( b, 2, b ) -> 2\n\t * ```\n\t *\n\t * @param viewParent Position parent.\n\t * @param viewOffset Position offset.\n\t * @param viewBlock Block used as a base to calculate offset.\n\t * @returns Offset in the model.\n\t */\n\tprivate _toModelOffset(\n\t\tviewParent: ViewElement,\n\t\tviewOffset: number,\n\t\tviewBlock: ViewElement\n\t): number {\n\t\tif ( viewBlock != viewParent ) {\n\t\t\t// See example.\n\t\t\tconst offsetToParentStart = this._toModelOffset( viewParent.parent as any, viewParent.index!, viewBlock );\n\t\t\tconst offsetInParent = this._toModelOffset( viewParent, viewOffset, viewParent );\n\n\t\t\treturn offsetToParentStart + offsetInParent;\n\t\t}\n\n\t\t// viewBlock == viewParent, so we need to calculate the offset in the parent element.\n\n\t\t// If the position is a text it is simple (\"ba|r\" -> 2).\n\t\tif ( viewParent.is( '$text' ) ) {\n\t\t\treturn viewOffset;\n\t\t}\n\n\t\t// If the position is in an element we need to sum lengths of siblings ( <b> bar </b> foo | -> 3 + 3 = 6 ).\n\t\tlet modelOffset = 0;\n\n\t\tfor ( let i = 0; i < viewOffset; i++ ) {\n\t\t\tmodelOffset += this.getModelLength( viewParent.getChild( i ) as any );\n\t\t}\n\n\t\treturn modelOffset;\n\t}\n\n\t/**\n\t * Gets the length of the view element in the model.\n\t *\n\t * The length is calculated as follows:\n\t * * if a {@link #registerViewToModelLength length mapping callback} is provided for the given `viewNode`, it is used to\n\t * evaluate the model length (`viewNode` is used as first and only parameter passed to the callback),\n\t * * length of a {@link module:engine/view/text~Text text node} is equal to the length of its\n\t * {@link module:engine/view/text~Text#data data},\n\t * * length of a {@link module:engine/view/uielement~UIElement ui element} is equal to 0,\n\t * * length of a mapped {@link module:engine/view/element~Element element} is equal to 1,\n\t * * length of a non-mapped {@link module:engine/view/element~Element element} is equal to the length of its children.\n\t *\n\t * Examples:\n\t *\n\t * ```\n\t * foo -> 3 // Text length is equal to its data length.\n\t * <p>foo</p> -> 1 // Length of an element which is mapped is by default equal to 1.\n\t * <b>foo</b> -> 3 // Length of an element which is not mapped is a length of its children.\n\t * <div><p>x</p><p>y</p></div> -> 2 // Assuming that <div> is not mapped and <p> are mapped.\n\t * ```\n\t *\n\t * @param viewNode View node.\n\t * @returns Length of the node in the tree model.\n\t */\n\tpublic getModelLength( viewNode: ViewNode | ViewDocumentFragment ): number {\n\t\tif ( this._viewToModelLengthCallbacks.get( ( viewNode as any ).name ) ) {\n\t\t\tconst callback = this._viewToModelLengthCallbacks.get( ( viewNode as any ).name )!;\n\n\t\t\treturn callback( viewNode as ViewElement );\n\t\t} else if ( this._viewToModelMapping.has( viewNode as ViewElement ) ) {\n\t\t\treturn 1;\n\t\t} else if ( viewNode.is( '$text' ) ) {\n\t\t\treturn viewNode.data.length;\n\t\t} else if ( viewNode.is( 'uiElement' ) ) {\n\t\t\treturn 0;\n\t\t} else {\n\t\t\tlet len = 0;\n\n\t\t\tfor ( const child of ( viewNode as ViewElement ).getChildren() ) {\n\t\t\t\tlen += this.getModelLength( child );\n\t\t\t}\n\n\t\t\treturn len;\n\t\t}\n\t}\n\n\t/**\n\t * Finds the position in the view node (or in its children) with the expected model offset.\n\t *\n\t * Example:\n\t *\n\t * ```\n\t * <p>fo<b>bar</b>bom</p> -> expected offset: 4\n\t *\n\t * findPositionIn( p, 4 ):\n\t * <p>|fo<b>bar</b>bom</p> -> expected offset: 4, actual offset: 0\n\t * <p>fo|<b>bar</b>bom</p> -> expected offset: 4, actual offset: 2\n\t * <p>fo<b>bar</b>|bom</p> -> expected offset: 4, actual offset: 5 -> we are too far\n\t *\n\t * findPositionIn( b, 4 - ( 5 - 3 ) ):\n\t * <p>fo<b>|bar</b>bom</p> -> expected offset: 2, actual offset: 0\n\t * <p>fo<b>bar|</b>bom</p> -> expected offset: 2, actual offset: 3 -> we are too far\n\t *\n\t * findPositionIn( bar, 2 - ( 3 - 3 ) ):\n\t * We are in the text node so we can simple find the offset.\n\t * <p>fo<b>ba|r</b>bom</p> -> expected offset: 2, actual offset: 2 -> position found\n\t * ```\n\t *\n\t * @param viewParent Tree view element in which we are looking for the position.\n\t * @param expectedOffset Expected offset.\n\t * @returns Found position.\n\t */\n\tpublic findPositionIn( viewParent: ViewNode | ViewDocumentFragment, expectedOffset: number ): ViewPosition {\n\t\t// Last scanned view node.\n\t\tlet viewNode: ViewNode;\n\t\t// Length of the last scanned view node.\n\t\tlet lastLength = 0;\n\n\t\tlet modelOffset = 0;\n\t\tlet viewOffset = 0;\n\n\t\t// In the text node it is simple: the offset in the model equals the offset in the text.\n\t\tif ( viewParent.is( '$text' ) ) {\n\t\t\treturn new ViewPosition( viewParent, expectedOffset );\n\t\t}\n\n\t\t// In other cases we add lengths of child nodes to find the proper offset.\n\n\t\t// If it is smaller we add the length.\n\t\twhile ( modelOffset < expectedOffset ) {\n\t\t\tviewNode = ( viewParent as ViewElement | ViewDocumentFragment ).getChild( viewOffset )!;\n\t\t\tlastLength = this.getModelLength( viewNode );\n\t\t\tmodelOffset += lastLength;\n\t\t\tviewOffset++;\n\t\t}\n\n\t\t// If it equals we found the position.\n\t\tif ( modelOffset == expectedOffset ) {\n\t\t\treturn this._moveViewPositionToTextNode( new ViewPosition( viewParent, viewOffset ) );\n\t\t}\n\t\t// If it is higher we need to enter last child.\n\t\telse {\n\t\t\t// ( modelOffset - lastLength ) is the offset to the child we enter,\n\t\t\t// so we subtract it from the expected offset to fine the offset in the child.\n\t\t\treturn this.findPositionIn( viewNode!, expectedOffset - ( modelOffset - lastLength ) );\n\t\t}\n\t}\n\n\t/**\n\t * Because we prefer positions in the text nodes over positions next to text nodes, if the view position was next to a text node,\n\t * it moves it into the text node instead.\n\t *\n\t * ```\n\t * <p>[]<b>foo</b></p> -> <p>[]<b>foo</b></p> // do not touch if position is not directly next to text\n\t * <p>foo[]<b>foo</b></p> -> <p>foo{}<b>foo</b></p> // move to text node\n\t * <p><b>[]foo</b></p> -> <p><b>{}foo</b></p> // move to text node\n\t * ```\n\t *\n\t * @param viewPosition Position potentially next to the text node.\n\t * @returns Position in the text node if possible.\n\t */\n\tprivate _moveViewPositionToTextNode( viewPosition: ViewPosition ): ViewPosition {\n\t\t// If the position is just after a text node, put it at the end of that text node.\n\t\t// If the position is just before a text node, put it at the beginning of that text node.\n\t\tconst nodeBefore = viewPosition.nodeBefore;\n\t\tconst nodeAfter = viewPosition.nodeAfter;\n\n\t\tif ( nodeBefore instanceof ViewText ) {\n\t\t\treturn new ViewPosition( nodeBefore, nodeBefore.data.length );\n\t\t} else if ( nodeAfter instanceof ViewText ) {\n\t\t\treturn new ViewPosition( nodeAfter, 0 );\n\t\t}\n\n\t\t// Otherwise, just return the given position.\n\t\treturn viewPosition;\n\t}\n}\n\n/**\n * Fired for each model-to-view position mapping request. The purpose of this event is to enable custom model-to-view position\n * mapping. Callbacks added to this event take {@link module:engine/model/position~Position model position} and are expected to\n * calculate the {@link module:engine/view/position~Position view position}. The calculated view position should be added as\n * a `viewPosition` value in the `data` object that is passed as one of parameters to the event callback.\n *\n * ```ts\n * // Assume that \"captionedImage\" model element is converted to <img> and following <span> elements in view,\n * // and the model element is bound to <img> element. Force mapping model positions inside \"captionedImage\" to that\n * // <span> element.\n * mapper.on( 'modelToViewPosition', ( evt, data ) => {\n * \tconst positionParent = modelPosition.parent;\n *\n * \tif ( positionParent.name == 'captionedImage' ) {\n * \t\tconst viewImg = data.mapper.toViewElement( positionParent );\n * \t\tconst viewCaption = viewImg.nextSibling; // The <span> element.\n *\n * \t\tdata.viewPosition = new ViewPosition( viewCaption, modelPosition.offset );\n *\n * \t\t// Stop the event if other callbacks should not modify calculated value.\n * \t\tevt.stop();\n * \t}\n * } );\n * ```\n *\n * **Note:** keep in mind that sometimes a \"phantom\" model position is being converted. A \"phantom\" model position is\n * a position that points to a nonexistent place in model. Such a position might still be valid for conversion, though\n * (it would point to a correct place in the view when converted). One example of such a situation is when a range is\n * removed from the model, there may be a need to map the range's end (which is no longer a valid model position). To\n * handle such situations, check the `data.isPhantom` flag:\n *\n * ```ts\n * // Assume that there is a \"customElement\" model element and whenever the position is before it,\n * // we want to move it to the inside of the view element bound to \"customElement\".\n * mapper.on( 'modelToViewPosition', ( evt, data ) => {\n * \tif ( data.isPhantom ) {\n * \t\treturn;\n * \t}\n *\n * \t// Below line might crash for phantom position that does not exist in model.\n * \tconst sibling = data.modelPosition.nodeBefore;\n *\n * \t// Check if this is the element we are interested in.\n * \tif ( !sibling.is( 'element', 'customElement' ) ) {\n * \t\treturn;\n * \t}\n *\n * \tconst viewElement = data.mapper.toViewElement( sibling );\n *\n * \tdata.viewPosition = new ViewPosition( sibling, 0 );\n *\n * \tevt.stop();\n * } );\n * ```\n *\n * **Note:** the default mapping callback is provided with a `low` priority setting and does not cancel the event, so it is possible to\n * attach a custom callback after a default callback and also use `data.viewPosition` calculated by the default callback\n * (for example to fix it).\n *\n * **Note:** the default mapping callback will not fire if `data.viewPosition` is already set.\n *\n * **Note:** these callbacks are called **very often**. For efficiency reasons, it is advised to use them only when position\n * mapping between the given model and view elements is unsolvable by using just elements mapping and default algorithm.\n * Also, the condition that checks if a special case scenario happened should be as simple as possible.\n *\n * @eventName ~Mapper#modelToViewPosition\n */\nexport type MapperModelToViewPositionEvent = {\n\tname: 'modelToViewPosition';\n\targs: [ MapperModelToViewPositionEventData ];\n};\n\n/**\n * Data pipeline object that can store and pass data between callbacks. The callback should add\n * the `viewPosition` value to that object with calculated the {@link module:engine/view/position~Position view position}.\n */\nexport type MapperModelToViewPositionEventData = {\n\n\t/**\n\t * Mapper instance that fired the event.\n\t */\n\tmapper: Mapper;\n\n\t/**\n\t * The model position.\n\t */\n\tmodelPosition: ModelPosition;\n\n\t/**\n\t * The callback should add the `viewPosition` value to that object with calculated the\n\t * {@link module:engine/view/position~Position view position}.\n\t */\n\tviewPosition?: ViewPosition;\n\n\t/**\n\t * Should be set to `true` if the model position to map is pointing to a place\n\t * in model tree which no longer exists. For example, it could be an end of a removed model range.\n\t */\n\tisPhantom?: boolean;\n};\n\n/**\n * Fired for each view-to-model position mapping request. See {@link module:engine/conversion/mapper~Mapper#event:modelToViewPosition}.\n *\n * ```ts\n * // See example in `modelToViewPosition` event description.\n * // This custom mapping will map positions from <span> element next to <img> to the \"captionedImage\" element.\n * mapper.on( 'viewToModelPosition', ( evt, data ) => {\n * \tconst positionParent = viewPosition.parent;\n *\n * \tif ( positionParent.hasClass( 'image-caption' ) ) {\n * \t\tconst viewImg = positionParent.previousSibling;\n * \t\tconst modelImg = data.mapper.toModelElement( viewImg );\n *\n * \t\tdata.modelPosition = new ModelPosition( modelImg, viewPosition.offset );\n * \t\tevt.stop();\n * \t}\n * } );\n * ```\n *\n * **Note:** the default mapping callback is provided with a `low` priority setting and does not cancel the event, so it is possible to\n * attach a custom callback after the default callback and also use `data.modelPosition` calculated by the default callback\n * (for example to fix it).\n *\n * **Note:** the default mapping callback will not fire if `data.modelPosition` is already set.\n *\n * **Note:** these callbacks are called **very often**. For efficiency reasons, it is advised to use them only when position\n * mapping between the given model and view elements is unsolvable by using just elements mapping and default algorithm.\n * Also, the condition that checks if special case scenario happened should be as simple as possible.\n *\n * @eventName ~Mapper#viewToModelPosition\n */\nexport type MapperViewToModelPositionEvent = {\n\tname: 'viewToModelPosition';\n\targs: [ MapperViewToModelPositionEventData ];\n};\n\n/**\n * Data pipeline object that can store and pass data between callbacks. The callback should add\n * `modelPosition` value to that object with calculated {@link module:engine/model/position~Position model position}.\n */\nexport type MapperViewToModelPositionEventData = {\n\n\t/**\n\t * Mapper instance that fired the event.\n\t */\n\tmapper: Mapper;\n\n\t/**\n\t * The callback should add `modelPosition` value to that object with calculated\n\t * {@link module:engine/model/position~Position model position}.\n\t */\n\tmodelPosition?: ModelPosition;\n\n\t/**\n\t * The view position.\n\t */\n\tviewPosition: ViewPosition;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/conversion/modelconsumable\n */\n\nimport TextProxy from '../model/textproxy';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type Item from '../model/item';\nimport type Selection from '../model/selection';\nimport type DocumentSelection from '../model/documentselection';\nimport type Range from '../model/range';\n\n/**\n * Manages a list of consumable values for the {@link module:engine/model/item~Item model items}.\n *\n * Consumables are various aspects of the model. A model item can be broken down into separate, single properties that might be\n * taken into consideration when converting that item.\n *\n * `ModelConsumable` is used by {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher} while analyzing the changed\n * parts of {@link module:engine/model/document~Document the document}. The added / changed / removed model items are broken down\n * into singular properties (the item itself and its attributes). All those parts are saved in `ModelConsumable`. Then,\n * during conversion, when the given part of a model item is converted (i.e. the view element has been inserted into the view,\n * but without attributes), the consumable value is removed from `ModelConsumable`.\n *\n * For model items, `ModelConsumable` stores consumable values of one of following types: `insert`, `addattribute:<attributeKey>`,\n * `changeattributes:<attributeKey>`, `removeattributes:<attributeKey>`.\n *\n * In most cases, it is enough to let th {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher}\n * gather consumable values, so there is no need to use\n * the {@link module:engine/conversion/modelconsumable~ModelConsumable#add add method} directly.\n * However, it is important to understand how consumable values can be\n * {@link module:engine/conversion/modelconsumable~ModelConsumable#consume consumed}.\n * See {@link module:engine/conversion/downcasthelpers default downcast converters} for more information.\n *\n * Keep in mind that one conversion event may have multiple callbacks (converters) attached to it. Each of those is\n * able to convert one or more parts of the model. However, when one of those callbacks actually converts\n * something, the others should not, because they would duplicate the results. Using `ModelConsumable` helps to avoid\n * this situation, because callbacks should only convert these values that were not yet consumed from `ModelConsumable`.\n *\n * Consuming multiple values in a single callback:\n *\n * ```ts\n * // Converter for custom `imageBlock` element that might have a `caption` element inside which changes\n * // how the image is displayed in the view:\n * //\n * // Model:\n * //\n * // [imageBlock]\n * // └─ [caption]\n * // └─ foo\n * //\n * // View:\n * //\n * // <figure>\n * // ├─ <img />\n * // └─ <caption>\n * // └─ foo\n * modelConversionDispatcher.on( 'insert:imageBlock', ( evt, data, conversionApi ) => {\n * \t// First, consume the `imageBlock` element.\n * \tconversionApi.consumable.consume( data.item, 'insert' );\n *\n * \t// Just create normal image element for the view.\n * \t// Maybe it will be \"decorated\" later.\n * \tconst viewImage = new ViewElement( 'img' );\n * \tconst insertPosition = conversionApi.mapper.toViewPosition( data.range.start );\n * \tconst viewWriter = conversionApi.writer;\n *\n * \t// Check if the `imageBlock` element has children.\n * \tif ( data.item.childCount > 0 ) {\n * \t\tconst modelCaption = data.item.getChild( 0 );\n *\n * \t\t// `modelCaption` insertion change is consumed from consumable values.\n * \t\t// It will not be converted by other converters, but it's children (probably some text) will be.\n * \t\t// Through mapping, converters for text will know where to insert contents of `modelCaption`.\n * \t\tif ( conversionApi.consumable.consume( modelCaption, 'insert' ) ) {\n * \t\t\tconst viewCaption = new ViewElement( 'figcaption' );\n *\n * \t\t\tconst viewImageHolder = new ViewElement( 'figure', null, [ viewImage, viewCaption ] );\n *\n * \t\t\tconversionApi.mapper.bindElements( modelCaption, viewCaption );\n * \t\t\tconversionApi.mapper.bindElements( data.item, viewImageHolder );\n * \t\t\tviewWriter.insert( insertPosition, viewImageHolder );\n * \t\t}\n * \t} else {\n * \t\tconversionApi.mapper.bindElements( data.item, viewImage );\n * \t\tviewWriter.insert( insertPosition, viewImage );\n * \t}\n *\n * \tevt.stop();\n * } );\n * ```\n */\nexport default class ModelConsumable {\n\t/**\n\t * Contains list of consumable values.\n\t */\n\tprivate _consumable = new Map<any, Map<string, boolean>>();\n\n\t/**\n\t * For each {@link module:engine/model/textproxy~TextProxy} added to `ModelConsumable`, this registry holds a parent\n\t * of that `TextProxy` and the start and end indices of that `TextProxy`. This allows identification of the `TextProxy`\n\t * instances that point to the same part of the model but are different instances. Each distinct `TextProxy`\n\t * is given a unique `Symbol` which is then registered as consumable. This process is transparent for the `ModelConsumable`\n\t * API user because whenever `TextProxy` is added, tested, consumed or reverted, the internal mechanisms of\n\t * `ModelConsumable` translate `TextProxy` to that unique `Symbol`.\n\t */\n\tprivate _textProxyRegistry = new Map<number | null, Map<number | null, Map<unknown, symbol>>>();\n\n\t/**\n\t * Adds a consumable value to the consumables list and links it with a given model item.\n\t *\n\t * ```ts\n\t * modelConsumable.add( modelElement, 'insert' ); // Add `modelElement` insertion change to consumable values.\n\t * modelConsumable.add( modelElement, 'addAttribute:bold' ); // Add `bold` attribute insertion on `modelElement` change.\n\t * modelConsumable.add( modelElement, 'removeAttribute:bold' ); // Add `bold` attribute removal on `modelElement` change.\n\t * modelConsumable.add( modelSelection, 'selection' ); // Add `modelSelection` to consumable values.\n\t * modelConsumable.add( modelRange, 'range' ); // Add `modelRange` to consumable values.\n\t * ```\n\t *\n\t * @param item Model item, range or selection that has the consumable.\n\t * @param type Consumable type. Will be normalized to a proper form, that is either `<word>` or `<part>:<part>`.\n\t * Second colon and everything after will be cut. Passing event name is a safe and good practice.\n\t */\n\tpublic add(\n\t\titem: Item | Selection | DocumentSelection | Range,\n\t\ttype: string\n\t): void {\n\t\ttype = _normalizeConsumableType( type );\n\n\t\tif ( item instanceof TextProxy ) {\n\t\t\titem = this._getSymbolForTextProxy( item ) as any;\n\t\t}\n\n\t\tif ( !this._consumable.has( item ) ) {\n\t\t\tthis._consumable.set( item, new Map() );\n\t\t}\n\n\t\tthis._consumable.get( item )!.set( type, true );\n\t}\n\n\t/**\n\t * Removes a given consumable value from a given model item.\n\t *\n\t * ```ts\n\t * modelConsumable.consume( modelElement, 'insert' ); // Remove `modelElement` insertion change from consumable values.\n\t * modelConsumable.consume( modelElement, 'addAttribute:bold' ); // Remove `bold` attribute insertion on `modelElement` change.\n\t * modelConsumable.consume( modelElement, 'removeAttribute:bold' ); // Remove `bold` attribute removal on `modelElement` change.\n\t * modelConsumable.consume( modelSelection, 'selection' ); // Remove `modelSelection` from consumable values.\n\t * modelConsumable.consume( modelRange, 'range' ); // Remove 'modelRange' from consumable values.\n\t * ```\n\t *\n\t * @param item Model item, range or selection from which consumable will be consumed.\n\t * @param type Consumable type. Will be normalized to a proper form, that is either `<word>` or `<part>:<part>`.\n\t * Second colon and everything after will be cut. Passing event name is a safe and good practice.\n\t * @returns `true` if consumable value was available and was consumed, `false` otherwise.\n\t */\n\tpublic consume(\n\t\titem: Item | Selection | DocumentSelection | Range,\n\t\ttype: string\n\t): boolean {\n\t\ttype = _normalizeConsumableType( type );\n\n\t\tif ( item instanceof TextProxy ) {\n\t\t\titem = this._getSymbolForTextProxy( item ) as any;\n\t\t}\n\n\t\tif ( this.test( item, type ) ) {\n\t\t\tthis._consumable.get( item )!.set( type, false );\n\n\t\t\treturn true;\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n\n\t/**\n\t * Tests whether there is a consumable value of a given type connected with a given model item.\n\t *\n\t * ```ts\n\t * modelConsumable.test( modelElement, 'insert' ); // Check for `modelElement` insertion change.\n\t * modelConsumable.test( modelElement, 'addAttribute:bold' ); // Check for `bold` attribute insertion on `modelElement` change.\n\t * modelConsumable.test( modelElement, 'removeAttribute:bold' ); // Check for `bold` attribute removal on `modelElement` change.\n\t * modelConsumable.test( modelSelection, 'selection' ); // Check if `modelSelection` is consumable.\n\t * modelConsumable.test( modelRange, 'range' ); // Check if `modelRange` is consumable.\n\t * ```\n\t *\n\t * @param item Model item, range or selection to be tested.\n\t * @param type Consumable type. Will be normalized to a proper form, that is either `<word>` or `<part>:<part>`.\n\t * Second colon and everything after will be cut. Passing event name is a safe and good practice.\n\t * @returns `null` if such consumable was never added, `false` if the consumable values was\n\t * already consumed or `true` if it was added and not consumed yet.\n\t */\n\tpublic test(\n\t\titem: Item | Selection | DocumentSelection | Range,\n\t\ttype: string\n\t): boolean | null {\n\t\ttype = _normalizeConsumableType( type );\n\n\t\tif ( item instanceof TextProxy ) {\n\t\t\titem = this._getSymbolForTextProxy( item ) as any;\n\t\t}\n\n\t\tconst itemConsumables = this._consumable.get( item );\n\n\t\tif ( itemConsumables === undefined ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst value = itemConsumables.get( type );\n\n\t\tif ( value === undefined ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn value;\n\t}\n\n\t/**\n\t * Reverts consuming of a consumable value.\n\t *\n\t * ```ts\n\t * modelConsumable.revert( modelElement, 'insert' ); // Revert consuming `modelElement` insertion change.\n\t * modelConsumable.revert( modelElement, 'addAttribute:bold' ); // Revert consuming `bold` attribute insert from `modelElement`.\n\t * modelConsumable.revert( modelElement, 'removeAttribute:bold' ); // Revert consuming `bold` attribute remove from `modelElement`.\n\t * modelConsumable.revert( modelSelection, 'selection' ); // Revert consuming `modelSelection`.\n\t * modelConsumable.revert( modelRange, 'range' ); // Revert consuming `modelRange`.\n\t * ```\n\t *\n\t * @param item Model item, range or selection to be reverted.\n\t * @param type Consumable type.\n\t * @returns `true` if consumable has been reversed, `false` otherwise. `null` if the consumable has\n\t * never been added.\n\t */\n\tpublic revert(\n\t\titem: Item | Selection | DocumentSelection | Range,\n\t\ttype: string\n\t): boolean | null {\n\t\ttype = _normalizeConsumableType( type );\n\n\t\tif ( item instanceof TextProxy ) {\n\t\t\titem = this._getSymbolForTextProxy( item ) as any;\n\t\t}\n\n\t\tconst test = this.test( item, type );\n\n\t\tif ( test === false ) {\n\t\t\tthis._consumable.get( item )!.set( type, true );\n\n\t\t\treturn true;\n\t\t} else if ( test === true ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Verifies if all events from the specified group were consumed.\n\t *\n\t * @param eventGroup The events group to verify.\n\t */\n\tpublic verifyAllConsumed( eventGroup: string ): void {\n\t\tconst items = [];\n\n\t\tfor ( const [ item, consumables ] of this._consumable ) {\n\t\t\tfor ( const [ event, canConsume ] of consumables ) {\n\t\t\t\tconst eventPrefix = event.split( ':' )[ 0 ];\n\n\t\t\t\tif ( canConsume && eventGroup == eventPrefix ) {\n\t\t\t\t\titems.push( {\n\t\t\t\t\t\tevent,\n\t\t\t\t\t\titem: item.name || item.description\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif ( items.length ) {\n\t\t\t/**\n\t\t\t * Some of the {@link module:engine/model/item~Item model items} were not consumed while downcasting the model to view.\n\t\t\t *\n\t\t\t * This might be the effect of:\n\t\t\t *\n\t\t\t * * A missing converter for some model elements. Make sure that you registered downcast converters for all model elements.\n\t\t\t * * A custom converter that does not consume converted items. Make sure that you\n\t\t\t * {@link module:engine/conversion/modelconsumable~ModelConsumable#consume consumed} all model elements that you converted\n\t\t\t * from the model to the view.\n\t\t\t * * A custom converter that called `event.stop()`. When providing a custom converter, keep in mind that you should not stop\n\t\t\t * the event. If you stop it then the default converter at the `lowest` priority will not trigger the conversion of this node's\n\t\t\t * attributes and child nodes.\n\t\t\t *\n\t\t\t * @error conversion-model-consumable-not-consumed\n\t\t\t * @param {Array.<module:engine/model/item~Item>} items Items that were not consumed.\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'conversion-model-consumable-not-consumed', null, { items } );\n\t\t}\n\t}\n\n\t/**\n\t * Gets a unique symbol for the passed {@link module:engine/model/textproxy~TextProxy} instance. All `TextProxy` instances that\n\t * have same parent, same start index and same end index will get the same symbol.\n\t *\n\t * Used internally to correctly consume `TextProxy` instances.\n\t *\n\t * @internal\n\t * @param textProxy `TextProxy` instance to get a symbol for.\n\t * @returns Symbol representing all equal instances of `TextProxy`.\n\t */\n\tpublic _getSymbolForTextProxy( textProxy: TextProxy ): symbol {\n\t\tlet symbol = null;\n\n\t\tconst startMap = this._textProxyRegistry.get( textProxy.startOffset );\n\n\t\tif ( startMap ) {\n\t\t\tconst endMap = startMap.get( textProxy.endOffset );\n\n\t\t\tif ( endMap ) {\n\t\t\t\tsymbol = endMap.get( textProxy.parent );\n\t\t\t}\n\t\t}\n\n\t\tif ( !symbol ) {\n\t\t\tsymbol = this._addSymbolForTextProxy( textProxy );\n\t\t}\n\n\t\treturn symbol;\n\t}\n\n\t/**\n\t * Adds a symbol for the given {@link module:engine/model/textproxy~TextProxy} instance.\n\t *\n\t * Used internally to correctly consume `TextProxy` instances.\n\t *\n\t * @param textProxy Text proxy instance.\n\t * @returns Symbol generated for given `TextProxy`.\n\t */\n\tprivate _addSymbolForTextProxy( textProxy: TextProxy ): symbol {\n\t\tconst start = textProxy.startOffset;\n\t\tconst end = textProxy.endOffset;\n\t\tconst parent = textProxy.parent;\n\n\t\tconst symbol = Symbol( '$textProxy:' + textProxy.data );\n\t\tlet startMap: Map<number | null, Map<unknown, symbol>> | undefined;\n\t\tlet endMap: Map<unknown, symbol> | undefined;\n\n\t\tstartMap = this._textProxyRegistry.get( start );\n\n\t\tif ( !startMap ) {\n\t\t\tstartMap = new Map();\n\t\t\tthis._textProxyRegistry.set( start, startMap );\n\t\t}\n\n\t\tendMap = startMap.get( end );\n\n\t\tif ( !endMap ) {\n\t\t\tendMap = new Map();\n\t\t\tstartMap.set( end, endMap );\n\t\t}\n\n\t\tendMap.set( parent, symbol );\n\n\t\treturn symbol;\n\t}\n}\n\n/**\n * Returns a normalized consumable type name from the given string. A normalized consumable type name is a string that has\n * at most one colon, for example: `insert` or `addMarker:highlight`. If a string to normalize has more \"parts\" (more colons),\n * the further parts are dropped, for example: `addattribute:bold:$text` -> `addattributes:bold`.\n *\n * @param type Consumable type.\n * @returns Normalized consumable type.\n */\nfunction _normalizeConsumableType( type: string ) {\n\tconst parts = type.split( ':' );\n\n\t// For inserts allow passing event name, it's stored in the context of a specified element so the element name is not needed.\n\tif ( parts[ 0 ] == 'insert' ) {\n\t\treturn parts[ 0 ];\n\t}\n\n\t// Markers are identified by the whole name (otherwise we would consume the whole markers group).\n\tif ( parts[ 0 ] == 'addMarker' || parts[ 0 ] == 'removeMarker' ) {\n\t\treturn type;\n\t}\n\n\treturn parts.length > 1 ? parts[ 0 ] + ':' + parts[ 1 ] : parts[ 0 ];\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/conversion/downcastdispatcher\n */\n\nimport Consumable from './modelconsumable';\nimport Range from '../model/range';\n\nimport { EmitterMixin } from '@ckeditor/ckeditor5-utils';\n\nimport type { default as Differ, DiffItem } from '../model/differ';\nimport type { default as MarkerCollection, Marker } from '../model/markercollection';\nimport type { TreeWalkerValue } from '../model/treewalker';\nimport type DocumentSelection from '../model/documentselection';\nimport type DowncastWriter from '../view/downcastwriter';\nimport type RootElement from '../model/rootelement';\nimport type Element from '../model/element';\nimport type Item from '../model/item';\nimport type Mapper from './mapper';\nimport type Position from '../model/position';\nimport type Schema from '../model/schema';\nimport type Selection from '../model/selection';\nimport type ViewElement from '../view/element';\n\n/**\n * The downcast dispatcher is a central point of downcasting (conversion from the model to the view), which is a process of reacting\n * to changes in the model and firing a set of events. The callbacks listening to these events are called converters. The\n * converters' role is to convert the model changes to changes in view (for example, adding view nodes or\n * changing attributes on view elements).\n *\n * During the conversion process, downcast dispatcher fires events basing on the state of the model and prepares\n * data for these events. It is important to understand that the events are connected with the changes done on the model,\n * for example: \"a node has been inserted\" or \"an attribute has changed\". This is in contrary to upcasting (a view-to-model conversion)\n * where you convert the view state (view nodes) to a model tree.\n *\n * The events are prepared basing on a diff created by the {@link module:engine/model/differ~Differ Differ}, which buffers them\n * and then passes to the downcast dispatcher as a diff between the old model state and the new model state.\n *\n * Note that because the changes are converted, there is a need to have a mapping between the model structure and the view structure.\n * To map positions and elements during the downcast (a model-to-view conversion), use {@link module:engine/conversion/mapper~Mapper}.\n *\n * Downcast dispatcher fires the following events for model tree changes:\n *\n * * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:insert `insert`} &ndash;\n * If a range of nodes was inserted to the model tree.\n * * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:remove `remove`} &ndash;\n * If a range of nodes was removed from the model tree.\n * * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute `attribute`} &ndash;\n * If an attribute was added, changed or removed from a model node.\n *\n * For {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:insert `insert`}\n * and {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute `attribute`},\n * the downcast dispatcher generates {@link module:engine/conversion/modelconsumable~ModelConsumable consumables}.\n * These are used to have control over which changes have already been consumed. It is useful when some converters\n * overwrite others or convert multiple changes (for example, it converts an insertion of an element and also converts that\n * element's attributes during the insertion).\n *\n * Additionally, downcast dispatcher fires events for {@link module:engine/model/markercollection~Marker marker} changes:\n *\n * * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:addMarker `addMarker`} &ndash; If a marker was added.\n * * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:removeMarker `removeMarker`} &ndash; If a marker was\n * removed.\n *\n * Note that changing a marker is done through removing the marker from the old range and adding it to the new range,\n * so both of these events are fired.\n *\n * Finally, a downcast dispatcher also handles firing events for the {@link module:engine/model/selection model selection}\n * conversion:\n *\n * * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:selection `selection`}\n * &ndash; Converts the selection from the model to the view.\n * * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute `attribute`}\n * &ndash; Fired for every selection attribute.\n * * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:addMarker `addMarker`}\n * &ndash; Fired for every marker that contains a selection.\n *\n * Unlike the model tree and the markers, the events for selection are not fired for changes but for a selection state.\n *\n * When providing custom listeners for a downcast dispatcher, remember to check whether a given change has not been\n * {@link module:engine/conversion/modelconsumable~ModelConsumable#consume consumed} yet.\n *\n * When providing custom listeners for a downcast dispatcher, keep in mind that you **should not** stop the event. If you stop it,\n * then the default converter at the `lowest` priority will not trigger the conversion of this node's attributes and child nodes.\n *\n * When providing custom listeners for a downcast dispatcher, remember to use the provided\n * {@link module:engine/view/downcastwriter~DowncastWriter view downcast writer} to apply changes to the view document.\n *\n * You can read more about conversion in the following guide:\n *\n * * {@glink framework/deep-dive/conversion/downcast Downcast conversion}\n *\n * An example of a custom converter for the downcast dispatcher:\n *\n * ```ts\n * // You will convert inserting a \"paragraph\" model element into the model.\n * downcastDispatcher.on( 'insert:paragraph', ( evt, data, conversionApi ) => {\n * \t// Remember to check whether the change has not been consumed yet and consume it.\n * \tif ( !conversionApi.consumable.consume( data.item, 'insert' ) ) {\n * \t\treturn;\n * \t}\n *\n * \t// Translate the position in the model to a position in the view.\n * \tconst viewPosition = conversionApi.mapper.toViewPosition( data.range.start );\n *\n * \t// Create a <p> element that will be inserted into the view at the `viewPosition`.\n * \tconst viewElement = conversionApi.writer.createContainerElement( 'p' );\n *\n * \t// Bind the newly created view element to the model element so positions will map accordingly in the future.\n * \tconversionApi.mapper.bindElements( data.item, viewElement );\n *\n * \t// Add the newly created view element to the view.\n * \tconversionApi.writer.insert( viewPosition, viewElement );\n * } );\n * ```\n */\nexport default class DowncastDispatcher extends EmitterMixin() {\n\t/**\n\t * A template for an interface passed by the dispatcher to the event callbacks.\n\t *\n\t * @internal\n\t */\n\tpublic readonly _conversionApi: Pick<DowncastConversionApi, 'dispatcher' | 'mapper' | 'schema'>;\n\n\t/**\n\t * A map of already fired events for a given `ModelConsumable`.\n\t */\n\tprivate readonly _firedEventsMap: WeakMap<DowncastConversionApi, Map<unknown, Set<string>>>;\n\n\t/**\n\t * Creates a downcast dispatcher instance.\n\t *\n\t * @see module:engine/conversion/downcastdispatcher~DowncastConversionApi\n\t *\n\t * @param conversionApi Additional properties for an interface that will be passed to events fired\n\t * by the downcast dispatcher.\n\t */\n\tconstructor( conversionApi: Pick<DowncastConversionApi, 'mapper' | 'schema'> ) {\n\t\tsuper();\n\n\t\tthis._conversionApi = { dispatcher: this, ...conversionApi };\n\t\tthis._firedEventsMap = new WeakMap();\n\t}\n\n\t/**\n\t * Converts changes buffered in the given {@link module:engine/model/differ~Differ model differ}\n\t * and fires conversion events based on it.\n\t *\n\t * @fires insert\n\t * @fires remove\n\t * @fires attribute\n\t * @fires addMarker\n\t * @fires removeMarker\n\t * @fires reduceChanges\n\t * @param differ The differ object with buffered changes.\n\t * @param markers Markers related to the model fragment to convert.\n\t * @param writer The view writer that should be used to modify the view document.\n\t */\n\tpublic convertChanges(\n\t\tdiffer: Differ,\n\t\tmarkers: MarkerCollection,\n\t\twriter: DowncastWriter\n\t): void {\n\t\tconst conversionApi = this._createConversionApi( writer, differ.getRefreshedItems() );\n\n\t\t// Before the view is updated, remove markers which have changed.\n\t\tfor ( const change of differ.getMarkersToRemove() ) {\n\t\t\tthis._convertMarkerRemove( change.name, change.range, conversionApi );\n\t\t}\n\n\t\t// Let features modify the change list (for example to allow reconversion).\n\t\tconst changes = this._reduceChanges( differ.getChanges() );\n\n\t\t// Convert changes that happened on model tree.\n\t\tfor ( const entry of changes ) {\n\t\t\tif ( entry.type === 'insert' ) {\n\t\t\t\tthis._convertInsert( Range._createFromPositionAndShift( entry.position, entry.length ), conversionApi );\n\t\t\t} else if ( entry.type === 'reinsert' ) {\n\t\t\t\tthis._convertReinsert( Range._createFromPositionAndShift( entry.position, entry.length ), conversionApi );\n\t\t\t} else if ( entry.type === 'remove' ) {\n\t\t\t\tthis._convertRemove( entry.position, entry.length, entry.name, conversionApi );\n\t\t\t} else {\n\t\t\t\t// Defaults to 'attribute' change.\n\t\t\t\tthis._convertAttribute( entry.range, entry.attributeKey, entry.attributeOldValue, entry.attributeNewValue, conversionApi );\n\t\t\t}\n\t\t}\n\n\t\tfor ( const markerName of conversionApi.mapper.flushUnboundMarkerNames() ) {\n\t\t\tconst markerRange = markers.get( markerName )!.getRange();\n\n\t\t\tthis._convertMarkerRemove( markerName, markerRange, conversionApi );\n\t\t\tthis._convertMarkerAdd( markerName, markerRange, conversionApi );\n\t\t}\n\n\t\t// After the view is updated, convert markers which have changed.\n\t\tfor ( const change of differ.getMarkersToAdd() ) {\n\t\t\tthis._convertMarkerAdd( change.name, change.range, conversionApi );\n\t\t}\n\n\t\t// Remove mappings for all removed view elements.\n\t\tconversionApi.mapper.flushDeferredBindings();\n\n\t\t// Verify if all insert consumables were consumed.\n\t\tconversionApi.consumable.verifyAllConsumed( 'insert' );\n\t}\n\n\t/**\n\t * Starts a conversion of a model range and the provided markers.\n\t *\n\t * @fires insert\n\t * @fires attribute\n\t * @fires addMarker\n\t * @param range The inserted range.\n\t * @param markers The map of markers that should be down-casted.\n\t * @param writer The view writer that should be used to modify the view document.\n\t * @param options Optional options object passed to `convertionApi.options`.\n\t */\n\tpublic convert(\n\t\trange: Range,\n\t\tmarkers: Map<string, Range>,\n\t\twriter: DowncastWriter,\n\t\toptions: DowncastConversionApi[ 'options' ] = {}\n\t): void {\n\t\tconst conversionApi = this._createConversionApi( writer, undefined, options );\n\n\t\tthis._convertInsert( range, conversionApi );\n\n\t\tfor ( const [ name, range ] of markers ) {\n\t\t\tthis._convertMarkerAdd( name, range, conversionApi );\n\t\t}\n\n\t\t// Verify if all insert consumables were consumed.\n\t\tconversionApi.consumable.verifyAllConsumed( 'insert' );\n\t}\n\n\t/**\n\t * Starts the model selection conversion.\n\t *\n\t * Fires events for a given {@link module:engine/model/selection~Selection selection} to start the selection conversion.\n\t *\n\t * @fires selection\n\t * @fires addMarker\n\t * @fires attribute\n\t * @param selection The selection to convert.\n\t * @param markers Markers connected with the converted model.\n\t * @param writer View writer that should be used to modify the view document.\n\t */\n\tpublic convertSelection(\n\t\tselection: Selection | DocumentSelection,\n\t\tmarkers: MarkerCollection,\n\t\twriter: DowncastWriter\n\t): void {\n\t\tconst conversionApi = this._createConversionApi( writer );\n\n\t\t// First perform a clean-up at the current position of the selection.\n\t\tthis.fire<DowncastCleanSelectionEvent>( 'cleanSelection', { selection }, conversionApi );\n\n\t\t// Don't convert selection if it is in a model root that does not have a view root (for now this is only the graveyard root).\n\t\tconst modelRoot = selection.getFirstPosition()!.root as RootElement;\n\n\t\tif ( !conversionApi.mapper.toViewElement( modelRoot ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Now, perform actual selection conversion.\n\t\tconst markersAtSelection = Array.from( markers.getMarkersAtPosition( selection.getFirstPosition()! ) );\n\n\t\tthis._addConsumablesForSelection( conversionApi.consumable, selection, markersAtSelection );\n\n\t\tthis.fire<DowncastSelectionEvent>( 'selection', { selection }, conversionApi );\n\n\t\tif ( !selection.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor ( const marker of markersAtSelection ) {\n\t\t\t// Do not fire event if the marker has been consumed.\n\t\t\tif ( conversionApi.consumable.test( selection, 'addMarker:' + marker.name ) ) {\n\t\t\t\tconst markerRange = marker.getRange();\n\n\t\t\t\tif ( !shouldMarkerChangeBeConverted( selection.getFirstPosition()!, marker, conversionApi.mapper ) ) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\tconst data = {\n\t\t\t\t\titem: selection,\n\t\t\t\t\tmarkerName: marker.name,\n\t\t\t\t\tmarkerRange\n\t\t\t\t};\n\n\t\t\t\tthis.fire<DowncastAddMarkerEvent>( `addMarker:${ marker.name }`, data, conversionApi );\n\t\t\t}\n\t\t}\n\n\t\tfor ( const key of selection.getAttributeKeys() ) {\n\t\t\t// Do not fire event if the attribute has been consumed.\n\t\t\tif ( conversionApi.consumable.test( selection, 'attribute:' + key ) ) {\n\t\t\t\tconst data = {\n\t\t\t\t\titem: selection,\n\t\t\t\t\trange: selection.getFirstRange()!,\n\t\t\t\t\tattributeKey: key,\n\t\t\t\t\tattributeOldValue: null,\n\t\t\t\t\tattributeNewValue: selection.getAttribute( key )\n\t\t\t\t};\n\n\t\t\t\tthis.fire<DowncastAttributeEvent>( `attribute:${ key }:$text`, data, conversionApi );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Fires insertion conversion of a range of nodes.\n\t *\n\t * For each node in the range, {@link #event:insert `insert` event is fired}. For each attribute on each node,\n\t * {@link #event:attribute `attribute` event is fired}.\n\t *\n\t * @fires insert\n\t * @fires attribute\n\t * @param range The inserted range.\n\t * @param conversionApi The conversion API object.\n\t * @param options.doNotAddConsumables Whether the ModelConsumable should not get populated\n\t * for items in the provided range.\n\t */\n\tprivate _convertInsert(\n\t\trange: Range,\n\t\tconversionApi: DowncastConversionApi,\n\t\toptions: { doNotAddConsumables?: boolean } = {}\n\t): void {\n\t\tif ( !options.doNotAddConsumables ) {\n\t\t\t// Collect a list of things that can be consumed, consisting of nodes and their attributes.\n\t\t\tthis._addConsumablesForInsert( conversionApi.consumable, Array.from( range ) );\n\t\t}\n\n\t\t// Fire a separate insert event for each node and text fragment contained in the range.\n\t\tfor ( const data of Array.from( range.getWalker( { shallow: true } ) ).map( walkerValueToEventData ) ) {\n\t\t\tthis._testAndFire( 'insert', data, conversionApi );\n\t\t}\n\t}\n\n\t/**\n\t * Fires conversion of a single node removal. Fires {@link #event:remove remove event} with provided data.\n\t *\n\t * @param position Position from which node was removed.\n\t * @param length Offset size of removed node.\n\t * @param name Name of removed node.\n\t * @param conversionApi The conversion API object.\n\t */\n\tprivate _convertRemove(\n\t\tposition: Position,\n\t\tlength: number,\n\t\tname: string,\n\t\tconversionApi: DowncastConversionApi\n\t): void {\n\t\tthis.fire<DowncastRemoveEvent>( `remove:${ name }`, { position, length }, conversionApi );\n\t}\n\n\t/**\n\t * Starts a conversion of an attribute change on a given `range`.\n\t *\n\t * For each node in the given `range`, {@link #event:attribute attribute event} is fired with the passed data.\n\t *\n\t * @fires attribute\n\t * @param range Changed range.\n\t * @param key Key of the attribute that has changed.\n\t * @param oldValue Attribute value before the change or `null` if the attribute has not been set before.\n\t * @param newValue New attribute value or `null` if the attribute has been removed.\n\t * @param conversionApi The conversion API object.\n\t */\n\tprivate _convertAttribute(\n\t\trange: Range,\n\t\tkey: string,\n\t\toldValue: unknown,\n\t\tnewValue: unknown,\n\t\tconversionApi: DowncastConversionApi\n\t): void {\n\t\t// Create a list with attributes to consume.\n\t\tthis._addConsumablesForRange( conversionApi.consumable, range, `attribute:${ key }` );\n\n\t\t// Create a separate attribute event for each node in the range.\n\t\tfor ( const value of range ) {\n\t\t\tconst data = {\n\t\t\t\titem: value.item,\n\t\t\t\trange: Range._createFromPositionAndShift( value.previousPosition, value.length! ),\n\t\t\t\tattributeKey: key,\n\t\t\t\tattributeOldValue: oldValue,\n\t\t\t\tattributeNewValue: newValue\n\t\t\t};\n\n\t\t\tthis._testAndFire( `attribute:${ key }`, data, conversionApi );\n\t\t}\n\t}\n\n\t/**\n\t * Fires re-insertion conversion (with a `reconversion` flag passed to `insert` events)\n\t * of a range of elements (only elements on the range depth, without children).\n\t *\n\t * For each node in the range on its depth (without children), {@link #event:insert `insert` event} is fired.\n\t * For each attribute on each node, {@link #event:attribute `attribute` event} is fired.\n\t *\n\t * @fires insert\n\t * @fires attribute\n\t * @param range The range to reinsert.\n\t * @param conversionApi The conversion API object.\n\t */\n\tprivate _convertReinsert( range: Range, conversionApi: DowncastConversionApi ): void {\n\t\t// Convert the elements - without converting children.\n\t\tconst walkerValues = Array.from( range.getWalker( { shallow: true } ) );\n\n\t\t// Collect a list of things that can be consumed, consisting of nodes and their attributes.\n\t\tthis._addConsumablesForInsert( conversionApi.consumable, walkerValues );\n\n\t\t// Fire a separate insert event for each node and text fragment contained shallowly in the range.\n\t\tfor ( const data of walkerValues.map( walkerValueToEventData ) ) {\n\t\t\tthis._testAndFire( 'insert', { ...data, reconversion: true }, conversionApi );\n\t\t}\n\t}\n\n\t/**\n\t * Converts the added marker. Fires the {@link #event:addMarker `addMarker`} event for each item\n\t * in the marker's range. If the range is collapsed, a single event is dispatched. See the event description for more details.\n\t *\n\t * @fires addMarker\n\t * @param markerName Marker name.\n\t * @param markerRange The marker range.\n\t * @param conversionApi The conversion API object.\n\t */\n\tprivate _convertMarkerAdd(\n\t\tmarkerName: string,\n\t\tmarkerRange: Range,\n\t\tconversionApi: DowncastConversionApi\n\t): void {\n\t\t// Do not convert if range is in graveyard.\n\t\tif ( markerRange.root.rootName == '$graveyard' ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// In markers' case, event name == consumable name.\n\t\tconst eventName = `addMarker:${ markerName }` as const;\n\n\t\t//\n\t\t// First, fire an event for the whole marker.\n\t\t//\n\t\tconversionApi.consumable.add( markerRange, eventName );\n\n\t\tthis.fire<DowncastAddMarkerEvent>( eventName, { markerName, markerRange }, conversionApi );\n\n\t\t//\n\t\t// Do not fire events for each item inside the range if the range got consumed.\n\t\t// Also consume the whole marker consumable if it wasn't consumed.\n\t\t//\n\t\tif ( !conversionApi.consumable.consume( markerRange, eventName ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t//\n\t\t// Then, fire an event for each item inside the marker range.\n\t\t//\n\t\tthis._addConsumablesForRange( conversionApi.consumable, markerRange, eventName );\n\n\t\tfor ( const item of markerRange.getItems() ) {\n\t\t\t// Do not fire event for already consumed items.\n\t\t\tif ( !conversionApi.consumable.test( item, eventName ) ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst data = { item, range: Range._createOn( item ), markerName, markerRange };\n\n\t\t\tthis.fire<DowncastAddMarkerEvent>( eventName, data, conversionApi );\n\t\t}\n\t}\n\n\t/**\n\t * Fires the conversion of the marker removal. Fires the {@link #event:removeMarker `removeMarker`} event with the provided data.\n\t *\n\t * @fires removeMarker\n\t * @param markerName Marker name.\n\t * @param markerRange The marker range.\n\t * @param conversionApi The conversion API object.\n\t */\n\tprivate _convertMarkerRemove( markerName: string, markerRange: Range, conversionApi: DowncastConversionApi ) {\n\t\t// Do not convert if range is in graveyard.\n\t\tif ( markerRange.root.rootName == '$graveyard' ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis.fire<DowncastRemoveMarkerEvent>( `removeMarker:${ markerName }`, { markerName, markerRange }, conversionApi );\n\t}\n\n\t/**\n\t * Fires the reduction of changes buffered in the {@link module:engine/model/differ~Differ `Differ`}.\n\t *\n\t * Features can replace selected {@link module:engine/model/differ~DiffItem `DiffItem`}s with `reinsert` entries to trigger\n\t * reconversion. The {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure\n\t * `DowncastHelpers.elementToStructure()`} is using this event to trigger reconversion.\n\t *\n\t * @fires reduceChanges\n\t */\n\tprivate _reduceChanges( changes: Iterable<DiffItem> ): Iterable<DiffItem | DiffItemReinsert> {\n\t\tconst data: { changes: Iterable<DiffItem | DiffItemReinsert> } = { changes };\n\n\t\tthis.fire<DowncastReduceChangesEvent>( 'reduceChanges', data );\n\n\t\treturn data.changes;\n\t}\n\n\t/**\n\t * Populates provided {@link module:engine/conversion/modelconsumable~ModelConsumable} with values to consume from a given range,\n\t * assuming that the range has just been inserted to the model.\n\t *\n\t * @param consumable The consumable.\n\t * @param walkerValues The walker values for the inserted range.\n\t * @returns The values to consume.\n\t */\n\tprivate _addConsumablesForInsert(\n\t\tconsumable: Consumable,\n\t\twalkerValues: Iterable<TreeWalkerValue>\n\t): Consumable {\n\t\tfor ( const value of walkerValues ) {\n\t\t\tconst item = value.item;\n\n\t\t\t// Add consumable if it wasn't there yet.\n\t\t\tif ( consumable.test( item, 'insert' ) === null ) {\n\t\t\t\tconsumable.add( item, 'insert' );\n\n\t\t\t\tfor ( const key of item.getAttributeKeys() ) {\n\t\t\t\t\tconsumable.add( item, 'attribute:' + key );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn consumable;\n\t}\n\n\t/**\n\t * Populates provided {@link module:engine/conversion/modelconsumable~ModelConsumable} with values to consume for a given range.\n\t *\n\t * @param consumable The consumable.\n\t * @param range The affected range.\n\t * @param type Consumable type.\n\t * @returns The values to consume.\n\t */\n\tprivate _addConsumablesForRange(\n\t\tconsumable: Consumable,\n\t\trange: Range,\n\t\ttype: string\n\t): Consumable {\n\t\tfor ( const item of range.getItems() ) {\n\t\t\tconsumable.add( item, type );\n\t\t}\n\n\t\treturn consumable;\n\t}\n\n\t/**\n\t * Populates provided {@link module:engine/conversion/modelconsumable~ModelConsumable} with selection consumable values.\n\t *\n\t * @param consumable The consumable.\n\t * @param selection The selection to create the consumable from.\n\t * @param markers Markers that contain the selection.\n\t * @returns The values to consume.\n\t */\n\tprivate _addConsumablesForSelection(\n\t\tconsumable: Consumable,\n\t\tselection: Selection | DocumentSelection,\n\t\tmarkers: Iterable<Marker>\n\t): Consumable {\n\t\tconsumable.add( selection, 'selection' );\n\n\t\tfor ( const marker of markers ) {\n\t\t\tconsumable.add( selection, 'addMarker:' + marker.name );\n\t\t}\n\n\t\tfor ( const key of selection.getAttributeKeys() ) {\n\t\t\tconsumable.add( selection, 'attribute:' + key );\n\t\t}\n\n\t\treturn consumable;\n\t}\n\n\t/**\n\t * Tests whether given event wasn't already fired and if so, fires it.\n\t *\n\t * @fires insert\n\t * @fires attribute\n\t * @param type Event type.\n\t * @param data Event data.\n\t * @param conversionApi The conversion API object.\n\t */\n\tprivate _testAndFire<TType extends 'insert' | 'attribute'>(\n\t\ttype: TType | `${ TType }:${ string }`,\n\t\tdata: EventMap[ TType ],\n\t\tconversionApi: DowncastConversionApi\n\t): void {\n\t\tconst eventName = getEventName( type, data );\n\t\tconst itemKey = data.item.is( '$textProxy' ) ? conversionApi.consumable._getSymbolForTextProxy( data.item ) : data.item;\n\n\t\tconst eventsFiredForConversion = this._firedEventsMap.get( conversionApi )!;\n\t\tconst eventsFiredForItem = eventsFiredForConversion.get( itemKey );\n\n\t\tif ( !eventsFiredForItem ) {\n\t\t\teventsFiredForConversion.set( itemKey, new Set( [ eventName ] ) );\n\t\t} else if ( !eventsFiredForItem.has( eventName ) ) {\n\t\t\teventsFiredForItem.add( eventName );\n\t\t} else {\n\t\t\treturn;\n\t\t}\n\n\t\tthis.fire<DowncastEvent<TType>>( eventName, data, conversionApi );\n\t}\n\n\t/**\n\t * Fires not already fired events for setting attributes on just inserted item.\n\t *\n\t * @param item The model item to convert attributes for.\n\t * @param conversionApi The conversion API object.\n\t */\n\tprivate _testAndFireAddAttributes(\n\t\titem: Item,\n\t\tconversionApi: DowncastConversionApi\n\t): void {\n\t\tconst data: EventMap[ 'attribute' ] = {\n\t\t\titem,\n\t\t\trange: Range._createOn( item )\n\t\t} as any;\n\n\t\tfor ( const key of data.item.getAttributeKeys() ) {\n\t\t\tdata.attributeKey = key;\n\t\t\tdata.attributeOldValue = null;\n\t\t\tdata.attributeNewValue = data.item.getAttribute( key );\n\n\t\t\tthis._testAndFire( `attribute:${ key }`, data, conversionApi );\n\t\t}\n\t}\n\n\t/**\n\t * Builds an instance of the {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi} from a template and a given\n\t * {@link module:engine/view/downcastwriter~DowncastWriter `DowncastWriter`} and options object.\n\t *\n\t * @param writer View writer that should be used to modify the view document.\n\t * @param refreshedItems A set of model elements that should not reuse their\n\t * previous view representations.\n\t * @param options Optional options passed to `convertionApi.options`.\n\t * @return The conversion API object.\n\t */\n\tprivate _createConversionApi(\n\t\twriter: DowncastWriter,\n\t\trefreshedItems: Set<Item> = new Set(),\n\t\toptions: DowncastConversionApi[ 'options' ] = {}\n\t): DowncastConversionApi {\n\t\tconst conversionApi: DowncastConversionApi = {\n\t\t\t...this._conversionApi,\n\t\t\tconsumable: new Consumable(),\n\t\t\twriter,\n\t\t\toptions,\n\t\t\tconvertItem: item => this._convertInsert( Range._createOn( item ), conversionApi ),\n\t\t\tconvertChildren: element => this._convertInsert( Range._createIn( element ), conversionApi, { doNotAddConsumables: true } ),\n\t\t\tconvertAttributes: item => this._testAndFireAddAttributes( item, conversionApi ),\n\t\t\tcanReuseView: viewElement => !refreshedItems.has( conversionApi.mapper.toModelElement( viewElement )! )\n\t\t};\n\n\t\tthis._firedEventsMap.set( conversionApi, new Map() );\n\n\t\treturn conversionApi;\n\t}\n}\n\n/**\n * Fired to enable reducing (transforming) changes buffered in the {@link module:engine/model/differ~Differ `Differ`} before\n * {@link ~DowncastDispatcher#convertChanges `convertChanges()`} will fire any conversion events.\n *\n * For instance, a feature can replace selected {@link module:engine/model/differ~DiffItem `DiffItem`}s with a `reinsert` entry\n * to trigger reconversion of an element when e.g. its attribute has changes.\n * The {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure\n\t * `DowncastHelpers.elementToStructure()`} helper is using this event to trigger reconversion of an element when the element,\n * its attributes or direct children changed.\n *\n * @eventName ~DowncastDispatcher#reduceChanges\n */\nexport type DowncastReduceChangesEvent = {\n\tname: 'reduceChanges';\n\targs: [ data: DowncastReduceChangesEventData ];\n};\n\nexport type DowncastReduceChangesEventData = {\n\n\t/**\n\t * A buffered changes to get reduced.\n\t */\n\tchanges: Iterable<DiffItem | DiffItemReinsert>;\n};\n\ntype EventMap<TItem = Item> = {\n\tinsert: {\n\t\titem: TItem;\n\t\trange: Range;\n\t\treconversion?: boolean;\n\t};\n\tremove: {\n\t\tposition: Position;\n\t\tlength: number;\n\t};\n\tattribute: {\n\t\titem: TItem;\n\t\trange: Range;\n\t\tattributeKey: string;\n\t\tattributeOldValue: unknown;\n\t\tattributeNewValue: unknown;\n\t};\n\tcleanSelection: {\n\t\tselection: Selection | DocumentSelection;\n\t};\n\tselection: {\n\t\tselection: Selection | DocumentSelection;\n\t};\n\taddMarker: {\n\t\titem?: Item | Selection | DocumentSelection;\n\t\trange?: Range;\n\t\tmarkerRange: Range;\n\t\tmarkerName: string;\n\t};\n\tremoveMarker: {\n\t\tmarkerRange: Range;\n\t\tmarkerName: string;\n\t};\n};\n\nexport type DowncastEvent<TName extends keyof EventMap<TItem>, TItem = Item> = {\n\tname: TName | `${ TName }:${ string }`;\n\targs: [ data: EventMap<TItem>[ TName ], conversionApi: DowncastConversionApi ];\n};\n\n/**\n * Fired for inserted nodes.\n *\n * `insert` is a namespace for a class of events. Names of actually called events follow this pattern:\n * `insert:name`. `name` is either `'$text'`, when {@link module:engine/model/text~Text a text node} has been inserted,\n * or {@link module:engine/model/element~Element#name name} of inserted element.\n *\n * This way, the listeners can either listen to a general `insert` event or specific event (for example `insert:paragraph`).\n *\n * @eventName ~DowncastDispatcher#insert\n * @param {Object} data Additional information about the change.\n * @param {module:engine/model/item~Item} data.item The inserted item.\n * @param {module:engine/model/range~Range} data.range Range spanning over inserted item.\n * @param {module:engine/conversion/downcastdispatcher~DowncastConversionApi} conversionApi Conversion interface\n * to be used by callback, passed in the `DowncastDispatcher` constructor.\n */\nexport type DowncastInsertEvent<TItem extends Item = Item> = DowncastEvent<'insert', TItem>;\n\n/**\n * Fired for removed nodes.\n *\n * `remove` is a namespace for a class of events. Names of actually called events follow this pattern:\n * `remove:name`. `name` is either `'$text'`, when a {@link module:engine/model/text~Text a text node} has been removed,\n * or the {@link module:engine/model/element~Element#name name} of removed element.\n *\n * This way, listeners can either listen to a general `remove` event or specific event (for example `remove:paragraph`).\n *\n * @eventName ~DowncastDispatcher#remove\n * @param {Object} data Additional information about the change.\n * @param {module:engine/model/position~Position} data.position Position from which the node has been removed.\n * @param {Number} data.length Offset size of the removed node.\n * @param {module:engine/conversion/downcastdispatcher~DowncastConversionApi} conversionApi Conversion interface\n * to be used by callback, passed in `DowncastDispatcher` constructor.\n */\nexport type DowncastRemoveEvent = DowncastEvent<'remove'>;\n\n/**\n * Fired in the following cases:\n *\n * * when an attribute has been added, changed, or removed from a node,\n * * when a node with an attribute is inserted,\n * * when a collapsed model selection attribute is converted.\n *\n * `attribute` is a namespace for a class of events. Names of actually called events follow this pattern:\n * `attribute:attributeKey:name`. `attributeKey` is the key of added/changed/removed attribute.\n * `name` is either `'$text'` if change was on {@link module:engine/model/text~Text a text node},\n * or the {@link module:engine/model/element~Element#name name} of element which attribute has changed.\n *\n * This way listeners can either listen to a general `attribute:bold` event or specific event (for example `attribute:src:imageBlock`).\n *\n * @eventName ~DowncastDispatcher#attribute\n * @param {Object} data Additional information about the change.\n * @param {module:engine/model/item~Item|module:engine/model/documentselection~DocumentSelection} data.item Changed item\n * or converted selection.\n * @param {module:engine/model/range~Range} data.range Range spanning over changed item or selection range.\n * @param {String} data.attributeKey Attribute key.\n * @param {*} data.attributeOldValue Attribute value before the change. This is `null` when selection attribute is converted.\n * @param {*} data.attributeNewValue New attribute value.\n * @param {module:engine/conversion/downcastdispatcher~DowncastConversionApi} conversionApi Conversion interface\n * to be used by callback, passed in `DowncastDispatcher` constructor.\n */\nexport type DowncastAttributeEvent<TItem = Item | Selection | DocumentSelection> = DowncastEvent<'attribute', TItem>;\n\n/**\n * Fired for {@link module:engine/model/selection~Selection selection} changes.\n *\n * @eventName ~DowncastDispatcher#selection\n * @param {module:engine/model/selection~Selection} selection Selection that is converted.\n * @param {module:engine/conversion/downcastdispatcher~DowncastConversionApi} conversionApi Conversion interface\n * to be used by callback, passed in `DowncastDispatcher` constructor.\n */\nexport type DowncastSelectionEvent = DowncastEvent<'selection'>;\n\n/**\n * Fired at the beginning of selection conversion, before\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:selection selection} events.\n *\n * Should be used to clean up the view state at the current selection position, before the selection is moved to another place.\n *\n * @eventName ~DowncastDispatcher#cleanSelection\n * @param {module:engine/model/selection~Selection} selection Selection that is converted.\n * @param {module:engine/conversion/downcastdispatcher~DowncastConversionApi} conversionApi Conversion interface\n * to be used by callback, passed in `DowncastDispatcher` constructor.\n */\nexport type DowncastCleanSelectionEvent = DowncastEvent<'cleanSelection'>;\n\n/**\n * Fired when a new marker is added to the model. Also fired when a collapsed model selection that is inside a marker is converted.\n *\n * `addMarker` is a namespace for a class of events. Names of actually called events follow this pattern:\n * `addMarker:markerName`. By specifying certain marker names, you can make the events even more gradual. For example,\n * if markers are named `foo:abc`, `foo:bar`, then it is possible to listen to `addMarker:foo` or `addMarker:foo:abc` and\n * `addMarker:foo:bar` events.\n *\n * If the marker range is not collapsed:\n *\n * * the event is fired for each item in the marker range one by one,\n * * `conversionApi.consumable` includes each item of the marker range and the consumable value is same as the event name.\n *\n * If the marker range is collapsed:\n *\n * * there is only one event,\n * * `conversionApi.consumable` includes marker range with the event name.\n *\n * If the selection inside a marker is converted:\n *\n * * there is only one event,\n * * `conversionApi.consumable` includes the selection instance with the event name.\n *\n * @eventName ~DowncastDispatcher#addMarker\n * @param {Object} data Additional information about the change.\n * @param {module:engine/model/item~Item|module:engine/model/selection~Selection} data.item Item inside the new marker or\n * the selection that is being converted.\n * @param {module:engine/model/range~Range} [data.range] Range spanning over converted item. Available only in marker conversion, if\n * the marker range was not collapsed.\n * @param {module:engine/model/range~Range} data.markerRange Marker range.\n * @param {String} data.markerName Marker name.\n * @param {module:engine/conversion/downcastdispatcher~DowncastConversionApi} conversionApi Conversion interface\n * to be used by callback, passed in `DowncastDispatcher` constructor.\n */\nexport type DowncastAddMarkerEvent = DowncastEvent<'addMarker'>;\n\n/**\n * Fired when a marker is removed from the model.\n *\n * `removeMarker` is a namespace for a class of events. Names of actually called events follow this pattern:\n * `removeMarker:markerName`. By specifying certain marker names, you can make the events even more gradual. For example,\n * if markers are named `foo:abc`, `foo:bar`, then it is possible to listen to `removeMarker:foo` or `removeMarker:foo:abc` and\n * `removeMarker:foo:bar` events.\n *\n * @eventName ~DowncastDispatcher#removeMarker\n * @param {Object} data Additional information about the change.\n * @param {module:engine/model/range~Range} data.markerRange Marker range.\n * @param {String} data.markerName Marker name.\n * @param {module:engine/conversion/downcastdispatcher~DowncastConversionApi} conversionApi Conversion interface\n * to be used by callback, passed in `DowncastDispatcher` constructor.\n */\nexport type DowncastRemoveMarkerEvent = DowncastEvent<'removeMarker'>;\n\nexport interface DiffItemReinsert {\n\ttype: 'reinsert';\n\tname: string;\n\tposition: Position;\n\tlength: number;\n}\n\n/**\n * Helper function, checks whether change of `marker` at `modelPosition` should be converted. Marker changes are not\n * converted if they happen inside an element with custom conversion method.\n */\nfunction shouldMarkerChangeBeConverted(\n\tmodelPosition: Position,\n\tmarker: Marker,\n\tmapper: Mapper\n): boolean {\n\tconst range = marker.getRange();\n\tconst ancestors = Array.from( modelPosition.getAncestors() );\n\tancestors.shift(); // Remove root element. It cannot be passed to `model.Range#containsItem`.\n\tancestors.reverse();\n\n\tconst hasCustomHandling = ( ancestors as Array<Element> ).some( element => {\n\t\tif ( range.containsItem( element ) ) {\n\t\t\tconst viewElement = mapper.toViewElement( element )!;\n\n\t\t\treturn !!viewElement.getCustomProperty( 'addHighlight' );\n\t\t}\n\t} );\n\n\treturn !hasCustomHandling;\n}\n\nfunction getEventName<TType extends string>( type: TType, data: { item: Item | Selection | DocumentSelection } ) {\n\tconst name = data.item.is( 'element' ) ? data.item.name : '$text';\n\n\treturn `${ type }:${ name }` as const;\n}\n\nfunction walkerValueToEventData( value: TreeWalkerValue ) {\n\tconst item = value.item;\n\tconst itemRange = Range._createFromPositionAndShift( value.previousPosition, value.length! );\n\n\treturn {\n\t\titem,\n\t\trange: itemRange\n\t};\n}\n\n/**\n * Conversion interface that is registered for given {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher}\n * and is passed as one of parameters when {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher dispatcher}\n * fires its events.\n */\nexport interface DowncastConversionApi {\n\n\t/**\n\t * The {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher} instance.\n\t */\n\tdispatcher: DowncastDispatcher;\n\n\t/**\n\t * Stores the information about what parts of a processed model item are still waiting to be handled. After a piece of a model item was\n\t * converted, an appropriate consumable value should be {@link module:engine/conversion/modelconsumable~ModelConsumable#consume\n\t * consumed}.\n\t */\n\tconsumable: Consumable;\n\n\t/**\n\t * The {@link module:engine/conversion/mapper~Mapper} instance.\n\t */\n\tmapper: Mapper;\n\n\t/**\n\t * The {@link module:engine/model/schema~Schema} instance set for the model that is downcast.\n\t */\n\tschema: Schema;\n\n\t/**\n\t * The {@link module:engine/view/downcastwriter~DowncastWriter} instance used to manipulate the data during conversion.\n\t */\n\twriter: DowncastWriter;\n\n\t/**\n\t * An object with an additional configuration which can be used during the conversion process.\n\t * Available only for data downcast conversion.\n\t */\n\toptions: Record<string, unknown>;\n\n\t/**\n\t * Triggers conversion of a specified item.\n\t * This conversion is triggered within (as a separate process of) the parent conversion.\n\t *\n\t * @param item The model item to trigger nested insert conversion on.\n\t */\n\tconvertItem( item: Item ): void;\n\n\t/**\n\t * Triggers conversion of children of a specified element.\n\t *\n\t * @param element The model element to trigger children insert conversion on.\n\t */\n\tconvertChildren( element: Element ): void;\n\n\t/**\n\t * Triggers conversion of attributes of a specified item.\n\t *\n\t * @param item The model item to trigger attribute conversion on.\n\t */\n\tconvertAttributes( item: Item ): void;\n\tcanReuseView( element: ViewElement ): boolean;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/selection\n */\n\nimport TypeCheckable from './typecheckable';\nimport Node from './node';\nimport Position, { type PositionOffset } from './position';\nimport Range from './range';\n\nimport type DocumentFragment from './documentfragment';\nimport type DocumentSelection from './documentselection';\nimport type Element from './element';\nimport type Item from './item';\n\nimport { CKEditorError, EmitterMixin, isIterable } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Selection is a set of {@link module:engine/model/range~Range ranges}. It has a direction specified by its\n * {@link module:engine/model/selection~Selection#anchor anchor} and {@link module:engine/model/selection~Selection#focus focus}\n * (it can be {@link module:engine/model/selection~Selection#isBackward forward or backward}).\n * Additionally, selection may have its own attributes (think whether text typed in in this selection\n * should have those attributes e.g. whether you type a bolded text).\n */\nexport default class Selection extends EmitterMixin( TypeCheckable ) {\n\t/**\n\t * Specifies whether the last added range was added as a backward or forward range.\n\t */\n\tprivate _lastRangeBackward: boolean = false;\n\n\t/**\n\t * List of attributes set on current selection.\n\t */\n\tprotected _attrs: Map<string, unknown> = new Map();\n\n\t/** @internal */\n\tpublic _ranges: Array<Range> = [];\n\n\t/**\n\t * Creates a new selection instance based on the given {@link module:engine/model/selection~Selectable selectable}\n\t * or creates an empty selection if no arguments were passed.\n\t *\n\t * ```ts\n\t * // Creates empty selection without ranges.\n\t * const selection = writer.createSelection();\n\t *\n\t * // Creates selection at the given range.\n\t * const range = writer.createRange( start, end );\n\t * const selection = writer.createSelection( range );\n\t *\n\t * // Creates selection at the given ranges\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( star2, end2 ) ];\n\t * const selection = writer.createSelection( ranges );\n\t *\n\t * // Creates selection from the other selection.\n\t * // Note: It doesn't copy selection attributes.\n\t * const otherSelection = writer.createSelection();\n\t * const selection = writer.createSelection( otherSelection );\n\t *\n\t * // Creates selection from the given document selection.\n\t * // Note: It doesn't copy selection attributes.\n\t * const documentSelection = model.document.selection;\n\t * const selection = writer.createSelection( documentSelection );\n\t *\n\t * // Creates selection at the given position.\n\t * const position = writer.createPositionFromPath( root, path );\n\t * const selection = writer.createSelection( position );\n\t *\n\t * // Creates selection at the given offset in the given element.\n\t * const paragraph = writer.createElement( 'paragraph' );\n\t * const selection = writer.createSelection( paragraph, offset );\n\t *\n\t * // Creates a range inside an {@link module:engine/model/element~Element element} which starts before the\n\t * // first child of that element and ends after the last child of that element.\n\t * const selection = writer.createSelection( paragraph, 'in' );\n\t *\n\t * // Creates a range on an {@link module:engine/model/item~Item item} which starts before the item and ends\n\t * // just after the item.\n\t * const selection = writer.createSelection( paragraph, 'on' );\n\t * ```\n\t *\n\t * Selection's constructor allow passing additional options (`'backward'`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = writer.createSelection( range, { backward: true } );\n\t * ```\n\t *\n\t * @internal\n\t */\n\tconstructor(\n\t\t...args: [] |\n\t\t[\n\t\t\tselectable: Node,\n\t\t\tplaceOrOffset: PlaceOrOffset,\n\t\t\toptions?: { backward?: boolean }\n\t\t] |\n\t\t[\n\t\t\tselectable?: Exclude<Selectable, Node>,\n\t\t\toptions?: { backward?: boolean }\n\t\t]\n\t) {\n\t\tsuper();\n\n\t\tif ( args.length ) {\n\t\t\tthis.setTo( ...args );\n\t\t}\n\t}\n\n\t/**\n\t * Selection anchor. Anchor is the position from which the selection was started. If a user is making a selection\n\t * by dragging the mouse, the anchor is where the user pressed the mouse button (the beginning of the selection).\n\t *\n\t * Anchor and {@link #focus} define the direction of the selection, which is important\n\t * when expanding/shrinking selection. The focus moves, while the anchor should remain in the same place.\n\t *\n\t * Anchor is always set to the {@link module:engine/model/range~Range#start start} or\n\t * {@link module:engine/model/range~Range#end end} position of the last of selection's ranges. Whether it is\n\t * the `start` or `end` depends on the specified `options.backward`. See the {@link #setTo `setTo()`} method.\n\t *\n\t * May be set to `null` if there are no ranges in the selection.\n\t *\n\t * @see #focus\n\t */\n\tpublic get anchor(): Position | null {\n\t\tif ( this._ranges.length > 0 ) {\n\t\t\tconst range = this._ranges[ this._ranges.length - 1 ];\n\n\t\t\treturn this._lastRangeBackward ? range.end : range.start;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Selection focus. Focus is the position where the selection ends. If a user is making a selection\n\t * by dragging the mouse, the focus is where the mouse cursor is.\n\t *\n\t * May be set to `null` if there are no ranges in the selection.\n\t *\n\t * @see #anchor\n\t */\n\tpublic get focus(): Position | null {\n\t\tif ( this._ranges.length > 0 ) {\n\t\t\tconst range = this._ranges[ this._ranges.length - 1 ];\n\n\t\t\treturn this._lastRangeBackward ? range.start : range.end;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Whether the selection is collapsed. Selection is collapsed when there is exactly one range in it\n\t * and it is collapsed.\n\t */\n\tpublic get isCollapsed(): boolean {\n\t\tconst length = this._ranges.length;\n\n\t\tif ( length === 1 ) {\n\t\t\treturn this._ranges[ 0 ].isCollapsed;\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n\n\t/**\n\t * Returns the number of ranges in the selection.\n\t */\n\tpublic get rangeCount(): number {\n\t\treturn this._ranges.length;\n\t}\n\n\t/**\n\t * Specifies whether the selection's {@link #focus} precedes the selection's {@link #anchor}.\n\t */\n\tpublic get isBackward(): boolean {\n\t\treturn !this.isCollapsed && this._lastRangeBackward;\n\t}\n\n\t/**\n\t * Checks whether this selection is equal to the given selection. Selections are equal if they have the same directions,\n\t * the same number of ranges and all ranges from one selection equal to ranges from the another selection.\n\t *\n\t * @param otherSelection Selection to compare with.\n\t * @returns `true` if selections are equal, `false` otherwise.\n\t */\n\tpublic isEqual( otherSelection: Selection | DocumentSelection ): boolean {\n\t\tif ( this.rangeCount != otherSelection.rangeCount ) {\n\t\t\treturn false;\n\t\t} else if ( this.rangeCount === 0 ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tif ( !this.anchor!.isEqual( otherSelection.anchor! ) || !this.focus!.isEqual( otherSelection.focus! ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tfor ( const thisRange of this._ranges ) {\n\t\t\tlet found = false;\n\n\t\t\tfor ( const otherRange of otherSelection._ranges ) {\n\t\t\t\tif ( thisRange.isEqual( otherRange ) ) {\n\t\t\t\t\tfound = true;\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( !found ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Returns an iterable object that iterates over copies of selection ranges.\n\t */\n\tpublic* getRanges(): IterableIterator<Range> {\n\t\tfor ( const range of this._ranges ) {\n\t\t\tyield new Range( range.start, range.end );\n\t\t}\n\t}\n\n\t/**\n\t * Returns a copy of the first range in the selection.\n\t * First range is the one which {@link module:engine/model/range~Range#start start} position\n\t * {@link module:engine/model/position~Position#isBefore is before} start position of all other ranges\n\t * (not to confuse with the first range added to the selection).\n\t *\n\t * Returns `null` if there are no ranges in selection.\n\t */\n\tpublic getFirstRange(): Range | null {\n\t\tlet first = null;\n\n\t\tfor ( const range of this._ranges ) {\n\t\t\tif ( !first || range.start.isBefore( first.start ) ) {\n\t\t\t\tfirst = range;\n\t\t\t}\n\t\t}\n\n\t\treturn first ? new Range( first.start, first.end ) : null;\n\t}\n\n\t/**\n\t * Returns a copy of the last range in the selection.\n\t * Last range is the one which {@link module:engine/model/range~Range#end end} position\n\t * {@link module:engine/model/position~Position#isAfter is after} end position of all other ranges (not to confuse with the range most\n\t * recently added to the selection).\n\t *\n\t * Returns `null` if there are no ranges in selection.\n\t */\n\tpublic getLastRange(): Range | null {\n\t\tlet last = null;\n\n\t\tfor ( const range of this._ranges ) {\n\t\t\tif ( !last || range.end.isAfter( last.end ) ) {\n\t\t\t\tlast = range;\n\t\t\t}\n\t\t}\n\n\t\treturn last ? new Range( last.start, last.end ) : null;\n\t}\n\n\t/**\n\t * Returns the first position in the selection.\n\t * First position is the position that {@link module:engine/model/position~Position#isBefore is before}\n\t * any other position in the selection.\n\t *\n\t * Returns `null` if there are no ranges in selection.\n\t */\n\tpublic getFirstPosition(): Position | null {\n\t\tconst first = this.getFirstRange();\n\n\t\treturn first ? first.start.clone() : null;\n\t}\n\n\t/**\n\t * Returns the last position in the selection.\n\t * Last position is the position that {@link module:engine/model/position~Position#isAfter is after}\n\t * any other position in the selection.\n\t *\n\t * Returns `null` if there are no ranges in selection.\n\t */\n\tpublic getLastPosition(): Position | null {\n\t\tconst lastRange = this.getLastRange();\n\n\t\treturn lastRange ? lastRange.end.clone() : null;\n\t}\n\n\t/**\n\t * Sets this selection's ranges and direction to the specified location based on the given\n\t * {@link module:engine/model/selection~Selectable selectable}.\n\t *\n\t * ```ts\n\t * // Removes all selection's ranges.\n\t * selection.setTo( null );\n\t *\n\t * // Sets selection to the given range.\n\t * const range = writer.createRange( start, end );\n\t * selection.setTo( range );\n\t *\n\t * // Sets selection to given ranges.\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( star2, end2 ) ];\n\t * selection.setTo( ranges );\n\t *\n\t * // Sets selection to other selection.\n\t * // Note: It doesn't copy selection attributes.\n\t * const otherSelection = writer.createSelection();\n\t * selection.setTo( otherSelection );\n\t *\n\t * // Sets selection to the given document selection.\n\t * // Note: It doesn't copy selection attributes.\n\t * const documentSelection = new DocumentSelection( doc );\n\t * selection.setTo( documentSelection );\n\t *\n\t * // Sets collapsed selection at the given position.\n\t * const position = writer.createPositionFromPath( root, path );\n\t * selection.setTo( position );\n\t *\n\t * // Sets collapsed selection at the position of the given node and an offset.\n\t * selection.setTo( paragraph, offset );\n\t * ```\n\t *\n\t * Creates a range inside an {@link module:engine/model/element~Element element} which starts before the first child of\n \t * that element and ends after the last child of that element.\n\t *\n\t * ```ts\n\t * selection.setTo( paragraph, 'in' );\n\t * ```\n\t *\n\t * Creates a range on an {@link module:engine/model/item~Item item} which starts before the item and ends just after the item.\n\t *\n\t * ```ts\n\t * selection.setTo( paragraph, 'on' );\n\t * ```\n\t *\n\t * `Selection#setTo()`' method allow passing additional options (`backward`) as the last argument.\n\t *\n\t * ```ts\n\t * // Sets backward selection.\n\t * const selection = writer.createSelection( range, { backward: true } );\n\t * ```\n\t */\n\tpublic setTo(\n\t\t...args: [\n\t\t\tselectable: Node,\n\t\t\tplaceOrOffset: PlaceOrOffset,\n\t\t\toptions?: { backward?: boolean }\n\t\t] | [\n\t\t\tselectable?: Exclude<Selectable, Node>,\n\t\t\toptions?: { backward?: boolean }\n\t\t]\n\t): void {\n\t\tlet [ selectable, placeOrOffset, options ] = args;\n\n\t\tif ( typeof placeOrOffset == 'object' ) {\n\t\t\toptions = placeOrOffset;\n\t\t\tplaceOrOffset = undefined;\n\t\t}\n\n\t\tif ( selectable === null ) {\n\t\t\tthis._setRanges( [] );\n\t\t} else if ( selectable instanceof Selection ) {\n\t\t\tthis._setRanges( selectable.getRanges(), selectable.isBackward );\n\t\t} else if ( selectable && typeof ( selectable as any ).getRanges == 'function' ) {\n\t\t\t// We assume that the selectable is a DocumentSelection.\n\t\t\t// It can't be imported here, because it would lead to circular imports.\n\t\t\tthis._setRanges( ( selectable as DocumentSelection ).getRanges(), ( selectable as DocumentSelection ).isBackward );\n\t\t} else if ( selectable instanceof Range ) {\n\t\t\tthis._setRanges( [ selectable ], !!options && !!options.backward );\n\t\t} else if ( selectable instanceof Position ) {\n\t\t\tthis._setRanges( [ new Range( selectable ) ] );\n\t\t} else if ( selectable instanceof Node ) {\n\t\t\tconst backward = !!options && !!options.backward;\n\t\t\tlet range;\n\n\t\t\tif ( placeOrOffset == 'in' ) {\n\t\t\t\trange = Range._createIn( selectable as Element );\n\t\t\t} else if ( placeOrOffset == 'on' ) {\n\t\t\t\trange = Range._createOn( selectable );\n\t\t\t} else if ( placeOrOffset !== undefined ) {\n\t\t\t\trange = new Range( Position._createAt( selectable, placeOrOffset ) );\n\t\t\t} else {\n\t\t\t\t/**\n\t\t\t\t * selection.setTo requires the second parameter when the first parameter is a node.\n\t\t\t\t *\n\t\t\t\t * @error model-selection-setto-required-second-parameter\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'model-selection-setto-required-second-parameter', [ this, selectable ] );\n\t\t\t}\n\n\t\t\tthis._setRanges( [ range ], backward );\n\t\t} else if ( isIterable( selectable ) ) {\n\t\t\t// We assume that the selectable is an iterable of ranges.\n\t\t\tthis._setRanges( selectable, options && !!options.backward );\n\t\t} else {\n\t\t\t/**\n\t\t\t * Cannot set the selection to the given place.\n\t\t\t *\n\t\t\t * Invalid parameters were specified when setting the selection. Common issues:\n\t\t\t *\n\t\t\t * * A {@link module:engine/model/textproxy~TextProxy} instance was passed instead of\n\t\t\t * a real {@link module:engine/model/text~Text}.\n\t\t\t * * View nodes were passed instead of model nodes.\n\t\t\t * * `null`/`undefined` was passed.\n\t\t\t *\n\t\t\t * @error model-selection-setto-not-selectable\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-selection-setto-not-selectable', [ this, selectable ] );\n\t\t}\n\t}\n\n\t/**\n\t * Replaces all ranges that were added to the selection with given array of ranges. Last range of the array\n\t * is treated like the last added range and is used to set {@link module:engine/model/selection~Selection#anchor} and\n\t * {@link module:engine/model/selection~Selection#focus}. Accepts a flag describing in which direction the selection is made.\n\t *\n\t * @fires change:range\n\t * @param newRanges Ranges to set.\n\t * @param isLastBackward Flag describing if last added range was selected forward - from start to end (`false`)\n\t * or backward - from end to start (`true`).\n\t */\n\tprotected _setRanges( newRanges: Iterable<Range>, isLastBackward: boolean = false ): void {\n\t\tconst ranges = Array.from( newRanges );\n\n\t\t// Check whether there is any range in new ranges set that is different than all already added ranges.\n\t\tconst anyNewRange = ranges.some( newRange => {\n\t\t\tif ( !( newRange instanceof Range ) ) {\n\t\t\t\t/**\n\t\t\t\t * Selection range set to an object that is not an instance of {@link module:engine/model/range~Range}.\n\t\t\t\t *\n\t\t\t\t * Only {@link module:engine/model/range~Range} instances can be used to set a selection.\n\t\t\t\t * Common mistakes leading to this error are:\n\t\t\t\t *\n\t\t\t\t * * using DOM `Range` object,\n\t\t\t\t * * incorrect CKEditor 5 installation with multiple `ckeditor5-engine` packages having different versions.\n\t\t\t\t *\n\t\t\t\t * @error model-selection-set-ranges-not-range\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'model-selection-set-ranges-not-range',\n\t\t\t\t\t[ this, newRanges ]\n\t\t\t\t);\n\t\t\t}\n\n\t\t\treturn this._ranges.every( oldRange => {\n\t\t\t\treturn !oldRange.isEqual( newRange );\n\t\t\t} );\n\t\t} );\n\n\t\t// Don't do anything if nothing changed.\n\t\tif ( ranges.length === this._ranges.length && !anyNewRange ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._replaceAllRanges( ranges );\n\n\t\tthis._lastRangeBackward = !!isLastBackward;\n\n\t\tthis.fire<SelectionChangeRangeEvent>( 'change:range', { directChange: true } );\n\t}\n\n\t/**\n\t * Moves {@link module:engine/model/selection~Selection#focus} to the specified location.\n\t *\n\t * The location can be specified in the same form as\n\t * {@link module:engine/model/writer~Writer#createPositionAt writer.createPositionAt()} parameters.\n\t *\n\t * @fires change:range\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic setFocus( itemOrPosition: Item | Position, offset?: PositionOffset ): void {\n\t\tif ( this.anchor === null ) {\n\t\t\t/**\n\t\t\t * Cannot set selection focus if there are no ranges in selection.\n\t\t\t *\n\t\t\t * @error model-selection-setfocus-no-ranges\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-selection-setfocus-no-ranges', [ this, itemOrPosition ] );\n\t\t}\n\n\t\tconst newFocus = Position._createAt( itemOrPosition, offset );\n\n\t\tif ( newFocus.compareWith( this.focus! ) == 'same' ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst anchor = this.anchor;\n\n\t\tif ( this._ranges.length ) {\n\t\t\tthis._popRange();\n\t\t}\n\n\t\tif ( newFocus.compareWith( anchor ) == 'before' ) {\n\t\t\tthis._pushRange( new Range( newFocus, anchor ) );\n\t\t\tthis._lastRangeBackward = true;\n\t\t} else {\n\t\t\tthis._pushRange( new Range( anchor, newFocus ) );\n\t\t\tthis._lastRangeBackward = false;\n\t\t}\n\n\t\tthis.fire<SelectionChangeRangeEvent>( 'change:range', { directChange: true } );\n\t}\n\n\t/**\n\t * Gets an attribute value for given key or `undefined` if that attribute is not set on the selection.\n\t *\n\t * @param key Key of attribute to look for.\n\t * @returns Attribute value or `undefined`.\n\t */\n\tpublic getAttribute( key: string ): unknown {\n\t\treturn this._attrs.get( key );\n\t}\n\n\t/**\n\t * Returns iterable that iterates over this selection's attributes.\n\t *\n\t * Attributes are returned as arrays containing two items. First one is attribute key and second is attribute value.\n\t * This format is accepted by native `Map` object and also can be passed in `Node` constructor.\n\t */\n\tpublic getAttributes(): IterableIterator<[ string, unknown ]> {\n\t\treturn this._attrs.entries();\n\t}\n\n\t/**\n\t * Returns iterable that iterates over this selection's attribute keys.\n\t */\n\tpublic getAttributeKeys(): IterableIterator<string> {\n\t\treturn this._attrs.keys();\n\t}\n\n\t/**\n\t * Checks if the selection has an attribute for given key.\n\t *\n\t * @param key Key of attribute to check.\n\t * @returns `true` if attribute with given key is set on selection, `false` otherwise.\n\t */\n\tpublic hasAttribute( key: string ): boolean {\n\t\treturn this._attrs.has( key );\n\t}\n\n\t/**\n\t * Removes an attribute with given key from the selection.\n\t *\n\t * If given attribute was set on the selection, fires the {@link #event:change:range} event with\n\t * removed attribute key.\n\t *\n\t * @fires change:attribute\n\t * @param key Key of attribute to remove.\n\t */\n\tpublic removeAttribute( key: string ): void {\n\t\tif ( this.hasAttribute( key ) ) {\n\t\t\tthis._attrs.delete( key );\n\n\t\t\tthis.fire<SelectionChangeAttributeEvent>( 'change:attribute', { attributeKeys: [ key ], directChange: true } );\n\t\t}\n\t}\n\n\t/**\n\t * Sets attribute on the selection. If attribute with the same key already is set, it's value is overwritten.\n\t *\n\t * If the attribute value has changed, fires the {@link #event:change:range} event with\n\t * the attribute key.\n\t *\n\t * @fires change:attribute\n\t * @param key Key of attribute to set.\n\t * @param value Attribute value.\n\t */\n\tpublic setAttribute( key: string, value: unknown ): void {\n\t\tif ( this.getAttribute( key ) !== value ) {\n\t\t\tthis._attrs.set( key, value );\n\n\t\t\tthis.fire<SelectionChangeAttributeEvent>( 'change:attribute', { attributeKeys: [ key ], directChange: true } );\n\t\t}\n\t}\n\n\t/**\n\t * Returns the selected element. {@link module:engine/model/element~Element Element} is considered as selected if there is only\n\t * one range in the selection, and that range contains exactly one element.\n\t * Returns `null` if there is no selected element.\n\t */\n\tpublic getSelectedElement(): Element | null {\n\t\tif ( this.rangeCount !== 1 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn this.getFirstRange()!.getContainedElement();\n\t}\n\n\t/**\n\t * Gets elements of type {@link module:engine/model/schema~Schema#isBlock \"block\"} touched by the selection.\n\t *\n\t * This method's result can be used for example to apply block styling to all blocks covered by this selection.\n\t *\n\t * **Note:** `getSelectedBlocks()` returns blocks that are nested in other non-block elements\n\t * but will not return blocks nested in other blocks.\n\t *\n\t * In this case the function will return exactly all 3 paragraphs (note: `<blockQuote>` is not a block itself):\n\t *\n\t * ```xml\n\t * <paragraph>[a</paragraph>\n\t * <blockQuote>\n\t * \t<paragraph>b</paragraph>\n\t * </blockQuote>\n\t * <paragraph>c]d</paragraph>\n\t * ```\n\t *\n\t * In this case the paragraph will also be returned, despite the collapsed selection:\n\t *\n\t * ```xml\n\t * <paragraph>[]a</paragraph>\n\t * ```\n\t *\n\t * In such a scenario, however, only blocks A, B & E will be returned as blocks C & D are nested in block B:\n\t *\n\t * ```xml\n\t * [<blockA></blockA>\n\t * <blockB>\n\t * \t<blockC></blockC>\n\t * \t<blockD></blockD>\n\t * </blockB>\n\t * <blockE></blockE>]\n\t * ```\n\t *\n\t * If the selection is inside a block all the inner blocks (A & B) are returned:\n\t *\n\t * ```xml\n\t * <block>\n\t * \t<blockA>[a</blockA>\n\t * \t<blockB>b]</blockB>\n\t * </block>\n\t * ```\n\t *\n\t * **Special case**: Selection ignores first and/or last blocks if nothing (from user perspective) is selected in them.\n\t *\n\t * ```xml\n\t * // Selection ends and the beginning of the last block.\n\t * <paragraph>[a</paragraph>\n\t * <paragraph>b</paragraph>\n\t * <paragraph>]c</paragraph> // This block will not be returned\n\t *\n\t * // Selection begins at the end of the first block.\n\t * <paragraph>a[</paragraph> // This block will not be returned\n\t * <paragraph>b</paragraph>\n\t * <paragraph>c]</paragraph>\n\t *\n\t * // Selection begings at the end of the first block and ends at the beginning of the last block.\n\t * <paragraph>a[</paragraph> // This block will not be returned\n\t * <paragraph>b</paragraph>\n\t * <paragraph>]c</paragraph> // This block will not be returned\n\t * ```\n\t */\n\tpublic* getSelectedBlocks(): IterableIterator<Element> {\n\t\tconst visited = new WeakSet<Node | DocumentFragment>();\n\n\t\tfor ( const range of this.getRanges() ) {\n\t\t\t// Get start block of range in case of a collapsed range.\n\t\t\tconst startBlock = getParentBlock( range.start, visited );\n\n\t\t\tif ( isStartBlockSelected( startBlock, range ) ) {\n\t\t\t\tyield startBlock as any;\n\t\t\t}\n\n\t\t\tfor ( const value of range.getWalker() ) {\n\t\t\t\tconst block = value.item;\n\n\t\t\t\tif ( value.type == 'elementEnd' && isUnvisitedTopBlock( block as any, visited, range ) ) {\n\t\t\t\t\tyield block as Element;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst endBlock = getParentBlock( range.end, visited );\n\n\t\t\tif ( isEndBlockSelected( endBlock, range ) ) {\n\t\t\t\tyield endBlock as any;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Checks whether the selection contains the entire content of the given element. This means that selection must start\n\t * at a position {@link module:engine/model/position~Position#isTouching touching} the element's start and ends at position\n\t * touching the element's end.\n\t *\n\t * By default, this method will check whether the entire content of the selection's current root is selected.\n\t * Useful to check if e.g. the user has just pressed <kbd>Ctrl</kbd> + <kbd>A</kbd>.\n\t */\n\tpublic containsEntireContent( element: Element = this.anchor!.root as Element ): boolean {\n\t\tconst limitStartPosition = Position._createAt( element, 0 );\n\t\tconst limitEndPosition = Position._createAt( element, 'end' );\n\n\t\treturn limitStartPosition.isTouching( this.getFirstPosition()! ) &&\n\t\t\tlimitEndPosition.isTouching( this.getLastPosition()! );\n\t}\n\n\t/**\n\t * Adds given range to internal {@link #_ranges ranges array}. Throws an error\n\t * if given range is intersecting with any range that is already stored in this selection.\n\t */\n\tprotected _pushRange( range: Range ): void {\n\t\tthis._checkRange( range );\n\t\tthis._ranges.push( new Range( range.start, range.end ) );\n\t}\n\n\t/**\n\t * Checks if given range intersects with ranges that are already in the selection. Throws an error if it does.\n\t */\n\tprotected _checkRange( range: Range ): void {\n\t\tfor ( let i = 0; i < this._ranges.length; i++ ) {\n\t\t\tif ( range.isIntersecting( this._ranges[ i ] ) ) {\n\t\t\t\t/**\n\t\t\t\t * Trying to add a range that intersects with another range in the selection.\n\t\t\t\t *\n\t\t\t\t * @error model-selection-range-intersects\n\t\t\t\t * @param addedRange Range that was added to the selection.\n\t\t\t\t * @param intersectingRange Range in the selection that intersects with `addedRange`.\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'model-selection-range-intersects',\n\t\t\t\t\t[ this, range ],\n\t\t\t\t\t{ addedRange: range, intersectingRange: this._ranges[ i ] }\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Replaces all the ranges by the given ones.\n\t * Uses {@link #_popRange _popRange} and {@link #_pushRange _pushRange} to ensure proper ranges removal and addition.\n\t */\n\tprotected _replaceAllRanges( ranges: Array<Range> ): void {\n\t\tthis._removeAllRanges();\n\n\t\tfor ( const range of ranges ) {\n\t\t\tthis._pushRange( range );\n\t\t}\n\t}\n\n\t/**\n\t * Deletes ranges from internal range array. Uses {@link #_popRange _popRange} to\n\t * ensure proper ranges removal.\n\t */\n\tprotected _removeAllRanges(): void {\n\t\twhile ( this._ranges.length > 0 ) {\n\t\t\tthis._popRange();\n\t\t}\n\t}\n\n\t/**\n\t * Removes most recently added range from the selection.\n\t */\n\tprotected _popRange(): void {\n\t\tthis._ranges.pop();\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nSelection.prototype.is = function( type: string ): boolean {\n\treturn type === 'selection' || type === 'model:selection';\n};\n\n/**\n * Describes one of the events: `change:range` or `change:attribute`.\n */\nexport type SelectionChangeEvent = {\n\tname: 'change' | 'change:range' | 'change:attribute';\n\targs: [ {\n\t\tdirectChange: boolean;\n\t\tattributeKeys?: Array<string>;\n\t} ];\n};\n\n/**\n * Fired when selection range(s) changed.\n *\n * @eventName ~Selection#change:range\n * @param directChange In case of {@link module:engine/model/selection~Selection} class it is always set\n * to `true` which indicates that the selection change was caused by a direct use of selection's API.\n * The {@link module:engine/model/documentselection~DocumentSelection}, however, may change because its position\n * was directly changed through the {@link module:engine/model/writer~Writer writer} or because its position was\n * changed because the structure of the model has been changed (which means an indirect change).\n * The indirect change does not occur in case of normal (detached) selections because they are \"static\" (as \"not live\")\n * which mean that they are not updated once the document changes.\n */\nexport type SelectionChangeRangeEvent = {\n\tname: 'change' | 'change:range';\n\targs: [ {\n\t\tdirectChange: boolean;\n\t} ];\n};\n\n/**\n * Fired when selection attribute changed.\n *\n * @eventName ~Selection#change:attribute\n * @param directChange In case of {@link module:engine/model/selection~Selection} class it is always set\n * to `true` which indicates that the selection change was caused by a direct use of selection's API.\n * The {@link module:engine/model/documentselection~DocumentSelection}, however, may change because its attributes\n * were directly changed through the {@link module:engine/model/writer~Writer writer} or because its position was\n * changed in the model and its attributes were refreshed (which means an indirect change).\n * The indirect change does not occur in case of normal (detached) selections because they are \"static\" (as \"not live\")\n * which mean that they are not updated once the document changes.\n * @param attributeKeys Array containing keys of attributes that changed.\n */\nexport type SelectionChangeAttributeEvent = {\n\tname: 'change' | 'change:attribute';\n\targs: [ {\n\t\tdirectChange: boolean;\n\t\tattributeKeys: Array<string>;\n\t} ];\n};\n\n/**\n * Checks whether the given element extends $block in the schema and has a parent (is not a root).\n * Marks it as already visited.\n */\nfunction isUnvisitedBlock( element: Node | DocumentFragment, visited: WeakSet<Node | DocumentFragment> ) {\n\tif ( visited.has( element ) ) {\n\t\treturn false;\n\t}\n\n\tvisited.add( element );\n\n\treturn element.root.document!.model.schema.isBlock( element ) && !!element.parent;\n}\n\n/**\n * Checks if the given element is a $block was not previously visited and is a top block in a range.\n */\nfunction isUnvisitedTopBlock( element: Element, visited: WeakSet<Node | DocumentFragment>, range: Range ) {\n\treturn isUnvisitedBlock( element, visited ) && isTopBlockInRange( element, range );\n}\n\n/**\n * Finds the lowest element in position's ancestors which is a block.\n * It will search until first ancestor that is a limit element.\n * Marks all ancestors as already visited to not include any of them later on.\n */\nfunction getParentBlock( position: Position, visited: WeakSet<Node | DocumentFragment> ) {\n\tconst element = position.parent;\n\tconst schema = element.root.document!.model.schema;\n\n\tconst ancestors = position.parent.getAncestors( { parentFirst: true, includeSelf: true } );\n\n\tlet hasParentLimit = false;\n\n\tconst block = ancestors.find( ( element ): element is Element => {\n\t\t// Stop searching after first parent node that is limit element.\n\t\tif ( hasParentLimit ) {\n\t\t\treturn false;\n\t\t}\n\n\t\thasParentLimit = schema.isLimit( element );\n\n\t\treturn !hasParentLimit && isUnvisitedBlock( element, visited );\n\t} );\n\n\t// Mark all ancestors of this position's parent, because find() might've stopped early and\n\t// the found block may be a child of another block.\n\tancestors.forEach( element => visited.add( element ) );\n\n\treturn block;\n}\n\n/**\n * Checks if the blocks is not nested in other block inside a range.\n */\nfunction isTopBlockInRange( block: Node, range: Range ) {\n\tconst parentBlock = findAncestorBlock( block );\n\n\tif ( !parentBlock ) {\n\t\treturn true;\n\t}\n\n\t// Add loose flag to check as parentRange can be equal to range.\n\tconst isParentInRange = range.containsRange( Range._createOn( parentBlock ), true );\n\n\treturn !isParentInRange;\n}\n\n/**\n * If a selection starts at the end of a block, that block is not returned as from the user's perspective this block wasn't selected.\n * See [#11585](https://github.com/ckeditor/ckeditor5/issues/11585) for more details.\n *\n * ```xml\n * <paragraph>a[</paragraph> // This block will not be returned\n * <paragraph>b</paragraph>\n * <paragraph>c]</paragraph>\n * ```\n *\n * Collapsed selection is not affected by it:\n *\n * ```xml\n * <paragraph>a[]</paragraph> // This block will be returned\n * ```\n */\nfunction isStartBlockSelected( startBlock: Element | undefined, range: Range ): boolean {\n\tif ( !startBlock ) {\n\t\treturn false;\n\t}\n\n\tif ( range.isCollapsed || startBlock.isEmpty ) {\n\t\treturn true;\n\t}\n\n\tif ( range.start.isTouching( Position._createAt( startBlock, startBlock.maxOffset ) ) ) {\n\t\treturn false;\n\t}\n\n\treturn isTopBlockInRange( startBlock, range );\n}\n\n/**\n * If a selection ends at the beginning of a block, that block is not returned as from the user's perspective this block wasn't selected.\n * See [#984](https://github.com/ckeditor/ckeditor5-engine/issues/984) for more details.\n *\n * ```xml\n * <paragraph>[a</paragraph>\n * <paragraph>b</paragraph>\n * <paragraph>]c</paragraph> // this block will not be returned\n * ```\n *\n * Collapsed selection is not affected by it:\n *\n * ```xml\n * <paragraph>[]a</paragraph> // this block will be returned\n * ```\n */\nfunction isEndBlockSelected( endBlock: Element | undefined, range: Range ): boolean {\n\tif ( !endBlock ) {\n\t\treturn false;\n\t}\n\n\tif ( range.isCollapsed || endBlock.isEmpty ) {\n\t\treturn true;\n\t}\n\n\tif ( range.end.isTouching( Position._createAt( endBlock, 0 ) ) ) {\n\t\treturn false;\n\t}\n\n\treturn isTopBlockInRange( endBlock, range );\n}\n\n/**\n * Returns first ancestor block of a node.\n */\nfunction findAncestorBlock( node: Node | DocumentFragment ) {\n\tconst schema = node.root.document!.model.schema;\n\n\tlet parent = node.parent;\n\n\twhile ( parent ) {\n\t\tif ( schema.isBlock( parent ) ) {\n\t\t\treturn parent as Element;\n\t\t}\n\n\t\tparent = parent.parent;\n\t}\n}\n\n/**\n * An entity that is used to set selection.\n *\n * See also {@link module:engine/model/selection~Selection#setTo}.\n */\nexport type Selectable = Selection | DocumentSelection | Position | Range | Node | Iterable<Range> | null;\n\n/**\n * The place or offset of the selection.\n */\nexport type PlaceOrOffset = number | 'before' | 'end' | 'after' | 'on' | 'in';\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/liverange\n */\n\nimport Range from './range';\n\nimport type { ModelApplyOperationEvent } from './model';\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\nimport type Item from './item';\nimport type MergeOperation from './operation/mergeoperation';\nimport type MoveOperation from './operation/moveoperation';\nimport type Operation from './operation/operation';\nimport type Position from './position';\n\nimport { EmitterMixin } from '@ckeditor/ckeditor5-utils';\n\n/**\n * `LiveRange` is a type of {@link module:engine/model/range~Range Range}\n * that updates itself as {@link module:engine/model/document~Document document}\n * is changed through operations. It may be used as a bookmark.\n *\n * **Note:** Be very careful when dealing with `LiveRange`. Each `LiveRange` instance bind events that might\n * have to be unbound. Use {@link module:engine/model/liverange~LiveRange#detach detach} whenever you don't need `LiveRange` anymore.\n */\nexport default class LiveRange extends EmitterMixin( Range ) {\n\t/**\n\t * Creates a live range.\n\t *\n\t * @see module:engine/model/range~Range\n\t */\n\tconstructor( start: Position, end?: Position | null ) {\n\t\tsuper( start, end );\n\n\t\tbindWithDocument.call( this );\n\t}\n\n\t/**\n\t * Unbinds all events previously bound by `LiveRange`. Use it whenever you don't need `LiveRange` instance\n\t * anymore (i.e. when leaving scope in which it was declared or before re-assigning variable that was\n\t * referring to it).\n\t */\n\tpublic detach(): void {\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Creates a {@link module:engine/model/range~Range range instance} that is equal to this live range.\n\t */\n\tpublic toRange(): Range {\n\t\treturn new Range( this.start, this.end );\n\t}\n\n\t/**\n\t * Creates a `LiveRange` instance that is equal to the given range.\n\t */\n\tpublic static fromRange( range: Range ): LiveRange {\n\t\treturn new LiveRange( range.start, range.end );\n\t}\n\n\t/**\n\t * @see module:engine/model/range~Range._createIn\n\t * @internal\n\t */\n\n\tdeclare public static readonly _createIn: ( element: Element | DocumentFragment ) => LiveRange;\n\n\t/**\n\t * @see module:engine/model/range~Range._createOn\n\t * @internal\n\t */\n\n\tdeclare public static readonly _createOn: ( element: Item | DocumentFragment ) => LiveRange;\n\n\t/**\n\t * @see module:engine/model/range~Range._createFromPositionAndShift\n\t * @internal\n\t */\n\n\tdeclare public static readonly _createFromPositionAndShift: ( position: Position, shift: number ) => LiveRange;\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nLiveRange.prototype.is = function( type: string ): boolean {\n\treturn type === 'liveRange' || type === 'model:liveRange' ||\n\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\ttype == 'range' || type === 'model:range';\n};\n\n/**\n * Fired when `LiveRange` instance boundaries have changed due to changes in the\n * {@link module:engine/model/document~Document document}.\n *\n * @eventName ~LiveRange#change:range\n * @param oldRange Range with start and end position equal to start and end position of this live\n * range before it got changed.\n * @param data Object with additional information about the change.\n * @param data.deletionPosition Source position for remove and merge changes.\n * Available if the range was moved to the graveyard root, `null` otherwise.\n */\nexport type LiveRangeChangeRangeEvent = {\n\tname: 'change' | 'change:range';\n\targs: [ range: Range, data: { deletionPosition: Position | null } ];\n};\n\n/**\n * Fired when `LiveRange` instance boundaries have not changed after a change in {@link module:engine/model/document~Document document}\n * but the change took place inside the range, effectively changing its content.\n *\n * @eventName ~LiveRange#change:content\n * @param range Range with start and end position equal to start and end position of\n * change range.\n * @param data Object with additional information about the change.\n * @param data.deletionPosition Due to the nature of this event, this property is always set to `null`. It is passed\n * for compatibility with the {@link module:engine/model/liverange~LiveRange#event:change:range} event.\n */\nexport type LiveRangeChangeContentEvent = {\n\tname: 'change' | 'change:content';\n\targs: [ range: Range, data: { deletionPosition: null } ];\n};\n\n/**\n * Describes `change:range` or `change:content` event.\n */\nexport type LiveRangeChangeEvent = {\n\tname: 'change' | 'change:range' | 'change:content';\n\targs: [ range: Range, data: { deletionPosition: Position | null } ];\n};\n\n/**\n * Binds this `LiveRange` to the {@link module:engine/model/document~Document document}\n * that owns this range's {@link module:engine/model/range~Range#root root}.\n */\nfunction bindWithDocument( this: LiveRange ) {\n\tthis.listenTo<ModelApplyOperationEvent>(\n\t\tthis.root.document!.model,\n\t\t'applyOperation',\n\t\t( event, args ) => {\n\t\t\tconst operation = args[ 0 ];\n\n\t\t\tif ( !operation.isDocumentOperation ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\ttransform.call( this, operation );\n\t\t},\n\t\t{ priority: 'low' }\n\t);\n}\n\n/**\n * Updates this range accordingly to the updates applied to the model. Bases on change events.\n */\nfunction transform( this: LiveRange, operation: Operation ) {\n\t// Transform the range by the operation. Join the result ranges if needed.\n\tconst ranges = this.getTransformedByOperation( operation );\n\tconst result = Range._createFromRanges( ranges );\n\n\tconst boundariesChanged = !result.isEqual( this );\n\tconst contentChanged = doesOperationChangeRangeContent( this, operation );\n\n\tlet deletionPosition = null;\n\n\tif ( boundariesChanged ) {\n\t\t// If range boundaries have changed, fire `change:range` event.\n\t\t//\n\t\tif ( result.root.rootName == '$graveyard' ) {\n\t\t\t// If the range was moved to the graveyard root, set `deletionPosition`.\n\t\t\tif ( operation.type == 'remove' ) {\n\t\t\t\tdeletionPosition = ( operation as MoveOperation ).sourcePosition;\n\t\t\t} else {\n\t\t\t\t// Merge operation.\n\t\t\t\tdeletionPosition = ( operation as MergeOperation ).deletionPosition;\n\t\t\t}\n\t\t}\n\n\t\tconst oldRange = this.toRange();\n\n\t\t( this as any ).start = result.start;\n\t\t( this as any ).end = result.end;\n\n\t\tthis.fire<LiveRangeChangeRangeEvent>( 'change:range', oldRange, { deletionPosition } );\n\t} else if ( contentChanged ) {\n\t\t// If range boundaries have not changed, but there was change inside the range, fire `change:content` event.\n\t\tthis.fire<LiveRangeChangeContentEvent>( 'change:content', this.toRange(), { deletionPosition } );\n\t}\n}\n\n/**\n * Checks whether given operation changes something inside the range (even if it does not change boundaries).\n */\nfunction doesOperationChangeRangeContent( range: Range, operation: any ) {\n\tswitch ( operation.type ) {\n\t\tcase 'insert':\n\t\t\treturn range.containsPosition( operation.position );\n\t\tcase 'move':\n\t\tcase 'remove':\n\t\tcase 'reinsert':\n\t\tcase 'merge':\n\t\t\treturn range.containsPosition( operation.sourcePosition ) ||\n\t\t\t\trange.start.isEqual( operation.sourcePosition ) ||\n\t\t\t\trange.containsPosition( operation.targetPosition );\n\t\tcase 'split':\n\t\t\treturn range.containsPosition( operation.splitPosition ) || range.containsPosition( operation.insertionPosition );\n\t}\n\n\treturn false;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/documentselection\n */\n\nimport TypeCheckable from './typecheckable';\nimport LiveRange from './liverange';\nimport Selection, {\n\ttype SelectionChangeAttributeEvent,\n\ttype SelectionChangeRangeEvent\n} from './selection';\nimport Text from './text';\nimport TextProxy from './textproxy';\n\nimport type { default as Document, DocumentChangeEvent } from './document';\nimport type { default as Model, ModelApplyOperationEvent } from './model';\nimport type { Marker, MarkerCollectionUpdateEvent } from './markercollection';\nimport type Batch from './batch';\nimport type Element from './element';\nimport type Item from './item';\nimport type { default as Position, PositionOffset } from './position';\nimport type Range from './range';\nimport type Schema from './schema';\n\nimport {\n\tCKEditorError,\n\tCollection,\n\tEmitterMixin,\n\ttoMap,\n\tuid\n} from '@ckeditor/ckeditor5-utils';\n\nconst storePrefix = 'selection:';\n\n/**\n * `DocumentSelection` is a special selection which is used as the\n * {@link module:engine/model/document~Document#selection document's selection}.\n * There can be only one instance of `DocumentSelection` per document.\n *\n * Document selection can only be changed by using the {@link module:engine/model/writer~Writer} instance\n * inside the {@link module:engine/model/model~Model#change `change()`} block, as it provides a secure way to modify model.\n *\n * `DocumentSelection` is automatically updated upon changes in the {@link module:engine/model/document~Document document}\n * to always contain valid ranges. Its attributes are inherited from the text unless set explicitly.\n *\n * Differences between {@link module:engine/model/selection~Selection} and `DocumentSelection` are:\n * * there is always a range in `DocumentSelection` - even if no ranges were added there is a \"default range\"\n * present in the selection,\n * * ranges added to this selection updates automatically when the document changes,\n * * attributes of `DocumentSelection` are updated automatically according to selection ranges.\n *\n * Since `DocumentSelection` uses {@link module:engine/model/liverange~LiveRange live ranges}\n * and is updated when {@link module:engine/model/document~Document document}\n * changes, it cannot be set on {@link module:engine/model/node~Node nodes}\n * that are inside {@link module:engine/model/documentfragment~DocumentFragment document fragment}.\n * If you need to represent a selection in document fragment,\n * use {@link module:engine/model/selection~Selection Selection class} instead.\n */\nexport default class DocumentSelection extends EmitterMixin( TypeCheckable ) {\n\t/**\n\t * Selection used internally by that class (`DocumentSelection` is a proxy to that selection).\n\t */\n\tprivate _selection: LiveSelection;\n\n\t/**\n\t * Creates an empty live selection for given {@link module:engine/model/document~Document}.\n\t *\n\t * @param doc Document which owns this selection.\n\t */\n\tconstructor( doc: Document ) {\n\t\tsuper();\n\n\t\tthis._selection = new LiveSelection( doc );\n\n\t\tthis._selection.delegate( 'change:range' ).to( this );\n\t\tthis._selection.delegate( 'change:attribute' ).to( this );\n\t\tthis._selection.delegate( 'change:marker' ).to( this );\n\t}\n\n\t/**\n\t * Describes whether the selection is collapsed. Selection is collapsed when there is exactly one range which is\n\t * collapsed.\n\t */\n\tpublic get isCollapsed(): boolean {\n\t\treturn this._selection.isCollapsed;\n\t}\n\n\t/**\n\t * Selection anchor. Anchor may be described as a position where the most recent part of the selection starts.\n\t * Together with {@link #focus} they define the direction of selection, which is important\n\t * when expanding/shrinking selection. Anchor is always {@link module:engine/model/range~Range#start start} or\n\t * {@link module:engine/model/range~Range#end end} position of the most recently added range.\n\t *\n\t * Is set to `null` if there are no ranges in selection.\n\t *\n\t * @see #focus\n\t */\n\tpublic get anchor(): Position | null {\n\t\treturn this._selection.anchor;\n\t}\n\n\t/**\n\t * Selection focus. Focus is a position where the selection ends.\n\t *\n\t * Is set to `null` if there are no ranges in selection.\n\t *\n\t * @see #anchor\n\t */\n\tpublic get focus(): Position | null {\n\t\treturn this._selection.focus;\n\t}\n\n\t/**\n\t * Number of ranges in selection.\n\t */\n\tpublic get rangeCount(): number {\n\t\treturn this._selection.rangeCount;\n\t}\n\n\t/**\n\t * Describes whether `Documentselection` has own range(s) set, or if it is defaulted to\n\t * {@link module:engine/model/document~Document#_getDefaultRange document's default range}.\n\t */\n\tpublic get hasOwnRange(): boolean {\n\t\treturn this._selection.hasOwnRange;\n\t}\n\n\t/**\n\t * Specifies whether the {@link #focus}\n\t * precedes {@link #anchor}.\n\t *\n\t * @readonly\n\t * @type {Boolean}\n\t */\n\tpublic get isBackward(): boolean {\n\t\treturn this._selection.isBackward;\n\t}\n\n\t/**\n\t * Describes whether the gravity is overridden (using {@link module:engine/model/writer~Writer#overrideSelectionGravity}) or not.\n\t *\n\t * Note that the gravity remains overridden as long as will not be restored the same number of times as it was overridden.\n\t */\n\tpublic get isGravityOverridden(): boolean {\n\t\treturn this._selection.isGravityOverridden;\n\t}\n\n\t/**\n\t * A collection of selection {@link module:engine/model/markercollection~Marker markers}.\n\t * Marker is a selection marker when selection range is inside the marker range.\n\t *\n\t * **Note**: Only markers from {@link ~DocumentSelection#observeMarkers observed markers groups} are collected.\n\t */\n\tpublic get markers(): Collection<Marker> {\n\t\treturn this._selection.markers;\n\t}\n\n\t/**\n\t * Used for the compatibility with the {@link module:engine/model/selection~Selection#isEqual} method.\n\t *\n\t * @internal\n\t */\n\tpublic get _ranges(): Array<Range> {\n\t\treturn this._selection._ranges;\n\t}\n\n\t/**\n\t * Returns an iterable that iterates over copies of selection ranges.\n\t */\n\tpublic getRanges(): IterableIterator<Range> {\n\t\treturn this._selection.getRanges();\n\t}\n\n\t/**\n\t * Returns the first position in the selection.\n\t * First position is the position that {@link module:engine/model/position~Position#isBefore is before}\n\t * any other position in the selection.\n\t *\n\t * Returns `null` if there are no ranges in selection.\n\t */\n\tpublic getFirstPosition(): Position | null {\n\t\treturn this._selection.getFirstPosition();\n\t}\n\n\t/**\n\t * Returns the last position in the selection.\n\t * Last position is the position that {@link module:engine/model/position~Position#isAfter is after}\n\t * any other position in the selection.\n\t *\n\t * Returns `null` if there are no ranges in selection.\n\t */\n\tpublic getLastPosition(): Position | null {\n\t\treturn this._selection.getLastPosition();\n\t}\n\n\t/**\n\t * Returns a copy of the first range in the selection.\n\t * First range is the one which {@link module:engine/model/range~Range#start start} position\n\t * {@link module:engine/model/position~Position#isBefore is before} start position of all other ranges\n\t * (not to confuse with the first range added to the selection).\n\t *\n\t * Returns `null` if there are no ranges in selection.\n\t */\n\tpublic getFirstRange(): Range | null {\n\t\treturn this._selection.getFirstRange();\n\t}\n\n\t/**\n\t * Returns a copy of the last range in the selection.\n\t * Last range is the one which {@link module:engine/model/range~Range#end end} position\n\t * {@link module:engine/model/position~Position#isAfter is after} end position of all other ranges (not to confuse with the range most\n\t * recently added to the selection).\n\t *\n\t * Returns `null` if there are no ranges in selection.\n\t */\n\tpublic getLastRange(): Range | null {\n\t\treturn this._selection.getLastRange();\n\t}\n\n\t/**\n\t * Gets elements of type {@link module:engine/model/schema~Schema#isBlock \"block\"} touched by the selection.\n\t *\n\t * This method's result can be used for example to apply block styling to all blocks covered by this selection.\n\t *\n\t * **Note:** `getSelectedBlocks()` returns blocks that are nested in other non-block elements\n\t * but will not return blocks nested in other blocks.\n\t *\n\t * In this case the function will return exactly all 3 paragraphs (note: `<blockQuote>` is not a block itself):\n\t *\n\t * ```\n\t * <paragraph>[a</paragraph>\n\t * <blockQuote>\n\t * \t<paragraph>b</paragraph>\n\t * </blockQuote>\n\t * <paragraph>c]d</paragraph>\n\t * ```\n\t *\n\t * In this case the paragraph will also be returned, despite the collapsed selection:\n\t *\n\t * ```\n\t * <paragraph>[]a</paragraph>\n\t * ```\n\t *\n\t * In such a scenario, however, only blocks A, B & E will be returned as blocks C & D are nested in block B:\n\t *\n\t * ```\n\t * [<blockA></blockA>\n\t * <blockB>\n\t * \t<blockC></blockC>\n\t * \t<blockD></blockD>\n\t * </blockB>\n\t * <blockE></blockE>]\n\t * ```\n\t *\n\t * If the selection is inside a block all the inner blocks (A & B) are returned:\n\t *\n\t * ```\n\t * <block>\n\t * \t<blockA>[a</blockA>\n\t * \t<blockB>b]</blockB>\n\t * </block>\n\t * ```\n\t *\n\t * **Special case**: If a selection ends at the beginning of a block, that block is not returned as from user perspective\n\t * this block wasn't selected. See [#984](https://github.com/ckeditor/ckeditor5-engine/issues/984) for more details.\n\t *\n\t * ```\n\t * <paragraph>[a</paragraph>\n\t * <paragraph>b</paragraph>\n\t * <paragraph>]c</paragraph> // this block will not be returned\n\t * ```\n\t */\n\tpublic getSelectedBlocks(): IterableIterator<Element> {\n\t\treturn this._selection.getSelectedBlocks();\n\t}\n\n\t/**\n\t * Returns the selected element. {@link module:engine/model/element~Element Element} is considered as selected if there is only\n\t * one range in the selection, and that range contains exactly one element.\n\t * Returns `null` if there is no selected element.\n\t */\n\tpublic getSelectedElement(): Element | null {\n\t\treturn this._selection.getSelectedElement();\n\t}\n\n\t/**\n\t * Checks whether the selection contains the entire content of the given element. This means that selection must start\n\t * at a position {@link module:engine/model/position~Position#isTouching touching} the element's start and ends at position\n\t * touching the element's end.\n\t *\n\t * By default, this method will check whether the entire content of the selection's current root is selected.\n\t * Useful to check if e.g. the user has just pressed <kbd>Ctrl</kbd> + <kbd>A</kbd>.\n\t */\n\tpublic containsEntireContent( element: Element ): boolean {\n\t\treturn this._selection.containsEntireContent( element );\n\t}\n\n\t/**\n\t * Unbinds all events previously bound by document selection.\n\t */\n\tpublic destroy(): void {\n\t\tthis._selection.destroy();\n\t}\n\n\t/**\n\t * Returns iterable that iterates over this selection's attribute keys.\n\t */\n\tpublic getAttributeKeys(): IterableIterator<string> {\n\t\treturn this._selection.getAttributeKeys();\n\t}\n\n\t/**\n\t * Returns iterable that iterates over this selection's attributes.\n\t *\n\t * Attributes are returned as arrays containing two items. First one is attribute key and second is attribute value.\n\t * This format is accepted by native `Map` object and also can be passed in `Node` constructor.\n\t */\n\tpublic getAttributes(): IterableIterator<[ string, unknown ]> {\n\t\treturn this._selection.getAttributes();\n\t}\n\n\t/**\n\t * Gets an attribute value for given key or `undefined` if that attribute is not set on the selection.\n\t *\n\t * @param key Key of attribute to look for.\n\t * @returns Attribute value or `undefined`.\n\t */\n\tpublic getAttribute( key: string ): unknown {\n\t\treturn this._selection.getAttribute( key );\n\t}\n\n\t/**\n\t * Checks if the selection has an attribute for given key.\n\t *\n\t * @param key Key of attribute to check.\n\t * @returns `true` if attribute with given key is set on selection, `false` otherwise.\n\t */\n\tpublic hasAttribute( key: string ): boolean {\n\t\treturn this._selection.hasAttribute( key );\n\t}\n\n\t/**\n\t * Refreshes selection attributes and markers according to the current position in the model.\n\t */\n\tpublic refresh(): void {\n\t\tthis._selection.updateMarkers();\n\t\tthis._selection._updateAttributes( false );\n\t}\n\n\t/**\n\t * Registers a marker group prefix or a marker name to be collected in the\n\t * {@link ~DocumentSelection#markers selection markers collection}.\n\t *\n\t * See also {@link module:engine/model/markercollection~MarkerCollection#getMarkersGroup `MarkerCollection#getMarkersGroup()`}.\n\t *\n\t * @param prefixOrName The marker group prefix or marker name.\n\t */\n\tpublic observeMarkers( prefixOrName: string ): void {\n\t\tthis._selection.observeMarkers( prefixOrName );\n\t}\n\n\t/**\n\t * Moves {@link module:engine/model/documentselection~DocumentSelection#focus} to the specified location.\n\t * Should be used only within the {@link module:engine/model/writer~Writer#setSelectionFocus} method.\n\t *\n\t * The location can be specified in the same form as\n\t * {@link module:engine/model/writer~Writer#createPositionAt writer.createPositionAt()} parameters.\n\t *\n\t * @see module:engine/model/writer~Writer#setSelectionFocus\n\t * @internal\n\t * @param offset Offset or one of the flags. Used only when\n\t * first parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic _setFocus( itemOrPosition: Item | Position, offset?: PositionOffset ): void {\n\t\tthis._selection.setFocus( itemOrPosition, offset );\n\t}\n\n\t/**\n\t * Sets this selection's ranges and direction to the specified location based on the given\n\t * {@link module:engine/model/selection~Selectable selectable}.\n\t * Should be used only within the {@link module:engine/model/writer~Writer#setSelection} method.\n\t *\n\t * @see module:engine/model/writer~Writer#setSelection\n\t * @internal\n\t */\n\tpublic _setTo( ...args: Parameters<Selection[ 'setTo' ]> ): void {\n\t\tthis._selection.setTo( ...args );\n\t}\n\n\t/**\n\t * Sets attribute on the selection. If attribute with the same key already is set, it's value is overwritten.\n\t * Should be used only within the {@link module:engine/model/writer~Writer#setSelectionAttribute} method.\n\t *\n\t * @see module:engine/model/writer~Writer#setSelectionAttribute\n\t * @internal\n\t * @param key Key of the attribute to set.\n\t * @param value Attribute value.\n\t */\n\tpublic _setAttribute( key: string, value: unknown ): void {\n\t\tthis._selection.setAttribute( key, value );\n\t}\n\n\t/**\n\t * Removes an attribute with given key from the selection.\n\t * If the given attribute was set on the selection, fires the {@link module:engine/model/selection~Selection#event:change:range}\n\t * event with removed attribute key.\n\t * Should be used only within the {@link module:engine/model/writer~Writer#removeSelectionAttribute} method.\n\t *\n\t * @see module:engine/model/writer~Writer#removeSelectionAttribute\n\t * @internal\n\t * @param key Key of the attribute to remove.\n\t */\n\tpublic _removeAttribute( key: string ): void {\n\t\tthis._selection.removeAttribute( key );\n\t}\n\n\t/**\n\t * Returns an iterable that iterates through all selection attributes stored in current selection's parent.\n\t *\n\t * @internal\n\t */\n\tpublic _getStoredAttributes(): Iterable<[ string, unknown ]> {\n\t\treturn this._selection.getStoredAttributes();\n\t}\n\n\t/**\n\t * Temporarily changes the gravity of the selection from the left to the right.\n\t *\n\t * The gravity defines from which direction the selection inherits its attributes. If it's the default left\n\t * gravity, the selection (after being moved by the the user) inherits attributes from its left hand side.\n\t * This method allows to temporarily override this behavior by forcing the gravity to the right.\n\t *\n\t * It returns an unique identifier which is required to restore the gravity. It guarantees the symmetry\n\t * of the process.\n\t *\n\t * @see module:engine/model/writer~Writer#overrideSelectionGravity\n\t * @internal\n\t * @returns The unique id which allows restoring the gravity.\n\t */\n\tpublic _overrideGravity(): string {\n\t\treturn this._selection.overrideGravity();\n\t}\n\n\t/**\n\t * Restores the {@link ~DocumentSelection#_overrideGravity overridden gravity}.\n\t *\n\t * Restoring the gravity is only possible using the unique identifier returned by\n\t * {@link ~DocumentSelection#_overrideGravity}. Note that the gravity remains overridden as long as won't be restored\n\t * the same number of times it was overridden.\n\t *\n\t * @see module:engine/model/writer~Writer#restoreSelectionGravity\n\t * @internal\n\t * @param uid The unique id returned by {@link #_overrideGravity}.\n\t */\n\tpublic _restoreGravity( uid: string ): void {\n\t\tthis._selection.restoreGravity( uid );\n\t}\n\n\t/**\n\t * Generates and returns an attribute key for selection attributes store, basing on original attribute key.\n\t *\n\t * @internal\n\t * @param key Attribute key to convert.\n\t * @returns Converted attribute key, applicable for selection store.\n\t */\n\tpublic static _getStoreAttributeKey( key: string ): string {\n\t\treturn storePrefix + key;\n\t}\n\n\t/**\n\t * Checks whether the given attribute key is an attribute stored on an element.\n\t *\n\t * @internal\n\t */\n\tpublic static _isStoreAttributeKey( key: string ): boolean {\n\t\treturn key.startsWith( storePrefix );\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nDocumentSelection.prototype.is = function( type: string ): boolean {\n\treturn type === 'selection' ||\n\t\ttype == 'model:selection' ||\n\t\ttype == 'documentSelection' ||\n\t\ttype == 'model:documentSelection';\n};\n\n/**\n * Fired when selection range(s) changed.\n *\n * @eventName ~DocumentSelection#change:range\n * @param directChange In case of {@link module:engine/model/selection~Selection} class it is always set\n * to `true` which indicates that the selection change was caused by a direct use of selection's API.\n * The {@link module:engine/model/documentselection~DocumentSelection}, however, may change because its position\n * was directly changed through the {@link module:engine/model/writer~Writer writer} or because its position was\n * changed because the structure of the model has been changed (which means an indirect change).\n * The indirect change does not occur in case of normal (detached) selections because they are \"static\" (as \"not live\")\n * which mean that they are not updated once the document changes.\n */\nexport type DocumentSelectionChangeRangeEvent = SelectionChangeRangeEvent;\n\n/**\n * Fired when selection attribute changed.\n *\n * @eventName ~DocumentSelection#change:attribute\n * @param directChange In case of {@link module:engine/model/selection~Selection} class it is always set\n * to `true` which indicates that the selection change was caused by a direct use of selection's API.\n * The {@link module:engine/model/documentselection~DocumentSelection}, however, may change because its attributes\n * were directly changed through the {@link module:engine/model/writer~Writer writer} or because its position was\n * changed in the model and its attributes were refreshed (which means an indirect change).\n * The indirect change does not occur in case of normal (detached) selections because they are \"static\" (as \"not live\")\n * which mean that they are not updated once the document changes.\n * @param attributeKeys Array containing keys of attributes that changed.\n*/\nexport type DocumentSelectionChangeAttributeEvent = SelectionChangeAttributeEvent;\n\n/**\n * Fired when selection marker(s) changed.\n *\n * @eventName ~DocumentSelection#change:marker\n * @param directChange This is always set to `false` in case of `change:marker` event as there is no possibility\n * to change markers directly through {@link module:engine/model/documentselection~DocumentSelection} API.\n * See also {@link module:engine/model/documentselection~DocumentSelection#event:change:range} and\n * {@link module:engine/model/documentselection~DocumentSelection#event:change:attribute}.\n * @param oldMarkers Markers in which the selection was before the change.\n */\nexport type DocumentSelectionChangeMarkerEvent = {\n\tname: 'change:marker';\n\targs: [ {\n\t\tdirectChange: boolean;\n\t\toldMarkers: Array<Marker>;\n\t} ];\n};\n\n/**\n * Fired when selection range(s), attribute(s) or marker(s) changed.\n *\n * @eventName ~DocumentSelection#change\n * @param directChange This is always set to `false` in case of `change:marker` event as there is no possibility\n * to change markers directly through {@link module:engine/model/documentselection~DocumentSelection} API.\n * See also {@link module:engine/model/documentselection~DocumentSelection#event:change:range} and\n * {@link module:engine/model/documentselection~DocumentSelection#event:change:attribute}.\n * @param attributeKeys Array containing keys of attributes that changed.\n * @param oldMarkers Markers in which the selection was before the change.\n */\nexport type DocumentSelectionChangeEvent = {\n\tname: 'change' | 'change:attribute' | 'change:marker' | 'change:range';\n\targs: [ {\n\t\tdirectChange: boolean;\n\t\tattributeKeys?: Array<string>;\n\t\toldMarkers?: Array<Marker>;\n\t} ];\n};\n\n/**\n * `LiveSelection` is used internally by {@link module:engine/model/documentselection~DocumentSelection} and shouldn't be used directly.\n *\n * LiveSelection` is automatically updated upon changes in the {@link module:engine/model/document~Document document}\n * to always contain valid ranges. Its attributes are inherited from the text unless set explicitly.\n *\n * Differences between {@link module:engine/model/selection~Selection} and `LiveSelection` are:\n * * there is always a range in `LiveSelection` - even if no ranges were added there is a \"default range\"\n * present in the selection,\n * * ranges added to this selection updates automatically when the document changes,\n * * attributes of `LiveSelection` are updated automatically according to selection ranges.\n */\nclass LiveSelection extends Selection {\n\t/**\n\t * List of selection markers.\n\t * Marker is a selection marker when selection range is inside the marker range.\n\t */\n\tpublic markers: Collection<Marker> = new Collection( { idProperty: 'name' } );\n\n\t/**\n\t * Document which owns this selection.\n\t */\n\tprivate _model: Model;\n\n\t/**\n\t * Document which owns this selection.\n\t */\n\tprivate _document: Document;\n\n\t/**\n\t * Stores selection ranges.\n\t *\n\t * @internal\n\t */\n\tpublic declare _ranges: Array<LiveRange>;\n\n\t/**\n\t * Keeps mapping of attribute name to priority with which the attribute got modified (added/changed/removed)\n\t * last time. Possible values of priority are: `'low'` and `'normal'`.\n\t *\n\t * Priorities are used by internal `LiveSelection` mechanisms. All attributes set using `LiveSelection`\n\t * attributes API are set with `'normal'` priority.\n\t */\n\tprivate _attributePriority: Map<string, 'low' | 'normal'> = new Map();\n\n\t/**\n\t * Position to which the selection should be set if the last selection range was moved to the graveyard.\n\t */\n\tprivate _selectionRestorePosition: Position | null = null;\n\n\t/**\n\t * Flag that informs whether the selection ranges have changed. It is changed on true when `LiveRange#change:range` event is fired.\n\t */\n\tprivate _hasChangedRange: boolean = false;\n\n\t/**\n\t * Each overriding gravity adds an UID to the set and each removal removes it.\n\t * Gravity is overridden when there's at least one UID in the set.\n\t * Gravity is restored when the set is empty.\n\t * This is to prevent conflicts when gravity is overridden by more than one feature at the same time.\n\t */\n\tprivate _overriddenGravityRegister: Set<string> = new Set();\n\n\t/**\n\t * Prefixes of marker names that should affect `LiveSelection#markers` collection.\n\t */\n\tprivate _observedMarkers: Set<string> = new Set();\n\n\t/**\n\t * Creates an empty live selection for given {@link module:engine/model/document~Document}.\n\t *\n\t * @param doc Document which owns this selection.\n\t */\n\tconstructor( doc: Document ) {\n\t\tsuper();\n\n\t\tthis._model = doc.model;\n\t\tthis._document = doc;\n\n\t\t// Ensure selection is correct after each operation.\n\t\tthis.listenTo<ModelApplyOperationEvent>( this._model, 'applyOperation', ( evt, args ) => {\n\t\t\tconst operation = args[ 0 ];\n\n\t\t\tif ( !operation.isDocumentOperation || operation.type == 'marker' || operation.type == 'rename' || operation.type == 'noop' ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Fix selection if the last range was removed from it and we have a position to which we can restore the selection.\n\t\t\tif ( this._ranges.length == 0 && this._selectionRestorePosition ) {\n\t\t\t\tthis._fixGraveyardSelection( this._selectionRestorePosition );\n\t\t\t}\n\n\t\t\t// \"Forget\" the restore position even if it was not \"used\".\n\t\t\tthis._selectionRestorePosition = null;\n\n\t\t\tif ( this._hasChangedRange ) {\n\t\t\t\tthis._hasChangedRange = false;\n\t\t\t\tthis.fire<DocumentSelectionChangeRangeEvent>( 'change:range', { directChange: false } );\n\t\t\t}\n\t\t}, { priority: 'lowest' } );\n\n\t\t// Ensure selection is correct and up to date after each range change.\n\t\tthis.on<DocumentSelectionChangeRangeEvent>( 'change:range', () => {\n\t\t\tthis._validateSelectionRanges( this.getRanges() );\n\t\t} );\n\n\t\t// Update markers data stored by the selection after each marker change.\n\t\t// This handles only marker changes done through marker operations (not model tree changes).\n\t\tthis.listenTo<MarkerCollectionUpdateEvent>( this._model.markers, 'update', ( evt, marker, oldRange, newRange ) => {\n\t\t\tthis._updateMarker( marker, newRange );\n\t\t} );\n\n\t\t// Ensure selection is up to date after each change block.\n\t\tthis.listenTo<DocumentChangeEvent>( this._document, 'change', ( evt, batch ) => {\n\t\t\tclearAttributesStoredInElement( this._model, batch );\n\t\t} );\n\t}\n\n\tpublic override get isCollapsed(): boolean {\n\t\tconst length = this._ranges.length;\n\n\t\treturn length === 0 ? this._document._getDefaultRange().isCollapsed : super.isCollapsed;\n\t}\n\n\tpublic override get anchor(): Position {\n\t\treturn super.anchor || this._document._getDefaultRange().start;\n\t}\n\n\tpublic override get focus(): Position {\n\t\treturn super.focus || this._document._getDefaultRange().end;\n\t}\n\n\tpublic override get rangeCount(): number {\n\t\treturn this._ranges.length ? this._ranges.length : 1;\n\t}\n\n\t/**\n\t * Describes whether `LiveSelection` has own range(s) set, or if it is defaulted to\n\t * {@link module:engine/model/document~Document#_getDefaultRange document's default range}.\n\t */\n\tpublic get hasOwnRange(): boolean {\n\t\treturn this._ranges.length > 0;\n\t}\n\n\t/**\n\t * When set to `true` then selection attributes on node before the caret won't be taken\n\t * into consideration while updating selection attributes.\n\t */\n\tpublic get isGravityOverridden(): boolean {\n\t\treturn !!this._overriddenGravityRegister.size;\n\t}\n\n\t/**\n\t * Unbinds all events previously bound by live selection.\n\t */\n\tpublic destroy(): void {\n\t\tfor ( let i = 0; i < this._ranges.length; i++ ) {\n\t\t\tthis._ranges[ i ].detach();\n\t\t}\n\n\t\tthis.stopListening();\n\t}\n\n\tpublic override* getRanges(): IterableIterator<Range> {\n\t\tif ( this._ranges.length ) {\n\t\t\tyield* super.getRanges();\n\t\t} else {\n\t\t\tyield this._document._getDefaultRange();\n\t\t}\n\t}\n\n\tpublic override getFirstRange(): Range {\n\t\treturn super.getFirstRange() || this._document._getDefaultRange();\n\t}\n\n\tpublic override getLastRange(): Range {\n\t\treturn super.getLastRange() || this._document._getDefaultRange();\n\t}\n\n\tpublic override setTo( ...args: Parameters<Selection[ 'setTo' ]> ) {\n\t\tsuper.setTo( ...args );\n\t\tthis._updateAttributes( true );\n\t\tthis.updateMarkers();\n\t}\n\n\tpublic override setFocus( itemOrPosition: Item | Position, offset?: PositionOffset ): void {\n\t\tsuper.setFocus( itemOrPosition, offset );\n\t\tthis._updateAttributes( true );\n\t\tthis.updateMarkers();\n\t}\n\n\tpublic override setAttribute( key: string, value: unknown ): void {\n\t\tif ( this._setAttribute( key, value ) ) {\n\t\t\t// Fire event with exact data.\n\t\t\tconst attributeKeys = [ key ];\n\t\t\tthis.fire<DocumentSelectionChangeAttributeEvent>( 'change:attribute', { attributeKeys, directChange: true } );\n\t\t}\n\t}\n\n\tpublic override removeAttribute( key: string ): void {\n\t\tif ( this._removeAttribute( key ) ) {\n\t\t\t// Fire event with exact data.\n\t\t\tconst attributeKeys = [ key ];\n\t\t\tthis.fire<DocumentSelectionChangeAttributeEvent>( 'change:attribute', { attributeKeys, directChange: true } );\n\t\t}\n\t}\n\n\tpublic overrideGravity(): string {\n\t\tconst overrideUid = uid();\n\n\t\t// Remember that another overriding has been requested. It will need to be removed\n\t\t// before the gravity is to be restored.\n\t\tthis._overriddenGravityRegister.add( overrideUid );\n\n\t\tif ( this._overriddenGravityRegister.size === 1 ) {\n\t\t\tthis._updateAttributes( true );\n\t\t}\n\n\t\treturn overrideUid;\n\t}\n\n\tpublic restoreGravity( uid: string ): void {\n\t\tif ( !this._overriddenGravityRegister.has( uid ) ) {\n\t\t\t/**\n\t\t\t * Restoring gravity for an unknown UID is not possible. Make sure you are using a correct\n\t\t\t * UID obtained from the {@link module:engine/model/writer~Writer#overrideSelectionGravity} to restore.\n\t\t\t *\n\t\t\t * @error document-selection-gravity-wrong-restore\n\t\t\t * @param uid The unique identifier returned by\n\t\t\t * {@link module:engine/model/documentselection~DocumentSelection#_overrideGravity}.\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'document-selection-gravity-wrong-restore',\n\t\t\t\tthis,\n\t\t\t\t{ uid }\n\t\t\t);\n\t\t}\n\n\t\tthis._overriddenGravityRegister.delete( uid );\n\n\t\t// Restore gravity only when all overriding have been restored.\n\t\tif ( !this.isGravityOverridden ) {\n\t\t\tthis._updateAttributes( true );\n\t\t}\n\t}\n\n\tpublic observeMarkers( prefixOrName: string ): void {\n\t\tthis._observedMarkers.add( prefixOrName );\n\t\tthis.updateMarkers();\n\t}\n\n\tprotected override _replaceAllRanges( ranges: Array<Range> ): void {\n\t\tthis._validateSelectionRanges( ranges );\n\n\t\tsuper._replaceAllRanges( ranges );\n\t}\n\n\tprotected override _popRange(): void {\n\t\tthis._ranges.pop()!.detach();\n\t}\n\n\tprotected override _pushRange( range: Range ): void {\n\t\tconst liveRange = this._prepareRange( range );\n\n\t\t// `undefined` is returned when given `range` is in graveyard root.\n\t\tif ( liveRange ) {\n\t\t\tthis._ranges.push( liveRange );\n\t\t}\n\t}\n\n\tprivate _validateSelectionRanges( ranges: Iterable<Range> ) {\n\t\tfor ( const range of ranges ) {\n\t\t\tif ( !this._document._validateSelectionRange( range ) ) {\n\t\t\t\t/**\n\t\t\t\t * Range from {@link module:engine/model/documentselection~DocumentSelection document selection}\n\t\t\t\t * starts or ends at incorrect position.\n\t\t\t\t *\n\t\t\t\t * @error document-selection-wrong-position\n\t\t\t\t * @param range\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'document-selection-wrong-position',\n\t\t\t\t\tthis,\n\t\t\t\t\t{ range }\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Prepares given range to be added to selection. Checks if it is correct,\n\t * converts it to {@link module:engine/model/liverange~LiveRange LiveRange}\n\t * and sets listeners listening to the range's change event.\n\t */\n\tprivate _prepareRange( range: Range ): LiveRange | undefined {\n\t\tthis._checkRange( range );\n\n\t\tif ( range.root == this._document.graveyard ) {\n\t\t\t// @if CK_DEBUG // console.warn( 'Trying to add a Range that is in the graveyard root. Range rejected.' );\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst liveRange = LiveRange.fromRange( range );\n\n\t\t// If selection range is moved to the graveyard remove it from the selection object.\n\t\t// Also, save some data that can be used to restore selection later, on `Model#applyOperation` event.\n\t\tliveRange.on( 'change:range', ( evt, oldRange, data ) => {\n\t\t\tthis._hasChangedRange = true;\n\n\t\t\tif ( liveRange.root == this._document.graveyard ) {\n\t\t\t\tthis._selectionRestorePosition = data.deletionPosition;\n\n\t\t\t\tconst index = this._ranges.indexOf( liveRange );\n\t\t\t\tthis._ranges.splice( index, 1 );\n\t\t\t\tliveRange.detach();\n\t\t\t}\n\t\t} );\n\n\t\treturn liveRange;\n\t}\n\n\tpublic updateMarkers(): void {\n\t\tif ( !this._observedMarkers.size ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst markers = [];\n\t\tlet changed = false;\n\n\t\tfor ( const marker of this._model.markers ) {\n\t\t\tconst markerGroup = marker.name.split( ':', 1 )[ 0 ];\n\n\t\t\tif ( !this._observedMarkers.has( markerGroup ) ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst markerRange = marker.getRange();\n\n\t\t\tfor ( const selectionRange of this.getRanges() ) {\n\t\t\t\tif ( markerRange.containsRange( selectionRange, !selectionRange.isCollapsed ) ) {\n\t\t\t\t\tmarkers.push( marker );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tconst oldMarkers = Array.from( this.markers );\n\n\t\tfor ( const marker of markers ) {\n\t\t\tif ( !this.markers.has( marker ) ) {\n\t\t\t\tthis.markers.add( marker );\n\n\t\t\t\tchanged = true;\n\t\t\t}\n\t\t}\n\n\t\tfor ( const marker of Array.from( this.markers ) ) {\n\t\t\tif ( !markers.includes( marker ) ) {\n\t\t\t\tthis.markers.remove( marker );\n\n\t\t\t\tchanged = true;\n\t\t\t}\n\t\t}\n\n\t\tif ( changed ) {\n\t\t\tthis.fire<DocumentSelectionChangeMarkerEvent>( 'change:marker', { oldMarkers, directChange: false } );\n\t\t}\n\t}\n\n\tprivate _updateMarker( marker: Marker, markerRange: Range | null ): void {\n\t\tconst markerGroup = marker.name.split( ':', 1 )[ 0 ];\n\n\t\tif ( !this._observedMarkers.has( markerGroup ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tlet changed = false;\n\n\t\tconst oldMarkers = Array.from( this.markers );\n\t\tconst hasMarker = this.markers.has( marker );\n\n\t\tif ( !markerRange ) {\n\t\t\tif ( hasMarker ) {\n\t\t\t\tthis.markers.remove( marker );\n\t\t\t\tchanged = true;\n\t\t\t}\n\t\t} else {\n\t\t\tlet contained = false;\n\n\t\t\tfor ( const selectionRange of this.getRanges() ) {\n\t\t\t\tif ( markerRange.containsRange( selectionRange, !selectionRange.isCollapsed ) ) {\n\t\t\t\t\tcontained = true;\n\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( contained && !hasMarker ) {\n\t\t\t\tthis.markers.add( marker );\n\n\t\t\t\tchanged = true;\n\t\t\t} else if ( !contained && hasMarker ) {\n\t\t\t\tthis.markers.remove( marker );\n\n\t\t\t\tchanged = true;\n\t\t\t}\n\t\t}\n\n\t\tif ( changed ) {\n\t\t\tthis.fire<DocumentSelectionChangeMarkerEvent>( 'change:marker', { oldMarkers, directChange: false } );\n\t\t}\n\t}\n\n\t/**\n\t * Updates this selection attributes according to its ranges and the {@link module:engine/model/document~Document model document}.\n\t */\n\tpublic _updateAttributes( clearAll: boolean ): void {\n\t\tconst newAttributes = toMap( this._getSurroundingAttributes() );\n\t\tconst oldAttributes = toMap( this.getAttributes() );\n\n\t\tif ( clearAll ) {\n\t\t\t// If `clearAll` remove all attributes and reset priorities.\n\t\t\tthis._attributePriority = new Map();\n\t\t\tthis._attrs = new Map();\n\t\t} else {\n\t\t\t// If not, remove only attributes added with `low` priority.\n\t\t\tfor ( const [ key, priority ] of this._attributePriority ) {\n\t\t\t\tif ( priority == 'low' ) {\n\t\t\t\t\tthis._attrs.delete( key );\n\t\t\t\t\tthis._attributePriority.delete( key );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tthis._setAttributesTo( newAttributes );\n\n\t\t// Let's evaluate which attributes really changed.\n\t\tconst changed = [];\n\n\t\t// First, loop through all attributes that are set on selection right now.\n\t\t// Check which of them are different than old attributes.\n\t\tfor ( const [ newKey, newValue ] of this.getAttributes() ) {\n\t\t\tif ( !oldAttributes.has( newKey ) || oldAttributes.get( newKey ) !== newValue ) {\n\t\t\t\tchanged.push( newKey );\n\t\t\t}\n\t\t}\n\n\t\t// Then, check which of old attributes got removed.\n\t\tfor ( const [ oldKey ] of oldAttributes ) {\n\t\t\tif ( !this.hasAttribute( oldKey ) ) {\n\t\t\t\tchanged.push( oldKey );\n\t\t\t}\n\t\t}\n\n\t\t// Fire event with exact data (fire only if anything changed).\n\t\tif ( changed.length > 0 ) {\n\t\t\tthis.fire<DocumentSelectionChangeAttributeEvent>( 'change:attribute', { attributeKeys: changed, directChange: false } );\n\t\t}\n\t}\n\n\t/**\n\t * Internal method for setting `LiveSelection` attribute. Supports attribute priorities (through `directChange`\n\t * parameter).\n\t */\n\tprivate _setAttribute( key: string, value: unknown, directChange: boolean = true ): boolean {\n\t\tconst priority = directChange ? 'normal' : 'low';\n\n\t\tif ( priority == 'low' && this._attributePriority.get( key ) == 'normal' ) {\n\t\t\t// Priority too low.\n\t\t\treturn false;\n\t\t}\n\n\t\tconst oldValue = super.getAttribute( key );\n\n\t\t// Don't do anything if value has not changed.\n\t\tif ( oldValue === value ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tthis._attrs.set( key, value );\n\n\t\t// Update priorities map.\n\t\tthis._attributePriority.set( key, priority );\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Internal method for removing `LiveSelection` attribute. Supports attribute priorities (through `directChange`\n\t * parameter).\n\t *\n\t * NOTE: Even if attribute is not present in the selection but is provided to this method, it's priority will\n\t * be changed according to `directChange` parameter.\n\t */\n\tprivate _removeAttribute( key: string, directChange: boolean = true ): boolean {\n\t\tconst priority = directChange ? 'normal' : 'low';\n\n\t\tif ( priority == 'low' && this._attributePriority.get( key ) == 'normal' ) {\n\t\t\t// Priority too low.\n\t\t\treturn false;\n\t\t}\n\n\t\t// Update priorities map.\n\t\tthis._attributePriority.set( key, priority );\n\n\t\t// Don't do anything if value has not changed.\n\t\tif ( !super.hasAttribute( key ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tthis._attrs.delete( key );\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Internal method for setting multiple `LiveSelection` attributes. Supports attribute priorities (through\n\t * `directChange` parameter).\n\t */\n\tprivate _setAttributesTo( attrs: Map<string, unknown> ): Set<string> {\n\t\tconst changed = new Set<string>();\n\n\t\tfor ( const [ oldKey, oldValue ] of this.getAttributes() ) {\n\t\t\t// Do not remove attribute if attribute with same key and value is about to be set.\n\t\t\tif ( attrs.get( oldKey ) === oldValue ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// All rest attributes will be removed so changed attributes won't change .\n\t\t\tthis._removeAttribute( oldKey, false );\n\t\t}\n\n\t\tfor ( const [ key, value ] of attrs ) {\n\t\t\t// Attribute may not be set because of attributes or because same key/value is already added.\n\t\t\tconst gotAdded = this._setAttribute( key, value, false );\n\n\t\t\tif ( gotAdded ) {\n\t\t\t\tchanged.add( key );\n\t\t\t}\n\t\t}\n\n\t\treturn changed;\n\t}\n\n\t/**\n\t * Returns an iterable that iterates through all selection attributes stored in current selection's parent.\n\t */\n\tpublic* getStoredAttributes(): IterableIterator<[ string, unknown ]> {\n\t\tconst selectionParent = this.getFirstPosition()!.parent as Element;\n\n\t\tif ( this.isCollapsed && selectionParent.isEmpty ) {\n\t\t\tfor ( const key of selectionParent.getAttributeKeys() ) {\n\t\t\t\tif ( key.startsWith( storePrefix ) ) {\n\t\t\t\t\tconst realKey = key.substr( storePrefix.length );\n\n\t\t\t\t\tyield [ realKey, selectionParent.getAttribute( key ) ];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Checks model text nodes that are closest to the selection's first position and returns attributes of first\n\t * found element. If there are no text nodes in selection's first position parent, it returns selection\n\t * attributes stored in that parent.\n\t */\n\tprivate _getSurroundingAttributes(): Iterable<[ string, unknown ]> | null {\n\t\tconst position = this.getFirstPosition()!;\n\t\tconst schema = this._model.schema;\n\n\t\tif ( position.root.rootName == '$graveyard' ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tlet attrs = null;\n\n\t\tif ( !this.isCollapsed ) {\n\t\t\t// 1. If selection is a range...\n\t\t\tconst range = this.getFirstRange();\n\n\t\t\t// ...look for a first character node in that range and take attributes from it.\n\t\t\tfor ( const value of range ) {\n\t\t\t\t// If the item is an object, we don't want to get attributes from its children...\n\t\t\t\tif ( value.item.is( 'element' ) && schema.isObject( value.item ) ) {\n\t\t\t\t\t// ...but collect attributes from inline object.\n\t\t\t\t\tattrs = getTextAttributes( value.item, schema );\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tif ( value.type == 'text' ) {\n\t\t\t\t\tattrs = value.item.getAttributes();\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\t// 2. If the selection is a caret or the range does not contain a character node...\n\n\t\t\tconst nodeBefore = position.textNode ? position.textNode : position.nodeBefore;\n\t\t\tconst nodeAfter = position.textNode ? position.textNode : position.nodeAfter;\n\n\t\t\t// When gravity is overridden then don't take node before into consideration.\n\t\t\tif ( !this.isGravityOverridden ) {\n\t\t\t\t// ...look at the node before caret and take attributes from it if it is a character node.\n\t\t\t\tattrs = getTextAttributes( nodeBefore, schema );\n\t\t\t}\n\n\t\t\t// 3. If not, look at the node after caret...\n\t\t\tif ( !attrs ) {\n\t\t\t\tattrs = getTextAttributes( nodeAfter, schema );\n\t\t\t}\n\n\t\t\t// 4. If not, try to find the first character on the left, that is in the same node.\n\t\t\t// When gravity is overridden then don't take node before into consideration.\n\t\t\tif ( !this.isGravityOverridden && !attrs ) {\n\t\t\t\tlet node = nodeBefore;\n\n\t\t\t\twhile ( node && !attrs ) {\n\t\t\t\t\tnode = node.previousSibling;\n\t\t\t\t\tattrs = getTextAttributes( node, schema );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// 5. If not found, try to find the first character on the right, that is in the same node.\n\t\t\tif ( !attrs ) {\n\t\t\t\tlet node = nodeAfter;\n\n\t\t\t\twhile ( node && !attrs ) {\n\t\t\t\t\tnode = node.nextSibling;\n\t\t\t\t\tattrs = getTextAttributes( node, schema );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// 6. If not found, selection should retrieve attributes from parent.\n\t\t\tif ( !attrs ) {\n\t\t\t\tattrs = this.getStoredAttributes();\n\t\t\t}\n\t\t}\n\n\t\treturn attrs;\n\t}\n\n\t/**\n\t * Fixes the selection after all its ranges got removed.\n\t * @param deletionPosition Position where the deletion happened.\n\t */\n\tprivate _fixGraveyardSelection( deletionPosition: Position ): void {\n\t\t// Find a range that is a correct selection range and is closest to the position where the deletion happened.\n\t\tconst selectionRange = this._model.schema.getNearestSelectionRange( deletionPosition );\n\n\t\t// If nearest valid selection range has been found - add it in the place of old range.\n\t\tif ( selectionRange ) {\n\t\t\t// Check the range, convert it to live range, bind events, etc.\n\t\t\tthis._pushRange( selectionRange );\n\t\t}\n\t\t// If nearest valid selection range cannot be found don't add any range. Selection will be set to the default range.\n\t}\n}\n\n/**\n * Helper function for {@link module:engine/model/liveselection~LiveSelection#_updateAttributes}.\n *\n * It checks if the passed model item is a text node (or text proxy) and, if so, returns it's attributes.\n * If not, it checks if item is an inline object and does the same. Otherwise it returns `null`.\n */\nfunction getTextAttributes( node: Item | null, schema: Schema ): Iterable<[string, unknown]> | null {\n\tif ( !node ) {\n\t\treturn null;\n\t}\n\n\tif ( node instanceof TextProxy || node instanceof Text ) {\n\t\treturn node.getAttributes();\n\t}\n\n\tif ( !schema.isInline( node ) ) {\n\t\treturn null;\n\t}\n\n\t// Stop on inline elements (such as `<softBreak>`) that are not objects (such as `<imageInline>` or `<mathml>`).\n\tif ( !schema.isObject( node ) ) {\n\t\treturn [];\n\t}\n\n\tconst attributes: Array<[string, unknown]> = [];\n\n\t// Collect all attributes that can be applied to the text node.\n\tfor ( const [ key, value ] of node.getAttributes() ) {\n\t\tif (\n\t\t\tschema.checkAttribute( '$text', key ) &&\n\t\t\tschema.getAttributeProperties( key ).copyFromObject !== false\n\t\t) {\n\t\t\tattributes.push( [ key, value ] );\n\t\t}\n\t}\n\n\treturn attributes;\n}\n\n/**\n * Removes selection attributes from element which is not empty anymore.\n */\nfunction clearAttributesStoredInElement( model: Model, batch: Batch ) {\n\tconst differ = model.document.differ;\n\n\tfor ( const entry of differ.getChanges() ) {\n\t\tif ( entry.type != 'insert' ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst changeParent = entry.position.parent as Element;\n\t\tconst isNoLongerEmpty = entry.length === changeParent.maxOffset;\n\n\t\tif ( isNoLongerEmpty ) {\n\t\t\tmodel.enqueueChange( batch, writer => {\n\t\t\t\tconst storedAttributes = Array.from( changeParent.getAttributeKeys() )\n\t\t\t\t\t.filter( key => key.startsWith( storePrefix ) );\n\n\t\t\t\tfor ( const key of storedAttributes ) {\n\t\t\t\t\twriter.removeAttribute( key, changeParent );\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/conversion/conversionhelpers\n */\n\n/**\n * Base class for conversion helpers.\n */\nexport default class ConversionHelpers<TDispatcher> {\n\tprivate readonly _dispatchers: Array<TDispatcher>;\n\n\t/**\n\t * Creates a conversion helpers instance.\n\t */\n\tconstructor( dispatchers: Array<TDispatcher> ) {\n\t\tthis._dispatchers = dispatchers;\n\t}\n\n\t/**\n\t * Registers a conversion helper.\n\t *\n\t * **Note**: See full usage example in the `{@link module:engine/conversion/conversion~Conversion#for conversion.for()}`\n\t * method description.\n\t *\n\t * @param conversionHelper The function to be called on event.\n\t */\n\tpublic add( conversionHelper: ( dispatcher: TDispatcher ) => void ): this {\n\t\tfor ( const dispatcher of this._dispatchers ) {\n\t\t\tconversionHelper( dispatcher );\n\t\t}\n\n\t\treturn this;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * Contains downcast (model-to-view) converters for {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher}.\n *\n * @module engine/conversion/downcasthelpers\n */\n\nimport ModelRange from '../model/range';\nimport ModelSelection from '../model/selection';\nimport ModelDocumentSelection from '../model/documentselection';\nimport ModelElement from '../model/element';\nimport ModelPosition from '../model/position';\n\nimport ViewAttributeElement from '../view/attributeelement';\nimport ConversionHelpers from './conversionhelpers';\n\nimport type {\n\tdefault as DowncastDispatcher,\n\tDiffItemReinsert,\n\tDowncastConversionApi,\n\tDowncastInsertEvent,\n\tDowncastAddMarkerEvent,\n\tDowncastAttributeEvent,\n\tDowncastReduceChangesEvent,\n\tDowncastRemoveMarkerEvent\n} from './downcastdispatcher';\nimport type ModelConsumable from './modelconsumable';\nimport type { DiffItem } from '../model/differ';\nimport type ModelNode from '../model/node';\nimport type ModelItem from '../model/item';\nimport type ModelTextProxy from '../model/textproxy';\nimport type ModelText from '../model/text';\n\nimport type DowncastWriter from '../view/downcastwriter';\nimport type ElementDefinition from '../view/elementdefinition';\nimport type ViewDocumentFragment from '../view/documentfragment';\nimport type UIElement from '../view/uielement';\nimport type ViewElement from '../view/element';\nimport type ViewNode from '../view/node';\nimport type ViewPosition from '../view/position';\nimport type ViewRange from '../view/range';\nimport type {\n\tdefault as Mapper,\n\tMapperModelToViewPositionEvent\n} from './mapper';\n\nimport {\n\tCKEditorError,\n\ttoArray,\n\ttype EventInfo,\n\ttype PriorityString\n} from '@ckeditor/ckeditor5-utils';\n\nimport { cloneDeep } from 'lodash-es';\n\n/**\n * Downcast conversion helper functions.\n *\n * Learn more about {@glink framework/deep-dive/conversion/downcast downcast helpers}.\n *\n * @extends module:engine/conversion/conversionhelpers~ConversionHelpers\n */\nexport default class DowncastHelpers extends ConversionHelpers<DowncastDispatcher> {\n\t/**\n\t * Model element to view element conversion helper.\n\t *\n\t * This conversion results in creating a view element. For example, model `<paragraph>Foo</paragraph>` becomes `<p>Foo</p>` in the view.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).elementToElement( {\n\t * \tmodel: 'paragraph',\n\t * \tview: 'p'\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).elementToElement( {\n\t * \tmodel: 'paragraph',\n\t * \tview: 'div',\n\t * \tconverterPriority: 'high'\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).elementToElement( {\n\t * \tmodel: 'fancyParagraph',\n\t * \tview: {\n\t * \t\tname: 'p',\n\t * \t\tclasses: 'fancy'\n\t * \t}\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).elementToElement( {\n\t * \tmodel: 'heading',\n\t * \tview: ( modelElement, conversionApi ) => {\n\t * \t\tconst { writer } = conversionApi;\n\t *\n\t * \t\treturn writer.createContainerElement( 'h' + modelElement.getAttribute( 'level' ) );\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * The element-to-element conversion supports the reconversion mechanism. It can be enabled by using either the `attributes` or\n\t * the `children` props on a model description. You will find a couple examples below.\n\t *\n\t * In order to reconvert an element if any of its direct children have been added or removed, use the `children` property on a `model`\n\t * description. For example, this model:\n\t *\n\t * ```xml\n\t * <box>\n\t * \t<paragraph>Some text.</paragraph>\n\t * </box>\n\t * ```\n\t *\n\t * will be converted into this structure in the view:\n\t *\n\t * ```html\n\t * <div class=\"box\" data-type=\"single\">\n\t * \t<p>Some text.</p>\n\t * </div>\n\t * ```\n\t *\n\t * But if more items were inserted in the model:\n\t *\n\t * ```xml\n\t * <box>\n\t * \t<paragraph>Some text.</paragraph>\n\t * \t<paragraph>Other item.</paragraph>\n\t * </box>\n\t * ```\n\t *\n\t * it will be converted into this structure in the view (note the element `data-type` change):\n\t *\n\t * ```html\n\t * <div class=\"box\" data-type=\"multiple\">\n\t * \t<p>Some text.</p>\n\t * \t<p>Other item.</p>\n\t * </div>\n\t * ```\n\t *\n\t * Such a converter would look like this (note that the `paragraph` elements are converted separately):\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).elementToElement( {\n\t * \tmodel: {\n\t * \t\tname: 'box',\n\t * \t\tchildren: true\n\t * \t},\n\t * \tview: ( modelElement, conversionApi ) => {\n\t * \t\tconst { writer } = conversionApi;\n\t *\n\t * \t\treturn writer.createContainerElement( 'div', {\n\t * \t\t\tclass: 'box',\n\t * \t\t\t'data-type': modelElement.childCount == 1 ? 'single' : 'multiple'\n\t * \t\t} );\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * In order to reconvert element if any of its attributes have been updated, use the `attributes` property on a `model`\n\t * description. For example, this model:\n\t *\n\t * ```xml\n\t * <heading level=\"2\">Some text.</heading>\n\t * ```\n\t *\n\t * will be converted into this structure in the view:\n\t *\n\t * ```html\n\t * <h2>Some text.</h2>\n\t * ```\n\t *\n\t * But if the `heading` element's `level` attribute has been updated to `3` for example, then\n\t * it will be converted into this structure in the view:\n\t *\n\t * ```html\n\t * <h3>Some text.</h3>\n\t * ```\n\t *\n\t * Such a converter would look as follows:\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).elementToElement( {\n\t * \tmodel: {\n\t * \t\tname: 'heading',\n\t * \t\tattributes: 'level'\n\t * \t},\n\t * \tview: ( modelElement, conversionApi ) => {\n\t * \t\tconst { writer } = conversionApi;\n\t *\n\t * \t\treturn writer.createContainerElement( 'h' + modelElement.getAttribute( 'level' ) );\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * You can read more about the element-to-element conversion in the\n\t * {@glink framework/deep-dive/conversion/downcast downcast conversion} guide.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.model The description or a name of the model element to convert.\n\t * @param config.model.attributes The list of attribute names that should be consumed while creating\n\t * the view element. Note that the view will be reconverted if any of the listed attributes changes.\n \t * @param config.model.children Specifies whether the view element requires reconversion if the list\n\t * of the model child nodes changed.\n\t * @param config.view A view element definition or a function that takes the model element and\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API}\n\t * as parameters and returns a view container element.\n\t */\n\tpublic elementToElement( config: {\n\t\tmodel: string | {\n\t\t\tname: string;\n\t\t\tattributes?: string | Array<string>;\n\t\t\tchildren?: boolean;\n\t\t};\n\t\tview: ElementDefinition | ElementCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( downcastElementToElement( config ) );\n\t}\n\n\t/**\n\t * The model element to view structure (several elements) conversion helper.\n\t *\n\t * This conversion results in creating a view structure with one or more slots defined for the child nodes.\n\t * For example, a model `<table>` may become this structure in the view:\n\t *\n\t * ```html\n\t * <figure class=\"table\">\n\t * \t<table>\n\t * \t\t<tbody>${ slot for table rows }</tbody>\n\t * \t</table>\n\t * </figure>\n\t * ```\n\t *\n\t * The children of the model's `<table>` element will be inserted into the `<tbody>` element.\n\t * If the `elementToElement()` helper was used, the children would be inserted into the `<figure>`.\n\t *\n\t * An example converter that converts the following model structure:\n\t *\n\t * ```xml\n\t * <wrappedParagraph>Some text.</wrappedParagraph>\n\t * ```\n\t *\n\t * into this structure in the view:\n\t *\n\t * ```html\n\t * <div class=\"wrapper\">\n\t * \t<p>Some text.</p>\n\t * </div>\n\t * ```\n\t *\n\t * would look like this:\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).elementToStructure( {\n\t * \tmodel: 'wrappedParagraph',\n\t * \tview: ( modelElement, conversionApi ) => {\n\t * \t\tconst { writer } = conversionApi;\n\t *\n\t * \t\tconst wrapperViewElement = writer.createContainerElement( 'div', { class: 'wrapper' } );\n\t * \t\tconst paragraphViewElement = writer.createContainerElement( 'p' );\n\t *\n\t * \t\twriter.insert( writer.createPositionAt( wrapperViewElement, 0 ), paragraphViewElement );\n\t * \t\twriter.insert( writer.createPositionAt( paragraphViewElement, 0 ), writer.createSlot() );\n\t *\n\t * \t\treturn wrapperViewElement;\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * The `createSlot()` function can also take a callback that allows filtering which children of the model element\n\t * should be converted into this slot.\n\t *\n\t * Imagine a table feature where for this model structure:\n\t *\n\t * ```xml\n\t * <table headingRows=\"1\">\n\t * \t<tableRow> ... table cells 1 ... </tableRow>\n\t * \t<tableRow> ... table cells 2 ... </tableRow>\n\t * \t<tableRow> ... table cells 3 ... </tableRow>\n\t * \t<caption>Caption text</caption>\n\t * </table>\n\t * ```\n\t *\n\t * we want to generate this view structure:\n\t *\n\t * ```html\n\t * <figure class=\"table\">\n\t * \t<table>\n\t * \t\t<thead>\n\t * \t\t\t<tr> ... table cells 1 ... </tr>\n\t * \t\t</thead>\n\t * \t\t<tbody>\n\t * \t\t\t<tr> ... table cells 2 ... </tr>\n\t * \t\t\t<tr> ... table cells 3 ... </tr>\n\t * \t\t</tbody>\n\t * \t</table>\n\t * \t<figcaption>Caption text</figcaption>\n\t * </figure>\n\t * ```\n\t *\n\t * The converter has to take the `headingRows` attribute into consideration when allocating the `<tableRow>` elements\n\t * into the `<tbody>` and `<thead>` elements. Hence, we need two slots and need to define proper filter callbacks for them.\n\t *\n\t * Additionally, all elements other than `<tableRow>` should be placed outside the `<table>` tag.\n\t * In the example above, this will handle the table caption.\n\t *\n\t * Such a converter would look like this:\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).elementToStructure( {\n\t * \tmodel: {\n\t * \t\tname: 'table',\n\t * \t\tattributes: [ 'headingRows' ]\n\t * \t},\n\t * \tview: ( modelElement, conversionApi ) => {\n\t * \t\tconst { writer } = conversionApi;\n\t *\n\t * \t\tconst figureElement = writer.createContainerElement( 'figure', { class: 'table' } );\n\t * \t\tconst tableElement = writer.createContainerElement( 'table' );\n\t *\n\t * \t\twriter.insert( writer.createPositionAt( figureElement, 0 ), tableElement );\n\t *\n\t * \t\tconst headingRows = modelElement.getAttribute( 'headingRows' ) || 0;\n\t *\n\t * \t\tif ( headingRows > 0 ) {\n\t * \t\t\tconst tableHead = writer.createContainerElement( 'thead' );\n\t *\n\t * \t\t\tconst headSlot = writer.createSlot( node => node.is( 'element', 'tableRow' ) && node.index < headingRows );\n\t *\n\t * \t\t\twriter.insert( writer.createPositionAt( tableElement, 'end' ), tableHead );\n\t * \t\t\twriter.insert( writer.createPositionAt( tableHead, 0 ), headSlot );\n\t * \t\t}\n\t *\n\t * \t\tif ( headingRows < tableUtils.getRows( table ) ) {\n\t * \t\t\tconst tableBody = writer.createContainerElement( 'tbody' );\n\t *\n\t * \t\t\tconst bodySlot = writer.createSlot( node => node.is( 'element', 'tableRow' ) && node.index >= headingRows );\n\t *\n\t * \t\t\twriter.insert( writer.createPositionAt( tableElement, 'end' ), tableBody );\n\t * \t\t\twriter.insert( writer.createPositionAt( tableBody, 0 ), bodySlot );\n\t * \t\t}\n\t *\n\t * \t\tconst restSlot = writer.createSlot( node => !node.is( 'element', 'tableRow' ) );\n\t *\n\t * \t\twriter.insert( writer.createPositionAt( figureElement, 'end' ), restSlot );\n\t *\n\t * \t\treturn figureElement;\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * Note: The children of a model element that's being converted must be allocated in the same order in the view\n\t * in which they are placed in the model.\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n \t * @param config.model The description or a name of the model element to convert.\n\t * @param config.model.name The name of the model element to convert.\n \t * @param config.model.attributes The list of attribute names that should be consumed while creating\n\t * the view structure. Note that the view will be reconverted if any of the listed attributes will change.\n\t * @param config.view A function that takes the model element and\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API} as parameters\n\t * and returns a view container element with slots for model child nodes to be converted into.\n\t */\n\tpublic elementToStructure( config: {\n\t\tmodel: string | {\n\t\t\tname: string;\n\t\t\tattributes?: string | Array<string>;\n\t\t};\n\t\tview: StructureCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( downcastElementToStructure( config ) );\n\t}\n\n\t/**\n\t * Model attribute to view element conversion helper.\n\t *\n\t * This conversion results in wrapping view nodes with a view attribute element. For example, a model text node with\n\t * `\"Foo\"` as data and the `bold` attribute becomes `<strong>Foo</strong>` in the view.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).attributeToElement( {\n\t * \tmodel: 'bold',\n\t * \tview: 'strong'\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToElement( {\n\t * \tmodel: 'bold',\n\t * \tview: 'b',\n\t * \tconverterPriority: 'high'\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToElement( {\n\t * \tmodel: 'invert',\n\t * \tview: {\n\t * \t\tname: 'span',\n\t * \t\tclasses: [ 'font-light', 'bg-dark' ]\n\t * \t}\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToElement( {\n\t * \tmodel: {\n\t * \t\tkey: 'fontSize',\n\t * \t\tvalues: [ 'big', 'small' ]\n\t * \t},\n\t * \tview: {\n\t * \t\tbig: {\n\t * \t\t\tname: 'span',\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'font-size': '1.2em'\n\t * \t\t\t}\n\t * \t\t},\n\t * \t\tsmall: {\n\t * \t\t\tname: 'span',\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'font-size': '0.8em'\n\t * \t\t\t}\n\t * \t\t}\n\t * \t}\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToElement( {\n\t * \tmodel: 'bold',\n\t * \tview: ( modelAttributeValue, conversionApi ) => {\n\t * \t\tconst { writer } = conversionApi;\n\t *\n\t * \t\treturn writer.createAttributeElement( 'span', {\n\t * \t\t\tstyle: 'font-weight:' + modelAttributeValue\n\t * \t\t} );\n\t * \t}\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToElement( {\n\t * \tmodel: {\n\t * \t\tkey: 'color',\n\t * \t\tname: '$text'\n\t * \t},\n\t * \tview: ( modelAttributeValue, conversionApi ) => {\n\t * \t\tconst { writer } = conversionApi;\n\t *\n\t * \t\treturn writer.createAttributeElement( 'span', {\n\t * \t\t\tstyle: 'color:' + modelAttributeValue\n\t * \t\t} );\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.model The key of the attribute to convert from or a `{ key, values }` object. `values` is an array\n\t * of `String`s with possible values if the model attribute is an enumerable.\n\t * @param config.view A view element definition or a function\n\t * that takes the model attribute value and\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API} as parameters and returns a view\n\t * attribute element. If `config.model.values` is given, `config.view` should be an object assigning values from `config.model.values`\n\t * to view element definitions or functions.\n\t * @param config.converterPriority Converter priority.\n\t */\n\tpublic attributeToElement<TValues extends string>(\n\t\tconfig: {\n\t\t\tmodel: string | {\n\t\t\t\tkey: string;\n\t\t\t\tname?: string;\n\t\t\t};\n\t\t\tview: ElementDefinition | AttributeElementCreatorFunction;\n\t\t\tconverterPriority?: PriorityString;\n\t\t} | {\n\t\t\tmodel: {\n\t\t\t\tkey: string;\n\t\t\t\tname?: string;\n\t\t\t\tvalues: Array<TValues>;\n\t\t\t};\n\t\t\tview: Record<TValues, ElementDefinition | AttributeElementCreatorFunction>;\n\t\t\tconverterPriority?: PriorityString;\n\t\t}\n\t): this {\n\t\treturn this.add( downcastAttributeToElement( config ) );\n\t}\n\n\t/**\n\t * Model attribute to view attribute conversion helper.\n\t *\n\t * This conversion results in adding an attribute to a view node, basing on an attribute from a model node. For example,\n\t * `<imageInline src='foo.jpg'></imageInline>` is converted to `<img src='foo.jpg'></img>`.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).attributeToAttribute( {\n\t * \tmodel: 'source',\n\t * \tview: 'src'\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToAttribute( {\n\t * \tmodel: 'source',\n\t * \tview: 'href',\n\t * \tconverterPriority: 'high'\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToAttribute( {\n\t * \tmodel: {\n\t * \t\tname: 'imageInline',\n\t * \t\tkey: 'source'\n\t * \t},\n\t * \tview: 'src'\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToAttribute( {\n\t * \tmodel: {\n\t * \t\tname: 'styled',\n\t * \t\tvalues: [ 'dark', 'light' ]\n\t * \t},\n\t * \tview: {\n\t * \t\tdark: {\n\t * \t\t\tkey: 'class',\n\t * \t\t\tvalue: [ 'styled', 'styled-dark' ]\n\t * \t\t},\n\t * \t\tlight: {\n\t * \t\t\tkey: 'class',\n\t * \t\t\tvalue: [ 'styled', 'styled-light' ]\n\t * \t\t}\n\t * \t}\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).attributeToAttribute( {\n\t * \tmodel: 'styled',\n\t * \tview: modelAttributeValue => ( {\n\t * \t\tkey: 'class',\n\t * \t\tvalue: 'styled-' + modelAttributeValue\n\t * \t} )\n\t * } );\n\t * ```\n\t *\n\t * **Note**: Downcasting to a style property requires providing `value` as an object:\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).attributeToAttribute( {\n\t * \tmodel: 'lineHeight',\n\t * \tview: modelAttributeValue => ( {\n\t * \t\tkey: 'style',\n\t * \t\tvalue: {\n\t * \t\t\t'line-height': modelAttributeValue,\n\t * \t\t\t'border-bottom': '1px dotted #ba2'\n\t * \t\t}\n\t * \t} )\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.model The key of the attribute to convert from or a `{ key, values, [ name ] }` object describing\n\t * the attribute key, possible values and, optionally, an element name to convert from.\n\t * @param config.view A view attribute key, or a `{ key, value }` object or a function that takes the model attribute value and\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API}\n\t * as parameters and returns a `{ key, value }` object. If the `key` is `'class'`, the `value` can be a `String` or an\n\t * array of `String`s. If the `key` is `'style'`, the `value` is an object with key-value pairs. In other cases, `value` is a `String`.\n\t * If `config.model.values` is set, `config.view` should be an object assigning values from `config.model.values` to\n\t * `{ key, value }` objects or a functions.\n\t * @param config.converterPriority Converter priority.\n\t */\n\tpublic attributeToAttribute<TValues extends string>(\n\t\tconfig: {\n\t\t\tmodel: string | {\n\t\t\t\tkey: string;\n\t\t\t\tname?: string;\n\t\t\t};\n\t\t\tview: string | AttributeDescriptor | AttributeCreatorFunction;\n\t\t\tconverterPriority?: PriorityString;\n\t\t} | {\n\t\t\tmodel: {\n\t\t\t\tkey: string;\n\t\t\t\tname?: string;\n\t\t\t\tvalues?: Array<TValues>;\n\t\t\t};\n\t\t\tview: Record<TValues, AttributeDescriptor | AttributeCreatorFunction>;\n\t\t\tconverterPriority?: PriorityString;\n\t\t}\n\t): this {\n\t\treturn this.add( downcastAttributeToAttribute( config ) );\n\t}\n\n\t/**\n\t * Model marker to view element conversion helper.\n\t *\n\t * **Note**: This method should be used mainly for editing the downcast and it is recommended\n\t * to use the {@link #markerToData `#markerToData()`} helper instead.\n\t *\n\t * This helper may produce invalid HTML code (e.g. a span between table cells).\n\t * It should only be used when you are sure that the produced HTML will be semantically correct.\n\t *\n\t * This conversion results in creating a view element on the boundaries of the converted marker. If the converted marker\n\t * is collapsed, only one element is created. For example, a model marker set like this: `<paragraph>F[oo b]ar</paragraph>`\n\t * becomes `<p>F<span data-marker=\"search\"></span>oo b<span data-marker=\"search\"></span>ar</p>` in the view.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'editingDowncast' ).markerToElement( {\n\t * \tmodel: 'search',\n\t * \tview: 'marker-search'\n\t * } );\n\t *\n\t * editor.conversion.for( 'editingDowncast' ).markerToElement( {\n\t * \tmodel: 'search',\n\t * \tview: 'search-result',\n\t * \tconverterPriority: 'high'\n\t * } );\n\t *\n\t * editor.conversion.for( 'editingDowncast' ).markerToElement( {\n\t * \tmodel: 'search',\n\t * \tview: {\n\t * \t\tname: 'span',\n\t * \t\tattributes: {\n\t * \t\t\t'data-marker': 'search'\n\t * \t\t}\n\t * \t}\n\t * } );\n\t *\n\t * editor.conversion.for( 'editingDowncast' ).markerToElement( {\n\t * \tmodel: 'search',\n\t * \tview: ( markerData, conversionApi ) => {\n\t * \t\tconst { writer } = conversionApi;\n\t *\n\t * \t\treturn writer.createUIElement( 'span', {\n\t * \t\t\t'data-marker': 'search',\n\t * \t\t\t'data-start': markerData.isOpening\n\t * \t\t} );\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * If a function is passed as the `config.view` parameter, it will be used to generate both boundary elements. The function\n\t * receives the `data` object and {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API}\n\t * as a parameters and should return an instance of the\n\t * {@link module:engine/view/uielement~UIElement view UI element}. The `data` object and\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi `conversionApi`} are passed from\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:addMarker}. Additionally,\n\t * the `data.isOpening` parameter is passed, which is set to `true` for the marker start boundary element, and `false` for\n\t * the marker end boundary element.\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.model The name of the model marker (or model marker group) to convert.\n\t * @param config.view A view element definition or a function that takes the model marker data and\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API} as a parameters\n\t * and returns a view UI element.\n\t * @param config.converterPriority Converter priority.\n\t */\n\tpublic markerToElement( config: {\n\t\tmodel: string;\n\t\tview: ElementDefinition | MarkerElementCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( downcastMarkerToElement( config ) );\n\t}\n\n\t/**\n\t * Model marker to highlight conversion helper.\n\t *\n\t * This conversion results in creating a highlight on view nodes. For this kind of conversion,\n\t * the {@link module:engine/conversion/downcasthelpers~HighlightDescriptor} should be provided.\n\t *\n\t * For text nodes, a `<span>` {@link module:engine/view/attributeelement~AttributeElement} is created and it wraps all text nodes\n\t * in the converted marker range. For example, a model marker set like this: `<paragraph>F[oo b]ar</paragraph>` becomes\n\t * `<p>F<span class=\"comment\">oo b</span>ar</p>` in the view.\n\t *\n\t * {@link module:engine/view/containerelement~ContainerElement} may provide a custom way of handling highlight. Most often,\n\t * the element itself is given classes and attributes described in the highlight descriptor (instead of being wrapped in `<span>`).\n\t * For example, a model marker set like this:\n\t * `[<imageInline src=\"foo.jpg\"></imageInline>]` becomes `<img src=\"foo.jpg\" class=\"comment\"></img>` in the view.\n\t *\n\t * For container elements, the conversion is two-step. While the converter processes the highlight descriptor and passes it\n\t * to a container element, it is the container element instance itself that applies values from the highlight descriptor.\n\t * So, in a sense, the converter takes care of stating what should be applied on what, while the element decides how to apply that.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' ).markerToHighlight( { model: 'comment', view: { classes: 'comment' } } );\n\t *\n\t * editor.conversion.for( 'downcast' ).markerToHighlight( {\n\t * \tmodel: 'comment',\n\t * \tview: { classes: 'comment' },\n\t * \tconverterPriority: 'high'\n\t * } );\n\t *\n\t * editor.conversion.for( 'downcast' ).markerToHighlight( {\n\t * \tmodel: 'comment',\n\t * \tview: ( data, conversionApi ) => {\n\t * \t\t// Assuming that the marker name is in a form of comment:commentType:commentId.\n\t * \t\tconst [ , commentType, commentId ] = data.markerName.split( ':' );\n\t *\n\t * \t\treturn {\n\t * \t\t\tclasses: [ 'comment', 'comment-' + commentType ],\n\t * \t\t\tattributes: { 'data-comment-id': commentId }\n\t * \t\t};\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * If a function is passed as the `config.view` parameter, it will be used to generate the highlight descriptor. The function\n\t * receives the `data` object and {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API}\n\t * as the parameters and should return a\n\t * {@link module:engine/conversion/downcasthelpers~HighlightDescriptor highlight descriptor}.\n\t * The `data` object properties are passed from {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:addMarker}.\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.model The name of the model marker (or model marker group) to convert.\n\t * @param config.view A highlight descriptor that will be used for highlighting or a function that takes the model marker data and\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API} as a parameters\n\t * and returns a highlight descriptor.\n\t * @param config.converterPriority Converter priority.\n\t */\n\tpublic markerToHighlight( config: {\n\t\tmodel: string;\n\t\tview: HighlightDescriptor | HighlightDescriptorCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( downcastMarkerToHighlight( config ) );\n\t}\n\n\t/**\n\t * Model marker converter for data downcast.\n\t *\n\t * This conversion creates a representation for model marker boundaries in the view:\n\t *\n\t * * If the marker boundary is before or after a model element, a view attribute is set on a corresponding view element.\n\t * * In other cases, a view element with the specified tag name is inserted at the corresponding view position.\n\t *\n\t * Typically, the marker names use the `group:uniqueId:otherData` convention. For example: `comment:e34zfk9k2n459df53sjl34:zx32c`.\n\t * The default configuration for this conversion is that the first part is the `group` part and the rest of\n\t * the marker name becomes the `name` part.\n\t *\n\t * Tag and attribute names and values are generated from the marker name:\n\t *\n\t * * The templates for attributes are `data-[group]-start-before=\"[name]\"`, `data-[group]-start-after=\"[name]\"`,\n\t * `data-[group]-end-before=\"[name]\"` and `data-[group]-end-after=\"[name]\"`.\n\t * * The templates for view elements are `<[group]-start name=\"[name]\">` and `<[group]-end name=\"[name]\">`.\n\t *\n\t * Attributes mark whether the given marker's start or end boundary is before or after the given element.\n\t * The `data-[group]-start-before` and `data-[group]-end-after` attributes are favored.\n\t * The other two are used when the former two cannot be used.\n\t *\n\t * The conversion configuration can take a function that will generate different group and name parts.\n\t * If such a function is set as the `config.view` parameter, it is passed a marker name and it is expected to return an object with two\n\t * properties: `group` and `name`. If the function returns a falsy value, the conversion will not take place.\n\t *\n\t * Basic usage:\n\t *\n\t * ```ts\n\t * // Using the default conversion.\n\t * // In this case, all markers with names starting with 'comment:' will be converted.\n\t * // The `group` parameter will be set to `comment`.\n\t * // The `name` parameter will be the rest of the marker name (without the `:`).\n\t * editor.conversion.for( 'dataDowncast' ).markerToData( {\n\t * \tmodel: 'comment'\n\t * } );\n\t * ```\n\t *\n\t * An example of a view that may be generated by this conversion (assuming a marker with the name `comment:commentId:uid` marked\n\t * by `[]`):\n\t *\n\t * ```\n\t * // Model:\n\t * <paragraph>Foo[bar</paragraph>\n\t * <imageBlock src=\"abc.jpg\"></imageBlock>]\n\t *\n\t * // View:\n\t * <p>Foo<comment-start name=\"commentId:uid\"></comment-start>bar</p>\n\t * <figure data-comment-end-after=\"commentId:uid\" class=\"image\"><img src=\"abc.jpg\" /></figure>\n\t * ```\n\t *\n\t * In the example above, the comment starts before \"bar\" and ends after the image.\n\t *\n\t * If the `name` part is empty, the following view may be generated:\n\t *\n\t * ```html\n\t * <p>Foo <myMarker-start></myMarker-start>bar</p>\n\t * <figure data-myMarker-end-after=\"\" class=\"image\"><img src=\"abc.jpg\" /></figure>\n\t * ```\n\t *\n\t * **Note:** A situation where some markers have the `name` part and some do not, is incorrect and should be avoided.\n\t *\n\t * Examples where `data-group-start-after` and `data-group-end-before` are used:\n\t *\n\t * ```\n\t * // Model:\n\t * <blockQuote>[]<paragraph>Foo</paragraph></blockQuote>\n\t *\n\t * // View:\n\t * <blockquote><p data-group-end-before=\"name\" data-group-start-before=\"name\">Foo</p></blockquote>\n\t * ```\n\t *\n\t * Similarly, when a marker is collapsed after the last element:\n\t *\n\t * ```\n\t * // Model:\n\t * <blockQuote><paragraph>Foo</paragraph>[]</blockQuote>\n\t *\n\t * // View:\n\t * <blockquote><p data-group-end-after=\"name\" data-group-start-after=\"name\">Foo</p></blockquote>\n\t * ```\n\t *\n\t * When there are multiple markers from the same group stored in the same attribute of the same element, their\n\t * name parts are put together in the attribute value, for example: `data-group-start-before=\"name1,name2,name3\"`.\n\t *\n\t * Other examples of usage:\n\t *\n\t * ```ts\n\t * // Using a custom function which is the same as the default conversion:\n\t * editor.conversion.for( 'dataDowncast' ).markerToData( {\n\t * \tmodel: 'comment',\n\t * \tview: markerName => ( {\n\t * \t\tgroup: 'comment',\n\t * \t\tname: markerName.substr( 8 ) // Removes 'comment:' part.\n\t * \t} )\n\t * } );\n\t *\n\t * // Using the converter priority:\n\t * editor.conversion.for( 'dataDowncast' ).markerToData( {\n\t * \tmodel: 'comment',\n\t * \tview: markerName => ( {\n\t * \t\tgroup: 'comment',\n\t * \t\tname: markerName.substr( 8 ) // Removes 'comment:' part.\n\t * \t} ),\n\t * \tconverterPriority: 'high'\n\t * } );\n\t * ```\n\t *\n\t * This kind of conversion is useful for saving data into the database, so it should be used in the data conversion pipeline.\n\t *\n\t * See the {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} API guide to learn how to\n\t * add a converter to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.model The name of the model marker (or the model marker group) to convert.\n\t * @param config.view A function that takes the model marker name and\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API} as the parameters\n\t * and returns an object with the `group` and `name` properties.\n\t * @param config.converterPriority Converter priority.\n\t */\n\tpublic markerToData( config: {\n\t\tmodel: string;\n\t\tview?: MarkerDataCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( downcastMarkerToData( config ) );\n\t}\n}\n\n/**\n * Function factory that creates a default downcast converter for text insertion changes.\n *\n * The converter automatically consumes the corresponding value from the consumables list and stops the event (see\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher}).\n *\n * ```ts\n * modelDispatcher.on( 'insert:$text', insertText() );\n * ```\n *\n * @returns Insert text event converter.\n */\nexport function insertText() {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: { item: ModelText | ModelTextProxy; range: ModelRange },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tif ( !conversionApi.consumable.consume( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst viewPosition = conversionApi.mapper.toViewPosition( data.range.start );\n\t\tconst viewText = viewWriter.createText( data.item.data );\n\n\t\tviewWriter.insert( viewPosition, viewText );\n\t};\n}\n\n/**\n * Function factory that creates a default downcast converter for triggering attributes and children conversion.\n *\n * @returns The converter.\n */\nexport function insertAttributesAndChildren() {\n\treturn (\n\t\tevt: unknown,\n\t\tdata: { item: ModelItem; reconversion?: boolean },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tconversionApi.convertAttributes( data.item );\n\n\t\t// Start converting children of the current item.\n\t\t// In case of reconversion children were already re-inserted or converted separately.\n\t\tif ( !data.reconversion && data.item.is( 'element' ) && !data.item.isEmpty ) {\n\t\t\tconversionApi.convertChildren( data.item );\n\t\t}\n\t};\n}\n\n/**\n * Function factory that creates a default downcast converter for node remove changes.\n *\n * ```ts\n * modelDispatcher.on( 'remove', remove() );\n * ```\n *\n * @returns Remove event converter.\n */\nexport function remove() {\n\treturn (\n\t\tevt: unknown,\n\t\tdata: { position: ModelPosition; length: number },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\t// Find the view range start position by mapping the model position at which the remove happened.\n\t\tconst viewStart = conversionApi.mapper.toViewPosition( data.position );\n\n\t\tconst modelEnd = data.position.getShiftedBy( data.length );\n\t\tconst viewEnd = conversionApi.mapper.toViewPosition( modelEnd, { isPhantom: true } );\n\n\t\tconst viewRange = conversionApi.writer.createRange( viewStart, viewEnd );\n\n\t\t// Trim the range to remove in case some UI elements are on the view range boundaries.\n\t\tconst removed = conversionApi.writer.remove( viewRange.getTrimmed() );\n\n\t\t// After the range is removed, unbind all view elements from the model.\n\t\t// Range inside view document fragment is used to unbind deeply.\n\t\tfor ( const child of conversionApi.writer.createRangeIn( removed ).getItems() ) {\n\t\t\tconversionApi.mapper.unbindViewElement( child as ViewElement, { defer: true } );\n\t\t}\n\t};\n}\n\n/**\n * Creates a `<span>` {@link module:engine/view/attributeelement~AttributeElement view attribute element} from the information\n * provided by the {@link module:engine/conversion/downcasthelpers~HighlightDescriptor highlight descriptor} object. If the priority\n * is not provided in the descriptor, the default priority will be used.\n */\nexport function createViewElementFromHighlightDescriptor( writer: DowncastWriter, descriptor: HighlightDescriptor ): ViewAttributeElement {\n\tconst viewElement = writer.createAttributeElement( 'span', descriptor.attributes );\n\n\tif ( descriptor.classes ) {\n\t\tviewElement._addClass( descriptor.classes );\n\t}\n\n\tif ( typeof descriptor.priority === 'number' ) {\n\t\t( viewElement as any )._priority = descriptor.priority;\n\t}\n\n\t( viewElement as any )._id = descriptor.id;\n\n\treturn viewElement;\n}\n\n/**\n * Function factory that creates a converter which converts a non-collapsed {@link module:engine/model/selection~Selection model selection}\n * to a {@link module:engine/view/documentselection~DocumentSelection view selection}. The converter consumes appropriate\n * value from the `consumable` object and maps model positions from the selection to view positions.\n *\n * ```ts\n * modelDispatcher.on( 'selection', convertRangeSelection() );\n * ```\n *\n * @returns Selection converter.\n */\nexport function convertRangeSelection() {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: { selection: ModelSelection | ModelDocumentSelection },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tconst selection = data.selection;\n\n\t\tif ( selection.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !conversionApi.consumable.consume( selection, 'selection' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewRanges: Array<ViewRange> = [];\n\n\t\tfor ( const range of selection.getRanges() ) {\n\t\t\tviewRanges.push( conversionApi.mapper.toViewRange( range ) );\n\t\t}\n\n\t\tconversionApi.writer.setSelection( viewRanges, { backward: selection.isBackward } );\n\t};\n}\n\n/**\n * Function factory that creates a converter which converts a collapsed {@link module:engine/model/selection~Selection model selection} to\n * a {@link module:engine/view/documentselection~DocumentSelection view selection}. The converter consumes appropriate\n * value from the `consumable` object, maps the model selection position to the view position and breaks\n * {@link module:engine/view/attributeelement~AttributeElement attribute elements} at the selection position.\n *\n * ```ts\n * modelDispatcher.on( 'selection', convertCollapsedSelection() );\n * ```\n *\n * An example of the view state before and after converting the collapsed selection:\n *\n * ```\n * <p><strong>f^oo<strong>bar</p>\n * -> <p><strong>f</strong>^<strong>oo</strong>bar</p>\n * ```\n *\n * By breaking attribute elements like `<strong>`, the selection is in a correct element. Then, when the selection attribute is\n * converted, broken attributes might be merged again, or the position where the selection is may be wrapped\n * with different, appropriate attribute elements.\n *\n * See also {@link module:engine/conversion/downcasthelpers~cleanSelection} which does a clean-up\n * by merging attributes.\n *\n * @returns Selection converter.\n */\nexport function convertCollapsedSelection() {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: { selection: ModelSelection | ModelDocumentSelection },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tconst selection = data.selection;\n\n\t\tif ( !selection.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !conversionApi.consumable.consume( selection, 'selection' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst modelPosition = selection.getFirstPosition()!;\n\t\tconst viewPosition = conversionApi.mapper.toViewPosition( modelPosition );\n\t\tconst brokenPosition = viewWriter.breakAttributes( viewPosition );\n\n\t\tviewWriter.setSelection( brokenPosition );\n\t};\n}\n\n/**\n * Function factory that creates a converter which cleans artifacts after the previous\n * {@link module:engine/model/selection~Selection model selection} conversion. It removes all empty\n * {@link module:engine/view/attributeelement~AttributeElement view attribute elements} and merges sibling attributes at all start and end\n * positions of all ranges.\n *\n * ```\n * <p><strong>^</strong></p>\n * -> <p>^</p>\n *\n * <p><strong>foo</strong>^<strong>bar</strong>bar</p>\n * -> <p><strong>foo^bar<strong>bar</p>\n *\n * <p><strong>foo</strong><em>^</em><strong>bar</strong>bar</p>\n * -> <p><strong>foo^bar<strong>bar</p>\n * ```\n *\n * This listener should be assigned before any converter for the new selection:\n *\n * ```ts\n * modelDispatcher.on( 'cleanSelection', cleanSelection() );\n * ```\n *\n * See {@link module:engine/conversion/downcasthelpers~convertCollapsedSelection}\n * which does the opposite by breaking attributes in the selection position.\n *\n * @returns Selection converter.\n */\nexport function cleanSelection() {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: unknown,\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst viewSelection = viewWriter.document.selection;\n\n\t\tfor ( const range of viewSelection.getRanges() ) {\n\t\t\t// Not collapsed selection should not have artifacts.\n\t\t\tif ( range.isCollapsed ) {\n\t\t\t\t// Position might be in the node removed by the view writer.\n\t\t\t\tif ( ( range.end.parent as ViewNode ).isAttached() ) {\n\t\t\t\t\tconversionApi.writer.mergeAttributes( range.start );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tviewWriter.setSelection( null );\n\t};\n}\n\n/**\n * Function factory that creates a converter which converts the set/change/remove attribute changes from the model to the view.\n * It can also be used to convert selection attributes. In that case, an empty attribute element will be created and the\n * selection will be put inside it.\n *\n * Attributes from the model are converted to a view element that will be wrapping these view nodes that are bound to\n * model elements having the given attribute. This is useful for attributes like `bold` that may be set on text nodes in the model\n * but are represented as an element in the view:\n *\n * ```\n * [paragraph] MODEL ====> VIEW <p>\n * \t|- a {bold: true} |- <b>\n * \t|- b {bold: true} | |- ab\n * \t|- c |- c\n * \t```\n *\n * Passed `Function` will be provided with the attribute value and then all the parameters of the\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute `attribute` event}.\n * It is expected that the function returns an {@link module:engine/view/element~Element}.\n * The result of the function will be the wrapping element.\n * When the provided `Function` does not return any element, no conversion will take place.\n *\n * The converter automatically consumes the corresponding value from the consumables list and stops the event (see\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher}).\n *\n * ```ts\n * modelDispatcher.on( 'attribute:bold', wrap( ( modelAttributeValue, { writer } ) => {\n * \treturn writer.createAttributeElement( 'strong' );\n * } );\n * ```\n *\n * @internal\n * @param elementCreator Function returning a view element that will be used for wrapping.\n * @returns Set/change attribute converter.\n */\nexport function wrap( elementCreator: AttributeElementCreatorFunction ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: {\n\t\t\titem: ModelItem | ModelSelection | ModelDocumentSelection;\n\t\t\trange: ModelRange;\n\t\t\tattributeKey: string;\n\t\t\tattributeOldValue: unknown;\n\t\t\tattributeNewValue: unknown;\n\t\t},\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tif ( !conversionApi.consumable.test( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Recreate current wrapping node. It will be used to unwrap view range if the attribute value has changed\n\t\t// or the attribute was removed.\n\t\tconst oldViewElement = elementCreator( data.attributeOldValue, conversionApi, data );\n\n\t\t// Create node to wrap with.\n\t\tconst newViewElement = elementCreator( data.attributeNewValue, conversionApi, data );\n\n\t\tif ( !oldViewElement && !newViewElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconversionApi.consumable.consume( data.item, evt.name );\n\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst viewSelection = viewWriter.document.selection;\n\n\t\tif ( data.item instanceof ModelSelection || data.item instanceof ModelDocumentSelection ) {\n\t\t\t// Selection attribute conversion.\n\t\t\tviewWriter.wrap( viewSelection.getFirstRange()!, newViewElement! );\n\t\t} else {\n\t\t\t// Node attribute conversion.\n\t\t\tlet viewRange = conversionApi.mapper.toViewRange( data.range );\n\n\t\t\t// First, unwrap the range from current wrapper.\n\t\t\tif ( data.attributeOldValue !== null && oldViewElement ) {\n\t\t\t\tviewRange = viewWriter.unwrap( viewRange, oldViewElement );\n\t\t\t}\n\n\t\t\tif ( data.attributeNewValue !== null && newViewElement ) {\n\t\t\t\tviewWriter.wrap( viewRange, newViewElement );\n\t\t\t}\n\t\t}\n\t};\n}\n\n/**\n * Function factory that creates a converter which converts node insertion changes from the model to the view.\n * The function passed will be provided with all the parameters of the dispatcher's\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:insert `insert` event}.\n * It is expected that the function returns an {@link module:engine/view/element~Element}.\n * The result of the function will be inserted into the view.\n *\n * The converter automatically consumes the corresponding value from the consumables list and binds the model and view elements.\n *\n * ```ts\n * downcastDispatcher.on(\n * \t'insert:myElem',\n * \tinsertElement( ( modelItem, { writer } ) => {\n * \t\tconst text = writer.createText( 'myText' );\n * \t\tconst myElem = writer.createElement( 'myElem', { myAttr: 'my-' + modelItem.getAttribute( 'myAttr' ) }, text );\n *\n * \t\t// Do something fancy with `myElem` using `modelItem` or other parameters.\n *\n * \t\treturn myElem;\n * \t}\n * ) );\n * ```\n *\n * @internal\n * @param elementCreator Function returning a view element, which will be inserted.\n * @param consumer Function defining element consumption process.\n * By default this function just consume passed item insertion.\n * @returns Insert element event converter.\n */\nexport function insertElement( elementCreator: ElementCreatorFunction, consumer: ConsumerFunction = defaultConsumer ) {\n\treturn (\n\t\tevt: unknown,\n\t\tdata: { item: ModelElement; range: ModelRange; reconversion?: boolean },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tif ( !consumer( data.item, conversionApi.consumable, { preflight: true } ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewElement = elementCreator( data.item, conversionApi, data );\n\n\t\tif ( !viewElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Consume an element insertion and all present attributes that are specified as a reconversion triggers.\n\t\tconsumer( data.item, conversionApi.consumable );\n\n\t\tconst viewPosition = conversionApi.mapper.toViewPosition( data.range.start );\n\n\t\tconversionApi.mapper.bindElements( data.item, viewElement );\n\t\tconversionApi.writer.insert( viewPosition, viewElement );\n\n\t\t// Convert attributes before converting children.\n\t\tconversionApi.convertAttributes( data.item );\n\n\t\t// Convert children or reinsert previous view elements.\n\t\treinsertOrConvertNodes( viewElement, data.item.getChildren(), conversionApi, { reconversion: data.reconversion } );\n\t};\n}\n\n/**\n * Function factory that creates a converter which converts a single model node insertion to a view structure.\n *\n * It is expected that the passed element creator function returns an {@link module:engine/view/element~Element} with attached slots\n * created with `writer.createSlot()` to indicate where child nodes should be converted.\n *\n * @see module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure\n *\n * @internal\n * @param elementCreator Function returning a view structure, which will be inserted.\n * @param consumer A callback that is expected to consume all the consumables\n * that were used by the element creator.\n * @returns Insert element event converter.\n*/\nexport function insertStructure( elementCreator: StructureCreatorFunction, consumer: ConsumerFunction ) {\n\treturn (\n\t\tevt: unknown,\n\t\tdata: { item: ModelElement; range: ModelRange; reconversion?: boolean },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tif ( !consumer( data.item, conversionApi.consumable, { preflight: true } ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst slotsMap = new Map<ViewElement, Array<ModelNode>>();\n\n\t\tconversionApi.writer._registerSlotFactory( createSlotFactory( data.item, slotsMap, conversionApi ) );\n\n\t\t// View creation.\n\t\tconst viewElement = elementCreator( data.item, conversionApi, data );\n\n\t\tconversionApi.writer._clearSlotFactory();\n\n\t\tif ( !viewElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Check if all children are covered by slots and there is no child that landed in multiple slots.\n\t\tvalidateSlotsChildren( data.item, slotsMap, conversionApi );\n\n\t\t// Consume an element insertion and all present attributes that are specified as a reconversion triggers.\n\t\tconsumer( data.item, conversionApi.consumable );\n\n\t\tconst viewPosition = conversionApi.mapper.toViewPosition( data.range.start );\n\n\t\tconversionApi.mapper.bindElements( data.item, viewElement );\n\t\tconversionApi.writer.insert( viewPosition, viewElement );\n\n\t\t// Convert attributes before converting children.\n\t\tconversionApi.convertAttributes( data.item );\n\n\t\t// Fill view slots with previous view elements or create new ones.\n\t\tfillSlots( viewElement, slotsMap, conversionApi, { reconversion: data.reconversion } );\n\t};\n}\n\n/**\n * Function factory that creates a converter which converts marker adding change to the\n * {@link module:engine/view/uielement~UIElement view UI element}.\n *\n * The view UI element that will be added to the view depends on the passed parameter. See {@link ~insertElement}.\n * In case of a non-collapsed range, the UI element will not wrap nodes but separate elements will be placed at the beginning\n * and at the end of the range.\n *\n * This converter binds created UI elements with the marker name using {@link module:engine/conversion/mapper~Mapper#bindElementToMarker}.\n *\n * @internal\n * @param elementCreator A view UI element or a function returning the view element that will be inserted.\n * @returns Insert element event converter.\n */\nexport function insertUIElement( elementCreator: MarkerElementCreatorFunction ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: {\n\t\t\tmarkerRange: ModelRange;\n\t\t\tmarkerName: string;\n\t\t\tisOpening?: boolean;\n\t\t},\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\t// Create two view elements. One will be inserted at the beginning of marker, one at the end.\n\t\t// If marker is collapsed, only \"opening\" element will be inserted.\n\t\tdata.isOpening = true;\n\t\tconst viewStartElement = elementCreator( data, conversionApi );\n\n\t\tdata.isOpening = false;\n\t\tconst viewEndElement = elementCreator( data, conversionApi );\n\n\t\tif ( !viewStartElement || !viewEndElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst markerRange = data.markerRange;\n\n\t\t// Marker that is collapsed has consumable build differently that non-collapsed one.\n\t\t// For more information see `addMarker` event description.\n\t\t// If marker's range is collapsed - check if it can be consumed.\n\t\tif ( markerRange.isCollapsed && !conversionApi.consumable.consume( markerRange, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// If marker's range is not collapsed - consume all items inside.\n\t\tfor ( const value of markerRange ) {\n\t\t\tif ( !conversionApi.consumable.consume( value.item, evt.name ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\n\t\tconst mapper = conversionApi.mapper;\n\t\tconst viewWriter = conversionApi.writer;\n\n\t\t// Add \"opening\" element.\n\t\tviewWriter.insert( mapper.toViewPosition( markerRange.start ), viewStartElement );\n\t\tconversionApi.mapper.bindElementToMarker( viewStartElement, data.markerName );\n\n\t\t// Add \"closing\" element only if range is not collapsed.\n\t\tif ( !markerRange.isCollapsed ) {\n\t\t\tviewWriter.insert( mapper.toViewPosition( markerRange.end ), viewEndElement );\n\t\t\tconversionApi.mapper.bindElementToMarker( viewEndElement, data.markerName );\n\t\t}\n\n\t\tevt.stop();\n\t};\n}\n\n/**\n * Function factory that returns a default downcast converter for removing a {@link module:engine/view/uielement~UIElement UI element}\n * based on marker remove change.\n *\n * This converter unbinds elements from the marker name.\n *\n * @returns Removed UI element converter.\n */\nfunction removeUIElement() {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: { markerName: string },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tconst elements = conversionApi.mapper.markerNameToElements( data.markerName );\n\n\t\tif ( !elements ) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor ( const element of elements ) {\n\t\t\tconversionApi.mapper.unbindElementFromMarkerName( element, data.markerName );\n\t\t\tconversionApi.writer.clear( conversionApi.writer.createRangeOn( element ), element );\n\t\t}\n\n\t\tconversionApi.writer.clearClonedElementsGroup( data.markerName );\n\n\t\tevt.stop();\n\t};\n}\n\n/**\n * Function factory that creates a default converter for model markers.\n *\n * See {@link DowncastHelpers#markerToData} for more information what type of view is generated.\n *\n * This converter binds created UI elements and affected view elements with the marker name\n * using {@link module:engine/conversion/mapper~Mapper#bindElementToMarker}.\n *\n * @returns Add marker converter.\n */\nfunction insertMarkerData( viewCreator: MarkerDataCreatorFunction ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: {\n\t\t\tmarkerName: string;\n\t\t\tmarkerRange: ModelRange;\n\t\t},\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tconst viewMarkerData = viewCreator( data.markerName, conversionApi );\n\n\t\tif ( !viewMarkerData ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst markerRange = data.markerRange;\n\n\t\tif ( !conversionApi.consumable.consume( markerRange, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Adding closing data first to keep the proper order in the view.\n\t\thandleMarkerBoundary( markerRange, false, conversionApi, data, viewMarkerData );\n\t\thandleMarkerBoundary( markerRange, true, conversionApi, data, viewMarkerData );\n\n\t\tevt.stop();\n\t};\n}\n\n/**\n * Helper function for `insertMarkerData()` that marks a marker boundary at the beginning or end of given `range`.\n */\nfunction handleMarkerBoundary(\n\trange: ModelRange,\n\tisStart: boolean,\n\tconversionApi: DowncastConversionApi,\n\tdata: { markerName: string },\n\tviewMarkerData: { name: string; group: string }\n): void {\n\tconst modelPosition = isStart ? range.start : range.end;\n\tconst elementAfter = modelPosition.nodeAfter && modelPosition.nodeAfter.is( 'element' ) ? modelPosition.nodeAfter : null;\n\tconst elementBefore = modelPosition.nodeBefore && modelPosition.nodeBefore.is( 'element' ) ? modelPosition.nodeBefore : null;\n\n\tif ( elementAfter || elementBefore ) {\n\t\tlet modelElement;\n\t\tlet isBefore;\n\n\t\t// If possible, we want to add `data-group-start-before` and `data-group-end-after` attributes.\n\t\tif ( isStart && elementAfter || !isStart && !elementBefore ) {\n\t\t\t// [<elementAfter>...</elementAfter> -> <elementAfter data-group-start-before=\"...\">...</elementAfter>\n\t\t\t// <parent>]<elementAfter> -> <parent><elementAfter data-group-end-before=\"...\">\n\t\t\tmodelElement = elementAfter;\n\t\t\tisBefore = true;\n\t\t} else {\n\t\t\t// <elementBefore>...</elementBefore>] -> <elementBefore data-group-end-after=\"...\">...</elementBefore>\n\t\t\t// </elementBefore>[</parent> -> </elementBefore data-group-start-after=\"...\"></parent>\n\t\t\tmodelElement = elementBefore;\n\t\t\tisBefore = false;\n\t\t}\n\n\t\tconst viewElement = conversionApi.mapper.toViewElement( modelElement! );\n\n\t\t// In rare circumstances, the model element may be not mapped to any view element and that would cause an error.\n\t\t// One of those situations is a soft break inside code block.\n\t\tif ( viewElement ) {\n\t\t\tinsertMarkerAsAttribute( viewElement, isStart, isBefore, conversionApi, data, viewMarkerData );\n\n\t\t\treturn;\n\t\t}\n\t}\n\n\tconst viewPosition = conversionApi.mapper.toViewPosition( modelPosition );\n\n\tinsertMarkerAsElement( viewPosition, isStart, conversionApi, data, viewMarkerData );\n}\n\n/**\n * Helper function for `insertMarkerData()` that marks a marker boundary in the view as an attribute on a view element.\n */\nfunction insertMarkerAsAttribute(\n\tviewElement: ViewElement,\n\tisStart: boolean,\n\tisBefore: boolean,\n\tconversionApi: DowncastConversionApi,\n\tdata: { markerName: string },\n\tviewMarkerData: { name: string; group: string }\n) {\n\tconst attributeName = `data-${ viewMarkerData.group }-${ isStart ? 'start' : 'end' }-${ isBefore ? 'before' : 'after' }`;\n\n\tconst markerNames = viewElement.hasAttribute( attributeName ) ? viewElement.getAttribute( attributeName )!.split( ',' ) : [];\n\n\t// Adding marker name at the beginning to have the same order in the attribute as there is with marker elements.\n\tmarkerNames.unshift( viewMarkerData.name );\n\n\tconversionApi.writer.setAttribute( attributeName, markerNames.join( ',' ), viewElement );\n\tconversionApi.mapper.bindElementToMarker( viewElement, data.markerName );\n}\n\n/**\n * Helper function for `insertMarkerData()` that marks a marker boundary in the view as a separate view ui element.\n */\nfunction insertMarkerAsElement(\n\tposition: ViewPosition,\n\tisStart: boolean,\n\tconversionApi: DowncastConversionApi,\n\tdata: { markerName: string },\n\tviewMarkerData: { name: string; group: string }\n) {\n\tconst viewElementName = `${ viewMarkerData.group }-${ isStart ? 'start' : 'end' }`;\n\n\tconst attrs = viewMarkerData.name ? { 'name': viewMarkerData.name } : null;\n\tconst viewElement = conversionApi.writer.createUIElement( viewElementName, attrs );\n\n\tconversionApi.writer.insert( position, viewElement );\n\tconversionApi.mapper.bindElementToMarker( viewElement, data.markerName );\n}\n\n/**\n * Function factory that creates a converter for removing a model marker data added by the {@link #insertMarkerData} converter.\n *\n * @returns Remove marker converter.\n */\nfunction removeMarkerData( viewCreator: MarkerDataCreatorFunction ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: { markerName: string },\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tconst viewData = viewCreator( data.markerName, conversionApi );\n\n\t\tif ( !viewData ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst elements = conversionApi.mapper.markerNameToElements( data.markerName );\n\n\t\tif ( !elements ) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor ( const element of elements ) {\n\t\t\tconversionApi.mapper.unbindElementFromMarkerName( element, data.markerName );\n\n\t\t\tif ( element.is( 'containerElement' ) ) {\n\t\t\t\tremoveMarkerFromAttribute( `data-${ viewData.group }-start-before`, element );\n\t\t\t\tremoveMarkerFromAttribute( `data-${ viewData.group }-start-after`, element );\n\t\t\t\tremoveMarkerFromAttribute( `data-${ viewData.group }-end-before`, element );\n\t\t\t\tremoveMarkerFromAttribute( `data-${ viewData.group }-end-after`, element );\n\t\t\t} else {\n\t\t\t\tconversionApi.writer.clear( conversionApi.writer.createRangeOn( element ), element );\n\t\t\t}\n\t\t}\n\n\t\tconversionApi.writer.clearClonedElementsGroup( data.markerName );\n\n\t\tevt.stop();\n\n\t\tfunction removeMarkerFromAttribute( attributeName: string, element: ViewElement ): void {\n\t\t\tif ( element.hasAttribute( attributeName ) ) {\n\t\t\t\tconst markerNames = new Set( element.getAttribute( attributeName )!.split( ',' ) );\n\n\t\t\t\tmarkerNames.delete( viewData!.name );\n\n\t\t\t\tif ( markerNames.size == 0 ) {\n\t\t\t\t\tconversionApi.writer.removeAttribute( attributeName, element );\n\t\t\t\t} else {\n\t\t\t\t\tconversionApi.writer.setAttribute( attributeName, Array.from( markerNames ).join( ',' ), element );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t};\n}\n\n/**\n * Function factory that creates a converter which converts the set/change/remove attribute changes from the model to the view.\n *\n * Attributes from the model are converted to the view element attributes in the view. You may provide a custom function to generate\n * a key-value attribute pair to add/change/remove. If not provided, model attributes will be converted to view element\n * attributes on a one-to-one basis.\n *\n * *Note:** The provided attribute creator should always return the same `key` for a given attribute from the model.\n *\n * The converter automatically consumes the corresponding value from the consumables list and stops the event (see\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher}).\n *\n * ```ts\n * modelDispatcher.on( 'attribute:customAttr:myElem', changeAttribute( ( value, data ) => {\n * \t// Change attribute key from `customAttr` to `class` in the view.\n * \tconst key = 'class';\n * \tlet value = data.attributeNewValue;\n *\n * \t// Force attribute value to 'empty' if the model element is empty.\n * \tif ( data.item.childCount === 0 ) {\n * \t\tvalue = 'empty';\n * \t}\n *\n * \t// Return the key-value pair.\n * \treturn { key, value };\n * } ) );\n * ```\n *\n * @param attributeCreator Function returning an object with two properties: `key` and `value`, which\n * represent the attribute key and attribute value to be set on a {@link module:engine/view/element~Element view element}.\n * The function is passed the model attribute value as the first parameter and additional data about the change as the second parameter.\n * @returns Set/change attribute converter.\n */\nfunction changeAttribute( attributeCreator: AttributeCreatorFunction ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: {\n\t\t\titem: ModelElement;\n\t\t\trange: ModelRange;\n\t\t\tattributeKey: string;\n\t\t\tattributeOldValue: unknown;\n\t\t\tattributeNewValue: unknown;\n\t\t},\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tif ( !conversionApi.consumable.test( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst oldAttribute = attributeCreator( data.attributeOldValue, conversionApi, data );\n\t\tconst newAttribute = attributeCreator( data.attributeNewValue, conversionApi, data );\n\n\t\tif ( !oldAttribute && !newAttribute ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconversionApi.consumable.consume( data.item, evt.name );\n\n\t\tconst viewElement = conversionApi.mapper.toViewElement( data.item );\n\t\tconst viewWriter = conversionApi.writer;\n\n\t\t// If model item cannot be mapped to a view element, it means item is not an `Element` instance but a `TextProxy` node.\n\t\t// Only elements can have attributes in a view so do not proceed for anything else (#1587).\n\t\tif ( !viewElement ) {\n\t\t\t/**\n\t\t\t * This error occurs when a {@link module:engine/model/textproxy~TextProxy text node's} attribute is to be downcasted\n\t\t\t * by an {@link module:engine/conversion/conversion~Conversion#attributeToAttribute `Attribute to Attribute converter`}.\n\t\t\t * In most cases it is caused by converters misconfiguration when only \"generic\" converter is defined:\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * editor.conversion.for( 'downcast' ).attributeToAttribute( {\n\t\t\t * \tmodel: 'attribute-name',\n\t\t\t * \tview: 'attribute-name'\n\t\t\t * } ) );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * and given attribute is used on text node, for example:\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * model.change( writer => {\n\t\t\t * \twriter.insertText( 'Foo', { 'attribute-name': 'bar' }, parent, 0 );\n\t\t\t * } );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * In such cases, to convert the same attribute for both {@link module:engine/model/element~Element}\n\t\t\t * and {@link module:engine/model/textproxy~TextProxy `Text`} nodes, text specific\n\t\t\t * {@link module:engine/conversion/conversion~Conversion#attributeToElement `Attribute to Element converter`}\n\t\t\t * with higher {@link module:utils/priorities~PriorityString priority} must also be defined:\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * editor.conversion.for( 'downcast' ).attributeToElement( {\n\t\t\t * \tmodel: {\n\t\t\t * \t\tkey: 'attribute-name',\n\t\t\t * \t\tname: '$text'\n\t\t\t * \t},\n\t\t\t * \tview: ( value, { writer } ) => {\n\t\t\t * \t\treturn writer.createAttributeElement( 'span', { 'attribute-name': value } );\n\t\t\t * \t},\n\t\t\t * \tconverterPriority: 'high'\n\t\t\t * } ) );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * @error conversion-attribute-to-attribute-on-text\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'conversion-attribute-to-attribute-on-text', conversionApi.dispatcher, data );\n\t\t}\n\n\t\t// First remove the old attribute if there was one.\n\t\tif ( data.attributeOldValue !== null && oldAttribute ) {\n\t\t\tif ( oldAttribute.key == 'class' ) {\n\t\t\t\tconst classes = toArray( oldAttribute.value );\n\n\t\t\t\tfor ( const className of classes ) {\n\t\t\t\t\tviewWriter.removeClass( className, viewElement );\n\t\t\t\t}\n\t\t\t} else if ( oldAttribute.key == 'style' ) {\n\t\t\t\tconst keys = Object.keys( oldAttribute.value );\n\n\t\t\t\tfor ( const key of keys ) {\n\t\t\t\t\tviewWriter.removeStyle( key, viewElement );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tviewWriter.removeAttribute( oldAttribute.key, viewElement );\n\t\t\t}\n\t\t}\n\n\t\t// Then set the new attribute.\n\t\tif ( data.attributeNewValue !== null && newAttribute ) {\n\t\t\tif ( newAttribute.key == 'class' ) {\n\t\t\t\tconst classes = toArray( newAttribute.value );\n\n\t\t\t\tfor ( const className of classes ) {\n\t\t\t\t\tviewWriter.addClass( className, viewElement );\n\t\t\t\t}\n\t\t\t} else if ( newAttribute.key == 'style' ) {\n\t\t\t\tconst keys = Object.keys( newAttribute.value );\n\n\t\t\t\tfor ( const key of keys ) {\n\t\t\t\t\tviewWriter.setStyle( key, ( newAttribute.value as Record<string, string> )[ key ], viewElement );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tviewWriter.setAttribute( newAttribute.key, newAttribute.value as string, viewElement );\n\t\t\t}\n\t\t}\n\t};\n}\n\n/**\n * Function factory that creates a converter which converts the text inside marker's range. The converter wraps the text with\n * {@link module:engine/view/attributeelement~AttributeElement} created from the provided descriptor.\n * See {link module:engine/conversion/downcasthelpers~createViewElementFromHighlightDescriptor}.\n *\n * It can also be used to convert the selection that is inside a marker. In that case, an empty attribute element will be\n * created and the selection will be put inside it.\n *\n * If the highlight descriptor does not provide the `priority` property, `10` will be used.\n *\n * If the highlight descriptor does not provide the `id` property, the name of the marker will be used.\n *\n * This converter binds the created {@link module:engine/view/attributeelement~AttributeElement attribute elemens} with the marker name\n * using the {@link module:engine/conversion/mapper~Mapper#bindElementToMarker} method.\n */\nfunction highlightText( highlightDescriptor: HighlightDescriptor | HighlightDescriptorCreatorFunction ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: {\n\t\t\titem?: ModelItem | ModelSelection | ModelDocumentSelection;\n\t\t\trange?: ModelRange;\n\t\t\tmarkerRange: ModelRange;\n\t\t\tmarkerName: string;\n\t\t},\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tif ( !data.item ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !( data.item instanceof ModelSelection || data.item instanceof ModelDocumentSelection ) && !data.item.is( '$textProxy' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst descriptor = prepareDescriptor( highlightDescriptor, data, conversionApi );\n\n\t\tif ( !descriptor ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !conversionApi.consumable.consume( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst viewElement = createViewElementFromHighlightDescriptor( viewWriter, descriptor );\n\t\tconst viewSelection = viewWriter.document.selection;\n\n\t\tif ( data.item instanceof ModelSelection || data.item instanceof ModelDocumentSelection ) {\n\t\t\tviewWriter.wrap( viewSelection.getFirstRange()!, viewElement );\n\t\t} else {\n\t\t\tconst viewRange = conversionApi.mapper.toViewRange( data.range! );\n\t\t\tconst rangeAfterWrap = viewWriter.wrap( viewRange, viewElement );\n\n\t\t\tfor ( const element of rangeAfterWrap.getItems() ) {\n\t\t\t\tif ( element.is( 'attributeElement' ) && element.isSimilar( viewElement ) ) {\n\t\t\t\t\tconversionApi.mapper.bindElementToMarker( element, data.markerName );\n\n\t\t\t\t\t// One attribute element is enough, because all of them are bound together by the view writer.\n\t\t\t\t\t// Mapper uses this binding to get all the elements no matter how many of them are registered in the mapper.\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t};\n}\n\n/**\n * Converter function factory. It creates a function which applies the marker's highlight to an element inside the marker's range.\n *\n * The converter checks if an element has the `addHighlight` function stored as a\n * {@link module:engine/view/element~Element#_setCustomProperty custom property} and, if so, uses it to apply the highlight.\n * In such case the converter will consume all element's children, assuming that they were handled by the element itself.\n *\n * When the `addHighlight` custom property is not present, the element is not converted in any special way.\n * This means that converters will proceed to convert the element's child nodes.\n *\n * If the highlight descriptor does not provide the `priority` property, `10` will be used.\n *\n * If the highlight descriptor does not provide the `id` property, the name of the marker will be used.\n *\n * This converter binds altered {@link module:engine/view/containerelement~ContainerElement container elements} with the marker name using\n * the {@link module:engine/conversion/mapper~Mapper#bindElementToMarker} method.\n */\nfunction highlightElement( highlightDescriptor: HighlightDescriptor | HighlightDescriptorCreatorFunction ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: {\n\t\t\titem?: ModelItem | ModelSelection | ModelDocumentSelection;\n\t\t\tmarkerName: string;\n\t\t\tmarkerRange: ModelRange;\n\t\t},\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\tif ( !data.item ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !( data.item instanceof ModelElement ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst descriptor = prepareDescriptor( highlightDescriptor, data, conversionApi );\n\n\t\tif ( !descriptor ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !conversionApi.consumable.test( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewElement = conversionApi.mapper.toViewElement( data.item );\n\n\t\tif ( viewElement && viewElement.getCustomProperty( 'addHighlight' ) ) {\n\t\t\t// Consume element itself.\n\t\t\tconversionApi.consumable.consume( data.item, evt.name );\n\n\t\t\t// Consume all children nodes.\n\t\t\tfor ( const value of ModelRange._createIn( data.item ) ) {\n\t\t\t\tconversionApi.consumable.consume( value.item, evt.name );\n\t\t\t}\n\n\t\t\tconst addHighlightCallback = viewElement.getCustomProperty( 'addHighlight' ) as AddHighlightCallback;\n\n\t\t\taddHighlightCallback( viewElement, descriptor, conversionApi.writer );\n\n\t\t\tconversionApi.mapper.bindElementToMarker( viewElement, data.markerName );\n\t\t}\n\t};\n}\n\n/**\n * Function factory that creates a converter which converts the removing model marker to the view.\n *\n * Both text nodes and elements are handled by this converter but they are handled a bit differently.\n *\n * Text nodes are unwrapped using the {@link module:engine/view/attributeelement~AttributeElement attribute element} created from the\n * provided highlight descriptor. See {link module:engine/conversion/downcasthelpers~HighlightDescriptor}.\n *\n * For elements, the converter checks if an element has the `removeHighlight` function stored as a\n * {@link module:engine/view/element~Element#_setCustomProperty custom property}. If so, it uses it to remove the highlight.\n * In such case, the children of that element will not be converted.\n *\n * When `removeHighlight` is not present, the element is not converted in any special way.\n * The converter will proceed to convert the element's child nodes instead.\n *\n * If the highlight descriptor does not provide the `priority` property, `10` will be used.\n *\n * If the highlight descriptor does not provide the `id` property, the name of the marker will be used.\n *\n * This converter unbinds elements from the marker name.\n */\nfunction removeHighlight( highlightDescriptor: HighlightDescriptor | HighlightDescriptorCreatorFunction ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: {\n\t\t\tmarkerName: string;\n\t\t\tmarkerRange: ModelRange;\n\t\t},\n\t\tconversionApi: DowncastConversionApi\n\t): void => {\n\t\t// This conversion makes sense only for non-collapsed range.\n\t\tif ( data.markerRange.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst descriptor = prepareDescriptor( highlightDescriptor, data, conversionApi );\n\n\t\tif ( !descriptor ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// View element that will be used to unwrap `AttributeElement`s.\n\t\tconst viewHighlightElement = createViewElementFromHighlightDescriptor( conversionApi.writer, descriptor );\n\n\t\t// Get all elements bound with given marker name.\n\t\tconst elements = conversionApi.mapper.markerNameToElements( data.markerName );\n\n\t\tif ( !elements ) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor ( const element of elements ) {\n\t\t\tconversionApi.mapper.unbindElementFromMarkerName( element, data.markerName );\n\n\t\t\tif ( element.is( 'attributeElement' ) ) {\n\t\t\t\tconversionApi.writer.unwrap( conversionApi.writer.createRangeOn( element ), viewHighlightElement );\n\t\t\t} else {\n\t\t\t\t// if element.is( 'containerElement' ).\n\t\t\t\tconst removeHighlightCallback = element.getCustomProperty( 'removeHighlight' ) as RemoveHighlightCallback;\n\n\t\t\t\tremoveHighlightCallback( element, descriptor.id!, conversionApi.writer );\n\t\t\t}\n\t\t}\n\n\t\tconversionApi.writer.clearClonedElementsGroup( data.markerName );\n\n\t\tevt.stop();\n\t};\n}\n\n/**\n * Model element to view element conversion helper.\n *\n * See {@link ~DowncastHelpers#elementToElement `.elementToElement()` downcast helper} for examples and config params description.\n *\n * @param config Conversion configuration.\n * @param config.model The description or a name of the model element to convert.\n * @param config.model.attributes List of attributes triggering element reconversion.\n * @param config.model.children Should reconvert element if the list of model child nodes changed.\n * @returns Conversion helper.\n */\nfunction downcastElementToElement( config: {\n\tmodel: string | {\n\t\tname: string;\n\t\tattributes?: string | Array<string>;\n\t\tchildren?: boolean;\n\t};\n\tview: ElementDefinition | ElementCreatorFunction;\n\tconverterPriority?: PriorityString;\n} ) {\n\tconst model = normalizeModelElementConfig( config.model );\n\tconst view = normalizeToElementConfig( config.view, 'container' );\n\n\t// Trigger reconversion on children list change if element is a subject to any reconversion.\n\t// This is required to be able to trigger Differ#refreshItem() on a direct child of the reconverted element.\n\tif ( model.attributes.length ) {\n\t\tmodel.children = true;\n\t}\n\n\treturn ( dispatcher: DowncastDispatcher ) => {\n\t\tdispatcher.on<DowncastInsertEvent<ModelElement>>(\n\t\t\t`insert:${ model.name }`,\n\t\t\tinsertElement( view, createConsumer( model ) ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\n\t\tif ( model.children || model.attributes.length ) {\n\t\t\tdispatcher.on<DowncastReduceChangesEvent>( 'reduceChanges', createChangeReducer( model ), { priority: 'low' } );\n\t\t}\n\t};\n}\n\n/**\n * Model element to view structure conversion helper.\n *\n * See {@link ~DowncastHelpers#elementToStructure `.elementToStructure()` downcast helper} for examples and config params description.\n *\n * @param config Conversion configuration.\n * @returns Conversion helper.\n */\nfunction downcastElementToStructure(\n\tconfig: {\n\t\tmodel: string | {\n\t\t\tname: string;\n\t\t\tattributes?: string | Array<string>;\n\t\t};\n\t\tview: StructureCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t}\n) {\n\tconst model = normalizeModelElementConfig( config.model );\n\tconst view = normalizeToElementConfig( config.view, 'container' );\n\n\t// Trigger reconversion on children list change because it always needs to use slots to put children in proper places.\n\t// This is required to be able to trigger Differ#refreshItem() on a direct child of the reconverted element.\n\tmodel.children = true;\n\n\treturn ( dispatcher: DowncastDispatcher ) => {\n\t\tif ( dispatcher._conversionApi.schema.checkChild( model.name, '$text' ) ) {\n\t\t\t/**\n\t\t\t * This error occurs when a {@link module:engine/model/element~Element model element} is downcasted\n\t\t\t * via {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure} helper but the element was\n\t\t\t * allowed to host `$text` by the {@link module:engine/model/schema~Schema model schema}.\n\t\t\t *\n\t\t\t * For instance, this may be the result of `myElement` allowing the content of\n\t\t\t * {@glink framework/deep-dive/schema#generic-items `$block`} in its schema definition:\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * // Element definition in schema.\n\t\t\t * schema.register( 'myElement', {\n\t\t\t * \tallowContentOf: '$block',\n\t\t\t *\n\t\t\t * \t// ...\n\t\t\t * } );\n\t\t\t *\n\t\t\t * // ...\n\t\t\t *\n\t\t\t * // Conversion of myElement with the use of elementToStructure().\n\t\t\t * editor.conversion.for( 'downcast' ).elementToStructure( {\n\t\t\t * \tmodel: 'myElement',\n\t\t\t * \tview: ( modelElement, { writer } ) => {\n\t\t\t * \t\t// ...\n\t\t\t * \t}\n\t\t\t * } );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * In such case, {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToElement `elementToElement()`} helper\n\t\t\t * can be used instead to get around this problem:\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * editor.conversion.for( 'downcast' ).elementToElement( {\n\t\t\t * \tmodel: 'myElement',\n\t\t\t * \tview: ( modelElement, { writer } ) => {\n\t\t\t * \t\t// ...\n\t\t\t * \t}\n\t\t\t * } );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * @error conversion-element-to-structure-disallowed-text\n\t\t\t * @param {String} elementName The name of the element the structure is to be created for.\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'conversion-element-to-structure-disallowed-text', dispatcher, { elementName: model.name } );\n\t\t}\n\n\t\tdispatcher.on<DowncastInsertEvent<ModelElement>>(\n\t\t\t`insert:${ model.name }`,\n\t\t\tinsertStructure( view, createConsumer( model ) ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\n\t\tdispatcher.on<DowncastReduceChangesEvent>( 'reduceChanges', createChangeReducer( model ), { priority: 'low' } );\n\t};\n}\n\n/**\n * Model attribute to view element conversion helper.\n *\n * See {@link ~DowncastHelpers#attributeToElement `.attributeToElement()` downcast helper} for examples.\n *\n * @param config Conversion configuration.\n * @param config.model The key of the attribute to convert from or a `{ key, values }` object. `values` is an array\n * of `String`s with possible values if the model attribute is an enumerable.\n * @param config.view A view element definition or a function that takes the model attribute value and\n * {@link module:engine/view/downcastwriter~DowncastWriter view downcast writer} as parameters and returns a view attribute element.\n * If `config.model.values` is given, `config.view` should be an object assigning values from `config.model.values` to view element\n * definitions or functions.\n * @param config.converterPriority Converter priority.\n * @returns Conversion helper.\n */\nfunction downcastAttributeToElement( config: {\n\tmodel: string | {\n\t\tkey: string;\n\t\tname?: string;\n\t\tvalues?: Array<string>;\n\t};\n\tview: ElementDefinition | AttributeElementCreatorFunction | Record<string, ElementDefinition | AttributeElementCreatorFunction>;\n\tconverterPriority?: PriorityString;\n} ) {\n\tconfig = cloneDeep( config );\n\n\tlet model = config.model;\n\n\tif ( typeof model == 'string' ) {\n\t\tmodel = { key: model };\n\t}\n\n\tlet eventName = `attribute:${ model.key }` as const;\n\n\tif ( model.name ) {\n\t\teventName += ':' + model.name;\n\t}\n\n\tif ( model.values ) {\n\t\tfor ( const modelValue of model.values ) {\n\t\t\t( config.view as any )[ modelValue ] = normalizeToElementConfig( ( config.view as any )[ modelValue ], 'attribute' );\n\t\t}\n\t} else {\n\t\tconfig.view = normalizeToElementConfig( config.view as any, 'attribute' );\n\t}\n\n\tconst elementCreator = getFromAttributeCreator<AttributeElementCreatorFunction>( config );\n\n\treturn ( dispatcher: DowncastDispatcher ) => {\n\t\tdispatcher.on<DowncastAttributeEvent>(\n\t\t\teventName,\n\t\t\twrap( elementCreator ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t};\n}\n\n/**\n * Model attribute to view attribute conversion helper.\n *\n * See {@link ~DowncastHelpers#attributeToAttribute `.attributeToAttribute()` downcast helper} for examples.\n *\n * @param config Conversion configuration.\n * @param config.model The key of the attribute to convert from or a `{ key, values, [ name ] }` object describing\n * the attribute key, possible values and, optionally, an element name to convert from.\n * @param config.view A view attribute key, or a `{ key, value }` object or a function that takes the model attribute value and returns\n * a `{ key, value }` object.\n * If `key` is `'class'`, `value` can be a `String` or an array of `String`s. If `key` is `'style'`, `value` is an object with\n * key-value pairs. In other cases, `value` is a `String`.\n * If `config.model.values` is set, `config.view` should be an object assigning values from `config.model.values` to\n * `{ key, value }` objects or a functions.\n * @param config.converterPriority Converter priority.\n * @returns Conversion helper.\n */\nfunction downcastAttributeToAttribute( config: {\n\tmodel: string | {\n\t\tkey: string;\n\t\tname?: string;\n\t\tvalues?: Array<string>;\n\t};\n\tview: string | AttributeDescriptor | AttributeCreatorFunction | Record<string, AttributeDescriptor | AttributeCreatorFunction>;\n\tconverterPriority?: PriorityString;\n} ) {\n\tconfig = cloneDeep( config );\n\n\tlet model = config.model;\n\n\tif ( typeof model == 'string' ) {\n\t\tmodel = { key: model };\n\t}\n\n\tlet eventName = `attribute:${ model.key }` as const;\n\n\tif ( model.name ) {\n\t\teventName += ':' + model.name;\n\t}\n\n\tif ( model.values ) {\n\t\tfor ( const modelValue of model.values ) {\n\t\t\t( config.view as any )[ modelValue ] = normalizeToAttributeConfig( ( config.view as any )[ modelValue ] );\n\t\t}\n\t} else {\n\t\tconfig.view = normalizeToAttributeConfig( config.view );\n\t}\n\n\tconst elementCreator = getFromAttributeCreator<AttributeCreatorFunction>( config );\n\n\treturn ( dispatcher: DowncastDispatcher ) => {\n\t\tdispatcher.on<DowncastAttributeEvent<ModelElement>>(\n\t\t\teventName,\n\t\t\tchangeAttribute( elementCreator ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t};\n}\n\n/**\n * Model marker to view element conversion helper.\n *\n * See {@link ~DowncastHelpers#markerToElement `.markerToElement()` downcast helper} for examples.\n *\n * @param config Conversion configuration.\n * @param config.model The name of the model marker (or model marker group) to convert.\n * @param config.view A view element definition or a function that takes the model marker data as a parameter and returns a view UI element.\n * @param config.converterPriority Converter priority.\n * @returns Conversion helper.\n */\nfunction downcastMarkerToElement( config: {\n\tmodel: string;\n\tview: ElementDefinition | MarkerElementCreatorFunction;\n\tconverterPriority?: PriorityString;\n} ) {\n\tconst view = normalizeToElementConfig( config.view, 'ui' );\n\n\treturn ( dispatcher: DowncastDispatcher ) => {\n\t\tdispatcher.on<DowncastAddMarkerEvent>(\n\t\t\t`addMarker:${ config.model }`,\n\t\t\tinsertUIElement( view ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t\tdispatcher.on<DowncastRemoveMarkerEvent>(\n\t\t\t`removeMarker:${ config.model }`,\n\t\t\tremoveUIElement(),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t};\n}\n\n/**\n * Model marker to view data conversion helper.\n *\n * See {@link ~DowncastHelpers#markerToData `markerToData()` downcast helper} to learn more.\n *\n * @returns Conversion helper.\n */\nfunction downcastMarkerToData( config: {\n\tmodel: string;\n\tview?: MarkerDataCreatorFunction;\n\tconverterPriority?: PriorityString;\n} ) {\n\tconfig = cloneDeep( config );\n\n\tconst group = config.model;\n\tlet view = config.view;\n\n\t// Default conversion.\n\tif ( !view ) {\n\t\tview = markerName => ( {\n\t\t\tgroup,\n\t\t\tname: markerName.substr( config.model.length + 1 )\n\t\t} );\n\t}\n\n\treturn ( dispatcher: DowncastDispatcher ) => {\n\t\tdispatcher.on<DowncastAddMarkerEvent>(\n\t\t\t`addMarker:${ group }`,\n\t\t\tinsertMarkerData( view! ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t\tdispatcher.on<DowncastRemoveMarkerEvent>(\n\t\t\t`removeMarker:${ group }`,\n\t\t\tremoveMarkerData( view! ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t};\n}\n\n/**\n * Model marker to highlight conversion helper.\n *\n * See {@link ~DowncastHelpers#markerToElement `.markerToElement()` downcast helper} for examples.\n *\n * @param config Conversion configuration.\n * @param config.model The name of the model marker (or model marker group) to convert.\n * @param config.view A highlight descriptor that will be used for highlighting or a function that takes\n * the model marker data as a parameter and returns a highlight descriptor.\n * @param config.converterPriority Converter priority.\n * @returns Conversion helper.\n */\nfunction downcastMarkerToHighlight( config: {\n\tmodel: string;\n\tview: HighlightDescriptor | HighlightDescriptorCreatorFunction;\n\tconverterPriority?: PriorityString;\n} ) {\n\treturn ( dispatcher: DowncastDispatcher ) => {\n\t\tdispatcher.on<DowncastAddMarkerEvent>(\n\t\t\t`addMarker:${ config.model }`,\n\t\t\thighlightText( config.view ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t\tdispatcher.on<DowncastAddMarkerEvent>(\n\t\t\t`addMarker:${ config.model }`,\n\t\t\thighlightElement( config.view ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t\tdispatcher.on<DowncastRemoveMarkerEvent>(\n\t\t\t`removeMarker:${ config.model }`,\n\t\t\tremoveHighlight( config.view ),\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t};\n}\n\n/**\n * Takes `config.model`, and converts it to an object with normalized structure.\n *\n * @param model Model configuration or element name.\n */\nfunction normalizeModelElementConfig( model: string | {\n\tname: string;\n\tattributes?: string | Array<string>;\n\tchildren?: boolean;\n} ): NormalizedModelElementConfig {\n\tif ( typeof model == 'string' ) {\n\t\tmodel = { name: model };\n\t}\n\n\t// List of attributes that should trigger reconversion.\n\tif ( !model.attributes ) {\n\t\tmodel.attributes = [];\n\t} else if ( !Array.isArray( model.attributes ) ) {\n\t\tmodel.attributes = [ model.attributes ];\n\t}\n\n\t// Whether a children insertion/deletion should trigger reconversion.\n\tmodel.children = !!model.children;\n\n\treturn model as any;\n}\n\ninterface NormalizedModelElementConfig {\n\tname: string;\n\tattributes: Array<string>;\n\tchildren: boolean;\n}\n\n/**\n * Takes `config.view`, and if it is an {@link module:engine/view/elementdefinition~ElementDefinition}, converts it\n * to a function (because lower level converters accept only element creator functions).\n *\n * @param view View configuration.\n * @param viewElementType View element type to create.\n * @returns Element creator function to use in lower level converters.\n */\nfunction normalizeToElementConfig<T extends Function>(\n\tview: ElementDefinition | T,\n\tviewElementType: 'container' | 'attribute' | 'ui'\n): T {\n\tif ( typeof view == 'function' ) {\n\t\t// If `view` is already a function, don't do anything.\n\t\treturn view as any;\n\t}\n\n\treturn ( ( modelData: unknown, conversionApi: DowncastConversionApi ) =>\n\t\tcreateViewElementFromDefinition( view, conversionApi, viewElementType ) ) as any;\n}\n\n/**\n * Creates a view element instance from the provided {@link module:engine/view/elementdefinition~ElementDefinition} and class.\n */\nfunction createViewElementFromDefinition(\n\tviewElementDefinition: ElementDefinition,\n\tconversionApi: DowncastConversionApi,\n\tviewElementType: 'container' | 'attribute' | 'ui'\n): ViewElement {\n\tif ( typeof viewElementDefinition == 'string' ) {\n\t\t// If `viewElementDefinition` is given as a `String`, normalize it to an object with `name` property.\n\t\tviewElementDefinition = { name: viewElementDefinition };\n\t}\n\n\tlet element: ViewElement;\n\tconst viewWriter = conversionApi.writer;\n\tconst attributes = Object.assign( {}, viewElementDefinition.attributes );\n\n\tif ( viewElementType == 'container' ) {\n\t\telement = viewWriter.createContainerElement( viewElementDefinition.name, attributes );\n\t} else if ( viewElementType == 'attribute' ) {\n\t\tconst options = {\n\t\t\tpriority: viewElementDefinition.priority || ViewAttributeElement.DEFAULT_PRIORITY\n\t\t};\n\n\t\telement = viewWriter.createAttributeElement( viewElementDefinition.name, attributes, options );\n\t} else {\n\t\t// 'ui'.\n\t\telement = viewWriter.createUIElement( viewElementDefinition.name, attributes );\n\t}\n\n\tif ( viewElementDefinition.styles ) {\n\t\tconst keys = Object.keys( viewElementDefinition.styles );\n\n\t\tfor ( const key of keys ) {\n\t\t\tviewWriter.setStyle( key, viewElementDefinition.styles[ key ], element );\n\t\t}\n\t}\n\n\tif ( viewElementDefinition.classes ) {\n\t\tconst classes = viewElementDefinition.classes;\n\n\t\tif ( typeof classes == 'string' ) {\n\t\t\tviewWriter.addClass( classes, element );\n\t\t} else {\n\t\t\tfor ( const className of classes ) {\n\t\t\t\tviewWriter.addClass( className, element );\n\t\t\t}\n\t\t}\n\t}\n\n\treturn element;\n}\n\nfunction getFromAttributeCreator<T extends AttributeElementCreatorFunction | AttributeCreatorFunction>( config: any ): T {\n\tif ( config.model.values ) {\n\t\treturn ( ( modelAttributeValue: any, conversionApi: DowncastConversionApi, data: any ) => {\n\t\t\tconst view = config.view[ modelAttributeValue ];\n\n\t\t\tif ( view ) {\n\t\t\t\treturn view( modelAttributeValue, conversionApi, data );\n\t\t\t}\n\n\t\t\treturn null;\n\t\t} ) as any;\n\t} else {\n\t\treturn config.view;\n\t}\n}\n\n/**\n * Takes the configuration, adds default parameters if they do not exist and normalizes other parameters to be used in downcast converters\n * for generating a view attribute.\n *\n * @param view View configuration.\n */\nfunction normalizeToAttributeConfig( view: any ): AttributeCreatorFunction {\n\tif ( typeof view == 'string' ) {\n\t\treturn modelAttributeValue => ( { key: view, value: modelAttributeValue as string } );\n\t} else if ( typeof view == 'object' ) {\n\t\t// { key, value, ... }\n\t\tif ( view.value ) {\n\t\t\treturn () => view;\n\t\t}\n\t\t// { key, ... }\n\t\telse {\n\t\t\treturn modelAttributeValue => ( { key: view.key, value: modelAttributeValue as string } );\n\t\t}\n\t} else {\n\t\t// function.\n\t\treturn view;\n\t}\n}\n\n/**\n * Helper function for `highlight`. Prepares the actual descriptor object using value passed to the converter.\n */\nfunction prepareDescriptor(\n\thighlightDescriptor: HighlightDescriptor | HighlightDescriptorCreatorFunction,\n\tdata: {\n\t\tmarkerName: string;\n\t\tmarkerRange: ModelRange;\n\t},\n\tconversionApi: DowncastConversionApi\n): HighlightDescriptor | null {\n\t// If passed descriptor is a creator function, call it. If not, just use passed value.\n\tconst descriptor = typeof highlightDescriptor == 'function' ?\n\t\thighlightDescriptor( data, conversionApi ) :\n\t\thighlightDescriptor;\n\n\tif ( !descriptor ) {\n\t\treturn null;\n\t}\n\n\t// Apply default descriptor priority.\n\tif ( !descriptor.priority ) {\n\t\tdescriptor.priority = 10;\n\t}\n\n\t// Default descriptor id is marker name.\n\tif ( !descriptor.id ) {\n\t\tdescriptor.id = data.markerName;\n\t}\n\n\treturn descriptor;\n}\n\n/**\n * Creates a function that checks a single differ diff item whether it should trigger reconversion.\n *\n * @param model A normalized `config.model` converter configuration.\n * @param model.name The name of element.\n * @param model.attributes The list of attribute names that should trigger reconversion.\n * @param model.children Whether the child list change should trigger reconversion.\n */\nfunction createChangeReducerCallback( model: NormalizedModelElementConfig ) {\n\treturn ( node: ModelNode, change: DiffItem | DiffItemReinsert ): boolean => {\n\t\tif ( !node.is( 'element', model.name ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( change.type == 'attribute' ) {\n\t\t\tif ( model.attributes.includes( change.attributeKey ) ) {\n\t\t\t\treturn true;\n\t\t\t}\n\t\t} else {\n\t\t\t/* istanbul ignore else: This is always true because otherwise it would not register a reducer callback. -- @preserve */\n\t\t\tif ( model.children ) {\n\t\t\t\treturn true;\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t};\n}\n\n/**\n * Creates a `reduceChanges` event handler for reconversion.\n *\n * @param model A normalized `config.model` converter configuration.\n * @param model.name The name of element.\n * @param model.attributes The list of attribute names that should trigger reconversion.\n * @param model.children Whether the child list change should trigger reconversion.\n */\nfunction createChangeReducer( model: NormalizedModelElementConfig ) {\n\tconst shouldReplace = createChangeReducerCallback( model );\n\n\treturn (\n\t\tevt: unknown,\n\t\tdata: { changes: Iterable<DiffItem | DiffItemReinsert>; reconvertedElements?: Set<ModelNode> }\n\t) => {\n\t\tconst reducedChanges: Array<DiffItem | DiffItemReinsert> = [];\n\n\t\tif ( !data.reconvertedElements ) {\n\t\t\tdata.reconvertedElements = new Set();\n\t\t}\n\n\t\tfor ( const change of data.changes ) {\n\t\t\t// For attribute use node affected by the change.\n\t\t\t// For insert or remove use parent element because we need to check if it's added/removed child.\n\t\t\tconst node = change.type == 'attribute' ? change.range.start.nodeAfter : change.position.parent as ModelNode;\n\n\t\t\tif ( !node || !shouldReplace( node, change ) ) {\n\t\t\t\treducedChanges.push( change );\n\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// If it's already marked for reconversion, so skip this change, otherwise add the diff items.\n\t\t\tif ( !data.reconvertedElements.has( node ) ) {\n\t\t\t\tdata.reconvertedElements.add( node );\n\n\t\t\t\tconst position = ModelPosition._createBefore( node );\n\t\t\t\tlet changeIndex = reducedChanges.length;\n\n\t\t\t\t// We need to insert remove+reinsert before any other change on and inside the re-converted element.\n\t\t\t\t// This is important because otherwise we would remove element that had already been modified by the previous change.\n\t\t\t\t// Note that there could be some element removed before the re-converted element, so we must not break this behavior.\n\t\t\t\tfor ( let i = reducedChanges.length - 1; i >= 0; i-- ) {\n\t\t\t\t\tconst change = reducedChanges[ i ];\n\t\t\t\t\tconst changePosition = change.type == 'attribute' ? change.range.start : change.position;\n\t\t\t\t\tconst positionRelation = changePosition.compareWith( position );\n\n\t\t\t\t\tif ( positionRelation == 'before' || change.type == 'remove' && positionRelation == 'same' ) {\n\t\t\t\t\t\tbreak;\n\t\t\t\t\t}\n\n\t\t\t\t\tchangeIndex = i;\n\t\t\t\t}\n\n\t\t\t\treducedChanges.splice( changeIndex, 0, {\n\t\t\t\t\ttype: 'remove',\n\t\t\t\t\tname: ( node as ModelElement ).name,\n\t\t\t\t\tposition,\n\t\t\t\t\tlength: 1\n\t\t\t\t} as any, {\n\t\t\t\t\ttype: 'reinsert',\n\t\t\t\t\tname: ( node as ModelElement ).name,\n\t\t\t\t\tposition,\n\t\t\t\t\tlength: 1\n\t\t\t\t} );\n\t\t\t}\n\t\t}\n\n\t\tdata.changes = reducedChanges;\n\t};\n}\n\n/**\n * Creates a function that checks if an element and its watched attributes can be consumed and consumes them.\n *\n * @param model A normalized `config.model` converter configuration.\n * @param model.name The name of element.\n * @param model.attributes The list of attribute names that should trigger reconversion.\n * @param model.children Whether the child list change should trigger reconversion.\n */\nfunction createConsumer( model: NormalizedModelElementConfig ): ConsumerFunction {\n\treturn ( node, consumable, options = {} ) => {\n\t\tconst events = [ 'insert' ];\n\n\t\t// Collect all set attributes that are triggering conversion.\n\t\tfor ( const attributeName of model.attributes ) {\n\t\t\tif ( node.hasAttribute( attributeName ) ) {\n\t\t\t\tevents.push( `attribute:${ attributeName }` );\n\t\t\t}\n\t\t}\n\n\t\tif ( !events.every( event => consumable.test( node, event ) ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( !options.preflight ) {\n\t\t\tevents.forEach( event => consumable.consume( node, event ) );\n\t\t}\n\n\t\treturn true;\n\t};\n}\n\n/**\n * Creates a function that create view slots.\n *\n * @returns Function exposed by writer as createSlot().\n */\nfunction createSlotFactory( element: ModelElement, slotsMap: Map<ViewElement, Array<ModelNode>>, conversionApi: DowncastConversionApi ) {\n\treturn ( writer: DowncastWriter, modeOrFilter: string | SlotFilter ) => {\n\t\tconst slot = writer.createContainerElement( '$slot' );\n\n\t\tlet children: Array<ModelNode> | null = null;\n\n\t\tif ( modeOrFilter === 'children' ) {\n\t\t\tchildren = Array.from( element.getChildren() );\n\t\t} else if ( typeof modeOrFilter == 'function' ) {\n\t\t\tchildren = Array.from( element.getChildren() ).filter( element => modeOrFilter( element ) );\n\t\t} else {\n\t\t\t/**\n\t\t\t * Unknown slot mode was provided to `writer.createSlot()` in downcast converter.\n\t\t\t *\n\t\t\t * @error conversion-slot-mode-unknown\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'conversion-slot-mode-unknown', conversionApi.dispatcher, { modeOrFilter } );\n\t\t}\n\n\t\tslotsMap.set( slot, children );\n\n\t\treturn slot;\n\t};\n}\n\n/**\n * Checks if all children are covered by slots and there is no child that landed in multiple slots.\n */\nfunction validateSlotsChildren(\n\telement: ModelElement,\n\tslotsMap: Map<ViewElement, Array<ModelNode>>,\n\tconversionApi: DowncastConversionApi\n) {\n\tconst childrenInSlots = Array.from( slotsMap.values() ).flat();\n\tconst uniqueChildrenInSlots = new Set( childrenInSlots );\n\n\tif ( uniqueChildrenInSlots.size != childrenInSlots.length ) {\n\t\t/**\n\t\t * Filters provided to `writer.createSlot()` overlap (at least two filters accept the same child element).\n\t\t *\n\t\t * @error conversion-slot-filter-overlap\n\t\t * @param {module:engine/model/element~Element} element The element of which children would not be properly\n\t\t * allocated to multiple slots.\n\t\t */\n\t\tthrow new CKEditorError( 'conversion-slot-filter-overlap', conversionApi.dispatcher, { element } );\n\t}\n\n\tif ( uniqueChildrenInSlots.size != element.childCount ) {\n\t\t/**\n\t\t * Filters provided to `writer.createSlot()` are incomplete and exclude at least one children element (one of\n\t\t * the children elements would not be assigned to any of the slots).\n\t\t *\n\t\t * @error conversion-slot-filter-incomplete\n\t\t * @param {module:engine/model/element~Element} element The element of which children would not be properly\n\t\t * allocated to multiple slots.\n\t\t */\n\t\tthrow new CKEditorError( 'conversion-slot-filter-incomplete', conversionApi.dispatcher, { element } );\n\t}\n}\n\n/**\n * Fill slots with appropriate view elements.\n */\nfunction fillSlots(\n\tviewElement: ViewElement,\n\tslotsMap: Map<ViewElement, Array<ModelNode>>,\n\tconversionApi: DowncastConversionApi,\n\toptions: { reconversion?: boolean }\n): void {\n\t// Set temporary position mapping to redirect child view elements into a proper slots.\n\tconversionApi.mapper.on<MapperModelToViewPositionEvent>( 'modelToViewPosition', toViewPositionMapping, { priority: 'highest' } );\n\n\tlet currentSlot: ViewElement | null = null;\n\tlet currentSlotNodes: Array<ModelNode> | null = null;\n\n\t// Fill slots with nested view nodes.\n\tfor ( [ currentSlot, currentSlotNodes ] of slotsMap ) {\n\t\treinsertOrConvertNodes( viewElement, currentSlotNodes, conversionApi, options );\n\n\t\tconversionApi.writer.move(\n\t\t\tconversionApi.writer.createRangeIn( currentSlot ),\n\t\t\tconversionApi.writer.createPositionBefore( currentSlot )\n\t\t);\n\t\tconversionApi.writer.remove( currentSlot );\n\t}\n\n\tconversionApi.mapper.off( 'modelToViewPosition', toViewPositionMapping );\n\n\tfunction toViewPositionMapping( evt: unknown, data: {\n\t\tmapper: Mapper;\n\t\tmodelPosition: ModelPosition;\n\t\tviewPosition?: ViewPosition;\n\t\tisPhantom?: boolean;\n\t} ) {\n\t\tconst element = data.modelPosition.nodeAfter!;\n\n\t\t// Find the proper offset within the slot.\n\t\tconst index = currentSlotNodes!.indexOf( element );\n\n\t\tif ( index < 0 ) {\n\t\t\treturn;\n\t\t}\n\n\t\tdata.viewPosition = data.mapper.findPositionIn( currentSlot!, index );\n\t}\n}\n\n/**\n * Inserts view representation of `nodes` into the `viewElement` either by bringing back just removed view nodes\n * or by triggering conversion for them.\n */\nfunction reinsertOrConvertNodes(\n\tviewElement: ViewElement,\n\tmodelNodes: Iterable<ModelNode>,\n\tconversionApi: DowncastConversionApi,\n\toptions: { reconversion?: boolean }\n) {\n\t// Fill with nested view nodes.\n\tfor ( const modelChildNode of modelNodes ) {\n\t\t// Try reinserting the view node for the specified model node...\n\t\tif ( !reinsertNode( viewElement.root, modelChildNode, conversionApi, options ) ) {\n\t\t\t// ...or else convert the model element to the view.\n\t\t\tconversionApi.convertItem( modelChildNode );\n\t\t}\n\t}\n}\n\n/**\n * Checks if the view for the given model element could be reused and reinserts it to the view.\n *\n * @returns `false` if view element can't be reused.\n */\nfunction reinsertNode(\n\tviewRoot: ViewElement | ViewDocumentFragment,\n\tmodelNode: ModelNode,\n\tconversionApi: DowncastConversionApi,\n\toptions: { reconversion?: boolean }\n): boolean {\n\tconst { writer, mapper } = conversionApi;\n\n\t// Don't reinsert if this is not a reconversion...\n\tif ( !options.reconversion ) {\n\t\treturn false;\n\t}\n\n\tconst viewChildNode = mapper.toViewElement( modelNode as ModelElement );\n\n\t// ...or there is no view to reinsert or it was already inserted to the view structure...\n\tif ( !viewChildNode || viewChildNode.root == viewRoot ) {\n\t\treturn false;\n\t}\n\n\t// ...or it was strictly marked as not to be reused.\n\tif ( !conversionApi.canReuseView( viewChildNode ) ) {\n\t\treturn false;\n\t}\n\n\t// Otherwise reinsert the view node.\n\twriter.move(\n\t\twriter.createRangeOn( viewChildNode ),\n\t\tmapper.toViewPosition( ModelPosition._createBefore( modelNode ) )\n\t);\n\n\treturn true;\n}\n\n/**\n * The default consumer for insert events.\n *\n * @param item Model item.\n * @param consumable The model consumable.\n * @param options.preflight Whether should consume or just check if can be consumed.\n */\nfunction defaultConsumer(\n\titem: ModelItem,\n\tconsumable: ModelConsumable,\n\t{ preflight }: { preflight?: boolean } = {}\n): boolean | null {\n\tif ( preflight ) {\n\t\treturn consumable.test( item, 'insert' );\n\t} else {\n\t\treturn consumable.consume( item, 'insert' );\n\t}\n}\n\n/**\n * An object describing how the marker highlight should be represented in the view.\n *\n * Each text node contained in a highlighted range will be wrapped in a `<span>`\n * {@link module:engine/view/attributeelement~AttributeElement view attribute element} with CSS class(es), attributes and a priority\n * described by this object.\n *\n * Additionally, each {@link module:engine/view/containerelement~ContainerElement container element} can handle displaying the highlight\n * separately by providing the `addHighlight` and `removeHighlight` custom properties. In this case:\n *\n * * The `HighlightDescriptor` object is passed to the `addHighlight` function upon conversion and should be used to apply the highlight to\n * the element.\n * * The descriptor `id` is passed to the `removeHighlight` function upon conversion and should be used to remove the highlight with the\n * given ID from the element.\n */\nexport interface HighlightDescriptor {\n\n\t/**\n\t * A CSS class or an array of classes to set. If the descriptor is used to\n\t * create an {@link module:engine/view/attributeelement~AttributeElement attribute element} over text nodes, these classes will be set\n\t * on that attribute element. If the descriptor is applied to an element, usually these classes will be set on that element, however,\n\t * this depends on how the element converts the descriptor.\n\t */\n\tclasses: string | Array<string>;\n\n\t/**\n\t * Descriptor identifier. If not provided, it defaults to the converted marker's name.\n\t */\n\tid?: string;\n\n\t/**\n\t * Descriptor priority. If not provided, it defaults to `10`. If the descriptor is used to create\n\t * an {@link module:engine/view/attributeelement~AttributeElement attribute element}, it will be that element's\n\t * {@link module:engine/view/attributeelement~AttributeElement#priority priority}. If the descriptor is applied to an element,\n\t * the priority will be used to determine which descriptor is more important.\n\t */\n\tpriority?: number;\n\n\t/**\n\t * Attributes to set. If the descriptor is used to create\n\t * an {@link module:engine/view/attributeelement~AttributeElement attribute element} over text nodes, these attributes will be set\n\t * on that attribute element. If the descriptor is applied to an element, usually these attributes will be set on that element, however,\n\t * this depends on how the element converts the descriptor.\n\t */\n\tattributes?: Record<string, string>;\n}\n\n/**\n * A filtering function used to choose model child nodes to be downcasted into the specific view\n * {@link module:engine/view/downcastwriter~DowncastWriter#createSlot \"slot\"} while executing the\n * {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure `elementToStructure()`} converter.\n *\n * @callback module:engine/conversion/downcasthelpers~SlotFilter\n *\n * @param node A model node.\n * @returns Whether the provided model node should be downcasted into this slot.\n *\n * @see module:engine/view/downcastwriter~DowncastWriter#createSlot\n * @see module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure\n * @see module:engine/conversion/downcasthelpers~insertStructure\n */\nexport type SlotFilter = ( node: ModelNode ) => boolean;\n\n/**\n * A view element creator function that takes the model element and {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi\n * downcast conversion API} as parameters and returns a view container element.\n *\n * @callback module:engine/conversion/downcasthelpers~ElementCreatorFunction\n *\n * @param element The model element to be converted to the view structure.\n * @param conversionApi The conversion interface.\n * @param data Additional information about the change (same as for\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:insert `insert`} event).\n * @param data.item Inserted item.\n * @param data.range Range spanning over inserted item.\n * @returns The view element.\n *\n * @see module:engine/conversion/downcasthelpers~DowncastHelpers#elementToElement\n * @see module:engine/conversion/downcasthelpers~insertElement\n */\nexport type ElementCreatorFunction = (\n\telement: ModelElement,\n\tconversionApi: DowncastConversionApi,\n\tdata: {\n\t\titem: ModelItem;\n\t\trange: ModelRange;\n\t}\n) => ViewElement | null;\n\n/**\n * A function that takes the model element and {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast\n * conversion API} as parameters and returns a view container element with slots for model child nodes to be converted into.\n *\n * @callback module:engine/conversion/downcasthelpers~StructureCreatorFunction\n *\n * @param element The model element to be converted to the view structure.\n * @param conversionApi The conversion interface.\n * @param data Additional information about the change (same as for\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:insert `insert`} event).\n * @param data.item Inserted item.\n * @param data.range Range spanning over inserted item.\n * @returns The view structure with slots for model child nodes.\n *\n * @see module:engine/conversion/downcasthelpers~DowncastHelpers#elementToStructure\n * @see module:engine/conversion/downcasthelpers~insertStructure\n */\nexport type StructureCreatorFunction = ElementCreatorFunction;\n\n/**\n * A view element creator function that takes the model attribute value and\n * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API} as parameters and returns a view\n * attribute element.\n *\n * @callback module:engine/conversion/downcasthelpers~AttributeElementCreatorFunction\n *\n * @param attributeValue The model attribute value to be converted to the view attribute element.\n * @param conversionApi The conversion interface.\n * @param data Additional information about the change (same as for\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute `attribute`} event).\n * @param data.item Changed item or converted selection.\n * @param data.range Range spanning over changed item or selection range.\n * @param data.attributeKey Attribute key.\n * @param data.attributeOldValue Attribute value before the change. This is `null` when selection attribute is converted.\n * @param data.attributeNewValue New attribute value.\n * @returns The view attribute element.\n *\n * @see module:engine/conversion/downcasthelpers~DowncastHelpers#attributeToElement\n * @see module:engine/conversion/downcasthelpers~wrap\n */\nexport type AttributeElementCreatorFunction = (\n\tattributeValue: any,\n\tconversionApi: DowncastConversionApi,\n\tdata: {\n\t\titem: ModelItem | ModelSelection | ModelDocumentSelection;\n\t\trange: ModelRange;\n\t\tattributeKey: string;\n\t\tattributeOldValue: unknown;\n\t\tattributeNewValue: unknown;\n\t}\n) => ViewAttributeElement | null;\n\n/**\n * A function that takes the model attribute value and\n * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi downcast conversion API}\n * as parameters.\n *\n * @callback module:engine/conversion/downcasthelpers~AttributeCreatorFunction\n *\n * @param attributeValue The model attribute value to be converted to the view attribute element.\n * @param conversionApi The conversion interface.\n * @param data Additional information about the change (same as for\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute `attribute`} event).\n * @param data.item Changed item or converted selection.\n * @param data.range Range spanning over changed item or selection range.\n * @param data.attributeKey Attribute key.\n * @param data.attributeOldValue Attribute value before the change. This is `null` when selection attribute is converted.\n * @param data.attributeNewValue New attribute value.\n * @returns A `{ key, value }` object. If `key` is `'class'`, `value` can be a `String` or an\n * array of `String`s. If `key` is `'style'`, `value` is an object with key-value pairs. In other cases, `value` is a `String`.\n *\n * @see module:engine/conversion/downcasthelpers~DowncastHelpers#attributeToAttribute\n */\nexport type AttributeCreatorFunction = (\n\tattributeValue: unknown,\n\tconversionApi: DowncastConversionApi,\n\tdata: {\n\t\titem: ModelItem;\n\t\trange: ModelRange;\n\t\tattributeKey: string;\n\t\tattributeOldValue: unknown;\n\t\tattributeNewValue: unknown;\n\t}\n) => AttributeDescriptor | null;\n\nexport type AttributeDescriptor = {\n\tkey: 'class';\n\tvalue: string | Array<string>;\n} | {\n\tkey: 'style';\n\tvalue: Record<string, string>;\n} | {\n\tkey: Exclude<string, 'class' | 'style'>;\n\tvalue: string;\n};\n\nexport type MarkerElementCreatorFunction = (\n\tdata: {\n\t\tmarkerRange: ModelRange;\n\t\tmarkerName: string;\n\t\tisOpening?: boolean;\n\t},\n\tconversionApi: DowncastConversionApi\n) => UIElement | null;\n\nexport type HighlightDescriptorCreatorFunction = (\n\tdata: {\n\t\tmarkerRange: ModelRange;\n\t\tmarkerName: string;\n\t},\n\tconversionApi: DowncastConversionApi\n) => HighlightDescriptor | null;\n\nexport type AddHighlightCallback = (\n\tviewElement: ViewElement,\n\tdescriptor: HighlightDescriptor,\n\twriter: DowncastWriter\n) => void;\n\nexport type RemoveHighlightCallback = (\n\tviewElement: ViewElement,\n\tid: string,\n\twriter: DowncastWriter\n) => void;\n\nexport type MarkerDataCreatorFunction = (\n\tmarkerName: string,\n\tconversionApi: DowncastConversionApi\n) => { name: string; group: string } | null;\n\n/**\n * A function that is expected to consume all the consumables that were used by the element creator.\n *\n * @callback module:engine/conversion/downcasthelpers~ConsumerFunction\n *\n * @param element The model element to be converted to the view structure.\n * @param consumable The `ModelConsumable` same as in\n * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi#consumable `DowncastConversionApi.consumable`}.\n * @param options.preflight Whether should consume or just check if can be consumed.\n * @returns `true` if all consumable values were available and were consumed, `false` otherwise.\n *\n * @see module:engine/conversion/downcasthelpers~insertStructure\n */\nexport type ConsumerFunction = (\n\telement: ModelElement,\n\tconsumable: ModelConsumable,\n\toptions?: { preflight?: boolean }\n) => boolean | null;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport type Node from '../node';\nimport type Position from '../position';\nimport type Schema from '../schema';\nimport type Writer from '../writer';\n\n/**\n * @module engine/model/utils/autoparagraphing\n */\n\n/**\n * Fixes all empty roots.\n *\n * @internal\n * @param writer The model writer.\n * @returns `true` if any change has been applied, `false` otherwise.\n */\nexport function autoParagraphEmptyRoots( writer: Writer ): boolean {\n\tconst { schema, document } = writer.model;\n\n\tfor ( const root of document.getRoots() ) {\n\t\tif ( root.isEmpty && !schema.checkChild( root, '$text' ) ) {\n\t\t\t// If paragraph element is allowed in the root, create paragraph element.\n\t\t\tif ( schema.checkChild( root, 'paragraph' ) ) {\n\t\t\t\twriter.insertElement( 'paragraph', root );\n\n\t\t\t\t// Other roots will get fixed in the next post-fixer round. Those will be triggered\n\t\t\t\t// in the same batch no matter if this method was triggered by the post-fixing or not\n\t\t\t\t// (the above insertElement call will trigger the post-fixers).\n\t\t\t\treturn true;\n\t\t\t}\n\t\t}\n\t}\n\n\treturn false;\n}\n\n/**\n * Checks if the given node wrapped with a paragraph would be accepted by the schema in the given position.\n *\n * @internal\n * @param position The position at which to check.\n * @param nodeOrType The child node or child type to check.\n * @param schema A schema instance used for element validation.\n */\nexport function isParagraphable(\n\tposition: Position,\n\tnodeOrType: Node | string,\n\tschema: Schema\n): boolean {\n\tconst context = schema.createContext( position );\n\n\t// When paragraph is allowed in this context...\n\tif ( !schema.checkChild( context, 'paragraph' ) ) {\n\t\treturn false;\n\t}\n\n\t// And a node would be allowed in this paragraph...\n\tif ( !schema.checkChild( context.push( 'paragraph' ), nodeOrType ) ) {\n\t\treturn false;\n\t}\n\n\treturn true;\n}\n\n/**\n * Inserts a new paragraph at the given position and returns a position inside that paragraph.\n *\n * @internal\n * @param position The position where a paragraph should be inserted.\n * @param writer The model writer.\n * @returns Position inside the created paragraph.\n */\nexport function wrapInParagraph( position: Position, writer: Writer ): Position {\n\tconst paragraph = writer.createElement( 'paragraph' );\n\n\twriter.insert( paragraph, position );\n\n\treturn writer.createPositionAt( paragraph, 0 );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport Matcher, { type ClassPatterns, type MatcherPattern, type PropertyPatterns } from '../view/matcher';\nimport ConversionHelpers from './conversionhelpers';\n\nimport type { default as UpcastDispatcher, UpcastElementEvent, UpcastConversionApi, UpcastConversionData } from './upcastdispatcher';\nimport type ModelElement from '../model/element';\nimport type ModelRange from '../model/range';\nimport type ModelPosition from '../model/position';\nimport type { ViewDocumentFragment, ViewElement, ViewText } from '../index';\nimport type Mapper from './mapper';\nimport type Model from '../model/model';\nimport type ViewSelection from '../view/selection';\nimport type ViewDocumentSelection from '../view/documentselection';\nimport { isParagraphable, wrapInParagraph } from '../model/utils/autoparagraphing';\n\nimport { priorities, type EventInfo, type PriorityString } from '@ckeditor/ckeditor5-utils';\n\nimport { cloneDeep } from 'lodash-es';\n\n/**\n * Contains the {@link module:engine/view/view view} to {@link module:engine/model/model model} converters for\n * {@link module:engine/conversion/upcastdispatcher~UpcastDispatcher}.\n *\n * @module engine/conversion/upcasthelpers\n */\n\n/**\n * Upcast conversion helper functions.\n *\n * Learn more about {@glink framework/deep-dive/conversion/upcast upcast helpers}.\n *\n * @extends module:engine/conversion/conversionhelpers~ConversionHelpers\n */\nexport default class UpcastHelpers extends ConversionHelpers<UpcastDispatcher> {\n\t/**\n\t * View element to model element conversion helper.\n\t *\n\t * This conversion results in creating a model element. For example,\n\t * view `<p>Foo</p>` becomes `<paragraph>Foo</paragraph>` in the model.\n\t *\n\t * Keep in mind that the element will be inserted only if it is allowed\n\t * by {@link module:engine/model/schema~Schema schema} configuration.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'upcast' ).elementToElement( {\n\t * \tview: 'p',\n\t * \tmodel: 'paragraph'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToElement( {\n\t * \tview: 'p',\n\t * \tmodel: 'paragraph',\n\t * \tconverterPriority: 'high'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToElement( {\n\t * \tview: {\n\t * \t\tname: 'p',\n\t * \t\tclasses: 'fancy'\n\t * \t},\n\t * \tmodel: 'fancyParagraph'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToElement( {\n\t * \tview: {\n\t * \t\tname: 'p',\n\t * \t\tclasses: 'heading'\n\t * \t},\n\t * \tmodel: ( viewElement, conversionApi ) => {\n\t * \t\tconst modelWriter = conversionApi.writer;\n\t *\n\t * \t\treturn modelWriter.createElement( 'heading', { level: viewElement.getAttribute( 'data-level' ) } );\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.view Pattern matching all view elements which should be converted. If not set, the converter\n\t * will fire for every view element.\n\t * @param config.model Name of the model element, a model element instance or a function that takes a view element\n\t * and {@link module:engine/conversion/upcastdispatcher~UpcastConversionApi upcast conversion API}\n\t * and returns a model element. The model element will be inserted in the model.\n\t * @param config.converterPriority Converter priority.\n\t */\n\tpublic elementToElement( config: {\n\t\tview: MatcherPattern;\n\t\tmodel: string | ElementCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( upcastElementToElement( config ) );\n\t}\n\n\t/**\n\t * View element to model attribute conversion helper.\n\t *\n\t * This conversion results in setting an attribute on a model node. For example, view `<strong>Foo</strong>` becomes\n\t * `Foo` {@link module:engine/model/text~Text model text node} with `bold` attribute set to `true`.\n\t *\n\t * This helper is meant to set a model attribute on all the elements that are inside the converted element:\n\t *\n\t * ```\n\t * <strong>Foo</strong> --> <strong><p>Foo</p></strong> --> <paragraph><$text bold=\"true\">Foo</$text></paragraph>\n\t * ```\n\t *\n\t * Above is a sample of HTML code, that goes through autoparagraphing (first step) and then is converted (second step).\n\t * Even though `<strong>` is over `<p>` element, `bold=\"true\"` was added to the text. See\n\t * {@link module:engine/conversion/upcasthelpers~UpcastHelpers#attributeToAttribute} for comparison.\n\t *\n\t * Keep in mind that the attribute will be set only if it is allowed by {@link module:engine/model/schema~Schema schema} configuration.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'upcast' ).elementToAttribute( {\n\t * \tview: 'strong',\n\t * \tmodel: 'bold'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToAttribute( {\n\t * \tview: 'strong',\n\t * \tmodel: 'bold',\n\t * \tconverterPriority: 'high'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToAttribute( {\n\t * \tview: {\n\t * \t\tname: 'span',\n\t * \t\tclasses: 'bold'\n\t * \t},\n\t * \tmodel: 'bold'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToAttribute( {\n\t * \tview: {\n\t * \t\tname: 'span',\n\t * \t\tclasses: [ 'styled', 'styled-dark' ]\n\t * \t},\n\t * \tmodel: {\n\t * \t\tkey: 'styled',\n\t * \t\tvalue: 'dark'\n\t * \t}\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToAttribute( {\n\t * \tview: {\n\t * \t\tname: 'span',\n\t * \t\tstyles: {\n\t * \t\t\t'font-size': /[\\s\\S]+/\n\t * \t\t}\n\t * \t},\n\t * \tmodel: {\n\t * \t\tkey: 'fontSize',\n\t * \t\tvalue: ( viewElement, conversionApi ) => {\n\t * \t\t\tconst fontSize = viewElement.getStyle( 'font-size' );\n\t * \t\t\tconst value = fontSize.substr( 0, fontSize.length - 2 );\n\t *\n\t * \t\t\tif ( value <= 10 ) {\n\t * \t\t\t\treturn 'small';\n\t * \t\t\t} else if ( value > 12 ) {\n\t * \t\t\t\treturn 'big';\n\t * \t\t\t}\n\t *\n\t * \t\t\treturn null;\n\t * \t\t}\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.view Pattern matching all view elements which should be converted.\n\t * @param config.model Model attribute key or an object with `key` and `value` properties, describing\n\t * the model attribute. `value` property may be set as a function that takes a view element and\n\t * {@link module:engine/conversion/upcastdispatcher~UpcastConversionApi upcast conversion API} and returns the value.\n\t * If `String` is given, the model attribute value will be set to `true`.\n\t * @param config.converterPriority Converter priority. Defaults to `low`.\n\t */\n\tpublic elementToAttribute( config: {\n\t\tview: MatcherPattern;\n\t\tmodel: string | {\n\t\t\tkey: string;\n\t\t\tvalue?: unknown;\n\t\t};\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( upcastElementToAttribute( config ) );\n\t}\n\n\t/**\n\t * View attribute to model attribute conversion helper.\n\t *\n\t * This conversion results in setting an attribute on a model node. For example, view `<img src=\"foo.jpg\"></img>` becomes\n\t * `<imageBlock source=\"foo.jpg\"></imageBlock>` in the model.\n\t *\n\t * This helper is meant to convert view attributes from view elements which got converted to the model, so the view attribute\n\t * is set only on the corresponding model node:\n\t *\n\t * ```\n\t * <div class=\"dark\"><div>foo</div></div> --> <div dark=\"true\"><div>foo</div></div>\n\t * ```\n\t *\n\t * Above, `class=\"dark\"` attribute is added only to the `<div>` elements that has it. This is in contrary to\n\t * {@link module:engine/conversion/upcasthelpers~UpcastHelpers#elementToAttribute} which sets attributes for\n\t * all the children in the model:\n\t *\n\t * ```\n\t * <strong>Foo</strong> --> <strong><p>Foo</p></strong> --> <paragraph><$text bold=\"true\">Foo</$text></paragraph>\n\t * ```\n\t *\n\t * Above is a sample of HTML code, that goes through autoparagraphing (first step) and then is converted (second step).\n\t * Even though `<strong>` is over `<p>` element, `bold=\"true\"` was added to the text.\n\t *\n\t * Keep in mind that the attribute will be set only if it is allowed by {@link module:engine/model/schema~Schema schema} configuration.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'upcast' ).attributeToAttribute( {\n\t * \tview: 'src',\n\t * \tmodel: 'source'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).attributeToAttribute( {\n\t * \tview: { key: 'src' },\n\t * \tmodel: 'source'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).attributeToAttribute( {\n\t * \tview: { key: 'src' },\n\t * \tmodel: 'source',\n\t * \tconverterPriority: 'normal'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).attributeToAttribute( {\n\t * \tview: {\n\t * \t\tkey: 'data-style',\n\t * \t\tvalue: /[\\s\\S]+/\n\t * \t},\n\t * \tmodel: 'styled'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).attributeToAttribute( {\n\t * \tview: {\n\t * \t\tname: 'img',\n\t * \t\tkey: 'class',\n\t * \t\tvalue: 'styled-dark'\n\t * \t},\n\t * \tmodel: {\n\t * \t\tkey: 'styled',\n\t * \t\tvalue: 'dark'\n\t * \t}\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).attributeToAttribute( {\n\t * \tview: {\n\t * \t\tkey: 'class',\n\t * \t\tvalue: /styled-[\\S]+/\n\t * \t},\n\t * \tmodel: {\n\t * \t\tkey: 'styled'\n\t * \t\tvalue: ( viewElement, conversionApi ) => {\n\t * \t\t\tconst regexp = /styled-([\\S]+)/;\n\t * \t\t\tconst match = viewElement.getAttribute( 'class' ).match( regexp );\n\t *\n\t * \t\t\treturn match[ 1 ];\n\t * \t\t}\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * Converting styles works a bit differently as it requires `view.styles` to be an object and by default\n\t * a model attribute will be set to `true` by such a converter. You can set the model attribute to any value by providing the `value`\n\t * callback that returns the desired value.\n\t *\n\t * ```ts\n\t * // Default conversion of font-weight style will result in setting bold attribute to true.\n\t * editor.conversion.for( 'upcast' ).attributeToAttribute( {\n\t * \tview: {\n\t * \t\tstyles: {\n\t * \t\t\t'font-weight': 'bold'\n\t * \t\t}\n\t * \t},\n\t * \tmodel: 'bold'\n\t * } );\n\t *\n\t * // This converter will pass any style value to the `lineHeight` model attribute.\n\t * editor.conversion.for( 'upcast' ).attributeToAttribute( {\n\t * \tview: {\n\t * \t\tstyles: {\n\t * \t\t\t'line-height': /[\\s\\S]+/\n\t * \t\t}\n\t * \t},\n\t * \tmodel: {\n\t * \t\tkey: 'lineHeight',\n\t * \t\tvalue: ( viewElement, conversionApi ) => viewElement.getStyle( 'line-height' )\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.view Specifies which view attribute will be converted. If a `String` is passed,\n\t * attributes with given key will be converted. If an `Object` is passed, it must have a required `key` property,\n\t * specifying view attribute key, and may have an optional `value` property, specifying view attribute value and optional `name`\n\t * property specifying a view element name from/on which the attribute should be converted. `value` can be given as a `String`,\n\t * a `RegExp` or a function callback, that takes view attribute value as the only parameter and returns `Boolean`.\n\t * @param config.model Model attribute key or an object with `key` and `value` properties, describing\n\t * the model attribute. `value` property may be set as a function that takes a view element and\n\t * {@link module:engine/conversion/upcastdispatcher~UpcastConversionApi upcast conversion API} and returns the value.\n\t * If `String` is given, the model attribute value will be same as view attribute value.\n\t * @param config.converterPriority Converter priority. Defaults to `low`.\n\t */\n\tpublic attributeToAttribute( config: {\n\t\tview: string | {\n\t\t\tkey: string;\n\t\t\tvalue?: string | RegExp | Array<string> | Record<string, string> | Record<string, RegExp> | ( ( value: unknown ) => boolean );\n\t\t\tname?: string;\n\t\t} | {\n\t\t\tname?: string | RegExp;\n\t\t\tstyles?: PropertyPatterns;\n\t\t\tclasses?: ClassPatterns;\n\t\t\tattributes?: PropertyPatterns;\n\t\t};\n\t\tmodel: string | {\n\t\t\tkey: string;\n\t\t\tvalue: unknown | ( ( viewElement: ViewElement, conversionApi: UpcastConversionApi ) => unknown );\n\t\t\tname?: string;\n\t\t};\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( upcastAttributeToAttribute( config ) );\n\t}\n\n\t/**\n\t * View element to model marker conversion helper.\n\t *\n\t * This conversion results in creating a model marker. For example, if the marker was stored in a view as an element:\n\t * `<p>Fo<span data-marker=\"comment\" data-comment-id=\"7\"></span>o</p><p>B<span data-marker=\"comment\" data-comment-id=\"7\"></span>ar</p>`,\n\t * after the conversion is done, the marker will be available in\n\t * {@link module:engine/model/model~Model#markers model document markers}.\n\t *\n\t * **Note**: When this helper is used in the data upcast in combination with\n\t * {@link module:engine/conversion/downcasthelpers~DowncastHelpers#markerToData `#markerToData()`} in the data downcast,\n\t * then invalid HTML code (e.g. a span between table cells) may be produced by the latter converter.\n\t *\n\t * In most of the cases, the {@link #dataToMarker} should be used instead.\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'upcast' ).elementToMarker( {\n\t * \tview: 'marker-search',\n\t * \tmodel: 'search'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToMarker( {\n\t * \tview: 'marker-search',\n\t * \tmodel: 'search',\n\t * \tconverterPriority: 'high'\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToMarker( {\n\t * \tview: 'marker-search',\n\t * \tmodel: ( viewElement, conversionApi ) => 'comment:' + viewElement.getAttribute( 'data-comment-id' )\n\t * } );\n\t *\n\t * editor.conversion.for( 'upcast' ).elementToMarker( {\n\t * \tview: {\n\t * \t\tname: 'span',\n\t * \t\tattributes: {\n\t * \t\t\t'data-marker': 'search'\n\t * \t\t}\n\t * \t},\n\t * \tmodel: 'search'\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.view Pattern matching all view elements which should be converted.\n\t * @param config.model Name of the model marker, or a function that takes a view element and returns\n\t * a model marker name.\n\t * @param config.converterPriority Converter priority.\n\t */\n\tpublic elementToMarker( config: {\n\t\tview: MatcherPattern;\n\t\tmodel: string | MarkerFromElementCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( upcastElementToMarker( config ) );\n\t}\n\n\t/**\n\t * View-to-model marker conversion helper.\n\t *\n\t * Converts view data created by {@link module:engine/conversion/downcasthelpers~DowncastHelpers#markerToData `#markerToData()`}\n\t * back to a model marker.\n\t *\n\t * This converter looks for specific view elements and view attributes that mark marker boundaries. See\n\t * {@link module:engine/conversion/downcasthelpers~DowncastHelpers#markerToData `#markerToData()`} to learn what view data\n\t * is expected by this converter.\n\t *\n\t * The `config.view` property is equal to the marker group name to convert.\n\t *\n\t * By default, this converter creates markers with the `group:name` name convention (to match the default `markerToData` conversion).\n\t *\n\t * The conversion configuration can take a function that will generate a marker name.\n\t * If such function is set as the `config.model` parameter, it is passed the `name` part from the view element or attribute and it is\n\t * expected to return a string with the marker name.\n\t *\n\t * Basic usage:\n\t *\n\t * ```ts\n\t * // Using the default conversion.\n\t * // In this case, all markers from the `comment` group will be converted.\n\t * // The conversion will look for `<comment-start>` and `<comment-end>` tags and\n\t * // `data-comment-start-before`, `data-comment-start-after`,\n\t * // `data-comment-end-before` and `data-comment-end-after` attributes.\n\t * editor.conversion.for( 'upcast' ).dataToMarker( {\n\t * \tview: 'comment'\n\t * } );\n\t * ```\n\t *\n\t * An example of a model that may be generated by this conversion:\n\t *\n\t * ```\n\t * // View:\n\t * <p>Foo<comment-start name=\"commentId:uid\"></comment-start>bar</p>\n\t * <figure data-comment-end-after=\"commentId:uid\" class=\"image\"><img src=\"abc.jpg\" /></figure>\n\t *\n\t * // Model:\n\t * <paragraph>Foo[bar</paragraph>\n\t * <imageBlock src=\"abc.jpg\"></imageBlock>]\n\t * ```\n\t *\n\t * Where `[]` are boundaries of a marker that will receive the `comment:commentId:uid` name.\n\t *\n\t * Other examples of usage:\n\t *\n\t * ```ts\n\t * // Using a custom function which is the same as the default conversion:\n\t * editor.conversion.for( 'upcast' ).dataToMarker( {\n\t * \tview: 'comment',\n\t * \tmodel: ( name, conversionApi ) => 'comment:' + name,\n\t * } );\n\t *\n\t * // Using the converter priority:\n\t * editor.conversion.for( 'upcast' ).dataToMarker( {\n\t * \tview: 'comment',\n\t * \tmodel: ( name, conversionApi ) => 'comment:' + name,\n\t * \tconverterPriority: 'high'\n\t * } );\n\t * ```\n\t *\n\t * See {@link module:engine/conversion/conversion~Conversion#for `conversion.for()`} to learn how to add a converter\n\t * to the conversion process.\n\t *\n\t * @param config Conversion configuration.\n\t * @param config.view The marker group name to convert.\n\t * @param config.model A function that takes the `name` part from the view element or attribute and\n\t * {@link module:engine/conversion/upcastdispatcher~UpcastConversionApi upcast conversion API} and returns the marker name.\n\t * @param config.converterPriority Converter priority.\n\t */\n\tpublic dataToMarker( config: {\n\t\tview: string;\n\t\tmodel?: MarkerFromAttributeCreatorFunction;\n\t\tconverterPriority?: PriorityString;\n\t} ): this {\n\t\treturn this.add( upcastDataToMarker( config ) );\n\t}\n}\n\n/**\n * Function factory, creates a converter that converts {@link module:engine/view/documentfragment~DocumentFragment view document fragment}\n * or all children of {@link module:engine/view/element~Element} into\n * {@link module:engine/model/documentfragment~DocumentFragment model document fragment}.\n * This is the \"entry-point\" converter for upcast (view to model conversion). This converter starts the conversion of all children\n * of passed view document fragment. Those children {@link module:engine/view/node~Node view nodes} are then handled by other converters.\n *\n * This also a \"default\", last resort converter for all view elements that has not been converted by other converters.\n * When a view element is being converted to the model but it does not have converter specified, that view element\n * will be converted to {@link module:engine/model/documentfragment~DocumentFragment model document fragment} and returned.\n *\n * @returns Universal converter for view {@link module:engine/view/documentfragment~DocumentFragment fragments} and\n * {@link module:engine/view/element~Element elements} that returns\n * {@link module:engine/model/documentfragment~DocumentFragment model fragment} with children of converted view item.\n */\nexport function convertToModelFragment() {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: UpcastConversionData<ViewElement | ViewDocumentFragment>,\n\t\tconversionApi: UpcastConversionApi\n\t): void => {\n\t\t// Second argument in `consumable.consume` is discarded for ViewDocumentFragment but is needed for ViewElement.\n\t\tif ( !data.modelRange && conversionApi.consumable.consume( data.viewItem, { name: true } ) ) {\n\t\t\tconst { modelRange, modelCursor } = conversionApi.convertChildren( data.viewItem, data.modelCursor );\n\n\t\t\tdata.modelRange = modelRange;\n\t\t\tdata.modelCursor = modelCursor;\n\t\t}\n\t};\n}\n\n/**\n * Function factory, creates a converter that converts {@link module:engine/view/text~Text} to {@link module:engine/model/text~Text}.\n *\n * @returns {@link module:engine/view/text~Text View text} converter.\n */\nexport function convertText() {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: UpcastConversionData<ViewText>,\n\t\t{ schema, consumable, writer }: UpcastConversionApi\n\t): void => {\n\t\tlet position = data.modelCursor;\n\n\t\t// When node is already converted then do nothing.\n\t\tif ( !consumable.test( data.viewItem ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !schema.checkChild( position, '$text' ) ) {\n\t\t\tif ( !isParagraphable( position, '$text', schema ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Do not auto-paragraph whitespaces.\n\t\t\tif ( data.viewItem.data.trim().length == 0 ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Wrap `$text` in paragraph and include any marker that is directly before `$text`. See #13053.\n\t\t\tconst nodeBefore = position.nodeBefore;\n\n\t\t\tposition = wrapInParagraph( position, writer );\n\n\t\t\tif ( nodeBefore && nodeBefore.is( 'element', '$marker' ) ) {\n\t\t\t\t// Move `$marker` to the paragraph.\n\t\t\t\twriter.move( writer.createRangeOn( nodeBefore ), position );\n\t\t\t\tposition = writer.createPositionAfter( nodeBefore );\n\t\t\t}\n\t\t}\n\n\t\tconsumable.consume( data.viewItem );\n\n\t\tconst text = writer.createText( data.viewItem.data );\n\n\t\twriter.insert( text, position );\n\n\t\tdata.modelRange = writer.createRange(\n\t\t\tposition,\n\t\t\tposition.getShiftedBy( text.offsetSize )\n\t\t);\n\t\tdata.modelCursor = data.modelRange.end;\n\t};\n}\n\n/**\n * Function factory, creates a callback function which converts a {@link module:engine/view/selection~Selection\n * view selection} taken from the {@link module:engine/view/document~Document#event:selectionChange} event\n * and sets in on the {@link module:engine/model/document~Document#selection model}.\n *\n * **Note**: because there is no view selection change dispatcher nor any other advanced view selection to model\n * conversion mechanism, the callback should be set directly on view document.\n *\n * ```ts\n * view.document.on( 'selectionChange', convertSelectionChange( modelDocument, mapper ) );\n * ```\n *\n * @param model Data model.\n * @param mapper Conversion mapper.\n * @returns {@link module:engine/view/document~Document#event:selectionChange} callback function.\n */\nexport function convertSelectionChange( model: Model, mapper: Mapper ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: { newSelection: ViewSelection | ViewDocumentSelection }\n\t): void => {\n\t\tconst viewSelection = data.newSelection;\n\n\t\tconst ranges: Array<ModelRange> = [];\n\n\t\tfor ( const viewRange of viewSelection.getRanges() ) {\n\t\t\tranges.push( mapper.toModelRange( viewRange ) );\n\t\t}\n\n\t\tconst modelSelection = model.createSelection( ranges, { backward: viewSelection.isBackward } );\n\n\t\tif ( !modelSelection.isEqual( model.document.selection ) ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\twriter.setSelection( modelSelection );\n\t\t\t} );\n\t\t}\n\t};\n}\n\n/**\n * View element to model element conversion helper.\n *\n * See {@link ~UpcastHelpers#elementToElement `.elementToElement()` upcast helper} for examples.\n *\n * @param config Conversion configuration.\n * @param config.view Pattern matching all view elements which should be converted. If not\n * set, the converter will fire for every view element.\n * @param config.model Name of the model element, a model element\n * instance or a function that takes a view element and returns a model element. The model element will be inserted in the model.\n * @param config.converterPriority Converter priority.\n * @returns Conversion helper.\n */\nfunction upcastElementToElement( config: {\n\tview: MatcherPattern;\n\tmodel: string | ElementCreatorFunction;\n\tconverterPriority?: PriorityString;\n} ) {\n\tconfig = cloneDeep( config );\n\n\tconst converter = prepareToElementConverter( config );\n\n\tconst elementName = getViewElementNameFromConfig( config.view );\n\tconst eventName = elementName ? `element:${ elementName }` as const : 'element';\n\n\treturn ( dispatcher: UpcastDispatcher ) => {\n\t\tdispatcher.on<UpcastElementEvent>( eventName, converter, { priority: config.converterPriority || 'normal' } );\n\t};\n}\n\n/**\n * View element to model attribute conversion helper.\n *\n * See {@link ~UpcastHelpers#elementToAttribute `.elementToAttribute()` upcast helper} for examples.\n *\n * @param config Conversion configuration.\n * @param config.view Pattern matching all view elements which should be converted.\n * @param config.model Model attribute key or an object with `key` and `value` properties, describing\n * the model attribute. `value` property may be set as a function that takes a view element and returns the value.\n * If `String` is given, the model attribute value will be set to `true`.\n * @param config.converterPriority Converter priority. Defaults to `low`.\n * @returns Conversion helper.\n */\nfunction upcastElementToAttribute( config: {\n\tview: MatcherPattern;\n\tmodel: string | {\n\t\tkey: string;\n\t\tvalue?: unknown | AttributeCreatorFunction;\n\t};\n\tconverterPriority?: PriorityString;\n} ) {\n\tconfig = cloneDeep( config );\n\n\tnormalizeModelAttributeConfig( config );\n\n\tconst converter = prepareToAttributeConverter( config as any, false );\n\n\tconst elementName = getViewElementNameFromConfig( config.view );\n\tconst eventName = elementName ? `element:${ elementName }` as const : 'element';\n\n\treturn ( dispatcher: UpcastDispatcher ) => {\n\t\tdispatcher.on<UpcastElementEvent>( eventName, converter, { priority: config.converterPriority || 'low' } );\n\t};\n}\n\n/**\n * View attribute to model attribute conversion helper.\n *\n * See {@link ~UpcastHelpers#attributeToAttribute `.attributeToAttribute()` upcast helper} for examples.\n *\n * @param config Conversion configuration.\n * @param config.view Specifies which view attribute will be converted. If a `String` is passed,\n * attributes with given key will be converted. If an `Object` is passed, it must have a required `key` property,\n * specifying view attribute key, and may have an optional `value` property, specifying view attribute value and optional `name`\n * property specifying a view element name from/on which the attribute should be converted. `value` can be given as a `String`,\n * a `RegExp` or a function callback, that takes view attribute value as the only parameter and returns `Boolean`.\n * @param config.model Model attribute key or an object with `key` and `value` properties, describing\n * the model attribute. `value` property may be set as a function that takes a view element and returns the value.\n * If `String` is given, the model attribute value will be same as view attribute value.\n * @param config.converterPriority Converter priority. Defaults to `low`.\n * @returns Conversion helper.\n */\nfunction upcastAttributeToAttribute( config: {\n\tview: string | {\n\t\tkey?: string;\n\t\tvalue?: string | RegExp | Array<string> | Record<string, string> | Record<string, RegExp> | ( ( value: unknown ) => boolean );\n\t\tname?: string | RegExp;\n\t\tstyles?: PropertyPatterns;\n\t\tclasses?: ClassPatterns;\n\t\tattributes?: PropertyPatterns;\n\t};\n\tmodel: string | {\n\t\tkey: string;\n\t\tvalue: unknown | ( ( viewElement: ViewElement, conversionApi: UpcastConversionApi ) => unknown );\n\t};\n\tconverterPriority?: PriorityString;\n} ) {\n\tconfig = cloneDeep( config );\n\n\tlet viewKey: string | null = null;\n\n\tif ( typeof config.view == 'string' || config.view.key ) {\n\t\tviewKey = normalizeViewAttributeKeyValueConfig( config );\n\t}\n\n\tnormalizeModelAttributeConfig( config, viewKey );\n\n\tconst converter = prepareToAttributeConverter( config as any, true );\n\n\treturn ( dispatcher: UpcastDispatcher ) => {\n\t\tdispatcher.on<UpcastElementEvent>( 'element', converter, { priority: config.converterPriority || 'low' } );\n\t};\n}\n\n/**\n * View element to model marker conversion helper.\n *\n * See {@link ~UpcastHelpers#elementToMarker `.elementToMarker()` upcast helper} for examples.\n *\n * @param config Conversion configuration.\n * @param config.view Pattern matching all view elements which should be converted.\n * @param config.model Name of the model marker, or a function that takes a view element and returns\n * a model marker name.\n * @param config.converterPriority Converter priority.\n * @returns Conversion helper.\n */\nfunction upcastElementToMarker( config: {\n\tview: MatcherPattern;\n\tmodel: string | MarkerFromElementCreatorFunction;\n\tconverterPriority?: PriorityString;\n} ) {\n\tconst model = normalizeElementToMarkerModelConfig( config.model );\n\n\treturn upcastElementToElement( { ...config, model } );\n}\n\n/**\n * View data to model marker conversion helper.\n *\n * See {@link ~UpcastHelpers#dataToMarker} to learn more.\n *\n * @returns Conversion helper.\n */\nfunction upcastDataToMarker( config: {\n\tview: string;\n\tmodel?: MarkerFromAttributeCreatorFunction;\n\tconverterPriority?: PriorityString;\n} ) {\n\tconfig = cloneDeep( config );\n\n\t// Default conversion.\n\tif ( !config.model ) {\n\t\tconfig.model = name => {\n\t\t\treturn name ? config.view + ':' + name : config.view;\n\t\t};\n\t}\n\n\tconst normalizedConfig = {\n\t\tview: config.view,\n\t\tmodel: config.model!\n\t};\n\n\tconst converterStart = prepareToElementConverter( normalizeDataToMarkerConfig( normalizedConfig, 'start' ) );\n\tconst converterEnd = prepareToElementConverter( normalizeDataToMarkerConfig( normalizedConfig, 'end' ) );\n\n\treturn ( dispatcher: UpcastDispatcher ): void => {\n\t\tdispatcher.on<UpcastElementEvent>(\n\t\t\t`element:${ config.view }-start`,\n\t\t\tconverterStart,\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\t\tdispatcher.on<UpcastElementEvent>(\n\t\t\t`element:${ config.view }-end`,\n\t\t\tconverterEnd,\n\t\t\t{ priority: config.converterPriority || 'normal' }\n\t\t);\n\n\t\t// Below is a hack that is needed to properly handle `converterPriority` for both elements and attributes.\n\t\t// Attribute conversion needs to be performed *after* element conversion.\n\t\t// This converter handles both element conversion and attribute conversion, which means that if a single\n\t\t// `config.converterPriority` is used, it will lead to problems. For example, if the `'high'` priority is used,\n\t\t// the attribute conversion will be performed before a lot of element upcast converters.\n\t\t// On the other hand, we want to support `config.converterPriority` and converter overwriting.\n\t\t//\n\t\t// To make it work, we need to do some extra processing for priority for attribute converter.\n\t\t// Priority `'low'` value should be the base value and then we will change it depending on `config.converterPriority` value.\n\t\t//\n\t\t// This hack probably would not be needed if attributes are upcasted separately.\n\t\t//\n\t\tconst basePriority = priorities.low;\n\t\tconst maxPriority = priorities.highest;\n\t\tconst priorityFactor = priorities.get( config.converterPriority ) / maxPriority; // Number in range [ -1, 1 ].\n\n\t\tdispatcher.on<UpcastElementEvent>(\n\t\t\t'element',\n\t\t\tupcastAttributeToMarker( normalizedConfig ),\n\t\t\t{ priority: basePriority + priorityFactor }\n\t\t);\n\t};\n}\n\n/**\n * Function factory, returns a callback function which converts view attributes to a model marker.\n *\n * The converter looks for elements with `data-group-start-before`, `data-group-start-after`, `data-group-end-before`\n * and `data-group-end-after` attributes and inserts `$marker` model elements before/after those elements.\n * `group` part is specified in `config.view`.\n *\n * @returns Marker converter.\n */\nfunction upcastAttributeToMarker( config: {\n\tview: string;\n\tmodel: MarkerFromAttributeCreatorFunction;\n} ) {\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: UpcastConversionData<ViewElement>,\n\t\tconversionApi: UpcastConversionApi\n\t) => {\n\t\tconst attrName = `data-${ config.view }`;\n\n\t\t// Check if any attribute for the given view item can be consumed before changing the conversion data\n\t\t// and consuming view items with these attributes.\n\t\tif (\n\t\t\t!conversionApi.consumable.test( data.viewItem, { attributes: attrName + '-end-after' } ) &&\n\t\t\t!conversionApi.consumable.test( data.viewItem, { attributes: attrName + '-start-after' } ) &&\n\t\t\t!conversionApi.consumable.test( data.viewItem, { attributes: attrName + '-end-before' } ) &&\n\t\t\t!conversionApi.consumable.test( data.viewItem, { attributes: attrName + '-start-before' } )\n\t\t) {\n\t\t\treturn;\n\t\t}\n\n\t\t// This converter wants to add a model element, marking a marker, before/after an element (or maybe even group of elements).\n\t\t// To do that, we can use `data.modelRange` which is set on an element (or a group of elements) that has been upcasted.\n\t\t// But, if the processed view element has not been upcasted yet (it does not have been converted), we need to\n\t\t// fire conversion for its children first, then we will have `data.modelRange` available.\n\t\tif ( !data.modelRange ) {\n\t\t\tObject.assign( data, conversionApi.convertChildren( data.viewItem, data.modelCursor ) );\n\t\t}\n\n\t\tif ( conversionApi.consumable.consume( data.viewItem, { attributes: attrName + '-end-after' } ) ) {\n\t\t\taddMarkerElements( data.modelRange!.end, data.viewItem.getAttribute( attrName + '-end-after' )!.split( ',' ) );\n\t\t}\n\n\t\tif ( conversionApi.consumable.consume( data.viewItem, { attributes: attrName + '-start-after' } ) ) {\n\t\t\taddMarkerElements( data.modelRange!.end, data.viewItem.getAttribute( attrName + '-start-after' )!.split( ',' ) );\n\t\t}\n\n\t\tif ( conversionApi.consumable.consume( data.viewItem, { attributes: attrName + '-end-before' } ) ) {\n\t\t\taddMarkerElements( data.modelRange!.start, data.viewItem.getAttribute( attrName + '-end-before' )!.split( ',' ) );\n\t\t}\n\n\t\tif ( conversionApi.consumable.consume( data.viewItem, { attributes: attrName + '-start-before' } ) ) {\n\t\t\taddMarkerElements( data.modelRange!.start, data.viewItem.getAttribute( attrName + '-start-before' )!.split( ',' ) );\n\t\t}\n\n\t\tfunction addMarkerElements( position: ModelPosition, markerViewNames: Array<string> ): void {\n\t\t\tfor ( const markerViewName of markerViewNames ) {\n\t\t\t\tconst markerName = config.model( markerViewName, conversionApi );\n\t\t\t\tconst element = conversionApi.writer.createElement( '$marker', { 'data-name': markerName } );\n\n\t\t\t\tconversionApi.writer.insert( element, position );\n\n\t\t\t\tif ( data.modelCursor.isEqual( position ) ) {\n\t\t\t\t\tdata.modelCursor = data.modelCursor.getShiftedBy( 1 );\n\t\t\t\t} else {\n\t\t\t\t\tdata.modelCursor = data.modelCursor._getTransformedByInsertion( position, 1 );\n\t\t\t\t}\n\n\t\t\t\tdata.modelRange = data.modelRange!._getTransformedByInsertion( position, 1 )[ 0 ];\n\t\t\t}\n\t\t}\n\t};\n}\n\n/**\n * Helper function for from-view-element conversion. Checks if `config.view` directly specifies converted view element's name\n * and if so, returns it.\n *\n * @param config Conversion view config.\n * @returns View element name or `null` if name is not directly set.\n */\nfunction getViewElementNameFromConfig( viewConfig: any ): string | null {\n\tif ( typeof viewConfig == 'string' ) {\n\t\treturn viewConfig;\n\t}\n\n\tif ( typeof viewConfig == 'object' && typeof viewConfig.name == 'string' ) {\n\t\treturn viewConfig.name;\n\t}\n\n\treturn null;\n}\n\n/**\n * Helper for to-model-element conversion. Takes a config object and returns a proper converter function.\n *\n * @param config Conversion configuration.\n * @returns View to model converter.\n */\nfunction prepareToElementConverter( config: {\n\tview: MatcherPattern;\n\tmodel: string | ElementCreatorFunction;\n} ) {\n\tconst matcher = new Matcher( config.view );\n\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: UpcastConversionData<ViewElement>,\n\t\tconversionApi: UpcastConversionApi\n\t): void => {\n\t\tconst matcherResult = matcher.match( data.viewItem );\n\n\t\tif ( !matcherResult ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst match = matcherResult.match;\n\n\t\t// Force consuming element's name.\n\t\tmatch.name = true;\n\n\t\tif ( !conversionApi.consumable.test( data.viewItem, match ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst modelElement = getModelElement( config.model, data.viewItem, conversionApi );\n\n\t\tif ( !modelElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !conversionApi.safeInsert( modelElement, data.modelCursor ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconversionApi.consumable.consume( data.viewItem, match );\n\t\tconversionApi.convertChildren( data.viewItem, modelElement );\n\t\tconversionApi.updateConversionResult( modelElement, data );\n\t};\n}\n\n/**\n * Helper function for upcasting-to-element converter. Takes the model configuration, the converted view element\n * and a writer instance and returns a model element instance to be inserted in the model.\n *\n * @param model Model conversion configuration.\n * @param input The converted view node.\n * @param conversionApi The upcast conversion API.\n */\nfunction getModelElement(\n\tmodel: string | ElementCreatorFunction,\n\tinput: ViewElement,\n\tconversionApi: UpcastConversionApi\n): ModelElement | null {\n\tif ( model instanceof Function ) {\n\t\treturn model( input, conversionApi );\n\t} else {\n\t\treturn conversionApi.writer.createElement( model );\n\t}\n}\n\n/**\n * Helper function view-attribute-to-model-attribute helper. Normalizes `config.view` which was set as `String` or\n * as an `Object` with `key`, `value` and `name` properties. Normalized `config.view` has is compatible with\n * {@link module:engine/view/matcher~MatcherPattern}.\n *\n * @param config Conversion config.\n * @returns Key of the converted view attribute.\n */\nfunction normalizeViewAttributeKeyValueConfig( config: any ) {\n\tif ( typeof config.view == 'string' ) {\n\t\tconfig.view = { key: config.view };\n\t}\n\n\tconst key: string = config.view.key;\n\tlet normalized: MatcherPattern;\n\n\tif ( key == 'class' || key == 'style' ) {\n\t\tconst keyName = key == 'class' ? 'classes' : 'styles';\n\n\t\tnormalized = {\n\t\t\t[ keyName ]: config.view.value\n\t\t};\n\t} else {\n\t\tconst value = typeof config.view.value == 'undefined' ? /[\\s\\S]*/ : config.view.value;\n\n\t\tnormalized = {\n\t\t\tattributes: {\n\t\t\t\t[ key ]: value\n\t\t\t}\n\t\t};\n\t}\n\n\tif ( config.view.name ) {\n\t\tnormalized.name = config.view.name;\n\t}\n\n\tconfig.view = normalized;\n\n\treturn key;\n}\n\n/**\n * Helper function that normalizes `config.model` in from-model-attribute conversion. `config.model` can be set\n * as a `String`, an `Object` with only `key` property or an `Object` with `key` and `value` properties. Normalized\n * `config.model` is an `Object` with `key` and `value` properties.\n *\n * @param config Conversion config.\n * @param viewAttributeKeyToCopy Key of the converted view attribute. If it is set, model attribute value\n * will be equal to view attribute value.\n */\nfunction normalizeModelAttributeConfig( config: any, viewAttributeKeyToCopy: string | null = null ) {\n\tconst defaultModelValue = viewAttributeKeyToCopy === null ? true :\n\t\t( viewElement: ViewElement ) => viewElement.getAttribute( viewAttributeKeyToCopy );\n\n\tconst key = typeof config.model != 'object' ? config.model : config.model.key;\n\tconst value = typeof config.model != 'object' || typeof config.model.value == 'undefined' ? defaultModelValue : config.model.value;\n\n\tconfig.model = { key, value };\n}\n\n/**\n * Helper for to-model-attribute conversion. Takes the model attribute name and conversion configuration and returns\n * a proper converter function.\n *\n * @param config Conversion configuration. It is possible to provide multiple configurations in an array.\n * @param shallow If set to `true` the attribute will be set only on top-level nodes. Otherwise, it will be set\n * on all elements in the range.\n */\nfunction prepareToAttributeConverter(\n\tconfig: {\n\t\tview: MatcherPattern;\n\t\tmodel: {\n\t\t\tkey: string;\n\t\t\tvalue: AttributeCreatorFunction | unknown;\n\t\t};\n\t},\n\tshallow: boolean\n) {\n\tconst matcher = new Matcher( config.view );\n\n\treturn (\n\t\tevt: EventInfo,\n\t\tdata: UpcastConversionData<ViewElement>,\n\t\tconversionApi: UpcastConversionApi\n\t): void => {\n\t\t// Converting an attribute of an element that has not been converted to anything does not make sense\n\t\t// because there will be nowhere to set that attribute on. At this stage, the element should've already\n\t\t// been converted (https://github.com/ckeditor/ckeditor5/issues/11000).\n\t\tif ( !data.modelRange && shallow ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst match = matcher.match( data.viewItem );\n\n\t\t// If there is no match, this callback should not do anything.\n\t\tif ( !match ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( onlyViewNameIsDefined( config.view, data.viewItem ) ) {\n\t\t\tmatch.match.name = true;\n\t\t} else {\n\t\t\t// Do not test `name` consumable because it could get consumed already while upcasting some other attribute\n\t\t\t// on the same element (for example <span class=\"big\" style=\"color: red\">foo</span>).\n\t\t\tdelete match.match.name;\n\t\t}\n\n\t\t// Try to consume appropriate values from consumable values list.\n\t\tif ( !conversionApi.consumable.test( data.viewItem, match.match ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst modelKey = config.model.key;\n\t\tconst modelValue: unknown = typeof config.model.value == 'function' ?\n\t\t\tconfig.model.value( data.viewItem, conversionApi ) : config.model.value;\n\n\t\t// Do not convert if attribute building function returned falsy value.\n\t\tif ( modelValue === null ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Since we are converting to attribute we need a range on which we will set the attribute.\n\t\t// If the range is not created yet, let's create it by converting children of the current node first.\n\t\tif ( !data.modelRange ) {\n\t\t\t// Convert children and set conversion result as a current data.\n\t\t\tObject.assign( data, conversionApi.convertChildren( data.viewItem, data.modelCursor ) );\n\t\t}\n\n\t\t// Set attribute on current `output`. `Schema` is checked inside this helper function.\n\t\tconst attributeWasSet = setAttributeOn( data.modelRange!, { key: modelKey, value: modelValue }, shallow, conversionApi );\n\n\t\t// It may happen that a converter will try to set an attribute that is not allowed in the given context.\n\t\t// In such a situation we cannot consume the attribute. See: https://github.com/ckeditor/ckeditor5/pull/9249#issuecomment-815658459.\n\t\tif ( attributeWasSet ) {\n\t\t\t// Verify if the element itself wasn't consumed yet. It could be consumed already while upcasting some other attribute\n\t\t\t// on the same element (for example <span class=\"big\" style=\"color: red\">foo</span>).\n\t\t\t// We need to consume it so other features (especially GHS) won't try to convert it.\n\t\t\t// Note that it's not tested by the other element-to-attribute converters whether an element was consumed before\n\t\t\t// (in case of converters that the element itself is just a context and not the primary information to convert).\n\t\t\tif ( conversionApi.consumable.test( data.viewItem, { name: true } ) ) {\n\t\t\t\tmatch.match.name = true;\n\t\t\t}\n\n\t\t\tconversionApi.consumable.consume( data.viewItem, match.match );\n\t\t}\n\t};\n}\n\n/**\n * Helper function that checks if element name should be consumed in attribute converters.\n *\n * @param viewConfig Conversion view config.\n */\nfunction onlyViewNameIsDefined( viewConfig: any, viewItem: ViewElement ): boolean {\n\t// https://github.com/ckeditor/ckeditor5-engine/issues/1786\n\tconst configToTest = typeof viewConfig == 'function' ? viewConfig( viewItem ) : viewConfig;\n\n\tif ( typeof configToTest == 'object' && !getViewElementNameFromConfig( configToTest ) ) {\n\t\treturn false;\n\t}\n\n\treturn !configToTest.classes && !configToTest.attributes && !configToTest.styles;\n}\n\n/**\n * Helper function for to-model-attribute converter. Sets model attribute on given range. Checks {@link module:engine/model/schema~Schema}\n * to ensure proper model structure.\n *\n * If any node on the given range has already defined an attribute with the same name, its value will not be updated.\n *\n * @param modelRange Model range on which attribute should be set.\n * @param modelAttribute Model attribute to set.\n * @param conversionApi Conversion API.\n * @param shallow If set to `true` the attribute will be set only on top-level nodes. Otherwise, it will be set\n * on all elements in the range.\n * @returns `true` if attribute was set on at least one node from given `modelRange`.\n */\nfunction setAttributeOn(\n\tmodelRange: ModelRange,\n\tmodelAttribute: {\n\t\tkey: string;\n\t\tvalue: unknown;\n\t},\n\tshallow: boolean,\n\tconversionApi: UpcastConversionApi\n): boolean {\n\tlet result = false;\n\n\t// Set attribute on each item in range according to Schema.\n\tfor ( const node of Array.from( modelRange.getItems( { shallow } ) ) ) {\n\t\t// Skip if not allowed.\n\t\tif ( !conversionApi.schema.checkAttribute( node, modelAttribute.key ) ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\t// Mark the node as consumed even if the attribute will not be updated because it's in a valid context (schema)\n\t\t// and would be converted if the attribute wouldn't be present. See #8921.\n\t\tresult = true;\n\n\t\t// Do not override the attribute if it's already present.\n\t\tif ( node.hasAttribute( modelAttribute.key ) ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tconversionApi.writer.setAttribute( modelAttribute.key, modelAttribute.value, node );\n\t}\n\n\treturn result;\n}\n\n/**\n * Helper function for upcasting-to-marker conversion. Takes the config in a format requested by `upcastElementToMarker()`\n * function and converts it to a format that is supported by `upcastElementToElement()` function.\n */\nfunction normalizeElementToMarkerModelConfig( model: string | MarkerFromElementCreatorFunction ): ElementCreatorFunction {\n\treturn ( viewElement, conversionApi ) => {\n\t\tconst markerName = typeof model == 'string' ? model : model( viewElement, conversionApi );\n\n\t\treturn conversionApi.writer.createElement( '$marker', { 'data-name': markerName } );\n\t};\n}\n\n/**\n * Helper function for upcasting-to-marker conversion. Takes the config in a format requested by `upcastDataToMarker()`\n * function and converts it to a format that is supported by `upcastElementToElement()` function.\n */\nfunction normalizeDataToMarkerConfig(\n\tconfig: {\n\t\tview: string;\n\t\tmodel: MarkerFromAttributeCreatorFunction;\n\t},\n\ttype: string\n) {\n\tconst elementCreatorFunction: ElementCreatorFunction = ( viewElement, conversionApi ) => {\n\t\tconst viewName = viewElement.getAttribute( 'name' )!;\n\t\tconst markerName = config.model( viewName, conversionApi );\n\n\t\treturn conversionApi.writer.createElement( '$marker', { 'data-name': markerName } );\n\t};\n\n\treturn {\n\t\t// Upcast <markerGroup-start> and <markerGroup-end> elements.\n\t\tview: `${ config.view }-${ type }`,\n\t\tmodel: elementCreatorFunction\n\t};\n}\n\nexport type ElementCreatorFunction = (\n\tviewElement: ViewElement,\n\tconversionApi: UpcastConversionApi\n) => ModelElement | null;\n\nexport type AttributeCreatorFunction = (\n\tmodelElement: ModelElement,\n\tconversionApi: UpcastConversionApi\n) => unknown;\n\nexport type MarkerFromElementCreatorFunction = (\n\tviewElement: ViewElement,\n\tconversionApi: UpcastConversionApi\n) => string;\n\nexport type MarkerFromAttributeCreatorFunction = (\n\tattributeValue: string,\n\tconversionApi: UpcastConversionApi\n) => string;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/utils/selection-post-fixer\n */\n\nimport Position from '../position';\nimport Range from '../range';\n\nimport type DocumentFragment from '../documentfragment';\nimport type Model from '../model';\nimport type Node from '../node';\nimport type Schema from '../schema';\nimport type Writer from '../writer';\nimport type Element from '../element';\n\n/**\n * Injects selection post-fixer to the model.\n *\n * The role of the selection post-fixer is to ensure that the selection is in a correct place\n * after a {@link module:engine/model/model~Model#change `change()`} block was executed.\n *\n * The correct position means that:\n *\n * * All collapsed selection ranges are in a place where the {@link module:engine/model/schema~Schema}\n * allows a `$text`.\n * * None of the selection's non-collapsed ranges crosses a {@link module:engine/model/schema~Schema#isLimit limit element}\n * boundary (a range must be rooted within one limit element).\n * * Only {@link module:engine/model/schema~Schema#isSelectable selectable elements} can be selected from the outside\n * (e.g. `[<paragraph>foo</paragraph>]` is invalid). This rule applies independently to both selection ends, so this\n * selection is correct: `<paragraph>f[oo</paragraph><imageBlock></imageBlock>]`.\n *\n * If the position is not correct, the post-fixer will automatically correct it.\n *\n * ## Fixing a non-collapsed selection\n *\n * See as an example a selection that starts in a P1 element and ends inside the text of a TD element\n * (`[` and `]` are range boundaries and `(l)` denotes an element defined as `isLimit=true`):\n *\n * ```\n * root\n * |- element P1\n * | |- \"foo\" root\n * |- element TABLE (l) P1 TABLE P2\n * | |- element TR (l) f o[o TR TR b a r\n * | | |- element TD (l) TD TD\n * | | |- \"aaa\" a]a a b b b\n * | |- element TR (l)\n * | | |- element TD (l) ||\n * | | |- \"bbb\" ||\n * |- element P2 VV\n * | |- \"bar\"\n * root\n * P1 TABLE] P2\n * f o[o TR TR b a r\n * TD TD\n * a a a b b b\n * ```\n *\n * In the example above, the TABLE, TR and TD are defined as `isLimit=true` in the schema. The range which is not contained within\n * a single limit element must be expanded to select the outermost limit element. The range end is inside the text node of the TD element.\n * As the TD element is a child of the TR and TABLE elements, where both are defined as `isLimit=true` in the schema, the range must be\n * expanded to select the whole TABLE element.\n *\n * **Note** If the selection contains multiple ranges, the method returns a minimal set of ranges that are not intersecting after expanding\n * them to select `isLimit=true` elements.\n */\nexport function injectSelectionPostFixer( model: Model ): void {\n\tmodel.document.registerPostFixer( writer => selectionPostFixer( writer, model ) );\n}\n\n/**\n * The selection post-fixer.\n */\nfunction selectionPostFixer( writer: Writer, model: Model ): boolean {\n\tconst selection = model.document.selection;\n\tconst schema = model.schema;\n\n\tconst ranges = [];\n\n\tlet wasFixed = false;\n\n\tfor ( const modelRange of selection.getRanges() ) {\n\t\t// Go through all ranges in selection and try fixing each of them.\n\t\t// Those ranges might overlap but will be corrected later.\n\t\tconst correctedRange = tryFixingRange( modelRange, schema );\n\n\t\t// \"Selection fixing\" algorithms sometimes get lost. In consequence, it may happen\n\t\t// that a new range is returned but, in fact, it has the same positions as the original\n\t\t// range anyway. If this range is not discarded, a new selection will be set and that,\n\t\t// for instance, would destroy the selection attributes. Let's make sure that the post-fixer\n\t\t// actually worked first before setting a new selection.\n\t\t//\n\t\t// https://github.com/ckeditor/ckeditor5/issues/6693\n\t\tif ( correctedRange && !correctedRange.isEqual( modelRange ) ) {\n\t\t\tranges.push( correctedRange );\n\t\t\twasFixed = true;\n\t\t} else {\n\t\t\tranges.push( modelRange );\n\t\t}\n\t}\n\n\t// If any of ranges were corrected update the selection.\n\tif ( wasFixed ) {\n\t\twriter.setSelection( mergeIntersectingRanges( ranges ), { backward: selection.isBackward } );\n\t}\n\n\treturn false;\n}\n\n/**\n * Tries fixing a range if it's incorrect.\n *\n * **Note:** This helper is used by the selection post-fixer and to fix the `beforeinput` target ranges.\n *\n * @returns Returns fixed range or null if range is valid.\n */\nexport function tryFixingRange( range: Range, schema: Schema ): Range | null {\n\tif ( range.isCollapsed ) {\n\t\treturn tryFixingCollapsedRange( range, schema );\n\t}\n\n\treturn tryFixingNonCollapsedRage( range, schema );\n}\n\n/**\n * Tries to fix collapsed ranges.\n *\n * * Fixes situation when a range is in a place where $text is not allowed\n *\n * @param range Collapsed range to fix.\n * @returns Returns fixed range or null if range is valid.\n */\nfunction tryFixingCollapsedRange( range: Range, schema: Schema ) {\n\tconst originalPosition = range.start;\n\n\tconst nearestSelectionRange = schema.getNearestSelectionRange( originalPosition );\n\n\t// This might be null, i.e. when the editor data is empty or the selection is inside a limit element\n\t// that doesn't allow text inside.\n\t// In the first case, there is no need to fix the selection range.\n\t// In the second, let's go up to the outer selectable element\n\tif ( !nearestSelectionRange ) {\n\t\tconst ancestorObject = originalPosition.getAncestors().reverse().find( ( item ): item is Element => schema.isObject( item ) );\n\n\t\tif ( ancestorObject ) {\n\t\t\treturn Range._createOn( ancestorObject );\n\t\t}\n\n\t\treturn null;\n\t}\n\n\tif ( !nearestSelectionRange.isCollapsed ) {\n\t\treturn nearestSelectionRange;\n\t}\n\n\tconst fixedPosition = nearestSelectionRange.start;\n\n\t// Fixed position is the same as original - no need to return corrected range.\n\tif ( originalPosition.isEqual( fixedPosition ) ) {\n\t\treturn null;\n\t}\n\n\treturn new Range( fixedPosition );\n}\n\n/**\n * Tries to fix an expanded range.\n *\n * @param range Expanded range to fix.\n * @returns Returns fixed range or null if range is valid.\n */\nfunction tryFixingNonCollapsedRage( range: Range, schema: Schema ) {\n\tconst { start, end } = range;\n\n\tconst isTextAllowedOnStart = schema.checkChild( start, '$text' );\n\tconst isTextAllowedOnEnd = schema.checkChild( end, '$text' );\n\n\tconst startLimitElement = schema.getLimitElement( start );\n\tconst endLimitElement = schema.getLimitElement( end );\n\n\t// Ranges which both end are inside the same limit element (or root) might needs only minor fix.\n\tif ( startLimitElement === endLimitElement ) {\n\t\t// Range is valid when both position allows to place a text:\n\t\t// - <block>f[oobarba]z</block>\n\t\t// This would be \"fixed\" by a next check but as it will be the same it's better to return null so the selection stays the same.\n\t\tif ( isTextAllowedOnStart && isTextAllowedOnEnd ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// Range that is on non-limit element (or is partially) must be fixed so it is placed inside the block around $text:\n\t\t// - [<block>foo</block>] -> <block>[foo]</block>\n\t\t// - [<block>foo]</block> -> <block>[foo]</block>\n\t\t// - <block>f[oo</block>] -> <block>f[oo]</block>\n\t\t// - [<block>foo</block><selectable></selectable>] -> <block>[foo</block><selectable></selectable>]\n\t\tif ( checkSelectionOnNonLimitElements( start, end, schema ) ) {\n\t\t\tconst isStartBeforeSelectable = start.nodeAfter && schema.isSelectable( start.nodeAfter );\n\t\t\tconst fixedStart = isStartBeforeSelectable ? null : schema.getNearestSelectionRange( start, 'forward' );\n\n\t\t\tconst isEndAfterSelectable = end.nodeBefore && schema.isSelectable( end.nodeBefore );\n\t\t\tconst fixedEnd = isEndAfterSelectable ? null : schema.getNearestSelectionRange( end, 'backward' );\n\n\t\t\t// The schema.getNearestSelectionRange might return null - if that happens use original position.\n\t\t\tconst rangeStart = fixedStart ? fixedStart.start : start;\n\t\t\tconst rangeEnd = fixedEnd ? fixedEnd.end : end;\n\n\t\t\treturn new Range( rangeStart, rangeEnd );\n\t\t}\n\t}\n\n\tconst isStartInLimit = startLimitElement && !startLimitElement.is( 'rootElement' );\n\tconst isEndInLimit = endLimitElement && !endLimitElement.is( 'rootElement' );\n\n\t// At this point we eliminated valid positions on text nodes so if one of range positions is placed inside a limit element\n\t// then the range crossed limit element boundaries and needs to be fixed.\n\tif ( isStartInLimit || isEndInLimit ) {\n\t\tconst bothInSameParent = ( start.nodeAfter && end.nodeBefore ) && start.nodeAfter.parent === end.nodeBefore.parent;\n\n\t\tconst expandStart = isStartInLimit && ( !bothInSameParent || !isSelectable( start.nodeAfter, schema ) );\n\t\tconst expandEnd = isEndInLimit && ( !bothInSameParent || !isSelectable( end.nodeBefore, schema ) );\n\n\t\t// Although we've already found limit element on start/end positions we must find the outer-most limit element.\n\t\t// as limit elements might be nested directly inside (ie table > tableRow > tableCell).\n\t\tlet fixedStart = start;\n\t\tlet fixedEnd = end;\n\n\t\tif ( expandStart ) {\n\t\t\tfixedStart = Position._createBefore( findOutermostLimitAncestor( startLimitElement, schema ) );\n\t\t}\n\n\t\tif ( expandEnd ) {\n\t\t\tfixedEnd = Position._createAfter( findOutermostLimitAncestor( endLimitElement, schema ) );\n\t\t}\n\n\t\treturn new Range( fixedStart, fixedEnd );\n\t}\n\n\t// Range was not fixed at this point so it is valid - ie it was placed around limit element already.\n\treturn null;\n}\n\n/**\n * Finds the outer-most ancestor.\n */\nfunction findOutermostLimitAncestor( startingNode: Node, schema: Schema ): Node {\n\tlet isLimitNode = startingNode;\n\tlet parent: Node | DocumentFragment = isLimitNode;\n\n\t// Find outer most isLimit block as such blocks might be nested (ie. in tables).\n\twhile ( schema.isLimit( parent ) && parent.parent ) {\n\t\tisLimitNode = parent;\n\t\tparent = parent.parent;\n\t}\n\n\treturn isLimitNode;\n}\n\n/**\n * Checks whether any of range boundaries is placed around non-limit elements.\n */\nfunction checkSelectionOnNonLimitElements( start: Position, end: Position, schema: Schema ) {\n\tconst startIsOnBlock = ( start.nodeAfter && !schema.isLimit( start.nodeAfter ) ) || schema.checkChild( start, '$text' );\n\tconst endIsOnBlock = ( end.nodeBefore && !schema.isLimit( end.nodeBefore ) ) || schema.checkChild( end, '$text' );\n\n\t// We should fix such selection when one of those nodes needs fixing.\n\treturn startIsOnBlock || endIsOnBlock;\n}\n\n/**\n * Returns a minimal non-intersecting array of ranges without duplicates.\n *\n * @param ranges Ranges to merge.\n * @returns Array of unique and non-intersecting ranges.\n */\nexport function mergeIntersectingRanges( ranges: Array<Range> ): Array<Range> {\n\tconst rangesToMerge = [ ...ranges ];\n\tconst rangeIndexesToRemove = new Set();\n\tlet currentRangeIndex = 1;\n\n\twhile ( currentRangeIndex < rangesToMerge.length ) {\n\t\tconst currentRange = rangesToMerge[ currentRangeIndex ];\n\t\tconst previousRanges = rangesToMerge.slice( 0, currentRangeIndex );\n\n\t\tfor ( const [ previousRangeIndex, previousRange ] of previousRanges.entries() ) {\n\t\t\tif ( rangeIndexesToRemove.has( previousRangeIndex ) ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tif ( currentRange.isEqual( previousRange ) ) {\n\t\t\t\trangeIndexesToRemove.add( previousRangeIndex );\n\t\t\t} else if ( currentRange.isIntersecting( previousRange ) ) {\n\t\t\t\trangeIndexesToRemove.add( previousRangeIndex );\n\t\t\t\trangeIndexesToRemove.add( currentRangeIndex );\n\n\t\t\t\tconst mergedRange = currentRange.getJoined( previousRange );\n\t\t\t\trangesToMerge.push( mergedRange! );\n\t\t\t}\n\t\t}\n\n\t\tcurrentRangeIndex++;\n\t}\n\n\tconst nonIntersectingRanges = rangesToMerge.filter( ( _, index ) => !rangeIndexesToRemove.has( index ) );\n\n\treturn nonIntersectingRanges;\n}\n\n/**\n * Checks if node exists and if it's a selectable.\n */\nfunction isSelectable( node: Node, schema: Schema ) {\n\treturn node && schema.isSelectable( node );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/controller/editingcontroller\n */\n\nimport {\n\tCKEditorError,\n\tObservableMixin,\n\tenv,\n\ttype GetCallback\n} from '@ckeditor/ckeditor5-utils';\n\nimport RootEditableElement from '../view/rooteditableelement';\nimport View from '../view/view';\nimport Mapper from '../conversion/mapper';\nimport DowncastDispatcher, {\n\ttype DowncastInsertEvent,\n\ttype DowncastRemoveEvent,\n\ttype DowncastSelectionEvent,\n\ttype DowncastCleanSelectionEvent\n} from '../conversion/downcastdispatcher';\nimport {\n\tcleanSelection,\n\tconvertCollapsedSelection,\n\tconvertRangeSelection,\n\tinsertAttributesAndChildren,\n\tinsertText,\n\tremove\n} from '../conversion/downcasthelpers';\n\nimport { convertSelectionChange } from '../conversion/upcasthelpers';\n\nimport { tryFixingRange } from '../model/utils/selection-post-fixer';\n\nimport type { default as Model, AfterChangesEvent, BeforeChangesEvent } from '../model/model';\nimport type ModelItem from '../model/item';\nimport type ModelText from '../model/text';\nimport type ModelTextProxy from '../model/textproxy';\nimport type Schema from '../model/schema';\nimport type { DocumentChangeEvent } from '../model/document';\nimport type { Marker } from '../model/markercollection';\nimport type { StylesProcessor } from '../view/stylesmap';\nimport type { ViewDocumentSelectionChangeEvent } from '../view/observer/selectionobserver';\nimport type { ViewDocumentInputEvent } from '../view/observer/inputobserver';\n\n// @if CK_DEBUG_ENGINE // const { dumpTrees, initDocumentDumping } = require( '../dev-utils/utils' );\n\n/**\n * A controller for the editing pipeline. The editing pipeline controls the {@link ~EditingController#model model} rendering,\n * including selection handling. It also creates the {@link ~EditingController#view view} which builds a\n * browser-independent virtualization over the DOM elements. The editing controller also attaches default converters.\n */\nexport default class EditingController extends ObservableMixin() {\n\t/**\n\t * Editor model.\n\t */\n\tpublic readonly model: Model;\n\n\t/**\n\t * Editing view controller.\n\t */\n\tpublic readonly view: View;\n\n\t/**\n\t * A mapper that describes the model-view binding.\n\t */\n\tpublic readonly mapper: Mapper;\n\n\t/**\n\t * Downcast dispatcher that converts changes from the model to the {@link #view editing view}.\n\t */\n\tpublic readonly downcastDispatcher: DowncastDispatcher;\n\n\t/**\n\t * Creates an editing controller instance.\n\t *\n\t * @param model Editing model.\n\t * @param stylesProcessor The styles processor instance.\n\t */\n\tconstructor( model: Model, stylesProcessor: StylesProcessor ) {\n\t\tsuper();\n\n\t\tthis.model = model;\n\t\tthis.view = new View( stylesProcessor );\n\t\tthis.mapper = new Mapper();\n\n\t\tthis.downcastDispatcher = new DowncastDispatcher( {\n\t\t\tmapper: this.mapper,\n\t\t\tschema: model.schema\n\t\t} );\n\n\t\tconst doc = this.model.document;\n\t\tconst selection = doc.selection;\n\t\tconst markers = this.model.markers;\n\n\t\t// When plugins listen on model changes (on selection change, post fixers, etc.) and change the view as a result of\n\t\t// the model's change, they might trigger view rendering before the conversion is completed (e.g. before the selection\n\t\t// is converted). We disable rendering for the length of the outermost model change() block to prevent that.\n\t\t//\n\t\t// See https://github.com/ckeditor/ckeditor5-engine/issues/1528\n\t\tthis.listenTo<BeforeChangesEvent>( this.model, '_beforeChanges', () => {\n\t\t\tthis.view._disableRendering( true );\n\t\t}, { priority: 'highest' } );\n\n\t\tthis.listenTo<AfterChangesEvent>( this.model, '_afterChanges', () => {\n\t\t\tthis.view._disableRendering( false );\n\t\t}, { priority: 'lowest' } );\n\n\t\t// Whenever model document is changed, convert those changes to the view (using model.Document#differ).\n\t\t// Do it on 'low' priority, so changes are converted after other listeners did their job.\n\t\t// Also convert model selection.\n\t\tthis.listenTo<DocumentChangeEvent>( doc, 'change', () => {\n\t\t\tthis.view.change( writer => {\n\t\t\t\tthis.downcastDispatcher.convertChanges( doc.differ, markers, writer );\n\t\t\t\tthis.downcastDispatcher.convertSelection( selection, markers, writer );\n\t\t\t} );\n\t\t}, { priority: 'low' } );\n\n\t\t// Convert selection from the view to the model when it changes in the view.\n\t\tthis.listenTo<ViewDocumentSelectionChangeEvent>( this.view.document, 'selectionChange',\n\t\t\tconvertSelectionChange( this.model, this.mapper )\n\t\t);\n\n\t\t// Fix `beforeinput` target ranges so that they map to the valid model ranges.\n\t\tthis.listenTo<ViewDocumentInputEvent>( this.view.document, 'beforeinput',\n\t\t\tfixTargetRanges( this.mapper, this.model.schema, this.view ),\n\t\t\t{ priority: 'high' }\n\t\t);\n\n\t\t// Attach default model converters.\n\t\tthis.downcastDispatcher.on<DowncastInsertEvent<ModelText | ModelTextProxy>>( 'insert:$text', insertText(), { priority: 'lowest' } );\n\t\tthis.downcastDispatcher.on<DowncastInsertEvent>( 'insert', insertAttributesAndChildren(), { priority: 'lowest' } );\n\t\tthis.downcastDispatcher.on<DowncastRemoveEvent>( 'remove', remove(), { priority: 'low' } );\n\n\t\t// Attach default model selection converters.\n\t\tthis.downcastDispatcher.on<DowncastCleanSelectionEvent>( 'cleanSelection', cleanSelection() );\n\t\tthis.downcastDispatcher.on<DowncastSelectionEvent>( 'selection', convertRangeSelection(), { priority: 'low' } );\n\t\tthis.downcastDispatcher.on<DowncastSelectionEvent>( 'selection', convertCollapsedSelection(), { priority: 'low' } );\n\n\t\t// Binds {@link module:engine/view/document~Document#roots view roots collection} to\n\t\t// {@link module:engine/model/document~Document#roots model roots collection} so creating\n\t\t// model root automatically creates corresponding view root.\n\t\tthis.view.document.roots.bindTo( this.model.document.roots ).using( root => {\n\t\t\t// $graveyard is a special root that has no reflection in the view.\n\t\t\tif ( root.rootName == '$graveyard' ) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tconst viewRoot = new RootEditableElement( this.view.document, root.name );\n\n\t\t\tviewRoot.rootName = root.rootName;\n\t\t\tthis.mapper.bindElements( root, viewRoot );\n\n\t\t\treturn viewRoot;\n\t\t} );\n\n\t\t// @if CK_DEBUG_ENGINE // initDocumentDumping( this.model.document );\n\t\t// @if CK_DEBUG_ENGINE // initDocumentDumping( this.view.document );\n\n\t\t// @if CK_DEBUG_ENGINE // dumpTrees( this.model.document, this.model.document.version );\n\t\t// @if CK_DEBUG_ENGINE // dumpTrees( this.view.document, this.model.document.version );\n\n\t\t// @if CK_DEBUG_ENGINE // this.model.document.on( 'change', () => {\n\t\t// @if CK_DEBUG_ENGINE //\tdumpTrees( this.view.document, this.model.document.version );\n\t\t// @if CK_DEBUG_ENGINE // }, { priority: 'lowest' } );\n\t}\n\n\t/**\n\t * Removes all event listeners attached to the `EditingController`. Destroys all objects created\n\t * by `EditingController` that need to be destroyed.\n\t */\n\tpublic destroy(): void {\n\t\tthis.view.destroy();\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Calling this method will refresh the marker by triggering the downcast conversion for it.\n\t *\n\t * Reconverting the marker is useful when you want to change its {@link module:engine/view/element~Element view element}\n\t * without changing any marker data. For instance:\n\t *\n\t * ```ts\n\t * let isCommentActive = false;\n\t *\n\t * model.conversion.markerToHighlight( {\n\t * \tmodel: 'comment',\n\t * \tview: data => {\n\t * \t\tconst classes = [ 'comment-marker' ];\n\t *\n\t * \t\tif ( isCommentActive ) {\n\t * \t\t\tclasses.push( 'comment-marker--active' );\n\t * \t\t}\n\t *\n\t * \t\treturn { classes };\n\t * \t}\n\t * } );\n\t *\n\t * // ...\n\t *\n\t * // Change the property that indicates if marker is displayed as active or not.\n\t * isCommentActive = true;\n\t *\n\t * // Reconverting will downcast and synchronize the marker with the new isCommentActive state value.\n\t * editor.editing.reconvertMarker( 'comment' );\n\t * ```\n\t *\n\t * **Note**: If you want to reconvert a model item, use {@link #reconvertItem} instead.\n\t *\n\t * @param markerOrName Name of a marker to update, or a marker instance.\n\t */\n\tpublic reconvertMarker( markerOrName: Marker | string ): void {\n\t\tconst markerName = typeof markerOrName == 'string' ? markerOrName : markerOrName.name;\n\t\tconst currentMarker = this.model.markers.get( markerName );\n\n\t\tif ( !currentMarker ) {\n\t\t\t/**\n\t\t\t * The marker with the provided name does not exist and cannot be reconverted.\n\t\t\t *\n\t\t\t * @error editingcontroller-reconvertmarker-marker-not-exist\n\t\t\t * @param {String} markerName The name of the reconverted marker.\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'editingcontroller-reconvertmarker-marker-not-exist', this, { markerName } );\n\t\t}\n\n\t\tthis.model.change( () => {\n\t\t\tthis.model.markers._refresh( currentMarker );\n\t\t} );\n\t}\n\n\t/**\n\t * Calling this method will downcast a model item on demand (by requesting a refresh in the {@link module:engine/model/differ~Differ}).\n\t *\n\t * You can use it if you want the view representation of a specific item updated as a response to external modifications. For instance,\n\t * when the view structure depends not only on the associated model data but also on some external state.\n\t *\n\t * **Note**: If you want to reconvert a model marker, use {@link #reconvertMarker} instead.\n\t *\n\t * @param item Item to refresh.\n\t */\n\tpublic reconvertItem( item: ModelItem ): void {\n\t\tthis.model.change( () => {\n\t\t\tthis.model.document.differ._refreshItem( item );\n\t\t} );\n\t}\n}\n\n/**\n * Checks whether the target ranges provided by the `beforeInput` event can be properly mapped to model ranges and fixes them if needed.\n *\n * This is using the same logic as the selection post-fixer.\n */\nfunction fixTargetRanges( mapper: Mapper, schema: Schema, view: View ): GetCallback<ViewDocumentInputEvent> {\n\treturn ( evt, data ) => {\n\t\t// The Renderer is disabled while composing on non-android browsers, so we can't be sure that target ranges\n\t\t// could be properly mapped to view and model because the DOM and view tree drifted apart.\n\t\tif ( view.document.isComposing && !env.isAndroid ) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor ( let i = 0; i < data.targetRanges.length; i++ ) {\n\t\t\tconst viewRange = data.targetRanges[ i ];\n\t\t\tconst modelRange = mapper.toModelRange( viewRange );\n\t\t\tconst correctedRange = tryFixingRange( modelRange, schema );\n\n\t\t\tif ( !correctedRange || correctedRange.isEqual( modelRange ) ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tdata.targetRanges[ i ] = mapper.toViewRange( correctedRange );\n\t\t}\n\t};\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/conversion/viewconsumable\n */\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type Element from '../view/element';\nimport type Node from '../view/node';\nimport type Text from '../view/text';\nimport type DocumentFragment from '../view/documentfragment';\nimport type { Match } from '../view/matcher';\n\nimport { isArray } from 'lodash-es';\n\n/**\n * Class used for handling consumption of view {@link module:engine/view/element~Element elements},\n * {@link module:engine/view/text~Text text nodes} and {@link module:engine/view/documentfragment~DocumentFragment document fragments}.\n * Element's name and its parts (attributes, classes and styles) can be consumed separately. Consuming an element's name\n * does not consume its attributes, classes and styles.\n * To add items for consumption use {@link module:engine/conversion/viewconsumable~ViewConsumable#add add method}.\n * To test items use {@link module:engine/conversion/viewconsumable~ViewConsumable#test test method}.\n * To consume items use {@link module:engine/conversion/viewconsumable~ViewConsumable#consume consume method}.\n * To revert already consumed items use {@link module:engine/conversion/viewconsumable~ViewConsumable#revert revert method}.\n *\n * ```ts\n * viewConsumable.add( element, { name: true } ); // Adds element's name as ready to be consumed.\n * viewConsumable.add( textNode ); // Adds text node for consumption.\n * viewConsumable.add( docFragment ); // Adds document fragment for consumption.\n * viewConsumable.test( element, { name: true } ); // Tests if element's name can be consumed.\n * viewConsumable.test( textNode ); // Tests if text node can be consumed.\n * viewConsumable.test( docFragment ); // Tests if document fragment can be consumed.\n * viewConsumable.consume( element, { name: true } ); // Consume element's name.\n * viewConsumable.consume( textNode ); // Consume text node.\n * viewConsumable.consume( docFragment ); // Consume document fragment.\n * viewConsumable.revert( element, { name: true } ); // Revert already consumed element's name.\n * viewConsumable.revert( textNode ); // Revert already consumed text node.\n * viewConsumable.revert( docFragment ); // Revert already consumed document fragment.\n * ```\n */\nexport default class ViewConsumable {\n\t/**\n\t * Map of consumable elements. If {@link module:engine/view/element~Element element} is used as a key,\n\t * {@link module:engine/conversion/viewconsumable~ViewElementConsumables ViewElementConsumables} instance is stored as value.\n\t * For {@link module:engine/view/text~Text text nodes} and\n\t * {@link module:engine/view/documentfragment~DocumentFragment document fragments} boolean value is stored as value.\n\t */\n\tprivate _consumables = new Map<Node | DocumentFragment, ViewElementConsumables | boolean>();\n\n\t/**\n\t * Adds {@link module:engine/view/text~Text text node} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment document fragment} as ready to be consumed.\n\t *\n\t * ```ts\n\t * viewConsumable.add( textNode ); // Adds text node to consume.\n\t * viewConsumable.add( docFragment ); // Adds document fragment to consume.\n\t * ```\n\t *\n\t * See also: {@link #add:ELEMENT `add( element, consumables )`}.\n\t *\n\t * @label TEXT_OR_FRAGMENT\n\t */\n\tpublic add(\n\t\ttextOrDocumentFragment: Text | DocumentFragment\n\t): void;\n\n\t/**\n\t * Adds {@link module:engine/view/element~Element view element} as ready to be consumed.\n\t *\n\t * ```ts\n\t * viewConsumable.add( p, { name: true } ); // Adds element's name to consume.\n\t * viewConsumable.add( p, { attributes: 'name' } ); // Adds element's attribute.\n\t * viewConsumable.add( p, { classes: 'foobar' } ); // Adds element's class.\n\t * viewConsumable.add( p, { styles: 'color' } ); // Adds element's style\n\t * viewConsumable.add( p, { attributes: 'name', styles: 'color' } ); // Adds attribute and style.\n\t * viewConsumable.add( p, { classes: [ 'baz', 'bar' ] } ); // Multiple consumables can be provided.\n\t * ```\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `viewconsumable-invalid-attribute` when `class` or `style`\n\t * attribute is provided - it should be handled separately by providing actual style/class.\n\t *\n\t * ```ts\n\t * viewConsumable.add( p, { attributes: 'style' } ); // This call will throw an exception.\n\t * viewConsumable.add( p, { styles: 'color' } ); // This is properly handled style.\n\t * ```\n\t *\n\t * See also: {@link #add:TEXT_OR_FRAGMENT `add( textOrDocumentFragment )`}.\n\t *\n\t * @label ELEMENT\n\t * @param consumables Used only if first parameter is {@link module:engine/view/element~Element view element} instance.\n\t * @param consumables.name If set to true element's name will be included.\n\t * @param consumables.attributes Attribute name or array of attribute names.\n\t * @param consumables.classes Class name or array of class names.\n\t * @param consumables.styles Style name or array of style names.\n\t */\n\tpublic add(\n\t\telement: Element,\n\t\tconsumables: Consumables\n\t): void;\n\n\tpublic add(\n\t\telement: Node | DocumentFragment,\n\t\tconsumables?: Consumables\n\t): void {\n\t\tlet elementConsumables: ViewElementConsumables;\n\n\t\t// For text nodes and document fragments just mark them as consumable.\n\t\tif ( element.is( '$text' ) || element.is( 'documentFragment' ) ) {\n\t\t\tthis._consumables.set( element, true );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// For elements create new ViewElementConsumables or update already existing one.\n\t\tif ( !this._consumables.has( element ) ) {\n\t\t\telementConsumables = new ViewElementConsumables( element );\n\t\t\tthis._consumables.set( element, elementConsumables );\n\t\t} else {\n\t\t\telementConsumables = this._consumables.get( element ) as any;\n\t\t}\n\n\t\telementConsumables.add( consumables! );\n\t}\n\n\t/**\n\t * Tests if {@link module:engine/view/element~Element view element}, {@link module:engine/view/text~Text text node} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment document fragment} can be consumed.\n\t * It returns `true` when all items included in method's call can be consumed. Returns `false` when\n\t * first already consumed item is found and `null` when first non-consumable item is found.\n\t *\n\t * ```ts\n\t * viewConsumable.test( p, { name: true } ); // Tests element's name.\n\t * viewConsumable.test( p, { attributes: 'name' } ); // Tests attribute.\n\t * viewConsumable.test( p, { classes: 'foobar' } ); // Tests class.\n\t * viewConsumable.test( p, { styles: 'color' } ); // Tests style.\n\t * viewConsumable.test( p, { attributes: 'name', styles: 'color' } ); // Tests attribute and style.\n\t * viewConsumable.test( p, { classes: [ 'baz', 'bar' ] } ); // Multiple consumables can be tested.\n\t * viewConsumable.test( textNode ); // Tests text node.\n\t * viewConsumable.test( docFragment ); // Tests document fragment.\n\t * ```\n\t *\n\t * Testing classes and styles as attribute will test if all added classes/styles can be consumed.\n\t *\n\t * ```ts\n\t * viewConsumable.test( p, { attributes: 'class' } ); // Tests if all added classes can be consumed.\n\t * viewConsumable.test( p, { attributes: 'style' } ); // Tests if all added styles can be consumed.\n\t * ```\n\t *\n\t * @param consumables Used only if first parameter is {@link module:engine/view/element~Element view element} instance.\n\t * @param consumables.name If set to true element's name will be included.\n\t * @param consumables.attributes Attribute name or array of attribute names.\n\t * @param consumables.classes Class name or array of class names.\n\t * @param consumables.styles Style name or array of style names.\n\t * @returns Returns `true` when all items included in method's call can be consumed. Returns `false`\n\t * when first already consumed item is found and `null` when first non-consumable item is found.\n\t */\n\tpublic test( element: Node | DocumentFragment, consumables?: Consumables | Match ): boolean | null {\n\t\tconst elementConsumables = this._consumables.get( element );\n\n\t\tif ( elementConsumables === undefined ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// For text nodes and document fragments return stored boolean value.\n\t\tif ( element.is( '$text' ) || element.is( 'documentFragment' ) ) {\n\t\t\treturn elementConsumables as boolean;\n\t\t}\n\n\t\t// For elements test consumables object.\n\t\treturn ( elementConsumables as ViewElementConsumables ).test( consumables! );\n\t}\n\n\t/**\n\t * Consumes {@link module:engine/view/element~Element view element}, {@link module:engine/view/text~Text text node} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment document fragment}.\n\t * It returns `true` when all items included in method's call can be consumed, otherwise returns `false`.\n\t *\n\t * ```ts\n\t * viewConsumable.consume( p, { name: true } ); // Consumes element's name.\n\t * viewConsumable.consume( p, { attributes: 'name' } ); // Consumes element's attribute.\n\t * viewConsumable.consume( p, { classes: 'foobar' } ); // Consumes element's class.\n\t * viewConsumable.consume( p, { styles: 'color' } ); // Consumes element's style.\n\t * viewConsumable.consume( p, { attributes: 'name', styles: 'color' } ); // Consumes attribute and style.\n\t * viewConsumable.consume( p, { classes: [ 'baz', 'bar' ] } ); // Multiple consumables can be consumed.\n\t * viewConsumable.consume( textNode ); // Consumes text node.\n\t * viewConsumable.consume( docFragment ); // Consumes document fragment.\n\t * ```\n\t *\n\t * Consuming classes and styles as attribute will test if all added classes/styles can be consumed.\n\t *\n\t * ```ts\n\t * viewConsumable.consume( p, { attributes: 'class' } ); // Consume only if all added classes can be consumed.\n\t * viewConsumable.consume( p, { attributes: 'style' } ); // Consume only if all added styles can be consumed.\n\t * ```\n\t *\n\t * @param consumables Used only if first parameter is {@link module:engine/view/element~Element view element} instance.\n\t * @param consumables.name If set to true element's name will be included.\n\t * @param consumables.attributes Attribute name or array of attribute names.\n\t * @param consumables.classes Class name or array of class names.\n\t * @param consumables.styles Style name or array of style names.\n\t * @returns Returns `true` when all items included in method's call can be consumed,\n\t * otherwise returns `false`.\n\t */\n\tpublic consume( element: Node | DocumentFragment, consumables?: Consumables | Match ): boolean {\n\t\tif ( this.test( element, consumables ) ) {\n\t\t\tif ( element.is( '$text' ) || element.is( 'documentFragment' ) ) {\n\t\t\t\t// For text nodes and document fragments set value to false.\n\t\t\t\tthis._consumables.set( element, false );\n\t\t\t} else {\n\t\t\t\t// For elements - consume consumables object.\n\t\t\t\t( this._consumables.get( element ) as ViewElementConsumables ).consume( consumables! );\n\t\t\t}\n\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Reverts {@link module:engine/view/element~Element view element}, {@link module:engine/view/text~Text text node} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment document fragment} so they can be consumed once again.\n\t * Method does not revert items that were never previously added for consumption, even if they are included in\n\t * method's call.\n\t *\n\t * ```ts\n\t * viewConsumable.revert( p, { name: true } ); // Reverts element's name.\n\t * viewConsumable.revert( p, { attributes: 'name' } ); // Reverts element's attribute.\n\t * viewConsumable.revert( p, { classes: 'foobar' } ); // Reverts element's class.\n\t * viewConsumable.revert( p, { styles: 'color' } ); // Reverts element's style.\n\t * viewConsumable.revert( p, { attributes: 'name', styles: 'color' } ); // Reverts attribute and style.\n\t * viewConsumable.revert( p, { classes: [ 'baz', 'bar' ] } ); // Multiple names can be reverted.\n\t * viewConsumable.revert( textNode ); // Reverts text node.\n\t * viewConsumable.revert( docFragment ); // Reverts document fragment.\n\t * ```\n\t *\n\t * Reverting classes and styles as attribute will revert all classes/styles that were previously added for\n\t * consumption.\n\t *\n\t * ```ts\n\t * viewConsumable.revert( p, { attributes: 'class' } ); // Reverts all classes added for consumption.\n\t * viewConsumable.revert( p, { attributes: 'style' } ); // Reverts all styles added for consumption.\n\t * ```\n\t *\n\t * @param consumables Used only if first parameter is {@link module:engine/view/element~Element view element} instance.\n\t * @param consumables.name If set to true element's name will be included.\n\t * @param consumables.attributes Attribute name or array of attribute names.\n\t * @param consumables.classes Class name or array of class names.\n\t * @param consumables.styles Style name or array of style names.\n\t */\n\tpublic revert( element: Node, consumables: Consumables ): void {\n\t\tconst elementConsumables = this._consumables.get( element );\n\n\t\tif ( elementConsumables !== undefined ) {\n\t\t\tif ( element.is( '$text' ) || element.is( 'documentFragment' ) ) {\n\t\t\t\t// For text nodes and document fragments - set consumable to true.\n\t\t\t\tthis._consumables.set( element, true );\n\t\t\t} else {\n\t\t\t\t// For elements - revert items from consumables object.\n\t\t\t\t( elementConsumables as ViewElementConsumables ).revert( consumables );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Creates consumable object from {@link module:engine/view/element~Element view element}. Consumable object will include\n\t * element's name and all its attributes, classes and styles.\n\t */\n\tpublic static consumablesFromElement( element: Element ): Consumables & { element: Element } {\n\t\tconst consumables = {\n\t\t\telement,\n\t\t\tname: true,\n\t\t\tattributes: [] as Array<string>,\n\t\t\tclasses: [] as Array<string>,\n\t\t\tstyles: [] as Array<string>\n\t\t};\n\n\t\tconst attributes = element.getAttributeKeys();\n\n\t\tfor ( const attribute of attributes ) {\n\t\t\t// Skip classes and styles - will be added separately.\n\t\t\tif ( attribute == 'style' || attribute == 'class' ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconsumables.attributes.push( attribute );\n\t\t}\n\n\t\tconst classes = element.getClassNames();\n\n\t\tfor ( const className of classes ) {\n\t\t\tconsumables.classes.push( className );\n\t\t}\n\n\t\tconst styles = element.getStyleNames();\n\n\t\tfor ( const style of styles ) {\n\t\t\tconsumables.styles.push( style );\n\t\t}\n\n\t\treturn consumables;\n\t}\n\n\t/**\n\t * Creates {@link module:engine/conversion/viewconsumable~ViewConsumable ViewConsumable} instance from\n\t * {@link module:engine/view/node~Node node} or {@link module:engine/view/documentfragment~DocumentFragment document fragment}.\n\t * Instance will contain all elements, child nodes, attributes, styles and classes added for consumption.\n\t *\n\t * @param from View node or document fragment from which `ViewConsumable` will be created.\n\t * @param instance If provided, given `ViewConsumable` instance will be used\n\t * to add all consumables. It will be returned instead of a new instance.\n\t */\n\tpublic static createFrom( from: Node | DocumentFragment, instance?: ViewConsumable ): ViewConsumable {\n\t\tif ( !instance ) {\n\t\t\tinstance = new ViewConsumable();\n\t\t}\n\n\t\tif ( from.is( '$text' ) ) {\n\t\t\tinstance.add( from );\n\n\t\t\treturn instance;\n\t\t}\n\n\t\t// Add `from` itself, if it is an element.\n\t\tif ( from.is( 'element' ) ) {\n\t\t\tinstance.add( from, ViewConsumable.consumablesFromElement( from ) );\n\t\t}\n\n\t\tif ( from.is( 'documentFragment' ) ) {\n\t\t\tinstance.add( from );\n\t\t}\n\n\t\tfor ( const child of ( from as Element | DocumentFragment ).getChildren() ) {\n\t\t\tinstance = ViewConsumable.createFrom( child, instance );\n\t\t}\n\n\t\treturn instance;\n\t}\n}\n\nexport interface Consumables {\n\n\t/**\n\t * If set to true element's name will be included.\n\t */\n\tname?: boolean;\n\n\t/**\n\t * Attribute name or array of attribute names.\n\t */\n\tattributes?: string | Array<string>;\n\n\t/**\n\t * Class name or array of class names.\n\t */\n\tclasses?: string | Array<string>;\n\n\t/**\n\t * Style name or array of style names.\n\t */\n\tstyles?: string | Array<string>;\n}\n\nconst CONSUMABLE_TYPES = [ 'attributes', 'classes', 'styles' ] as const;\n\ntype ConsumableType = ( typeof CONSUMABLE_TYPES )[ number ];\n\n/**\n * This is a private helper-class for {@link module:engine/conversion/viewconsumable~ViewConsumable}.\n * It represents and manipulates consumable parts of a single {@link module:engine/view/element~Element}.\n */\nexport class ViewElementConsumables {\n\tpublic readonly element: Node | DocumentFragment;\n\n\t/**\n\t * Flag indicating if name of the element can be consumed.\n\t */\n\tprivate _canConsumeName: boolean | null;\n\n\t/**\n\t * Contains maps of element's consumables: attributes, classes and styles.\n\t */\n\tprivate readonly _consumables: Record<ConsumableType, Map<string, boolean>>;\n\n\t/**\n\t * Creates ViewElementConsumables instance.\n\t *\n\t * @param from View node or document fragment from which `ViewElementConsumables` is being created.\n\t */\n\tconstructor( from: Node | DocumentFragment ) {\n\t\tthis.element = from;\n\t\tthis._canConsumeName = null;\n\t\tthis._consumables = {\n\t\t\tattributes: new Map(),\n\t\t\tstyles: new Map(),\n\t\t\tclasses: new Map()\n\t\t};\n\t}\n\n\t/**\n\t * Adds consumable parts of the {@link module:engine/view/element~Element view element}.\n\t * Element's name itself can be marked to be consumed (when element's name is consumed its attributes, classes and\n\t * styles still could be consumed):\n\t *\n\t * ```ts\n\t * consumables.add( { name: true } );\n\t * ```\n\t *\n\t * Attributes classes and styles:\n\t *\n\t * ```ts\n\t * consumables.add( { attributes: 'title', classes: 'foo', styles: 'color' } );\n\t * consumables.add( { attributes: [ 'title', 'name' ], classes: [ 'foo', 'bar' ] );\n\t * ```\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `viewconsumable-invalid-attribute` when `class` or `style`\n\t * attribute is provided - it should be handled separately by providing `style` and `class` in consumables object.\n\t *\n\t * @param consumables Object describing which parts of the element can be consumed.\n\t * @param consumables.name If set to `true` element's name will be added as consumable.\n\t * @param consumables.attributes Attribute name or array of attribute names to add as consumable.\n\t * @param consumables.classes Class name or array of class names to add as consumable.\n\t * @param consumables.styles Style name or array of style names to add as consumable.\n\t */\n\tpublic add( consumables: Consumables ): void {\n\t\tif ( consumables.name ) {\n\t\t\tthis._canConsumeName = true;\n\t\t}\n\n\t\tfor ( const type of CONSUMABLE_TYPES ) {\n\t\t\tif ( type in consumables ) {\n\t\t\t\tthis._add( type, consumables[ type ]! );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Tests if parts of the {@link module:engine/view/node~Node view node} can be consumed.\n\t *\n\t * Element's name can be tested:\n\t *\n\t * ```ts\n\t * consumables.test( { name: true } );\n\t * ```\n\t *\n\t * Attributes classes and styles:\n\t *\n\t * ```ts\n\t * consumables.test( { attributes: 'title', classes: 'foo', styles: 'color' } );\n\t * consumables.test( { attributes: [ 'title', 'name' ], classes: [ 'foo', 'bar' ] );\n\t * ```\n\t *\n\t * @param consumables Object describing which parts of the element should be tested.\n\t * @param consumables.name If set to `true` element's name will be tested.\n\t * @param consumables.attributes Attribute name or array of attribute names to test.\n\t * @param consumables.classes Class name or array of class names to test.\n\t * @param consumables.styles Style name or array of style names to test.\n\t * @returns `true` when all tested items can be consumed, `null` when even one of the items\n\t * was never marked for consumption and `false` when even one of the items was already consumed.\n\t */\n\tpublic test( consumables: Consumables | Match ): boolean | null {\n\t\t// Check if name can be consumed.\n\t\tif ( consumables.name && !this._canConsumeName ) {\n\t\t\treturn this._canConsumeName;\n\t\t}\n\n\t\tfor ( const type of CONSUMABLE_TYPES ) {\n\t\t\tif ( type in consumables ) {\n\t\t\t\tconst value = this._test( type, consumables[ type ]! );\n\n\t\t\t\tif ( value !== true ) {\n\t\t\t\t\treturn value;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// Return true only if all can be consumed.\n\t\treturn true;\n\t}\n\n\t/**\n\t * Consumes parts of {@link module:engine/view/element~Element view element}. This function does not check if consumable item\n\t * is already consumed - it consumes all consumable items provided.\n\t * Element's name can be consumed:\n\t *\n\t * ```ts\n\t * consumables.consume( { name: true } );\n\t * ```\n\t *\n\t * Attributes classes and styles:\n\t *\n\t * ```ts\n\t * consumables.consume( { attributes: 'title', classes: 'foo', styles: 'color' } );\n\t * consumables.consume( { attributes: [ 'title', 'name' ], classes: [ 'foo', 'bar' ] );\n\t * ```\n\t *\n\t * @param consumables Object describing which parts of the element should be consumed.\n\t * @param consumables.name If set to `true` element's name will be consumed.\n\t * @param consumables.attributes Attribute name or array of attribute names to consume.\n\t * @param consumables.classes Class name or array of class names to consume.\n\t * @param consumables.styles Style name or array of style names to consume.\n\t */\n\tpublic consume( consumables: Consumables | Match ): void {\n\t\tif ( consumables.name ) {\n\t\t\tthis._canConsumeName = false;\n\t\t}\n\n\t\tfor ( const type of CONSUMABLE_TYPES ) {\n\t\t\tif ( type in consumables ) {\n\t\t\t\tthis._consume( type, consumables[ type ]! );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Revert already consumed parts of {@link module:engine/view/element~Element view Element}, so they can be consumed once again.\n\t * Element's name can be reverted:\n\t *\n\t * ```ts\n\t * consumables.revert( { name: true } );\n\t * ```\n\t *\n\t * Attributes classes and styles:\n\t *\n\t * ```ts\n\t * consumables.revert( { attributes: 'title', classes: 'foo', styles: 'color' } );\n\t * consumables.revert( { attributes: [ 'title', 'name' ], classes: [ 'foo', 'bar' ] );\n\t * ```\n\t *\n\t * @param consumables Object describing which parts of the element should be reverted.\n\t * @param consumables.name If set to `true` element's name will be reverted.\n\t * @param consumables.attributes Attribute name or array of attribute names to revert.\n\t * @param consumables.classes Class name or array of class names to revert.\n\t * @param consumables.styles Style name or array of style names to revert.\n\t */\n\tpublic revert( consumables: Consumables ): void {\n\t\tif ( consumables.name ) {\n\t\t\tthis._canConsumeName = true;\n\t\t}\n\n\t\tfor ( const type of CONSUMABLE_TYPES ) {\n\t\t\tif ( type in consumables ) {\n\t\t\t\tthis._revert( type, consumables[ type ]! );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Helper method that adds consumables of a given type: attribute, class or style.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `viewconsumable-invalid-attribute` when `class` or `style`\n\t * type is provided - it should be handled separately by providing actual style/class type.\n\t *\n\t * @param type Type of the consumable item: `attributes`, `classes` or `styles`.\n\t * @param item Consumable item or array of items.\n\t */\n\tprivate _add( type: ConsumableType, item: string | Array<string> ) {\n\t\tconst items = isArray( item ) ? item : [ item ];\n\t\tconst consumables = this._consumables[ type ];\n\n\t\tfor ( const name of items ) {\n\t\t\tif ( type === 'attributes' && ( name === 'class' || name === 'style' ) ) {\n\t\t\t\t/**\n\t\t\t\t * Class and style attributes should be handled separately in\n\t\t\t\t * {@link module:engine/conversion/viewconsumable~ViewConsumable#add `ViewConsumable#add()`}.\n\t\t\t\t *\n\t\t\t\t * What you have done is trying to use:\n\t\t\t\t *\n\t\t\t\t *\t\tconsumables.add( { attributes: [ 'class', 'style' ] } );\n\t\t\t\t *\n\t\t\t\t * While each class and style should be registered separately:\n\t\t\t\t *\n\t\t\t\t *\t\tconsumables.add( { classes: 'some-class', styles: 'font-weight' } );\n\t\t\t\t *\n\t\t\t\t * @error viewconsumable-invalid-attribute\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError( 'viewconsumable-invalid-attribute', this );\n\t\t\t}\n\n\t\t\tconsumables.set( name, true );\n\n\t\t\tif ( type === 'styles' ) {\n\t\t\t\tfor ( const alsoName of this.element.document.stylesProcessor.getRelatedStyles( name ) ) {\n\t\t\t\t\tconsumables.set( alsoName, true );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Helper method that tests consumables of a given type: attribute, class or style.\n\t *\n\t * @param type Type of the consumable item: `attributes`, `classes` or `styles`.\n\t * @param item Consumable item or array of items.\n\t * @returns Returns `true` if all items can be consumed, `null` when one of the items cannot be\n\t * consumed and `false` when one of the items is already consumed.\n\t */\n\tprivate _test( type: ConsumableType, item: string | Array<string> ): boolean | null {\n\t\tconst items = isArray( item ) ? item : [ item ];\n\t\tconst consumables = this._consumables[ type ];\n\n\t\tfor ( const name of items ) {\n\t\t\tif ( type === 'attributes' && ( name === 'class' || name === 'style' ) ) {\n\t\t\t\tconst consumableName = name == 'class' ? 'classes' : 'styles';\n\n\t\t\t\t// Check all classes/styles if class/style attribute is tested.\n\t\t\t\tconst value = this._test( consumableName, [ ...this._consumables[ consumableName ].keys() ] );\n\n\t\t\t\tif ( value !== true ) {\n\t\t\t\t\treturn value;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconst value = consumables.get( name );\n\t\t\t\t// Return null if attribute is not found.\n\t\t\t\tif ( value === undefined ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\tif ( !value ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Helper method that consumes items of a given type: attribute, class or style.\n\t *\n\t * @param type Type of the consumable item: `attributes`, `classes` or `styles`.\n\t * @param item Consumable item or array of items.\n\t */\n\tprivate _consume( type: ConsumableType, item: string | Array<string> ) {\n\t\tconst items = isArray( item ) ? item : [ item ];\n\t\tconst consumables = this._consumables[ type ];\n\n\t\tfor ( const name of items ) {\n\t\t\tif ( type === 'attributes' && ( name === 'class' || name === 'style' ) ) {\n\t\t\t\tconst consumableName = name == 'class' ? 'classes' : 'styles';\n\n\t\t\t\t// If class or style is provided for consumption - consume them all.\n\t\t\t\tthis._consume( consumableName, [ ...this._consumables[ consumableName ].keys() ] );\n\t\t\t} else {\n\t\t\t\tconsumables.set( name, false );\n\n\t\t\t\tif ( type == 'styles' ) {\n\t\t\t\t\tfor ( const toConsume of this.element.document.stylesProcessor.getRelatedStyles( name ) ) {\n\t\t\t\t\t\tconsumables.set( toConsume, false );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Helper method that reverts items of a given type: attribute, class or style.\n\t *\n\t * @param type Type of the consumable item: `attributes`, `classes` or , `styles`.\n\t * @param item Consumable item or array of items.\n\t */\n\tprivate _revert( type: ConsumableType, item: string | Array<string> ) {\n\t\tconst items = isArray( item ) ? item : [ item ];\n\t\tconst consumables = this._consumables[ type ];\n\n\t\tfor ( const name of items ) {\n\t\t\tif ( type === 'attributes' && ( name === 'class' || name === 'style' ) ) {\n\t\t\t\tconst consumableName = name == 'class' ? 'classes' : 'styles';\n\n\t\t\t\t// If class or style is provided for reverting - revert them all.\n\t\t\t\tthis._revert( consumableName, [ ...this._consumables[ consumableName ].keys() ] );\n\t\t\t} else {\n\t\t\t\tconst value = consumables.get( name );\n\n\t\t\t\tif ( value === false ) {\n\t\t\t\t\tconsumables.set( name, true );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/schema\n */\n\nimport Element from './element';\nimport Position from './position';\nimport Range from './range';\nimport Text from './text';\nimport TreeWalker from './treewalker';\n\nimport type DocumentFragment from './documentfragment';\nimport type DocumentSelection from './documentselection';\nimport type Item from './item';\nimport type Node from './node';\nimport type Selection from './selection';\nimport type Writer from './writer';\n\nimport { CKEditorError, ObservableMixin } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The model's schema. It defines the allowed and disallowed structures of nodes as well as nodes' attributes.\n * The schema is usually defined by the features and based on them, the editing framework and features\n * make decisions on how to change and process the model.\n *\n * The instance of schema is available in {@link module:engine/model/model~Model#schema `editor.model.schema`}.\n *\n * Read more about the schema in:\n *\n * * The {@glink framework/architecture/editing-engine#schema schema section} of the\n * {@glink framework/architecture/editing-engine Introduction to the Editing engine architecture} guide.\n * * The {@glink framework/deep-dive/schema Schema deep-dive} guide.\n */\nexport default class Schema extends ObservableMixin() {\n\tprivate readonly _sourceDefinitions: Record<string, Array<SchemaItemDefinition>> = {};\n\n\t/**\n\t * A dictionary containing attribute properties.\n\t */\n\tprivate readonly _attributeProperties: Record<string, AttributeProperties> = {};\n\tprivate _compiledDefinitions?: Record<string, SchemaCompiledItemDefinition> | null;\n\n\t/**\n\t * Creates a schema instance.\n\t */\n\tconstructor() {\n\t\tsuper();\n\n\t\tthis.decorate( 'checkChild' );\n\t\tthis.decorate( 'checkAttribute' );\n\n\t\tthis.on( 'checkAttribute', ( evt, args ) => {\n\t\t\targs[ 0 ] = new SchemaContext( args[ 0 ] );\n\t\t}, { priority: 'highest' } );\n\n\t\tthis.on( 'checkChild', ( evt, args ) => {\n\t\t\targs[ 0 ] = new SchemaContext( args[ 0 ] );\n\t\t\targs[ 1 ] = this.getDefinition( args[ 1 ] );\n\t\t}, { priority: 'highest' } );\n\t}\n\n\t/**\n\t * Registers a schema item. Can only be called once for every item name.\n\t *\n\t * ```ts\n\t * schema.register( 'paragraph', {\n\t * \tinheritAllFrom: '$block'\n\t * } );\n\t * ```\n\t */\n\tpublic register( itemName: string, definition?: SchemaItemDefinition ): void {\n\t\tif ( this._sourceDefinitions[ itemName ] ) {\n\t\t\t/**\n\t\t\t * A single item cannot be registered twice in the schema.\n\t\t\t *\n\t\t\t * This situation may happen when:\n\t\t\t *\n\t\t\t * * Two or more plugins called {@link module:engine/model/schema~Schema#register `register()`} with the same name.\n\t\t\t * This will usually mean that there is a collision between plugins which try to use the same element in the model.\n\t\t\t * Unfortunately, the only way to solve this is by modifying one of these plugins to use a unique model element name.\n\t\t\t * * A single plugin was loaded twice. This happens when it is installed by npm/yarn in two versions\n\t\t\t * and usually means one or more of the following issues:\n\t\t\t * * a version mismatch (two of your dependencies require two different versions of this plugin),\n\t\t\t * * incorrect imports (this plugin is somehow imported twice in a way which confuses webpack),\n\t\t\t * * mess in `node_modules/` (`rm -rf node_modules/` may help).\n\t\t\t *\n\t\t\t * **Note:** Check the logged `itemName` to better understand which plugin was duplicated/conflicting.\n\t\t\t *\n\t\t\t * @param itemName The name of the model element that is being registered twice.\n\t\t\t * @error schema-cannot-register-item-twice\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'schema-cannot-register-item-twice',\n\t\t\t\tthis,\n\t\t\t\t{\n\t\t\t\t\titemName\n\t\t\t\t}\n\t\t\t);\n\t\t}\n\n\t\tthis._sourceDefinitions[ itemName ] = [\n\t\t\tObject.assign( {}, definition )\n\t\t];\n\n\t\tthis._clearCache();\n\t}\n\n\t/**\n\t * Extends a {@link #register registered} item's definition.\n\t *\n\t * Extending properties such as `allowIn` will add more items to the existing properties,\n\t * while redefining properties such as `isBlock` will override the previously defined ones.\n\t *\n\t * ```ts\n\t * schema.register( 'foo', {\n\t * \tallowIn: '$root',\n\t * \tisBlock: true;\n\t * } );\n\t * schema.extend( 'foo', {\n\t * \tallowIn: 'blockQuote',\n\t * \tisBlock: false\n\t * } );\n\t *\n\t * schema.getDefinition( 'foo' );\n\t * //\t{\n\t * //\t\tallowIn: [ '$root', 'blockQuote' ],\n\t * // \t\tisBlock: false\n\t * //\t}\n\t * ```\n\t */\n\tpublic extend( itemName: string, definition: SchemaItemDefinition ): void {\n\t\tif ( !this._sourceDefinitions[ itemName ] ) {\n\t\t\t/**\n\t\t\t * Cannot extend an item which was not registered yet.\n\t\t\t *\n\t\t\t * This error happens when a plugin tries to extend the schema definition of an item which was not\n\t\t\t * {@link module:engine/model/schema~Schema#register registered} yet.\n\t\t\t *\n\t\t\t * @param itemName The name of the model element which is being extended.\n\t\t\t * @error schema-cannot-extend-missing-item\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'schema-cannot-extend-missing-item', this, {\n\t\t\t\titemName\n\t\t\t} );\n\t\t}\n\n\t\tthis._sourceDefinitions[ itemName ].push( Object.assign( {}, definition ) );\n\n\t\tthis._clearCache();\n\t}\n\n\t/**\n\t * Returns data of all registered items.\n\t *\n\t * This method should normally be used for reflection purposes (e.g. defining a clone of a certain element,\n\t * checking a list of all block elements, etc).\n\t * Use specific methods (such as {@link #checkChild `checkChild()`} or {@link #isLimit `isLimit()`})\n\t * in other cases.\n\t */\n\tpublic getDefinitions(): Record<string, SchemaCompiledItemDefinition> {\n\t\tif ( !this._compiledDefinitions ) {\n\t\t\tthis._compile();\n\t\t}\n\n\t\treturn this._compiledDefinitions!;\n\t}\n\n\t/**\n\t * Returns a definition of the given item or `undefined` if an item is not registered.\n\t *\n\t * This method should normally be used for reflection purposes (e.g. defining a clone of a certain element,\n\t * checking a list of all block elements, etc).\n\t * Use specific methods (such as {@link #checkChild `checkChild()`} or {@link #isLimit `isLimit()`})\n\t * in other cases.\n\t */\n\tpublic getDefinition( item: string | Item | DocumentFragment | SchemaContextItem ): SchemaCompiledItemDefinition | undefined {\n\t\tlet itemName: string;\n\n\t\tif ( typeof item == 'string' ) {\n\t\t\titemName = item;\n\t\t} else if ( 'is' in item && ( item.is( '$text' ) || item.is( '$textProxy' ) ) ) {\n\t\t\titemName = '$text';\n\t\t}\n\t\t// Element or module:engine/model/schema~SchemaContextItem.\n\t\telse {\n\t\t\titemName = ( item as any ).name;\n\t\t}\n\n\t\treturn this.getDefinitions()[ itemName ];\n\t}\n\n\t/**\n\t * Returns `true` if the given item is registered in the schema.\n\t *\n\t * ```ts\n\t * schema.isRegistered( 'paragraph' ); // -> true\n\t * schema.isRegistered( editor.model.document.getRoot() ); // -> true\n\t * schema.isRegistered( 'foo' ); // -> false\n\t * ```\n\t */\n\tpublic isRegistered( item: string | Item | DocumentFragment | SchemaContextItem ): boolean {\n\t\treturn !!this.getDefinition( item );\n\t}\n\n\t/**\n\t * Returns `true` if the given item is defined to be\n\t * a block by the {@link module:engine/model/schema~SchemaItemDefinition}'s `isBlock` property.\n\t *\n\t * ```ts\n\t * schema.isBlock( 'paragraph' ); // -> true\n\t * schema.isBlock( '$root' ); // -> false\n\t *\n\t * const paragraphElement = writer.createElement( 'paragraph' );\n\t * schema.isBlock( paragraphElement ); // -> true\n\t * ```\n\t *\n\t * See the {@glink framework/deep-dive/schema#block-elements Block elements} section of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide for more details.\n\t */\n\tpublic isBlock( item: string | Item | DocumentFragment | SchemaContextItem ): boolean {\n\t\tconst def = this.getDefinition( item );\n\n\t\treturn !!( def && def.isBlock );\n\t}\n\n\t/**\n\t * Returns `true` if the given item should be treated as a limit element.\n\t *\n\t * It considers an item to be a limit element if its\n\t * {@link module:engine/model/schema~SchemaItemDefinition}'s\n\t * {@link module:engine/model/schema~SchemaItemDefinition#isLimit `isLimit`} or\n\t * {@link module:engine/model/schema~SchemaItemDefinition#isObject `isObject`} property\n\t * was set to `true`.\n\t *\n\t * ```ts\n\t * schema.isLimit( 'paragraph' ); // -> false\n\t * schema.isLimit( '$root' ); // -> true\n\t * schema.isLimit( editor.model.document.getRoot() ); // -> true\n\t * schema.isLimit( 'imageBlock' ); // -> true\n\t * ```\n\t *\n\t * See the {@glink framework/deep-dive/schema#limit-elements Limit elements} section of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide for more details.\n\t */\n\tpublic isLimit( item: string | Item | DocumentFragment | SchemaContextItem ): boolean {\n\t\tconst def = this.getDefinition( item );\n\n\t\tif ( !def ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn !!( def.isLimit || def.isObject );\n\t}\n\n\t/**\n\t * Returns `true` if the given item should be treated as an object element.\n\t *\n\t * It considers an item to be an object element if its\n\t * {@link module:engine/model/schema~SchemaItemDefinition}'s\n\t * {@link module:engine/model/schema~SchemaItemDefinition#isObject `isObject`} property\n\t * was set to `true`.\n\t *\n\t * ```ts\n\t * schema.isObject( 'paragraph' ); // -> false\n\t * schema.isObject( 'imageBlock' ); // -> true\n\t *\n\t * const imageElement = writer.createElement( 'imageBlock' );\n\t * schema.isObject( imageElement ); // -> true\n\t * ```\n\t *\n\t * See the {@glink framework/deep-dive/schema#object-elements Object elements} section of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide for more details.\n\t */\n\tpublic isObject( item: string | Item | DocumentFragment | SchemaContextItem ): boolean {\n\t\tconst def = this.getDefinition( item );\n\n\t\tif ( !def ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Note: Check out the implementation of #isLimit(), #isSelectable(), and #isContent()\n\t\t// to understand why these three constitute an object.\n\t\treturn !!( def.isObject || ( def.isLimit && def.isSelectable && def.isContent ) );\n\t}\n\n\t/**\n\t * Returns `true` if the given item is defined to be\n\t * an inline element by the {@link module:engine/model/schema~SchemaItemDefinition}'s `isInline` property.\n\t *\n\t * ```ts\n\t * schema.isInline( 'paragraph' ); // -> false\n\t * schema.isInline( 'softBreak' ); // -> true\n\t *\n\t * const text = writer.createText( 'foo' );\n\t * schema.isInline( text ); // -> true\n\t * ```\n\t *\n\t * See the {@glink framework/deep-dive/schema#inline-elements Inline elements} section of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide for more details.\n\t */\n\tpublic isInline( item: string | Item | DocumentFragment | SchemaContextItem ): boolean {\n\t\tconst def = this.getDefinition( item );\n\n\t\treturn !!( def && def.isInline );\n\t}\n\n\t/**\n\t * Returns `true` if the given item is defined to be\n\t * a selectable element by the {@link module:engine/model/schema~SchemaItemDefinition}'s `isSelectable` property.\n\t *\n\t * ```ts\n\t * schema.isSelectable( 'paragraph' ); // -> false\n\t * schema.isSelectable( 'heading1' ); // -> false\n\t * schema.isSelectable( 'imageBlock' ); // -> true\n\t * schema.isSelectable( 'tableCell' ); // -> true\n\t *\n\t * const text = writer.createText( 'foo' );\n\t * schema.isSelectable( text ); // -> false\n\t * ```\n\t *\n\t * See the {@glink framework/deep-dive/schema#selectable-elements Selectable elements section} of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide for more details.\n\t */\n\tpublic isSelectable( item: string | Item | DocumentFragment | SchemaContextItem ): boolean {\n\t\tconst def = this.getDefinition( item );\n\n\t\tif ( !def ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn !!( def.isSelectable || def.isObject );\n\t}\n\n\t/**\n\t * Returns `true` if the given item is defined to be\n\t * a content by the {@link module:engine/model/schema~SchemaItemDefinition}'s `isContent` property.\n\t *\n\t * ```ts\n\t * schema.isContent( 'paragraph' ); // -> false\n\t * schema.isContent( 'heading1' ); // -> false\n\t * schema.isContent( 'imageBlock' ); // -> true\n\t * schema.isContent( 'horizontalLine' ); // -> true\n\t *\n\t * const text = writer.createText( 'foo' );\n\t * schema.isContent( text ); // -> true\n\t * ```\n\t *\n\t * See the {@glink framework/deep-dive/schema#content-elements Content elements section} of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide for more details.\n\t */\n\tpublic isContent( item: string | Item | DocumentFragment | SchemaContextItem ): boolean {\n\t\tconst def = this.getDefinition( item );\n\n\t\tif ( !def ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn !!( def.isContent || def.isObject );\n\t}\n\n\t/**\n\t * Checks whether the given node (`child`) can be a child of the given context.\n\t *\n\t * ```ts\n\t * schema.checkChild( model.document.getRoot(), paragraph ); // -> false\n\t *\n\t * schema.register( 'paragraph', {\n\t * \tallowIn: '$root'\n\t * } );\n\t * schema.checkChild( model.document.getRoot(), paragraph ); // -> true\n\t * ```\n\t *\n\t * Note: When verifying whether the given node can be a child of the given context, the\n\t * schema also verifies the entire context &mdash; from its root to its last element. Therefore, it is possible\n\t * for `checkChild()` to return `false` even though the context's last element can contain the checked child.\n\t * It happens if one of the context's elements does not allow its child.\n\t *\n\t * @fires checkChild\n\t * @param context The context in which the child will be checked.\n\t * @param def The child to check.\n\t */\n\tpublic checkChild( context: SchemaContextDefinition, def: string | Node | DocumentFragment ): boolean {\n\t\t// Note: context and child are already normalized here to a SchemaContext and SchemaCompiledItemDefinition.\n\t\tif ( !def ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn this._checkContextMatch( def as any, context as any );\n\t}\n\n\t/**\n\t * Checks whether the given attribute can be applied in the given context (on the last\n\t * item of the context).\n\t *\n\t * ```ts\n\t * schema.checkAttribute( textNode, 'bold' ); // -> false\n\t *\n\t * schema.extend( '$text', {\n\t * \tallowAttributes: 'bold'\n\t * } );\n\t * schema.checkAttribute( textNode, 'bold' ); // -> true\n\t * ```\n\t *\n\t * @fires checkAttribute\n\t * @param context The context in which the attribute will be checked.\n\t */\n\tpublic checkAttribute( context: SchemaContextDefinition, attributeName: string ): boolean {\n\t\tconst def = this.getDefinition( ( context as any ).last );\n\n\t\tif ( !def ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn def.allowAttributes.includes( attributeName );\n\t}\n\n\t/**\n\t * Checks whether the given element (`elementToMerge`) can be merged with the specified base element (`positionOrBaseElement`).\n\t *\n\t * In other words &mdash; whether `elementToMerge`'s children {@link #checkChild are allowed} in the `positionOrBaseElement`.\n\t *\n\t * This check ensures that elements merged with {@link module:engine/model/writer~Writer#merge `Writer#merge()`}\n\t * will be valid.\n\t *\n\t * Instead of elements, you can pass the instance of the {@link module:engine/model/position~Position} class as the\n\t * `positionOrBaseElement`. It means that the elements before and after the position will be checked whether they can be merged.\n\t *\n\t * @param positionOrBaseElement The position or base element to which the `elementToMerge` will be merged.\n\t * @param elementToMerge The element to merge. Required if `positionOrBaseElement` is an element.\n\t */\n\tpublic checkMerge( positionOrBaseElement: Position | Element, elementToMerge: Element ): boolean {\n\t\tif ( positionOrBaseElement instanceof Position ) {\n\t\t\tconst nodeBefore = positionOrBaseElement.nodeBefore;\n\t\t\tconst nodeAfter = positionOrBaseElement.nodeAfter;\n\n\t\t\tif ( !( nodeBefore instanceof Element ) ) {\n\t\t\t\t/**\n\t\t\t\t * The node before the merge position must be an element.\n\t\t\t\t *\n\t\t\t\t * @error schema-check-merge-no-element-before\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'schema-check-merge-no-element-before',\n\t\t\t\t\tthis\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tif ( !( nodeAfter instanceof Element ) ) {\n\t\t\t\t/**\n\t\t\t\t * The node after the merge position must be an element.\n\t\t\t\t *\n\t\t\t\t * @error schema-check-merge-no-element-after\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'schema-check-merge-no-element-after',\n\t\t\t\t\tthis\n\t\t\t\t);\n\t\t\t}\n\n\t\t\treturn this.checkMerge( nodeBefore, nodeAfter );\n\t\t}\n\n\t\tfor ( const child of elementToMerge.getChildren() ) {\n\t\t\tif ( !this.checkChild( positionOrBaseElement, child ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Allows registering a callback to the {@link #checkChild} method calls.\n\t *\n\t * Callbacks allow you to implement rules which are not otherwise possible to achieve\n\t * by using the declarative API of {@link module:engine/model/schema~SchemaItemDefinition}.\n\t * For example, by using this method you can disallow elements in specific contexts.\n\t *\n\t * This method is a shorthand for using the {@link #event:checkChild} event. For even better control,\n\t * you can use that event instead.\n\t *\n\t * Example:\n\t *\n\t * ```ts\n\t * // Disallow heading1 directly inside a blockQuote.\n\t * schema.addChildCheck( ( context, childDefinition ) => {\n\t * \tif ( context.endsWith( 'blockQuote' ) && childDefinition.name == 'heading1' ) {\n\t * \t\treturn false;\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * Which translates to:\n\t *\n\t * ```ts\n\t * schema.on( 'checkChild', ( evt, args ) => {\n\t * \tconst context = args[ 0 ];\n\t * \tconst childDefinition = args[ 1 ];\n\t *\n\t * \tif ( context.endsWith( 'blockQuote' ) && childDefinition && childDefinition.name == 'heading1' ) {\n\t * \t\t// Prevent next listeners from being called.\n\t * \t\tevt.stop();\n\t * \t\t// Set the checkChild()'s return value.\n\t * \t\tevt.return = false;\n\t * \t}\n\t * }, { priority: 'high' } );\n\t * ```\n\t *\n\t * @param callback The callback to be called. It is called with two parameters:\n\t * {@link module:engine/model/schema~SchemaContext} (context) instance and\n\t * {@link module:engine/model/schema~SchemaCompiledItemDefinition} (child-to-check definition).\n\t * The callback may return `true/false` to override `checkChild()`'s return value. If it does not return\n\t * a boolean value, the default algorithm (or other callbacks) will define `checkChild()`'s return value.\n\t */\n\tpublic addChildCheck( callback: SchemaChildCheckCallback ): void {\n\t\tthis.on<SchemaCheckChildEvent>( 'checkChild', ( evt, [ ctx, childDef ] ) => {\n\t\t\t// checkChild() was called with a non-registered child.\n\t\t\t// In 99% cases such check should return false, so not to overcomplicate all callbacks\n\t\t\t// don't even execute them.\n\t\t\tif ( !childDef ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst retValue = callback( ctx, childDef );\n\n\t\t\tif ( typeof retValue == 'boolean' ) {\n\t\t\t\tevt.stop();\n\t\t\t\tevt.return = retValue;\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * Allows registering a callback to the {@link #checkAttribute} method calls.\n\t *\n\t * Callbacks allow you to implement rules which are not otherwise possible to achieve\n\t * by using the declarative API of {@link module:engine/model/schema~SchemaItemDefinition}.\n\t * For example, by using this method you can disallow attribute if node to which it is applied\n\t * is contained within some other element (e.g. you want to disallow `bold` on `$text` within `heading1`).\n\t *\n\t * This method is a shorthand for using the {@link #event:checkAttribute} event. For even better control,\n\t * you can use that event instead.\n\t *\n\t * Example:\n\t *\n\t * ```ts\n\t * // Disallow bold on $text inside heading1.\n\t * schema.addAttributeCheck( ( context, attributeName ) => {\n\t * \tif ( context.endsWith( 'heading1 $text' ) && attributeName == 'bold' ) {\n\t * \t\treturn false;\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * Which translates to:\n\t *\n\t * ```ts\n\t * schema.on( 'checkAttribute', ( evt, args ) => {\n\t * \tconst context = args[ 0 ];\n\t * \tconst attributeName = args[ 1 ];\n\t *\n\t * \tif ( context.endsWith( 'heading1 $text' ) && attributeName == 'bold' ) {\n\t * \t\t// Prevent next listeners from being called.\n\t * \t\tevt.stop();\n\t * \t\t// Set the checkAttribute()'s return value.\n\t * \t\tevt.return = false;\n\t * \t}\n\t * }, { priority: 'high' } );\n\t * ```\n\t *\n\t * @param callback The callback to be called. It is called with two parameters:\n\t * {@link module:engine/model/schema~SchemaContext} (context) instance and attribute name.\n\t * The callback may return `true/false` to override `checkAttribute()`'s return value. If it does not return\n\t * a boolean value, the default algorithm (or other callbacks) will define `checkAttribute()`'s return value.\n\t */\n\tpublic addAttributeCheck( callback: SchemaAttributeCheckCallback ): void {\n\t\tthis.on<SchemaCheckAttributeEvent>( 'checkAttribute', ( evt, [ ctx, attributeName ] ) => {\n\t\t\tconst retValue = callback( ctx, attributeName );\n\n\t\t\tif ( typeof retValue == 'boolean' ) {\n\t\t\t\tevt.stop();\n\t\t\t\tevt.return = retValue;\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * This method allows assigning additional metadata to the model attributes. For example,\n\t * {@link module:engine/model/schema~AttributeProperties `AttributeProperties#isFormatting` property} is\n\t * used to mark formatting attributes (like `bold` or `italic`).\n\t *\n\t * ```ts\n\t * // Mark bold as a formatting attribute.\n\t * schema.setAttributeProperties( 'bold', {\n\t * \tisFormatting: true\n\t * } );\n\t *\n\t * // Override code not to be considered a formatting markup.\n\t * schema.setAttributeProperties( 'code', {\n\t * \tisFormatting: false\n\t * } );\n\t * ```\n\t *\n\t * Properties are not limited to members defined in the\n\t * {@link module:engine/model/schema~AttributeProperties `AttributeProperties` type} and you can also use custom properties:\n\t *\n\t * ```ts\n\t * schema.setAttributeProperties( 'blockQuote', {\n\t * \tcustomProperty: 'value'\n\t * } );\n\t * ```\n\t *\n\t * Subsequent calls with the same attribute will extend its custom properties:\n\t *\n\t * ```ts\n\t * schema.setAttributeProperties( 'blockQuote', {\n\t * \tone: 1\n\t * } );\n\t *\n\t * schema.setAttributeProperties( 'blockQuote', {\n\t * \ttwo: 2\n\t * } );\n\t *\n\t * console.log( schema.getAttributeProperties( 'blockQuote' ) );\n\t * // Logs: { one: 1, two: 2 }\n\t * ```\n\t *\n\t * @param attributeName A name of the attribute to receive the properties.\n\t * @param properties A dictionary of properties.\n\t */\n\tpublic setAttributeProperties( attributeName: string, properties: AttributeProperties ): void {\n\t\tthis._attributeProperties[ attributeName ] = Object.assign( this.getAttributeProperties( attributeName ), properties );\n\t}\n\n\t/**\n\t * Returns properties associated with a given model attribute. See {@link #setAttributeProperties `setAttributeProperties()`}.\n\t *\n\t * @param attributeName A name of the attribute.\n\t */\n\tpublic getAttributeProperties( attributeName: string ): AttributeProperties {\n\t\treturn this._attributeProperties[ attributeName ] || {};\n\t}\n\n\t/**\n\t * Returns the lowest {@link module:engine/model/schema~Schema#isLimit limit element} containing the entire\n\t * selection/range/position or the root otherwise.\n\t *\n\t * @param selectionOrRangeOrPosition The selection/range/position to check.\n\t * @returns The lowest limit element containing the entire `selectionOrRangeOrPosition`.\n\t */\n\tpublic getLimitElement( selectionOrRangeOrPosition: Selection | DocumentSelection | Range | Position ): Element {\n\t\tlet element: Element;\n\n\t\tif ( selectionOrRangeOrPosition instanceof Position ) {\n\t\t\telement = selectionOrRangeOrPosition.parent as Element;\n\t\t} else {\n\t\t\tconst ranges = selectionOrRangeOrPosition instanceof Range ?\n\t\t\t\t[ selectionOrRangeOrPosition ] :\n\t\t\t\tArray.from( selectionOrRangeOrPosition.getRanges() );\n\n\t\t\t// Find the common ancestor for all selection's ranges.\n\t\t\telement = ranges\n\t\t\t\t.reduce<Element | null>( ( element, range ) => {\n\t\t\t\t\tconst rangeCommonAncestor = range.getCommonAncestor() as ( Element | null );\n\n\t\t\t\t\tif ( !element ) {\n\t\t\t\t\t\treturn rangeCommonAncestor;\n\t\t\t\t\t}\n\n\t\t\t\t\treturn element.getCommonAncestor( rangeCommonAncestor as Element, { includeSelf: true } ) as Element;\n\t\t\t\t}, null )!;\n\t\t}\n\n\t\twhile ( !this.isLimit( element ) ) {\n\t\t\tif ( element.parent ) {\n\t\t\t\telement = element.parent as Element;\n\t\t\t} else {\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn element;\n\t}\n\n\t/**\n\t * Checks whether the attribute is allowed in selection:\n\t *\n\t * * if the selection is not collapsed, then checks if the attribute is allowed on any of nodes in that range,\n\t * * if the selection is collapsed, then checks if on the selection position there's a text with the\n\t * specified attribute allowed.\n\t *\n\t * @param selection Selection which will be checked.\n\t * @param attribute The name of the attribute to check.\n\t */\n\tpublic checkAttributeInSelection( selection: Selection | DocumentSelection, attribute: string ): boolean {\n\t\tif ( selection.isCollapsed ) {\n\t\t\tconst firstPosition = selection.getFirstPosition()!;\n\t\t\tconst context = [\n\t\t\t\t...firstPosition.getAncestors() as Array<Element>,\n\t\t\t\tnew Text( '', selection.getAttributes() )\n\t\t\t];\n\n\t\t\t// Check whether schema allows for a text with the attribute in the selection.\n\t\t\treturn this.checkAttribute( context, attribute );\n\t\t} else {\n\t\t\tconst ranges = selection.getRanges();\n\n\t\t\t// For all ranges, check nodes in them until you find a node that is allowed to have the attribute.\n\t\t\tfor ( const range of ranges ) {\n\t\t\t\tfor ( const value of range ) {\n\t\t\t\t\tif ( this.checkAttribute( value.item, attribute ) ) {\n\t\t\t\t\t\t// If we found a node that is allowed to have the attribute, return true.\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// If we haven't found such node, return false.\n\t\treturn false;\n\t}\n\n\t/**\n\t * Transforms the given set of ranges into a set of ranges where the given attribute is allowed (and can be applied).\n\t *\n\t * @param ranges Ranges to be validated.\n\t * @param attribute The name of the attribute to check.\n\t * @returns Ranges in which the attribute is allowed.\n\t */\n\tpublic* getValidRanges( ranges: Iterable<Range>, attribute: string ): IterableIterator<Range> {\n\t\tranges = convertToMinimalFlatRanges( ranges );\n\n\t\tfor ( const range of ranges ) {\n\t\t\tyield* this._getValidRangesForRange( range, attribute );\n\t\t}\n\t}\n\n\t/**\n\t * Basing on given `position`, finds and returns a {@link module:engine/model/range~Range range} which is\n\t * nearest to that `position` and is a correct range for selection.\n\t *\n\t * The correct selection range might be collapsed when it is located in a position where the text node can be placed.\n\t * Non-collapsed range is returned when selection can be placed around element marked as an \"object\" in\n\t * the {@link module:engine/model/schema~Schema schema}.\n\t *\n\t * Direction of searching for the nearest correct selection range can be specified as:\n\t *\n\t * * `both` - searching will be performed in both ways,\n\t * * `forward` - searching will be performed only forward,\n\t * * `backward` - searching will be performed only backward.\n\t *\n\t * When valid selection range cannot be found, `null` is returned.\n\t *\n\t * @param position Reference position where new selection range should be looked for.\n\t * @param direction Search direction.\n\t * @returns Nearest selection range or `null` if one cannot be found.\n\t */\n\tpublic getNearestSelectionRange( position: Position, direction: 'both' | 'forward' | 'backward' = 'both' ): Range | null {\n\t\tif ( position.root.rootName == '$graveyard' ) {\n\t\t\t// No valid selection range in the graveyard.\n\t\t\t// This is important when getting the document selection default range.\n\t\t\treturn null;\n\t\t}\n\n\t\t// Return collapsed range if provided position is valid.\n\t\tif ( this.checkChild( position, '$text' ) ) {\n\t\t\treturn new Range( position );\n\t\t}\n\n\t\tlet backwardWalker, forwardWalker;\n\n\t\t// Never leave a limit element.\n\t\tconst limitElement = ( position.getAncestors() as Array<Element> ).reverse().find( item => this.isLimit( item ) ) ||\n\t\t\tposition.root as Element;\n\n\t\tif ( direction == 'both' || direction == 'backward' ) {\n\t\t\tbackwardWalker = new TreeWalker( {\n\t\t\t\tboundaries: Range._createIn( limitElement ),\n\t\t\t\tstartPosition: position,\n\t\t\t\tdirection: 'backward'\n\t\t\t} );\n\t\t}\n\n\t\tif ( direction == 'both' || direction == 'forward' ) {\n\t\t\tforwardWalker = new TreeWalker( {\n\t\t\t\tboundaries: Range._createIn( limitElement ),\n\t\t\t\tstartPosition: position\n\t\t\t} );\n\t\t}\n\n\t\tfor ( const data of combineWalkers( backwardWalker, forwardWalker ) ) {\n\t\t\tconst type = ( data.walker == backwardWalker ? 'elementEnd' : 'elementStart' );\n\t\t\tconst value = data.value;\n\n\t\t\tif ( value.type == type && this.isObject( value.item ) ) {\n\t\t\t\treturn Range._createOn( value.item );\n\t\t\t}\n\n\t\t\tif ( this.checkChild( value.nextPosition, '$text' ) ) {\n\t\t\t\treturn new Range( value.nextPosition );\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Tries to find position ancestors that allow to insert a given node.\n\t * It starts searching from the given position and goes node by node to the top of the model tree\n\t * as long as a {@link module:engine/model/schema~Schema#isLimit limit element}, an\n\t * {@link module:engine/model/schema~Schema#isObject object element} or a topmost ancestor is not reached.\n\t *\n\t * @param position The position that the search will start from.\n\t * @param node The node for which an allowed parent should be found or its name.\n\t * @returns Allowed parent or null if nothing was found.\n\t */\n\tpublic findAllowedParent( position: Position, node: Node | string ): Element | null {\n\t\tlet parent = position.parent as ( Element | null );\n\n\t\twhile ( parent ) {\n\t\t\tif ( this.checkChild( parent, node ) ) {\n\t\t\t\treturn parent;\n\t\t\t}\n\n\t\t\t// Do not split limit elements.\n\t\t\tif ( this.isLimit( parent ) ) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tparent = parent.parent as ( Element | null );\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Sets attributes allowed by the schema on a given node.\n\t *\n\t * @param node A node to set attributes on.\n\t * @param attributes Attributes keys and values.\n\t * @param writer An instance of the model writer.\n\t */\n\tpublic setAllowedAttributes(\n\t\tnode: Node,\n\t\tattributes: Record<string, unknown>,\n\t\twriter: Writer\n\t): void {\n\t\tconst model = writer.model;\n\n\t\tfor ( const [ attributeName, attributeValue ] of Object.entries( attributes ) ) {\n\t\t\tif ( model.schema.checkAttribute( node, attributeName ) ) {\n\t\t\t\twriter.setAttribute( attributeName, attributeValue, node );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Removes attributes disallowed by the schema.\n\t *\n\t * @param nodes Nodes that will be filtered.\n\t */\n\tpublic removeDisallowedAttributes( nodes: Iterable<Node>, writer: Writer ): void {\n\t\tfor ( const node of nodes ) {\n\t\t\t// When node is a `Text` it has no children, so just filter it out.\n\t\t\tif ( node.is( '$text' ) ) {\n\t\t\t\tremoveDisallowedAttributeFromNode( this, node, writer );\n\t\t\t}\n\t\t\t// In a case of `Element` iterates through positions between nodes inside this element\n\t\t\t// and filter out node before the current position, or position parent when position\n\t\t\t// is at start of an element. Using positions prevent from omitting merged nodes\n\t\t\t// see https://github.com/ckeditor/ckeditor5-engine/issues/1789.\n\t\t\telse {\n\t\t\t\tconst rangeInNode = Range._createIn( node as Element );\n\t\t\t\tconst positionsInRange = rangeInNode.getPositions();\n\n\t\t\t\tfor ( const position of positionsInRange ) {\n\t\t\t\t\tconst item = position.nodeBefore || position.parent;\n\n\t\t\t\t\tremoveDisallowedAttributeFromNode( this, item as any, writer );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Gets attributes of a node that have a given property.\n\t *\n\t * @param node Node to get attributes from.\n\t * @param propertyName Name of the property that attribute must have to return it.\n\t * @param propertyValue Desired value of the property that we want to check.\n\t * When `undefined` attributes will be returned if they have set a given property no matter what the value is. If specified it will\n\t * return attributes which given property's value is equal to this parameter.\n\t * @returns Object with attributes' names as key and attributes' values as value.\n\t */\n\tpublic getAttributesWithProperty( node: Node, propertyName: string, propertyValue: unknown ): Record<string, unknown> {\n\t\tconst attributes: Record<string, unknown> = {};\n\n\t\tfor ( const [ attributeName, attributeValue ] of node.getAttributes() ) {\n\t\t\tconst attributeProperties = this.getAttributeProperties( attributeName );\n\n\t\t\tif ( attributeProperties[ propertyName ] === undefined ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tif ( propertyValue === undefined || propertyValue === attributeProperties[ propertyName ] ) {\n\t\t\t\tattributes[ attributeName ] = attributeValue;\n\t\t\t}\n\t\t}\n\n\t\treturn attributes;\n\t}\n\n\t/**\n\t * Creates an instance of the schema context.\n\t */\n\tpublic createContext( context: SchemaContextDefinition ): SchemaContext {\n\t\treturn new SchemaContext( context );\n\t}\n\n\tprivate _clearCache(): void {\n\t\tthis._compiledDefinitions = null;\n\t}\n\n\tprivate _compile(): void {\n\t\tconst compiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal> = {};\n\t\tconst sourceRules = this._sourceDefinitions;\n\t\tconst itemNames = Object.keys( sourceRules );\n\n\t\tfor ( const itemName of itemNames ) {\n\t\t\tcompiledDefinitions[ itemName ] = compileBaseItemRule( sourceRules[ itemName ], itemName );\n\t\t}\n\n\t\tfor ( const itemName of itemNames ) {\n\t\t\tcompileAllowChildren( compiledDefinitions, itemName );\n\t\t}\n\n\t\tfor ( const itemName of itemNames ) {\n\t\t\tcompileAllowContentOf( compiledDefinitions, itemName );\n\t\t}\n\n\t\tfor ( const itemName of itemNames ) {\n\t\t\tcompileAllowWhere( compiledDefinitions, itemName );\n\t\t}\n\n\t\tfor ( const itemName of itemNames ) {\n\t\t\tcompileAllowAttributesOf( compiledDefinitions, itemName );\n\t\t\tcompileInheritPropertiesFrom( compiledDefinitions, itemName );\n\t\t}\n\n\t\tfor ( const itemName of itemNames ) {\n\t\t\tcleanUpAllowIn( compiledDefinitions, itemName );\n\t\t\tsetupAllowChildren( compiledDefinitions, itemName );\n\t\t\tcleanUpAllowAttributes( compiledDefinitions, itemName );\n\t\t}\n\n\t\tthis._compiledDefinitions = compiledDefinitions as any;\n\t}\n\n\tprivate _checkContextMatch(\n\t\tdef: SchemaCompiledItemDefinition,\n\t\tcontext: SchemaContext,\n\t\tcontextItemIndex: number = context.length - 1\n\t): boolean {\n\t\tconst contextItem = context.getItem( contextItemIndex );\n\n\t\tif ( def.allowIn.includes( contextItem.name ) ) {\n\t\t\tif ( contextItemIndex == 0 ) {\n\t\t\t\treturn true;\n\t\t\t} else {\n\t\t\t\tconst parentRule = this.getDefinition( contextItem );\n\n\t\t\t\treturn this._checkContextMatch( parentRule!, context, contextItemIndex - 1 );\n\t\t\t}\n\t\t} else {\n\t\t\treturn false;\n\t\t}\n\t}\n\n\t/**\n\t * Takes a flat range and an attribute name. Traverses the range recursively and deeply to find and return all ranges\n\t * inside the given range on which the attribute can be applied.\n\t *\n\t * This is a helper function for {@link ~Schema#getValidRanges}.\n\t *\n\t * @param range The range to process.\n\t * @param attribute The name of the attribute to check.\n\t * @returns Ranges in which the attribute is allowed.\n\t */\n\tprivate* _getValidRangesForRange( range: Range, attribute: string ): Iterable<Range> {\n\t\tlet start = range.start;\n\t\tlet end = range.start;\n\n\t\tfor ( const item of range.getItems( { shallow: true } ) ) {\n\t\t\tif ( item.is( 'element' ) ) {\n\t\t\t\tyield* this._getValidRangesForRange( Range._createIn( item ), attribute );\n\t\t\t}\n\n\t\t\tif ( !this.checkAttribute( item, attribute ) ) {\n\t\t\t\tif ( !start.isEqual( end ) ) {\n\t\t\t\t\tyield new Range( start, end );\n\t\t\t\t}\n\n\t\t\t\tstart = Position._createAfter( item );\n\t\t\t}\n\n\t\t\tend = Position._createAfter( item );\n\t\t}\n\n\t\tif ( !start.isEqual( end ) ) {\n\t\t\tyield new Range( start, end );\n\t\t}\n\t}\n}\n\n/**\n * Event fired when the {@link ~Schema#checkChild} method is called. It allows plugging in\n * additional behavior, for example implementing rules which cannot be defined using the declarative\n * {@link module:engine/model/schema~SchemaItemDefinition} interface.\n *\n * **Note:** The {@link ~Schema#addChildCheck} method is a more handy way to register callbacks. Internally,\n * it registers a listener to this event but comes with a simpler API and it is the recommended choice\n * in most of the cases.\n *\n * The {@link ~Schema#checkChild} method fires an event because it is\n * {@link module:utils/observablemixin~Observable#decorate decorated} with it. Thanks to that you can\n * use this event in various ways, but the most important use case is overriding standard behavior of the\n * `checkChild()` method. Let's see a typical listener template:\n *\n * ```ts\n * schema.on( 'checkChild', ( evt, args ) => {\n * \tconst context = args[ 0 ];\n * \tconst childDefinition = args[ 1 ];\n * }, { priority: 'high' } );\n * ```\n *\n * The listener is added with a `high` priority to be executed before the default method is really called. The `args` callback\n * parameter contains arguments passed to `checkChild( context, child )`. However, the `context` parameter is already\n * normalized to a {@link module:engine/model/schema~SchemaContext} instance and `child` to a\n * {@link module:engine/model/schema~SchemaCompiledItemDefinition} instance, so you do not have to worry about\n * the various ways how `context` and `child` may be passed to `checkChild()`.\n *\n * **Note:** `childDefinition` may be `undefined` if `checkChild()` was called with a non-registered element.\n *\n * So, in order to implement a rule \"disallow `heading1` in `blockQuote`\", you can add such a listener:\n *\n * ```ts\n * schema.on( 'checkChild', ( evt, args ) => {\n * \tconst context = args[ 0 ];\n * \tconst childDefinition = args[ 1 ];\n *\n * \tif ( context.endsWith( 'blockQuote' ) && childDefinition && childDefinition.name == 'heading1' ) {\n * \t\t// Prevent next listeners from being called.\n * \t\tevt.stop();\n * \t\t// Set the checkChild()'s return value.\n * \t\tevt.return = false;\n * \t}\n * }, { priority: 'high' } );\n * ```\n *\n * Allowing elements in specific contexts will be a far less common use case, because it is normally handled by the\n * `allowIn` rule from {@link module:engine/model/schema~SchemaItemDefinition}. But if you have a complex scenario\n * where `listItem` should be allowed only in element `foo` which must be in element `bar`, then this would be the way:\n *\n * ```ts\n * schema.on( 'checkChild', ( evt, args ) => {\n * \tconst context = args[ 0 ];\n * \tconst childDefinition = args[ 1 ];\n *\n * \tif ( context.endsWith( 'bar foo' ) && childDefinition.name == 'listItem' ) {\n * \t\t// Prevent next listeners from being called.\n * \t\tevt.stop();\n * \t\t// Set the checkChild()'s return value.\n * \t\tevt.return = true;\n * \t}\n * }, { priority: 'high' } );\n * ```\n *\n * @eventName ~Schema#checkChild\n * @param args The `checkChild()`'s arguments.\n */\nexport type SchemaCheckChildEvent = {\n\tname: 'checkChild';\n\targs: [ [ context: SchemaContext, def: SchemaCompiledItemDefinition ] ];\n};\n\n/**\n * Event fired when the {@link ~Schema#checkAttribute} method is called. It allows plugging in\n * additional behavior, for example implementing rules which cannot be defined using the declarative\n * {@link module:engine/model/schema~SchemaItemDefinition} interface.\n *\n * **Note:** The {@link ~Schema#addAttributeCheck} method is a more handy way to register callbacks. Internally,\n * it registers a listener to this event but comes with a simpler API and it is the recommended choice\n * in most of the cases.\n *\n * The {@link ~Schema#checkAttribute} method fires an event because it is\n * {@link module:utils/observablemixin~Observable#decorate decorated} with it. Thanks to that you can\n * use this event in various ways, but the most important use case is overriding the standard behavior of the\n * `checkAttribute()` method. Let's see a typical listener template:\n *\n * ```ts\n * schema.on( 'checkAttribute', ( evt, args ) => {\n * \tconst context = args[ 0 ];\n * \tconst attributeName = args[ 1 ];\n * }, { priority: 'high' } );\n * ```\n *\n * The listener is added with a `high` priority to be executed before the default method is really called. The `args` callback\n * parameter contains arguments passed to `checkAttribute( context, attributeName )`. However, the `context` parameter is already\n * normalized to a {@link module:engine/model/schema~SchemaContext} instance, so you do not have to worry about\n * the various ways how `context` may be passed to `checkAttribute()`.\n *\n * So, in order to implement a rule \"disallow `bold` in a text which is in a `heading1`, you can add such a listener:\n *\n * ```ts\n * schema.on( 'checkAttribute', ( evt, args ) => {\n * \tconst context = args[ 0 ];\n * \tconst attributeName = args[ 1 ];\n *\n * \tif ( context.endsWith( 'heading1 $text' ) && attributeName == 'bold' ) {\n * \t\t// Prevent next listeners from being called.\n * \t\tevt.stop();\n * \t\t// Set the checkAttribute()'s return value.\n * \t\tevt.return = false;\n * \t}\n * }, { priority: 'high' } );\n * ```\n *\n * Allowing attributes in specific contexts will be a far less common use case, because it is normally handled by the\n * `allowAttributes` rule from {@link module:engine/model/schema~SchemaItemDefinition}. But if you have a complex scenario\n * where `bold` should be allowed only in element `foo` which must be in element `bar`, then this would be the way:\n *\n * ```ts\n * schema.on( 'checkAttribute', ( evt, args ) => {\n * \tconst context = args[ 0 ];\n * \tconst attributeName = args[ 1 ];\n *\n * \tif ( context.endsWith( 'bar foo $text' ) && attributeName == 'bold' ) {\n * \t\t// Prevent next listeners from being called.\n * \t\tevt.stop();\n * \t\t// Set the checkAttribute()'s return value.\n * \t\tevt.return = true;\n * \t}\n * }, { priority: 'high' } );\n * ```\n *\n * @eventName ~Schema#checkAttribute\n * @param args The `checkAttribute()`'s arguments.\n */\nexport type SchemaCheckAttributeEvent = {\n\tname: 'checkAttribute';\n\targs: [ [ context: SchemaContext, attributeName: string ] ];\n};\n\n/**\n * A definition of a {@link module:engine/model/schema~Schema schema} item.\n *\n * You can define the following rules:\n *\n * * {@link ~SchemaItemDefinition#allowIn `allowIn`} &ndash; Defines in which other items this item will be allowed.\n * * {@link ~SchemaItemDefinition#allowChildren `allowChildren`} &ndash; Defines which other items are allowed inside this item.\n * * {@link ~SchemaItemDefinition#allowAttributes `allowAttributes`} &ndash; Defines allowed attributes of the given item.\n * * {@link ~SchemaItemDefinition#allowContentOf `allowContentOf`} &ndash; Inherits \"allowed children\" from other items.\n * * {@link ~SchemaItemDefinition#allowWhere `allowWhere`} &ndash; Inherits \"allowed in\" from other items.\n * * {@link ~SchemaItemDefinition#allowAttributesOf `allowAttributesOf`} &ndash; Inherits attributes from other items.\n * * {@link ~SchemaItemDefinition#inheritTypesFrom `inheritTypesFrom`} &ndash; Inherits `is*` properties of other items.\n * * {@link ~SchemaItemDefinition#inheritAllFrom `inheritAllFrom`} &ndash;\n * A shorthand for `allowContentOf`, `allowWhere`, `allowAttributesOf`, `inheritTypesFrom`.\n *\n * # The `is*` properties\n *\n * There are a couple commonly used `is*` properties. Their role is to assign additional semantics to schema items.\n * You can define more properties but you will also need to implement support for them in the existing editor features.\n *\n * * {@link ~SchemaItemDefinition#isBlock `isBlock`} &ndash; Whether this item is paragraph-like.\n * Generally speaking, content is usually made out of blocks like paragraphs, list items, images, headings, etc.\n * * {@link ~SchemaItemDefinition#isInline `isInline`} &ndash; Whether an item is \"text-like\" and should be treated as an inline node.\n * Examples of inline elements: `$text`, `softBreak` (`<br>`), etc.\n * * {@link ~SchemaItemDefinition#isLimit `isLimit`} &ndash; It can be understood as whether this element\n * should not be split by <kbd>Enter</kbd>. Examples of limit elements: `$root`, table cell, image caption, etc.\n * In other words, all actions that happen inside a limit element are limited to its content.\n * All objects are treated as limit elements, too.\n * * {@link ~SchemaItemDefinition#isObject `isObject`} &ndash; Whether an item is \"self-contained\" and should be treated as a whole.\n * Examples of object elements: `imageBlock`, `table`, `video`, etc. An object is also a limit, so\n * {@link module:engine/model/schema~Schema#isLimit `isLimit()`} returns `true` for object elements automatically.\n *\n * Read more about the meaning of these types in the\n * {@glink framework/deep-dive/schema#defining-additional-semantics dedicated section of the Schema deep-dive} guide.\n *\n * # Generic items\n *\n * There are several generic items (classes of elements) available: `$root`, `$container`, `$block`, `$blockObject`,\n * `$inlineObject`, and `$text`. They are defined as follows:\n *\n * ```ts\n * schema.register( '$root', {\n * \tisLimit: true\n * } );\n *\n * schema.register( '$container', {\n * \tallowIn: [ '$root', '$container' ]\n * } );\n *\n * schema.register( '$block', {\n * \tallowIn: [ '$root', '$container' ],\n * \tisBlock: true\n * } );\n *\n * schema.register( '$blockObject', {\n * \tallowWhere: '$block',\n * \tisBlock: true,\n * \tisObject: true\n * } );\n *\n * schema.register( '$inlineObject', {\n * \tallowWhere: '$text',\n * \tallowAttributesOf: '$text',\n * \tisInline: true,\n * \tisObject: true\n * } );\n *\n * schema.register( '$text', {\n * \tallowIn: '$block',\n * \tisInline: true,\n * \tisContent: true\n * } );\n * ```\n *\n * They reflect typical editor content that is contained within one root, consists of several blocks\n * (paragraphs, lists items, headings, images) which, in turn, may contain text inside.\n *\n * By inheriting from the generic items you can define new items which will get extended by other editor features.\n * Read more about generic types in the {@glink framework/deep-dive/schema Schema deep-dive} guide.\n *\n * # Example definitions\n *\n * Allow `paragraph` in roots and block quotes:\n *\n * ```ts\n * schema.register( 'paragraph', {\n * \tallowIn: [ '$root', 'blockQuote' ],\n * \tisBlock: true\n * } );\n * ```\n *\n * Allow `paragraph` everywhere where `$block` is allowed (i.e. in `$root`):\n *\n * ```ts\n * schema.register( 'paragraph', {\n * \tallowWhere: '$block',\n * \tisBlock: true\n * } );\n * ```\n *\n * Allow `paragraph` inside a `$root` and allow `$text` as a `paragraph` child:\n *\n * ```ts\n * schema.register( 'paragraph', {\n * \tallowIn: '$root',\n * \tallowChildren: '$text',\n * \tisBlock: true\n * } );\n * ```\n *\n * The previous rule can be written in a shorter form using inheritance:\n *\n * ```ts\n * schema.register( 'paragraph', {\n * \tinheritAllFrom: '$block'\n * } );\n * ```\n *\n * Make `imageBlock` a block object, which is allowed everywhere where `$block` is.\n * Also, allow `src` and `alt` attributes in it:\n *\n * ```ts\n * schema.register( 'imageBlock', {\n * \tinheritAllFrom: '$blockObject',\n * \tallowAttributes: [ 'src', 'alt' ],\n * } );\n * ```\n *\n * Make `caption` allowed in `imageBlock` and make it allow all the content of `$block`s (usually, `$text`).\n * Also, mark it as a limit element so it cannot be split:\n *\n * ```ts\n * schema.register( 'caption', {\n * \tallowIn: 'imageBlock',\n * \tallowContentOf: '$block',\n * \tisLimit: true\n * } );\n * ```\n *\n * Make `listItem` inherit all from `$block` but also allow additional attributes:\n *\n * ```ts\n * schema.register( 'listItem', {\n * \tinheritAllFrom: '$block',\n * \tallowAttributes: [ 'listType', 'listIndent' ]\n * } );\n * ```\n *\n * Which translates to:\n *\n * ```ts\n * schema.register( 'listItem', {\n * \tallowWhere: '$block',\n * \tallowContentOf: '$block',\n * \tallowAttributesOf: '$block',\n * \tinheritTypesFrom: '$block',\n * \tallowAttributes: [ 'listType', 'listIndent' ]\n * } );\n * ```\n *\n * # Tips\n *\n * * Check schema definitions of existing features to see how they are defined.\n * * If you want to publish your feature so other developers can use it, try to use\n * generic items as much as possible.\n * * Keep your model clean. Limit it to the actual data and store information in a normalized way.\n * * Remember about defining the `is*` properties. They do not affect the allowed structures, but they can\n * affect how the editor features treat your elements.\n */\nexport interface SchemaItemDefinition {\n\n\t/**\n\t * Defines in which other items this item will be allowed.\n\t */\n\tallowIn?: string | Array<string>;\n\n\t/**\n\t * Defines which other items are allowed inside this item.\n\t */\n\tallowChildren?: string | Array<string>;\n\n\t/**\n\t * Defines allowed attributes of the given item.\n\t */\n\tallowAttributes?: string | Array<string>;\n\n\t/**\n\t * Inherits \"allowed children\" from other items.\n\t */\n\tallowContentOf?: string | Array<string>;\n\n\t/**\n\t * Inherits \"allowed in\" from other items.\n\t */\n\tallowWhere?: string | Array<string>;\n\n\t/**\n\t * Inherits attributes from other items.\n\t */\n\tallowAttributesOf?: string | Array<string>;\n\n\t/**\n\t * Inherits `is*` properties of other items.\n\t */\n\tinheritTypesFrom?: string | Array<string>;\n\n\t/**\n\t * A shorthand for `allowContentOf`, `allowWhere`, `allowAttributesOf`, `inheritTypesFrom`.\n\t */\n\tinheritAllFrom?: string;\n\n\t/**\n\t * Whether this item is paragraph-like. Generally speaking, content is usually made out of blocks\n\t * like paragraphs, list items, images, headings, etc. All these elements are marked as blocks. A block\n\t * should not allow another block inside. Note: There is also the `$block` generic item which has `isBlock` set to `true`.\n\t * Most block type items will inherit from `$block` (through `inheritAllFrom`).\n\t *\n\t * Read more about the block elements in the\n\t * {@glink framework/deep-dive/schema#block-elements Block elements section} of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide.\n\t */\n\tisBlock?: boolean;\n\n\t/**\n\t * Whether an item is \"text-like\" and should be treated as an inline node. Examples of inline elements:\n\t * `$text`, `softBreak` (`<br>`), etc.\n\t *\n\t * Read more about the inline elements in the\n\t * {@glink framework/deep-dive/schema#inline-elements Inline elements section} of the Schema deep-dive guide.\n\t */\n\tisInline?: boolean;\n\n\t/**\n\t * It can be understood as whether this element should not be split by <kbd>Enter</kbd>.\n\t * Examples of limit elements: `$root`, table cell, image caption, etc. In other words, all actions that happen inside\n\t * a limit element are limited to its content.\n\t *\n\t * Read more about the limit elements in the\n\t * {@glink framework/deep-dive/schema#limit-elements Limit elements section} of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide.\n\t */\n\tisLimit?: boolean;\n\n\t/**\n\t * Whether an item is \"self-contained\" and should be treated as a whole. Examples of object elements:\n\t * `imageBlock`, `table`, `video`, etc.\n\t *\n\t * **Note:** An object is also a limit, so\n\t * {@link module:engine/model/schema~Schema#isLimit `isLimit()`} returns `true` for object elements automatically.\n\t *\n\t * Read more about the object elements in the\n\t * {@glink framework/deep-dive/schema#object-elements Object elements section} of the Schema deep-dive guide.\n\t */\n\tisObject?: boolean;\n\n\t/**\n\t * `true` when an element should be selectable as a whole by the user.\n\t * Examples of selectable elements: `imageBlock`, `table`, `tableCell`, etc.\n\t *\n\t * **Note:** An object is also a selectable element, so\n\t * {@link module:engine/model/schema~Schema#isSelectable `isSelectable()`} returns `true` for object elements automatically.\n\t *\n\t * Read more about selectable elements in the\n\t * {@glink framework/deep-dive/schema#selectable-elements Selectable elements section} of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide.\n\t */\n\tisSelectable?: boolean;\n\n\t/**\n\t * An item is a content when it always finds its way to the editor data output regardless of the number and type of its descendants.\n\t * Examples of content elements: `$text`, `imageBlock`, `table`, etc. (but not `paragraph`, `heading1` or `tableCell`).\n\t *\n\t * **Note:** An object is also a content element, so\n\t * {@link module:engine/model/schema~Schema#isContent `isContent()`} returns `true` for object elements automatically.\n\t *\n\t * Read more about content elements in the\n\t * {@glink framework/deep-dive/schema#content-elements Content elements section} of\n\t * the {@glink framework/deep-dive/schema Schema deep-dive} guide.\n\t */\n\tisContent?: boolean;\n}\n\n/**\n * A simplified version of {@link module:engine/model/schema~SchemaItemDefinition} after\n * compilation by the {@link module:engine/model/schema~Schema schema}.\n * Rules fed to the schema by {@link module:engine/model/schema~Schema#register}\n * and {@link module:engine/model/schema~Schema#extend} methods are defined in the\n * {@link module:engine/model/schema~SchemaItemDefinition} format.\n * Later on, they are compiled to `SchemaCompiledItemDefinition` so when you use e.g.\n * the {@link module:engine/model/schema~Schema#getDefinition} method you get the compiled version.\n *\n * The compiled version contains only the following properties:\n *\n * * The `name` property,\n * * The `is*` properties,\n * * The `allowIn` array,\n * * The `allowChildren` array,\n * * The `allowAttributes` array.\n */\nexport interface SchemaCompiledItemDefinition {\n\tname: string;\n\tisBlock: boolean;\n\tisContent: boolean;\n\tisInline: boolean;\n\tisLimit: boolean;\n\tisObject: boolean;\n\tisSelectable: boolean;\n\tallowIn: Array<string>;\n\tallowChildren: Array<string>;\n\tallowAttributes: Array<string>;\n}\n\ninterface SchemaCompiledItemDefinitionInternal {\n\tname: string;\n\n\tisBlock?: boolean;\n\tisContent?: boolean;\n\tisInline?: boolean;\n\tisLimit?: boolean;\n\tisObject?: boolean;\n\tisSelectable?: boolean;\n\n\tallowIn: Array<string>;\n\tallowChildren: Array<string>;\n\tallowAttributes: Array<string>;\n\n\tallowAttributesOf?: Array<string>;\n\tallowContentOf?: Array<string>;\n\tallowWhere?: Array<string>;\n\tinheritTypesFrom?: Array<string>;\n}\n\ntype TypeNames = Array<'isBlock' | 'isContent' | 'isInline' | 'isLimit' | 'isObject' | 'isSelectable'>;\n\n/**\n * A schema context &mdash; a list of ancestors of a given position in the document.\n *\n * Considering such position:\n *\n * ```xml\n * <$root>\n * \t<blockQuote>\n * \t\t<paragraph>\n * \t\t\t^\n * \t\t</paragraph>\n * \t</blockQuote>\n * </$root>\n * ```\n *\n * The context of this position is its {@link module:engine/model/position~Position#getAncestors lists of ancestors}:\n *\n *\t\t[ rootElement, blockQuoteElement, paragraphElement ]\n *\n * Contexts are used in the {@link module:engine/model/schema~Schema#event:checkChild `Schema#checkChild`} and\n * {@link module:engine/model/schema~Schema#event:checkAttribute `Schema#checkAttribute`} events as a definition\n * of a place in the document where the check occurs. The context instances are created based on the first arguments\n * of the {@link module:engine/model/schema~Schema#checkChild `Schema#checkChild()`} and\n * {@link module:engine/model/schema~Schema#checkAttribute `Schema#checkAttribute()`} methods so when\n * using these methods you need to use {@link module:engine/model/schema~SchemaContextDefinition}s.\n */\nexport class SchemaContext implements Iterable<SchemaContextItem> {\n\tprivate _items!: Array<SchemaContextItem>;\n\n\t/**\n\t * Creates an instance of the context.\n\t */\n\tconstructor( context: SchemaContextDefinition ) {\n\t\tif ( context instanceof SchemaContext ) {\n\t\t\treturn context;\n\t\t}\n\n\t\tlet items: Array<string | Item | DocumentFragment>;\n\n\t\tif ( typeof context == 'string' ) {\n\t\t\titems = [ context ];\n\t\t} else if ( !Array.isArray( context ) ) {\n\t\t\t// `context` is item or position.\n\t\t\t// Position#getAncestors() doesn't accept any parameters but it works just fine here.\n\t\t\titems = context.getAncestors( { includeSelf: true } );\n\t\t} else {\n\t\t\titems = context;\n\t\t}\n\n\t\tthis._items = items.map( mapContextItem );\n\t}\n\n\t/**\n\t * The number of items.\n\t */\n\tpublic get length(): number {\n\t\treturn this._items.length;\n\t}\n\n\t/**\n\t * The last item (the lowest node).\n\t */\n\tpublic get last(): SchemaContextItem {\n\t\treturn this._items[ this._items.length - 1 ]!;\n\t}\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * Iterates over all context items.\n\t */\n\tpublic [ Symbol.iterator ](): IterableIterator<SchemaContextItem> {\n\t\treturn this._items[ Symbol.iterator ]();\n\t}\n\n\t/**\n\t * Returns a new schema context instance with an additional item.\n\t *\n\t * Item can be added as:\n\t *\n\t * ```ts\n\t * const context = new SchemaContext( [ '$root' ] );\n\t *\n\t * // An element.\n\t * const fooElement = writer.createElement( 'fooElement' );\n\t * const newContext = context.push( fooElement ); // [ '$root', 'fooElement' ]\n\t *\n\t * // A text node.\n\t * const text = writer.createText( 'foobar' );\n\t * const newContext = context.push( text ); // [ '$root', '$text' ]\n\t *\n\t * // A string (element name).\n\t * const newContext = context.push( 'barElement' ); // [ '$root', 'barElement' ]\n\t * ```\n\t *\n\t * **Note** {@link module:engine/model/node~Node} that is already in the model tree will be added as the only item\n\t * (without ancestors).\n\t *\n\t * @param item An item that will be added to the current context.\n\t * @returns A new schema context instance with an additional item.\n\t */\n\tpublic push( item: string | Node ): SchemaContext {\n\t\tconst ctx = new SchemaContext( [ item ] );\n\n\t\tctx._items = [ ...this._items, ...ctx._items ];\n\n\t\treturn ctx;\n\t}\n\n\t/**\n\t * Gets an item on the given index.\n\t */\n\tpublic getItem( index: number ): SchemaContextItem {\n\t\treturn this._items[ index ];\n\t}\n\n\t/**\n\t * Returns the names of items.\n\t */\n\tpublic* getNames(): IterableIterator<string> {\n\t\tyield* this._items.map( item => item.name );\n\t}\n\n\t/**\n\t * Checks whether the context ends with the given nodes.\n\t *\n\t * ```ts\n\t * const ctx = new SchemaContext( [ rootElement, paragraphElement, textNode ] );\n\t *\n\t * ctx.endsWith( '$text' ); // -> true\n\t * ctx.endsWith( 'paragraph $text' ); // -> true\n\t * ctx.endsWith( '$root' ); // -> false\n\t * ctx.endsWith( 'paragraph' ); // -> false\n\t * ```\n\t */\n\tpublic endsWith( query: string ): boolean {\n\t\treturn Array.from( this.getNames() ).join( ' ' ).endsWith( query );\n\t}\n\n\t/**\n\t * Checks whether the context starts with the given nodes.\n\t *\n\t * ```ts\n\t * const ctx = new SchemaContext( [ rootElement, paragraphElement, textNode ] );\n\t *\n\t * ctx.endsWith( '$root' ); // -> true\n\t * ctx.endsWith( '$root paragraph' ); // -> true\n\t * ctx.endsWith( '$text' ); // -> false\n\t * ctx.endsWith( 'paragraph' ); // -> false\n\t * ```\n\t */\n\tpublic startsWith( query: string ): boolean {\n\t\treturn Array.from( this.getNames() ).join( ' ' ).startsWith( query );\n\t}\n}\n\n/**\n * The definition of a {@link module:engine/model/schema~SchemaContext schema context}.\n *\n * Contexts can be created in multiple ways:\n *\n * * By defining a **node** in this cases this node and all its ancestors will be used.\n * * By defining a **position** in the document in this case all its ancestors will be used.\n * * By defining an **array of nodes** in this case this array defines the entire context.\n * * By defining a **name of node** - in this case node will be \"mocked\". It is not recommended because context\n * will be unrealistic (e.g. attributes of these nodes are not specified). However, at times this may be the only\n * way to define the context (e.g. when checking some hypothetical situation).\n * * By defining an **array of node names** (potentially, mixed with real nodes) The same as **name of node**\n * but it is possible to create a path.\n * * By defining a {@link module:engine/model/schema~SchemaContext} instance - in this case the same instance as provided\n * will be return.\n *\n * Examples of context definitions passed to the {@link module:engine/model/schema~Schema#checkChild `Schema#checkChild()`}\n * method:\n *\n * ```ts\n * // Assuming that we have a $root > blockQuote > paragraph structure, the following code\n * // will check node 'foo' in the following context:\n * // [ rootElement, blockQuoteElement, paragraphElement ]\n * const contextDefinition = paragraphElement;\n * const childToCheck = 'foo';\n * schema.checkChild( contextDefinition, childToCheck );\n *\n * // Also check in [ rootElement, blockQuoteElement, paragraphElement ].\n * schema.checkChild( model.createPositionAt( paragraphElement, 0 ), 'foo' );\n *\n * // Check in [ rootElement, paragraphElement ].\n * schema.checkChild( [ rootElement, paragraphElement ], 'foo' );\n *\n * // Check only fakeParagraphElement.\n * schema.checkChild( 'paragraph', 'foo' );\n *\n * // Check in [ fakeRootElement, fakeBarElement, paragraphElement ].\n * schema.checkChild( [ '$root', 'bar', paragraphElement ], 'foo' );\n * ```\n *\n * All these `checkChild()` calls will fire {@link module:engine/model/schema~Schema#event:checkChild `Schema#checkChild`}\n * events in which `args[ 0 ]` is an instance of the context. Therefore, you can write a listener like this:\n *\n * ```ts\n * schema.on( 'checkChild', ( evt, args ) => {\n * \tconst ctx = args[ 0 ];\n *\n * \tconsole.log( Array.from( ctx.getNames() ) );\n * } );\n * ```\n *\n * Which will log the following:\n *\n * ```ts\n * [ '$root', 'blockQuote', 'paragraph' ]\n * [ '$root', 'paragraph' ]\n * [ '$root', 'bar', 'paragraph' ]\n * ```\n *\n * Note: When using the {@link module:engine/model/schema~Schema#checkAttribute `Schema#checkAttribute()`} method\n * you may want to check whether a text node may have an attribute. A {@link module:engine/model/text~Text} is a\n * correct way to define a context so you can do this:\n *\n * ```ts\n * schema.checkAttribute( textNode, 'bold' );\n * ```\n *\n * But sometimes you want to check whether a text at a given position might've had some attribute,\n * in which case you can create a context by mixing in an array of elements with a `'$text'` string:\n *\n * ```ts\n * // Check in [ rootElement, paragraphElement, textNode ].\n * schema.checkChild( [ ...positionInParagraph.getAncestors(), '$text' ], 'bold' );\n * ```\n */\nexport type SchemaContextDefinition = Item | Position | SchemaContext | string | Array<string | Item>;\n\n/**\n * An item of the {@link module:engine/model/schema~SchemaContext schema context}.\n *\n * It contains 3 properties:\n *\n * * `name` the name of this item,\n * * `* getAttributeKeys()` a generator of keys of item attributes,\n * * `getAttribute( keyName )` a method to get attribute values.\n *\n * The context item interface is a highly simplified version of {@link module:engine/model/node~Node} and its role\n * is to expose only the information which schema checks are able to provide (which is the name of the node and\n * node's attributes).\n *\n * ```ts\n * schema.on( 'checkChild', ( evt, args ) => {\n * \tconst ctx = args[ 0 ];\n * \tconst firstItem = ctx.getItem( 0 );\n *\n * \tconsole.log( firstItem.name ); // -> '$root'\n * \tconsole.log( firstItem.getAttribute( 'foo' ) ); // -> 'bar'\n * \tconsole.log( Array.from( firstItem.getAttributeKeys() ) ); // -> [ 'foo', 'faa' ]\n * } );\n * ```\n */\nexport interface SchemaContextItem {\n\tname: string;\n\tgetAttributeKeys(): Generator<string>;\n\tgetAttribute( keyName: string ): unknown;\n}\n\n/**\n * A structure containing additional metadata describing the attribute.\n *\n * See {@link module:engine/model/schema~Schema#setAttributeProperties `Schema#setAttributeProperties()`} for usage examples.\n */\nexport interface AttributeProperties {\n\n\t/**\n\t * Indicates that the attribute should be considered as a visual formatting, like `bold`, `italic` or\n\t * `fontSize` rather than semantic attribute (such as `src`, `listType`, etc.). For example, it is used by the \"Remove format\" feature.\n\t */\n\tisFormatting?: boolean;\n\n\t/**\n\t * Indicates that given text attribute should be copied to the next block when enter is pressed.\n\t */\n\tcopyOnEnter?: boolean;\n\n\t/**\n\t * Indicates that given attribute should be preserved while replacing the element.\n\t */\n\tcopyOnReplace?: boolean;\n\n\t/**\n\t * Indicates that given text attribute should be copied from an inline object to the next inserted inline content.\n\t *\n\t * @default true\n\t */\n\tcopyFromObject?: boolean;\n\n\t[ name: string ]: unknown;\n}\n\nexport type SchemaAttributeCheckCallback = ( context: SchemaContext, attributeName: string ) => unknown;\n\nexport type SchemaChildCheckCallback = ( ctx: SchemaContext, def: SchemaCompiledItemDefinition ) => unknown;\n\nfunction compileBaseItemRule( sourceItemRules: Array<SchemaItemDefinition>, itemName: string ): SchemaCompiledItemDefinitionInternal {\n\tconst itemRule = {\n\t\tname: itemName,\n\n\t\tallowIn: [],\n\t\tallowContentOf: [],\n\t\tallowWhere: [],\n\n\t\tallowAttributes: [],\n\t\tallowAttributesOf: [],\n\n\t\tallowChildren: [],\n\n\t\tinheritTypesFrom: []\n\t};\n\n\tcopyTypes( sourceItemRules, itemRule );\n\n\tcopyProperty( sourceItemRules, itemRule, 'allowIn' );\n\tcopyProperty( sourceItemRules, itemRule, 'allowContentOf' );\n\tcopyProperty( sourceItemRules, itemRule, 'allowWhere' );\n\n\tcopyProperty( sourceItemRules, itemRule, 'allowAttributes' );\n\tcopyProperty( sourceItemRules, itemRule, 'allowAttributesOf' );\n\n\tcopyProperty( sourceItemRules, itemRule, 'allowChildren' );\n\n\tcopyProperty( sourceItemRules, itemRule, 'inheritTypesFrom' );\n\n\tmakeInheritAllWork( sourceItemRules, itemRule );\n\n\treturn itemRule;\n}\n\nfunction compileAllowChildren(\n\tcompiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>,\n\titemName: string\n) {\n\tconst item = compiledDefinitions[ itemName ];\n\n\tfor ( const allowChildrenItem of item.allowChildren ) {\n\t\tconst allowedChildren = compiledDefinitions[ allowChildrenItem ];\n\n\t\t// The allowChildren property may point to an unregistered element.\n\t\tif ( !allowedChildren ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tallowedChildren.allowIn.push( itemName );\n\t}\n\n\t// The allowIn property already includes correct items, reset the allowChildren property\n\t// to avoid duplicates later when setting up compilation results.\n\titem.allowChildren.length = 0;\n}\n\nfunction compileAllowContentOf(\n\tcompiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>,\n\titemName: string\n) {\n\tfor ( const allowContentOfItemName of compiledDefinitions[ itemName ].allowContentOf! ) {\n\t\t// The allowContentOf property may point to an unregistered element.\n\t\tif ( compiledDefinitions[ allowContentOfItemName ] ) {\n\t\t\tconst allowedChildren = getAllowedChildren( compiledDefinitions, allowContentOfItemName );\n\n\t\t\tallowedChildren.forEach( allowedItem => {\n\t\t\t\tallowedItem.allowIn.push( itemName );\n\t\t\t} );\n\t\t}\n\t}\n\n\tdelete compiledDefinitions[ itemName ].allowContentOf;\n}\n\nfunction compileAllowWhere(\n\tcompiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>,\n\titemName: string\n) {\n\tfor ( const allowWhereItemName of compiledDefinitions[ itemName ].allowWhere! ) {\n\t\tconst inheritFrom = compiledDefinitions[ allowWhereItemName ];\n\n\t\t// The allowWhere property may point to an unregistered element.\n\t\tif ( inheritFrom ) {\n\t\t\tconst allowedIn = inheritFrom.allowIn;\n\n\t\t\tcompiledDefinitions[ itemName ].allowIn.push( ...allowedIn );\n\t\t}\n\t}\n\n\tdelete compiledDefinitions[ itemName ].allowWhere;\n}\n\nfunction compileAllowAttributesOf(\n\tcompiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>,\n\titemName: string\n) {\n\tfor ( const allowAttributeOfItem of compiledDefinitions[ itemName ].allowAttributesOf! ) {\n\t\tconst inheritFrom = compiledDefinitions[ allowAttributeOfItem ];\n\n\t\tif ( inheritFrom ) {\n\t\t\tconst inheritAttributes = inheritFrom.allowAttributes;\n\n\t\t\tcompiledDefinitions[ itemName ].allowAttributes.push( ...inheritAttributes );\n\t\t}\n\t}\n\n\tdelete compiledDefinitions[ itemName ].allowAttributesOf;\n}\n\nfunction compileInheritPropertiesFrom(\n\tcompiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>,\n\titemName: string\n) {\n\tconst item = compiledDefinitions[ itemName ];\n\n\tfor ( const inheritPropertiesOfItem of item.inheritTypesFrom! ) {\n\t\tconst inheritFrom = compiledDefinitions[ inheritPropertiesOfItem ];\n\n\t\tif ( inheritFrom ) {\n\t\t\tconst typeNames = Object.keys( inheritFrom ).filter( name => name.startsWith( 'is' ) ) as TypeNames;\n\n\t\t\tfor ( const name of typeNames ) {\n\t\t\t\tif ( !( name in item ) ) {\n\t\t\t\t\titem[ name ] = inheritFrom[ name ];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\tdelete item.inheritTypesFrom;\n}\n\n// Remove items which weren't registered (because it may break some checks or we'd need to complicate them).\n// Make sure allowIn doesn't contain repeated values.\nfunction cleanUpAllowIn(\n\tcompiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>,\n\titemName: string\n) {\n\tconst itemRule = compiledDefinitions[ itemName ];\n\tconst existingItems = itemRule.allowIn.filter( itemToCheck => compiledDefinitions[ itemToCheck ] );\n\n\titemRule.allowIn = Array.from( new Set( existingItems ) );\n}\n\n// Setup allowChildren items based on allowIn.\nfunction setupAllowChildren(\n\tcompiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>,\n\titemName: string\n) {\n\tconst itemRule = compiledDefinitions[ itemName ];\n\n\tfor ( const allowedParentItemName of itemRule.allowIn ) {\n\t\tconst allowedParentItem = compiledDefinitions[ allowedParentItemName ];\n\n\t\tallowedParentItem.allowChildren.push( itemName );\n\t}\n}\n\nfunction cleanUpAllowAttributes(\n\tcompiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>,\n\titemName: string\n) {\n\tconst itemRule = compiledDefinitions[ itemName ];\n\n\titemRule.allowAttributes = Array.from( new Set( itemRule.allowAttributes ) );\n}\n\nfunction copyTypes( sourceItemRules: Array<SchemaItemDefinition>, itemRule: SchemaCompiledItemDefinitionInternal ) {\n\tfor ( const sourceItemRule of sourceItemRules ) {\n\t\tconst typeNames = Object.keys( sourceItemRule ).filter( name => name.startsWith( 'is' ) ) as TypeNames;\n\n\t\tfor ( const name of typeNames ) {\n\t\t\titemRule[ name ] = !!sourceItemRule[ name ];\n\t\t}\n\t}\n}\n\nfunction copyProperty(\n\tsourceItemRules: Array<SchemaItemDefinition>,\n\titemRule: SchemaCompiledItemDefinitionInternal,\n\tpropertyName: 'allowIn' |\n\t\t'allowContentOf' |\n\t\t'allowWhere' |\n\t\t'allowAttributes' |\n\t\t'allowAttributesOf' |\n\t\t'allowChildren' |\n\t\t'inheritTypesFrom'\n) {\n\tfor ( const sourceItemRule of sourceItemRules ) {\n\t\tconst value = sourceItemRule[ propertyName ];\n\n\t\tif ( typeof value == 'string' ) {\n\t\t\titemRule[ propertyName ]!.push( value );\n\t\t} else if ( Array.isArray( value ) ) {\n\t\t\titemRule[ propertyName ]!.push( ...value );\n\t\t}\n\t}\n}\n\nfunction makeInheritAllWork( sourceItemRules: Array<SchemaItemDefinition>, itemRule: SchemaCompiledItemDefinitionInternal ) {\n\tfor ( const sourceItemRule of sourceItemRules ) {\n\t\tconst inheritFrom = sourceItemRule.inheritAllFrom;\n\n\t\tif ( inheritFrom ) {\n\t\t\titemRule.allowContentOf!.push( inheritFrom );\n\t\t\titemRule.allowWhere!.push( inheritFrom );\n\t\t\titemRule.allowAttributesOf!.push( inheritFrom );\n\t\t\titemRule.inheritTypesFrom!.push( inheritFrom );\n\t\t}\n\t}\n}\n\nfunction getAllowedChildren( compiledDefinitions: Record<string, SchemaCompiledItemDefinitionInternal>, itemName: string ) {\n\tconst itemRule = compiledDefinitions[ itemName ];\n\n\treturn getValues( compiledDefinitions ).filter( def => def.allowIn.includes( itemRule.name ) );\n}\n\nfunction getValues( obj: Record<string, SchemaCompiledItemDefinitionInternal> ) {\n\treturn Object.keys( obj ).map( key => obj[ key ] );\n}\n\nfunction mapContextItem( ctxItem: string | Item | DocumentFragment ): SchemaContextItem {\n\tif ( typeof ctxItem == 'string' || ctxItem.is( 'documentFragment' ) ) {\n\t\treturn {\n\t\t\tname: typeof ctxItem == 'string' ? ctxItem : '$documentFragment',\n\n\t\t\t* getAttributeKeys() {},\n\n\t\t\tgetAttribute() {}\n\t\t};\n\t} else {\n\t\treturn {\n\t\t\t// '$text' means text nodes and text proxies.\n\t\t\tname: ctxItem.is( 'element' ) ? ctxItem.name : '$text',\n\n\t\t\t* getAttributeKeys() {\n\t\t\t\tyield* ctxItem.getAttributeKeys();\n\t\t\t},\n\n\t\t\tgetAttribute( key ) {\n\t\t\t\treturn ctxItem.getAttribute( key );\n\t\t\t}\n\t\t};\n\t}\n}\n\n/**\n * Generator function returning values from provided walkers, switching between them at each iteration. If only one walker\n * is provided it will return data only from that walker.\n *\n * @param backward Walker iterating in backward direction.\n * @param forward Walker iterating in forward direction.\n * @returns Object returned at each iteration contains `value` and `walker` (informing which walker returned\n * given value) fields.\n */\nfunction* combineWalkers( backward: TreeWalker | undefined, forward: TreeWalker | undefined ) {\n\tlet done = false;\n\n\twhile ( !done ) {\n\t\tdone = true;\n\n\t\tif ( backward ) {\n\t\t\tconst step = backward.next();\n\n\t\t\tif ( !step.done ) {\n\t\t\t\tdone = false;\n\t\t\t\tyield {\n\t\t\t\t\twalker: backward,\n\t\t\t\t\tvalue: step.value\n\t\t\t\t};\n\t\t\t}\n\t\t}\n\n\t\tif ( forward ) {\n\t\t\tconst step = forward.next();\n\n\t\t\tif ( !step.done ) {\n\t\t\t\tdone = false;\n\t\t\t\tyield {\n\t\t\t\t\twalker: forward,\n\t\t\t\t\tvalue: step.value\n\t\t\t\t};\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Takes an array of non-intersecting ranges. For each of them gets minimal flat ranges covering that range and returns\n * all those minimal flat ranges.\n *\n * @param ranges Ranges to process.\n * @returns Minimal flat ranges of given `ranges`.\n */\nfunction* convertToMinimalFlatRanges( ranges: Iterable<Range> ): Iterable<Range> {\n\tfor ( const range of ranges ) {\n\t\tyield* range.getMinimalFlatRanges();\n\t}\n}\n\nfunction removeDisallowedAttributeFromNode( schema: Schema, node: Node, writer: Writer ) {\n\tfor ( const attribute of node.getAttributeKeys() ) {\n\t\tif ( !schema.checkAttribute( node, attribute ) ) {\n\t\t\twriter.removeAttribute( attribute, node );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/conversion/upcastdispatcher\n */\n\nimport ViewConsumable from './viewconsumable';\nimport ModelRange from '../model/range';\nimport ModelPosition from '../model/position';\nimport type ModelElement from '../model/element';\nimport type ModelNode from '../model/node';\nimport type ViewElement from '../view/element';\nimport type ViewText from '../view/text';\nimport type ViewDocumentFragment from '../view/documentfragment';\nimport type ModelDocumentFragment from '../model/documentfragment';\nimport type { default as Schema, SchemaContextDefinition } from '../model/schema';\nimport { SchemaContext } from '../model/schema'; // eslint-disable-line no-duplicate-imports\nimport type ModelWriter from '../model/writer';\nimport { isParagraphable, wrapInParagraph } from '../model/utils/autoparagraphing';\n\nimport type ViewItem from '../view/item';\n\nimport { CKEditorError, EmitterMixin } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Upcast dispatcher is a central point of the view-to-model conversion, which is a process of\n * converting a given {@link module:engine/view/documentfragment~DocumentFragment view document fragment} or\n * {@link module:engine/view/element~Element view element} into a correct model structure.\n *\n * During the conversion process, the dispatcher fires events for all {@link module:engine/view/node~Node view nodes}\n * from the converted view document fragment.\n * Special callbacks called \"converters\" should listen to these events in order to convert the view nodes.\n *\n * The second parameter of the callback is the `data` object with the following properties:\n *\n * * `data.viewItem` contains a {@link module:engine/view/node~Node view node} or a\n * {@link module:engine/view/documentfragment~DocumentFragment view document fragment}\n * that is converted at the moment and might be handled by the callback.\n * * `data.modelRange` is used to point to the result\n * of the current conversion (e.g. the element that is being inserted)\n * and is always a {@link module:engine/model/range~Range} when the conversion succeeds.\n * * `data.modelCursor` is a {@link module:engine/model/position~Position position} on which the converter should insert\n * the newly created items.\n *\n * The third parameter of the callback is an instance of {@link module:engine/conversion/upcastdispatcher~UpcastConversionApi}\n * which provides additional tools for converters.\n *\n * You can read more about conversion in the {@glink framework/deep-dive/conversion/upcast Upcast conversion} guide.\n *\n * Examples of event-based converters:\n *\n * ```ts\n * // A converter for links (<a>).\n * editor.data.upcastDispatcher.on( 'element:a', ( evt, data, conversionApi ) => {\n * \tif ( conversionApi.consumable.consume( data.viewItem, { name: true, attributes: [ 'href' ] } ) ) {\n * \t\t// The <a> element is inline and is represented by an attribute in the model.\n * \t\t// This is why you need to convert only children.\n * \t\tconst { modelRange } = conversionApi.convertChildren( data.viewItem, data.modelCursor );\n *\n * \t\tfor ( let item of modelRange.getItems() ) {\n * \t\t\tif ( conversionApi.schema.checkAttribute( item, 'linkHref' ) ) {\n * \t\t\t\tconversionApi.writer.setAttribute( 'linkHref', data.viewItem.getAttribute( 'href' ), item );\n * \t\t\t}\n * \t\t}\n * \t}\n * } );\n *\n * // Convert <p> element's font-size style.\n * // Note: You should use a low-priority observer in order to ensure that\n * // it is executed after the element-to-element converter.\n * editor.data.upcastDispatcher.on( 'element:p', ( evt, data, conversionApi ) => {\n * \tconst { consumable, schema, writer } = conversionApi;\n *\n * \tif ( !consumable.consume( data.viewItem, { style: 'font-size' } ) ) {\n * \t\treturn;\n * \t}\n *\n * \tconst fontSize = data.viewItem.getStyle( 'font-size' );\n *\n * \t// Do not go for the model element after data.modelCursor because it might happen\n * \t// that a single view element was converted to multiple model elements. Get all of them.\n * \tfor ( const item of data.modelRange.getItems( { shallow: true } ) ) {\n * \t\tif ( schema.checkAttribute( item, 'fontSize' ) ) {\n * \t\t\twriter.setAttribute( 'fontSize', fontSize, item );\n * \t\t}\n * \t}\n * }, { priority: 'low' } );\n *\n * // Convert all elements which have no custom converter into a paragraph (autoparagraphing).\n * editor.data.upcastDispatcher.on( 'element', ( evt, data, conversionApi ) => {\n * \t// Check if an element can be converted.\n * \tif ( !conversionApi.consumable.test( data.viewItem, { name: data.viewItem.name } ) ) {\n * \t\t// When an element is already consumed by higher priority converters, do nothing.\n * \t\treturn;\n * \t}\n *\n * \tconst paragraph = conversionApi.writer.createElement( 'paragraph' );\n *\n * \t// Try to safely insert a paragraph at the model cursor - it will find an allowed parent for the current element.\n * \tif ( !conversionApi.safeInsert( paragraph, data.modelCursor ) ) {\n * \t\t// When an element was not inserted, it means that you cannot insert a paragraph at this position.\n * \t\treturn;\n * \t}\n *\n * \t// Consume the inserted element.\n * \tconversionApi.consumable.consume( data.viewItem, { name: data.viewItem.name } ) );\n *\n * \t// Convert the children to a paragraph.\n * \tconst { modelRange } = conversionApi.convertChildren( data.viewItem, paragraph ) );\n *\n * \t// Update `modelRange` and `modelCursor` in the `data` as a conversion result.\n * \tconversionApi.updateConversionResult( paragraph, data );\n * }, { priority: 'low' } );\n * ```\n *\n * @fires viewCleanup\n * @fires element\n * @fires text\n * @fires documentFragment\n */\nexport default class UpcastDispatcher extends EmitterMixin() {\n\t/**\n\t * An interface passed by the dispatcher to the event callbacks.\n\t */\n\tpublic conversionApi: UpcastConversionApi;\n\n\t/**\n\t * The list of elements that were created during splitting.\n\t *\n\t * After the conversion process, the list is cleared.\n\t */\n\tprivate _splitParts = new Map<ModelElement, Array<ModelElement>>();\n\n\t/**\n\t * The list of cursor parent elements that were created during splitting.\n\t *\n\t * After the conversion process the list is cleared.\n\t */\n\tprivate _cursorParents = new Map<ModelNode, ModelElement | ModelDocumentFragment>();\n\n\t/**\n\t * The position in the temporary structure where the converted content is inserted. The structure reflects the context of\n\t * the target position where the content will be inserted. This property is built based on the context parameter of the\n\t * convert method.\n\t */\n\tprivate _modelCursor: ModelPosition | null = null;\n\n\t/**\n\t * The list of elements that were created during the splitting but should not get removed on conversion end even if they are empty.\n\t *\n\t * The list is cleared after the conversion process.\n\t */\n\tprivate _emptyElementsToKeep = new Set<ModelElement>();\n\n\t/**\n\t * Creates an upcast dispatcher that operates using the passed API.\n\t *\n\t * @see module:engine/conversion/upcastdispatcher~UpcastConversionApi\n\t * @param conversionApi Additional properties for an interface that will be passed to events fired\n\t * by the upcast dispatcher.\n\t */\n\tconstructor( conversionApi: Pick<UpcastConversionApi, 'schema'> ) {\n\t\tsuper();\n\n\t\tthis.conversionApi = {\n\t\t\t...conversionApi,\n\t\t\tconsumable: null as any,\n\t\t\twriter: null as any,\n\t\t\tstore: null,\n\t\t\tconvertItem: ( viewItem, modelCursor ) => this._convertItem( viewItem, modelCursor ),\n\t\t\tconvertChildren: ( viewElement, positionOrElement ) => this._convertChildren( viewElement, positionOrElement ),\n\t\t\tsafeInsert: ( modelNode, position ) => this._safeInsert( modelNode, position ),\n\t\t\tupdateConversionResult: ( modelElement, data ) => this._updateConversionResult( modelElement, data ),\n\t\t\t// Advanced API - use only if custom position handling is needed.\n\t\t\tsplitToAllowedParent: ( modelNode, modelCursor ) => this._splitToAllowedParent( modelNode, modelCursor ),\n\t\t\tgetSplitParts: modelElement => this._getSplitParts( modelElement ),\n\t\t\tkeepEmptyElement: modelElement => this._keepEmptyElement( modelElement )\n\t\t};\n\t}\n\n\t/**\n\t * Starts the conversion process. The entry point for the conversion.\n\t *\n\t * @fires element\n\t * @fires text\n\t * @fires documentFragment\n\t * @param viewElement The part of the view to be converted.\n\t * @param writer An instance of the model writer.\n\t * @param context Elements will be converted according to this context.\n\t * @returns Model data that is the result of the conversion process\n\t * wrapped in `DocumentFragment`. Converted marker elements will be set as the document fragment's\n\t * {@link module:engine/model/documentfragment~DocumentFragment#markers static markers map}.\n\t */\n\tpublic convert(\n\t\tviewElement: ViewElement | ViewDocumentFragment,\n\t\twriter: ModelWriter,\n\t\tcontext: SchemaContextDefinition = [ '$root' ]\n\t): ModelDocumentFragment {\n\t\tthis.fire<UpcastViewCleanupEvent>( 'viewCleanup', viewElement );\n\n\t\t// Create context tree and set position in the top element.\n\t\t// Items will be converted according to this position.\n\t\tthis._modelCursor = createContextTree( context, writer )!;\n\n\t\t// Store writer in conversion as a conversion API\n\t\t// to be sure that conversion process will use the same batch.\n\t\tthis.conversionApi.writer = writer;\n\n\t\t// Create consumable values list for conversion process.\n\t\tthis.conversionApi.consumable = ViewConsumable.createFrom( viewElement );\n\n\t\t// Custom data stored by converter for conversion process.\n\t\tthis.conversionApi.store = {};\n\n\t\t// Do the conversion.\n\t\tconst { modelRange } = this._convertItem( viewElement, this._modelCursor );\n\n\t\t// Conversion result is always a document fragment so let's create it.\n\t\tconst documentFragment = writer.createDocumentFragment();\n\n\t\t// When there is a conversion result.\n\t\tif ( modelRange ) {\n\t\t\t// Remove all empty elements that were create while splitting.\n\t\t\tthis._removeEmptyElements();\n\n\t\t\t// Move all items that were converted in context tree to the document fragment.\n\t\t\tfor ( const item of Array.from( this._modelCursor.parent.getChildren() ) ) {\n\t\t\t\twriter.append( item, documentFragment );\n\t\t\t}\n\n\t\t\t// Extract temporary markers elements from model and set as static markers collection.\n\t\t\t( documentFragment as any ).markers = extractMarkersFromModelFragment( documentFragment, writer );\n\t\t}\n\n\t\t// Clear context position.\n\t\tthis._modelCursor = null;\n\n\t\t// Clear split elements & parents lists.\n\t\tthis._splitParts.clear();\n\t\tthis._cursorParents.clear();\n\t\tthis._emptyElementsToKeep.clear();\n\n\t\t// Clear conversion API.\n\t\t( this.conversionApi as any ).writer = null;\n\t\tthis.conversionApi.store = null;\n\n\t\t// Return fragment as conversion result.\n\t\treturn documentFragment;\n\t}\n\n\t/**\n\t * @see module:engine/conversion/upcastdispatcher~UpcastConversionApi#convertItem\n\t */\n\tprivate _convertItem( viewItem: ViewItem | ViewDocumentFragment, modelCursor: ModelPosition ): {\n\t\tmodelRange: ModelRange | null;\n\t\tmodelCursor: ModelPosition;\n\t} {\n\t\tconst data: UpcastConversionData = { viewItem, modelCursor, modelRange: null };\n\n\t\tif ( viewItem.is( 'element' ) ) {\n\t\t\tthis.fire<UpcastElementEvent>(\n\t\t\t\t`element:${ viewItem.name }`,\n\t\t\t\tdata as UpcastConversionData<ViewElement>,\n\t\t\t\tthis.conversionApi\n\t\t\t);\n\t\t} else if ( viewItem.is( '$text' ) ) {\n\t\t\tthis.fire<UpcastTextEvent>(\n\t\t\t\t'text',\n\t\t\t\tdata as UpcastConversionData<ViewText>,\n\t\t\t\tthis.conversionApi\n\t\t\t);\n\t\t} else {\n\t\t\tthis.fire<UpcastDocumentFragmentEvent>(\n\t\t\t\t'documentFragment',\n\t\t\t\tdata as UpcastConversionData<ViewDocumentFragment>,\n\t\t\t\tthis.conversionApi\n\t\t\t);\n\t\t}\n\n\t\t// Handle incorrect conversion result.\n\t\tif ( data.modelRange && !( data.modelRange instanceof ModelRange ) ) {\n\t\t\t/**\n\t\t\t * Incorrect conversion result was dropped.\n\t\t\t *\n\t\t\t * {@link module:engine/model/range~Range Model range} should be a conversion result.\n\t\t\t *\n\t\t\t * @error view-conversion-dispatcher-incorrect-result\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'view-conversion-dispatcher-incorrect-result', this );\n\t\t}\n\n\t\treturn { modelRange: data.modelRange, modelCursor: data.modelCursor };\n\t}\n\n\t/**\n\t * @see module:engine/conversion/upcastdispatcher~UpcastConversionApi#convertChildren\n\t */\n\tprivate _convertChildren(\n\t\tviewItem: ViewElement | ViewDocumentFragment,\n\t\telementOrModelCursor: ModelPosition | ModelElement\n\t): {\n\t\tmodelRange: ModelRange;\n\t\tmodelCursor: ModelPosition;\n\t} {\n\t\tlet nextModelCursor = elementOrModelCursor.is( 'position' ) ?\n\t\t\telementOrModelCursor : ModelPosition._createAt( elementOrModelCursor, 0 );\n\n\t\tconst modelRange = new ModelRange( nextModelCursor );\n\n\t\tfor ( const viewChild of Array.from( viewItem.getChildren() ) ) {\n\t\t\tconst result = this._convertItem( viewChild, nextModelCursor );\n\n\t\t\tif ( result.modelRange instanceof ModelRange ) {\n\t\t\t\t( modelRange as any ).end = result.modelRange.end;\n\t\t\t\tnextModelCursor = result.modelCursor;\n\t\t\t}\n\t\t}\n\n\t\treturn { modelRange, modelCursor: nextModelCursor };\n\t}\n\n\t/**\n\t * @see module:engine/conversion/upcastdispatcher~UpcastConversionApi#safeInsert\n\t */\n\tprivate _safeInsert(\n\t\tmodelNode: ModelNode,\n\t\tposition: ModelPosition\n\t): boolean {\n\t\t// Find allowed parent for element that we are going to insert.\n\t\t// If current parent does not allow to insert element but one of the ancestors does\n\t\t// then split nodes to allowed parent.\n\t\tconst splitResult = this._splitToAllowedParent( modelNode, position );\n\n\t\t// When there is no split result it means that we can't insert element to model tree, so let's skip it.\n\t\tif ( !splitResult ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Insert element on allowed position.\n\t\tthis.conversionApi.writer!.insert( modelNode, splitResult.position );\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * @see module:engine/conversion/upcastdispatcher~UpcastConversionApi#updateConversionResult\n\t */\n\tprivate _updateConversionResult( modelElement: ModelElement, data: UpcastConversionData ): void {\n\t\tconst parts = this._getSplitParts( modelElement );\n\n\t\tconst writer = this.conversionApi.writer!;\n\n\t\t// Set conversion result range - only if not set already.\n\t\tif ( !data.modelRange ) {\n\t\t\tdata.modelRange = writer.createRange(\n\t\t\t\twriter.createPositionBefore( modelElement ),\n\t\t\t\twriter.createPositionAfter( parts[ parts.length - 1 ] )\n\t\t\t);\n\t\t}\n\n\t\tconst savedCursorParent = this._cursorParents.get( modelElement );\n\n\t\t// Now we need to check where the `modelCursor` should be.\n\t\tif ( savedCursorParent ) {\n\t\t\t// If we split parent to insert our element then we want to continue conversion in the new part of the split parent.\n\t\t\t//\n\t\t\t// before: <allowed><notAllowed>foo[]</notAllowed></allowed>\n\t\t\t// after: <allowed><notAllowed>foo</notAllowed> <converted></converted> <notAllowed>[]</notAllowed></allowed>\n\n\t\t\tdata.modelCursor = writer.createPositionAt( savedCursorParent, 0 );\n\t\t} else {\n\t\t\t// Otherwise just continue after inserted element.\n\n\t\t\tdata.modelCursor = data.modelRange.end;\n\t\t}\n\t}\n\n\t/**\n\t * @see module:engine/conversion/upcastdispatcher~UpcastConversionApi#splitToAllowedParent\n\t */\n\tprivate _splitToAllowedParent( node: ModelNode, modelCursor: ModelPosition ): {\n\t\tposition: ModelPosition;\n\t\tcursorParent?: ModelElement | ModelDocumentFragment;\n\t} | null {\n\t\tconst { schema, writer } = this.conversionApi;\n\n\t\t// Try to find allowed parent.\n\t\tlet allowedParent = schema.findAllowedParent( modelCursor, node );\n\n\t\tif ( allowedParent ) {\n\t\t\t// When current position parent allows to insert node then return this position.\n\t\t\tif ( allowedParent === modelCursor.parent ) {\n\t\t\t\treturn { position: modelCursor };\n\t\t\t}\n\n\t\t\t// When allowed parent is in context tree (it's outside the converted tree).\n\t\t\tif ( this._modelCursor!.parent.getAncestors().includes( allowedParent ) ) {\n\t\t\t\tallowedParent = null;\n\t\t\t}\n\t\t}\n\n\t\tif ( !allowedParent ) {\n\t\t\t// Check if the node wrapped with a paragraph would be accepted by the schema.\n\t\t\tif ( !isParagraphable( modelCursor, node, schema ) ) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\treturn {\n\t\t\t\tposition: wrapInParagraph( modelCursor, writer! )\n\t\t\t};\n\t\t}\n\n\t\t// Split element to allowed parent.\n\t\tconst splitResult = this.conversionApi.writer!.split( modelCursor, allowedParent );\n\n\t\t// Using the range returned by `model.Writer#split`, we will pair original elements with their split parts.\n\t\t//\n\t\t// The range returned from the writer spans \"over the split\" or, precisely saying, from the end of the original element (the one\n\t\t// that got split) to the beginning of the other part of that element:\n\t\t//\n\t\t// <limit><a><b><c>X[]Y</c></b><a></limit> ->\n\t\t// <limit><a><b><c>X[</c></b></a><a><b><c>]Y</c></b></a>\n\t\t//\n\t\t// After the split there cannot be any full node between the positions in `splitRange`. The positions are touching.\n\t\t// Also, because of how splitting works, it is easy to notice, that \"closing tags\" are in the reverse order than \"opening tags\".\n\t\t// Also, since we split all those elements, each of them has to have the other part.\n\t\t//\n\t\t// With those observations in mind, we will pair the original elements with their split parts by saving \"closing tags\" and matching\n\t\t// them with \"opening tags\" in the reverse order. For that we can use a stack.\n\t\tconst stack: Array<ModelElement> = [];\n\n\t\tfor ( const treeWalkerValue of splitResult.range.getWalker() ) {\n\t\t\tif ( treeWalkerValue.type == 'elementEnd' ) {\n\t\t\t\tstack.push( treeWalkerValue.item as ModelElement );\n\t\t\t} else {\n\t\t\t\t// There should not be any text nodes after the element is split, so the only other value is `elementStart`.\n\t\t\t\tconst originalPart = stack.pop();\n\t\t\t\tconst splitPart = treeWalkerValue.item as ModelElement;\n\n\t\t\t\tthis._registerSplitPair( originalPart!, splitPart );\n\t\t\t}\n\t\t}\n\n\t\tconst cursorParent = splitResult.range.end.parent;\n\t\tthis._cursorParents.set( node, cursorParent );\n\n\t\treturn {\n\t\t\tposition: splitResult.position,\n\t\t\tcursorParent\n\t\t};\n\t}\n\n\t/**\n\t * Registers that a `splitPart` element is a split part of the `originalPart` element.\n\t *\n\t * The data set by this method is used by {@link #_getSplitParts} and {@link #_removeEmptyElements}.\n\t */\n\tprivate _registerSplitPair( originalPart: ModelElement, splitPart: ModelElement ): void {\n\t\tif ( !this._splitParts.has( originalPart ) ) {\n\t\t\tthis._splitParts.set( originalPart, [ originalPart ] );\n\t\t}\n\n\t\tconst list = this._splitParts.get( originalPart )!;\n\n\t\tthis._splitParts.set( splitPart, list );\n\t\tlist.push( splitPart );\n\t}\n\n\t/**\n\t * @see module:engine/conversion/upcastdispatcher~UpcastConversionApi#getSplitParts\n\t */\n\tprivate _getSplitParts( element: ModelElement ): Array<ModelElement> {\n\t\tlet parts: Array<ModelElement>;\n\n\t\tif ( !this._splitParts.has( element ) ) {\n\t\t\tparts = [ element ];\n\t\t} else {\n\t\t\tparts = this._splitParts.get( element )!;\n\t\t}\n\n\t\treturn parts;\n\t}\n\n\t/**\n\t * Mark an element that were created during the splitting to not get removed on conversion end even if it is empty.\n\t */\n\tprivate _keepEmptyElement( element: ModelElement ): void {\n\t\tthis._emptyElementsToKeep.add( element );\n\t}\n\n\t/**\n\t * Checks if there are any empty elements created while splitting and removes them.\n\t *\n\t * This method works recursively to re-check empty elements again after at least one element was removed in the initial call,\n\t * as some elements might have become empty after other empty elements were removed from them.\n\t */\n\tprivate _removeEmptyElements(): void {\n\t\tlet anyRemoved = false;\n\n\t\tfor ( const element of this._splitParts.keys() ) {\n\t\t\tif ( element.isEmpty && !this._emptyElementsToKeep.has( element ) ) {\n\t\t\t\tthis.conversionApi.writer!.remove( element );\n\t\t\t\tthis._splitParts.delete( element );\n\n\t\t\t\tanyRemoved = true;\n\t\t\t}\n\t\t}\n\n\t\tif ( anyRemoved ) {\n\t\t\tthis._removeEmptyElements();\n\t\t}\n\t}\n}\n\n/**\n * Fired before the first conversion event, at the beginning of the upcast (view-to-model conversion) process.\n *\n * @eventName ~UpcastDispatcher#viewCleanup\n * @param viewItem A part of the view to be converted.\n */\nexport type UpcastViewCleanupEvent = {\n\tname: 'viewCleanup';\n\targs: [ ViewElement | ViewDocumentFragment ];\n};\n\ntype UpcastEvent<TName extends string, TItem extends ViewItem | ViewDocumentFragment> = {\n\tname: TName | `${ TName }:${ string }`;\n\targs: [ data: UpcastConversionData<TItem>, conversionApi: UpcastConversionApi ];\n};\n\n/**\n * Conversion data.\n *\n * **Note:** Keep in mind that this object is shared by reference between all conversion callbacks that will be called.\n * This means that callbacks can override values if needed, and these values will be available in other callbacks.\n */\nexport interface UpcastConversionData<TItem extends ViewItem | ViewDocumentFragment = ViewItem | ViewDocumentFragment> {\n\n\t/**\n\t * The converted item.\n\t */\n\tviewItem: TItem;\n\n\t/**\n\t * The position where the converter should start changes.\n\t * Change this value for the next converter to tell where the conversion should continue.\n\t */\n\tmodelCursor: ModelPosition;\n\n\t/**\n\t * The current state of conversion result. Every change to\n\t * the converted element should be reflected by setting or modifying this property.\n\t */\n\tmodelRange: ModelRange | null;\n}\n\n/**\n * Fired when an {@link module:engine/view/element~Element} is converted.\n *\n * `element` is a namespace event for a class of events. Names of actually called events follow the pattern of\n * `element:<elementName>` where `elementName` is the name of the converted element. This way listeners may listen to\n * a conversion of all or just specific elements.\n *\n * @eventName ~UpcastDispatcher#element\n * @param data The conversion data. Keep in mind that this object is shared by reference between all callbacks\n * that will be called. This means that callbacks can override values if needed, and these values\n * will be available in other callbacks.\n * @param conversionApi Conversion utilities to be used by the callback.\n */\nexport type UpcastElementEvent = UpcastEvent<'element', ViewElement>;\n\n/**\n * Fired when a {@link module:engine/view/text~Text} is converted.\n *\n * @eventName ~UpcastDispatcher#text\n * @see ~UpcastDispatcher#event:element\n */\nexport type UpcastTextEvent = UpcastEvent<'text', ViewText>;\n\n/**\n * Fired when a {@link module:engine/view/documentfragment~DocumentFragment} is converted.\n *\n * @eventName ~UpcastDispatcher#documentFragment\n * @see ~UpcastDispatcher#event:element\n */\nexport type UpcastDocumentFragmentEvent = UpcastEvent<'documentFragment', ViewDocumentFragment>;\n\n/**\n * Traverses given model item and searches elements which marks marker range. Found element is removed from\n * DocumentFragment but path of this element is stored in a Map which is then returned.\n *\n * @param modelItem Fragment of model.\n * @returns List of static markers.\n */\nfunction extractMarkersFromModelFragment( modelItem: ModelDocumentFragment, writer: ModelWriter ): Map<string, ModelRange> {\n\tconst markerElements = new Set<ModelElement>();\n\tconst markers = new Map<string, ModelRange>();\n\n\t// Create ModelTreeWalker.\n\tconst range = ModelRange._createIn( modelItem ).getItems();\n\n\t// Walk through DocumentFragment and collect marker elements.\n\tfor ( const item of range ) {\n\t\t// Check if current element is a marker.\n\t\tif ( item.is( 'element', '$marker' ) ) {\n\t\t\tmarkerElements.add( item );\n\t\t}\n\t}\n\n\t// Walk through collected marker elements store its path and remove its from the DocumentFragment.\n\tfor ( const markerElement of markerElements ) {\n\t\tconst markerName = markerElement.getAttribute( 'data-name' ) as string;\n\t\tconst currentPosition = writer.createPositionBefore( markerElement );\n\n\t\t// When marker of given name is not stored it means that we have found the beginning of the range.\n\t\tif ( !markers.has( markerName ) ) {\n\t\t\tmarkers.set( markerName, new ModelRange( currentPosition.clone() ) );\n\t\t// Otherwise is means that we have found end of the marker range.\n\t\t} else {\n\t\t\t( markers.get( markerName ) as any ).end = currentPosition.clone();\n\t\t}\n\n\t\t// Remove marker element from DocumentFragment.\n\t\twriter.remove( markerElement );\n\t}\n\n\treturn markers;\n}\n\n/**\n * Creates model fragment according to given context and returns position in the bottom (the deepest) element.\n */\nfunction createContextTree(\n\tcontextDefinition: SchemaContextDefinition,\n\twriter: ModelWriter\n): ModelPosition | undefined {\n\tlet position: ModelPosition | undefined;\n\n\tfor ( const item of new SchemaContext( contextDefinition ) ) {\n\t\tconst attributes: Record<string, unknown> = {};\n\n\t\tfor ( const key of item.getAttributeKeys() ) {\n\t\t\tattributes[ key ] = item.getAttribute( key );\n\t\t}\n\n\t\tconst current = writer.createElement( item.name, attributes );\n\n\t\tif ( position ) {\n\t\t\twriter.insert( current, position );\n\t\t}\n\n\t\tposition = ModelPosition._createAt( current, 0 );\n\t}\n\n\treturn position;\n}\n\n/**\n * A set of conversion utilities available as the third parameter of the\n * {@link module:engine/conversion/upcastdispatcher~UpcastDispatcher upcast dispatcher}'s events.\n */\nexport interface UpcastConversionApi {\n\n\t/**\n\t * Stores information about what parts of the processed view item are still waiting to be handled. After a piece of view item\n\t * was converted, an appropriate consumable value should be\n\t * {@link module:engine/conversion/viewconsumable~ViewConsumable#consume consumed}.\n\t */\n\tconsumable: ViewConsumable;\n\n\t/**\n\t * The model's schema instance.\n\t */\n\tschema: Schema;\n\n\t/**\n\t * The {@link module:engine/model/writer~Writer} instance used to manipulate the data during conversion.\n\t */\n\twriter: ModelWriter;\n\n\t/**\n\t * Custom data stored by converters for the conversion process. Custom properties of this object can be defined and use to\n\t * pass parameters between converters.\n\t *\n\t * The difference between this property and the `data` parameter of\n\t * {@link module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:element} is that the `data` parameters allow you\n\t * to pass parameters within a single event and `store` within the whole conversion.\n\t */\n\tstore: unknown;\n\n\t/**\n\t * Starts the conversion of a given item by firing an appropriate event.\n\t *\n\t * Every fired event is passed (as the first parameter) an object with the `modelRange` property. Every event may set and/or\n\t * modify that property. When all callbacks are done, the final value of the `modelRange` property is returned by this method.\n\t * The `modelRange` must be a {@link module:engine/model/range~Range model range} or `null` (as set by default).\n\t *\n\t * @fires module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:element\n\t * @fires module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:text\n\t * @fires module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:documentFragment\n\t * @param viewItem Item to convert.\n\t * @param modelCursor The conversion position.\n\t * @returns The conversion result:\n\t * * `result.modelRange` The model range containing the result of the item conversion,\n\t * created and modified by callbacks attached to the fired event, or `null` if the conversion result was incorrect.\n\t * * `result.modelCursor` The position where the conversion should be continued.\n\t */\n\tconvertItem( viewItem: ViewItem, modelCursor: ModelPosition ): {\n\t\tmodelRange: ModelRange | null;\n\t\tmodelCursor: ModelPosition;\n\t};\n\n\t/**\n\t * Starts the conversion of all children of a given item by firing appropriate events for all the children.\n\t *\n\t * @fires module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:element\n\t * @fires module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:text\n\t * @fires module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:documentFragment\n\t * @param viewElement An element whose children should be converted.\n\t * @param positionOrElement A position or an element of\n\t * the conversion.\n\t * @returns The conversion result:\n\t * * `result.modelRange` The model range containing the results of the conversion of all children\n\t * of the given item. When no child was converted, the range is collapsed.\n\t * * `result.modelCursor` The position where the conversion should be continued.\n\t */\n\tconvertChildren( viewElement: ViewElement | ViewDocumentFragment, positionOrElement: ModelPosition | ModelElement ): {\n\t\tmodelRange: ModelRange | null;\n\t\tmodelCursor: ModelPosition;\n\t};\n\n\t/**\n\t * Safely inserts an element to the document, checking the {@link module:engine/model/schema~Schema schema} to find an allowed parent\n\t * for an element that you are going to insert, starting from the given position. If the current parent does not allow to insert\n\t * the element but one of the ancestors does, then splits the nodes to allowed parent.\n\t *\n\t * If the schema allows to insert the node in a given position, nothing is split.\n\t *\n\t * If it was not possible to find an allowed parent, `false` is returned and nothing is split.\n\t *\n\t * Otherwise, ancestors are split.\n\t *\n\t * For instance, if `<imageBlock>` is not allowed in `<paragraph>` but is allowed in `$root`:\n\t *\n\t * ```\n\t * <paragraph>foo[]bar</paragraph>\n\t *\n\t * -> safe insert for `<imageBlock>` will split ->\n\t *\n\t * <paragraph>foo</paragraph>[]<paragraph>bar</paragraph>\n\t *```\n\t *\n\t * Example usage:\n\t *\n\t * ```\n\t * const myElement = conversionApi.writer.createElement( 'myElement' );\n\t *\n\t * if ( !conversionApi.safeInsert( myElement, data.modelCursor ) ) {\n\t * \treturn;\n\t * }\n\t *```\n\t *\n\t * The split result is saved and {@link #updateConversionResult} should be used to update the\n\t * {@link module:engine/conversion/upcastdispatcher~UpcastConversionData conversion data}.\n\t *\n\t * @param modelNode The node to insert.\n\t * @param position The position where an element is going to be inserted.\n\t * @returns The split result. If it was not possible to find an allowed position, `false` is returned.\n\t */\n\tsafeInsert( modelNode: ModelNode, position: ModelPosition ): boolean;\n\n\t/**\n\t * Updates the conversion result and sets a proper {@link module:engine/conversion/upcastdispatcher~UpcastConversionData#modelRange} and\n\t * the next {@link module:engine/conversion/upcastdispatcher~UpcastConversionData#modelCursor} after the conversion.\n\t * Used together with {@link #safeInsert}, it enables you to easily convert elements without worrying if the node was split\n\t * during the conversion of its children.\n\t *\n\t * A usage example in converter code:\n\t *\n\t * ```ts\n\t * const myElement = conversionApi.writer.createElement( 'myElement' );\n\t *\n\t * if ( !conversionApi.safeInsert( myElement, data.modelCursor ) ) {\n\t * \treturn;\n\t * }\n\t *\n\t * // Children conversion may split `myElement`.\n\t * conversionApi.convertChildren( data.viewItem, myElement );\n\t *\n\t * conversionApi.updateConversionResult( myElement, data );\n\t * ```\n\t */\n\tupdateConversionResult( modelElement: ModelElement, data: UpcastConversionData ): void;\n\n\t/**\n\t * Checks the {@link module:engine/model/schema~Schema schema} to find an allowed parent for an element that is going to be inserted\n\t * starting from the given position. If the current parent does not allow inserting an element but one of the ancestors does, the method\n\t * splits nodes to allowed parent.\n\t *\n\t * If the schema allows inserting the node in the given position, nothing is split and an object with that position is returned.\n\t *\n\t * If it was not possible to find an allowed parent, `null` is returned and nothing is split.\n\t *\n\t * Otherwise, ancestors are split and an object with a position and the copy of the split element is returned.\n\t *\n\t * For instance, if `<imageBlock>` is not allowed in `<paragraph>` but is allowed in `$root`:\n\t *\n\t * ```\n\t * <paragraph>foo[]bar</paragraph>\n\t *\n\t * -> split for `<imageBlock>` ->\n\t *\n\t * <paragraph>foo</paragraph>[]<paragraph>bar</paragraph>\n\t * ```\n\t *\n\t * In the example above, the position between `<paragraph>` elements will be returned as `position` and the second `paragraph`\n\t * as `cursorParent`.\n\t *\n\t * **Note:** This is an advanced method. For most cases {@link #safeInsert} and {@link #updateConversionResult} should be used.\n\t *\n\t * @param modelNode The node to insert.\n\t * @param modelCursor The position where the element is going to be inserted.\n\t * @returns The split result. If it was not possible to find an allowed position, `null` is returned.\n\t * * `position` The position between split elements.\n\t * * `cursorParent` The element inside which the cursor should be placed to\n\t * continue the conversion. When the element is not defined it means that there was no split.\n\t */\n\tsplitToAllowedParent( modelNode: ModelNode, modelCursor: ModelPosition ): {\n\t\tposition: ModelPosition;\n\t\tcursorParent?: ModelElement | ModelDocumentFragment;\n\t} | null;\n\n\t/**\n\t * Returns all the split parts of the given `element` that were created during upcasting through using {@link #splitToAllowedParent}.\n\t * It enables you to easily track these elements and continue processing them after they are split during the conversion of their\n\t * children.\n\t *\n\t * ```\n\t * <paragraph>Foo<imageBlock />bar<imageBlock />baz</paragraph> ->\n\t * <paragraph>Foo</paragraph><imageBlock /><paragraph>bar</paragraph><imageBlock /><paragraph>baz</paragraph>\n\t * ```\n\t *\n\t * For a reference to any of above paragraphs, the function will return all three paragraphs (the original element included),\n\t * sorted in the order of their creation (the original element is the first one).\n\t *\n\t * If the given `element` was not split, an array with a single element is returned.\n\t *\n\t * A usage example in the converter code:\n\t *\n\t * ```ts\n\t * const myElement = conversionApi.writer.createElement( 'myElement' );\n\t *\n\t * // Children conversion may split `myElement`.\n\t * conversionApi.convertChildren( data.viewItem, data.modelCursor );\n\t *\n\t * const splitParts = conversionApi.getSplitParts( myElement );\n\t * const lastSplitPart = splitParts[ splitParts.length - 1 ];\n\t *\n\t * // Setting `data.modelRange` basing on split parts:\n\t * data.modelRange = conversionApi.writer.createRange(\n\t * \tconversionApi.writer.createPositionBefore( myElement ),\n\t * \tconversionApi.writer.createPositionAfter( lastSplitPart )\n\t * );\n\t *\n\t * // Setting `data.modelCursor` to continue after the last split element:\n\t * data.modelCursor = conversionApi.writer.createPositionAfter( lastSplitPart );\n\t * ```\n\t *\n\t * **Tip:** If you are unable to get a reference to the original element (for example because the code is split into multiple converters\n\t * or even classes) but it has already been converted, you may want to check the first element in `data.modelRange`. This is a common\n\t * situation if an attribute converter is separated from an element converter.\n\t *\n\t * **Note:** This is an advanced method. For most cases {@link #safeInsert} and {@link #updateConversionResult} should be used.\n\t */\n\tgetSplitParts( modelElement: ModelElement ): Array<ModelElement>;\n\n\t/**\n\t * Mark an element that was created during splitting to not get removed on conversion end even if it is empty.\n\t *\n\t * **Note:** This is an advanced method. For most cases you will not need to keep the split empty element.\n\t */\n\tkeepEmptyElement( modelElement: ModelElement ): void;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/dataprocessor/basichtmlwriter\n */\n\n/* globals document */\n\nimport type HtmlWriter from './htmlwriter';\n\n/**\n * Basic HTML writer. It uses the native `innerHTML` property for basic conversion\n * from a document fragment to an HTML string.\n */\nexport default class BasicHtmlWriter implements HtmlWriter {\n\t/**\n\t * Returns an HTML string created from the document fragment.\n\t */\n\tpublic getHtml( fragment: DocumentFragment ): string {\n\t\tconst doc = document.implementation.createHTMLDocument( '' );\n\t\tconst container = doc.createElement( 'div' );\n\n\t\tcontainer.appendChild( fragment );\n\n\t\treturn container.innerHTML;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/dataprocessor/htmldataprocessor\n */\n\n/* globals DOMParser */\n\nimport BasicHtmlWriter from './basichtmlwriter';\nimport DomConverter from '../view/domconverter';\n\nimport type DataProcessor from './dataprocessor';\nimport type HtmlWriter from './htmlwriter';\nimport type ViewDocument from '../view/document';\nimport type ViewDocumentFragment from '../view/documentfragment';\nimport type { MatcherPattern } from '../view/matcher';\n\n/**\n * The HTML data processor class.\n * This data processor implementation uses HTML as input and output data.\n */\nexport default class HtmlDataProcessor implements DataProcessor {\n\t/**\n\t * A DOM parser instance used to parse an HTML string to an HTML document.\n\t */\n\tpublic domParser: DOMParser;\n\n\t/**\n\t * A DOM converter used to convert DOM elements to view elements.\n\t */\n\tpublic domConverter: DomConverter;\n\n\t/**\n\t * A basic HTML writer instance used to convert DOM elements to an HTML string.\n\t */\n\tpublic htmlWriter: HtmlWriter;\n\n\tpublic skipComments: boolean = true;\n\n\t/**\n\t * Creates a new instance of the HTML data processor class.\n\t *\n\t * @param document The view document instance.\n\t */\n\tconstructor( document: ViewDocument ) {\n\t\tthis.domParser = new DOMParser();\n\t\tthis.domConverter = new DomConverter( document, { renderingMode: 'data' } );\n\t\tthis.htmlWriter = new BasicHtmlWriter();\n\t}\n\n\t/**\n\t * Converts a provided {@link module:engine/view/documentfragment~DocumentFragment document fragment}\n\t * to data format &mdash; in this case to an HTML string.\n\t *\n\t * @returns HTML string.\n\t */\n\tpublic toData( viewFragment: ViewDocumentFragment ): string {\n\t\t// Convert view DocumentFragment to DOM DocumentFragment.\n\t\tconst domFragment = this.domConverter.viewToDom( viewFragment ) as DocumentFragment;\n\n\t\t// Convert DOM DocumentFragment to HTML output.\n\t\treturn this.htmlWriter.getHtml( domFragment );\n\t}\n\n\t/**\n\t * Converts the provided HTML string to a view tree.\n\t *\n\t * @param data An HTML string.\n\t * @returns A converted view element.\n\t */\n\tpublic toView( data: string ): ViewDocumentFragment {\n\t\t// Convert input HTML data to DOM DocumentFragment.\n\t\tconst domFragment = this._toDom( data );\n\n\t\t// Convert DOM DocumentFragment to view DocumentFragment.\n\t\treturn this.domConverter.domToView( domFragment, { skipComments: this.skipComments } ) as ViewDocumentFragment;\n\t}\n\n\t/**\n\t * Registers a {@link module:engine/view/matcher~MatcherPattern} for view elements whose content should be treated as raw data\n\t * and not processed during the conversion from the DOM to the view elements.\n\t *\n\t * The raw data can be later accessed by a\n\t * {@link module:engine/view/element~Element#getCustomProperty custom property of a view element} called `\"$rawContent\"`.\n\t *\n\t * @param pattern Pattern matching all view elements whose content should be treated as raw data.\n\t */\n\tpublic registerRawContentMatcher( pattern: MatcherPattern ): void {\n\t\tthis.domConverter.registerRawContentMatcher( pattern );\n\t}\n\n\t/**\n\t * If the processor is set to use marked fillers, it will insert `&nbsp;` fillers wrapped in `<span>` elements\n\t * (`<span data-cke-filler=\"true\">&nbsp;</span>`) instead of regular `&nbsp;` characters.\n\t *\n\t * This mode allows for a more precise handling of the block fillers (so they do not leak into the editor content) but\n\t * bloats the editor data with additional markup.\n\t *\n\t * This mode may be required by some features and will be turned on by them automatically.\n\t *\n\t * @param type Whether to use the default or the marked `&nbsp;` block fillers.\n\t */\n\tpublic useFillerType( type: 'default' | 'marked' ): void {\n\t\tthis.domConverter.blockFillerMode = type == 'marked' ? 'markedNbsp' : 'nbsp';\n\t}\n\n\t/**\n\t * Converts an HTML string to its DOM representation. Returns a document fragment containing nodes parsed from\n\t * the provided data.\n\t */\n\tprotected _toDom( data: string ): DocumentFragment {\n\t\t// Wrap data with a <body> tag so leading non-layout nodes (like <script>, <style>, HTML comment)\n\t\t// will be preserved in the body collection.\n\t\t// Do it only for data that is not a full HTML document.\n\t\tif ( !data.match( /<(?:html|body|head|meta)(?:\\s[^>]*)?>/i ) ) {\n\t\t\tdata = `<body>${ data }</body>`;\n\t\t}\n\n\t\tconst document = this.domParser.parseFromString( data, 'text/html' );\n\t\tconst fragment = document.createDocumentFragment();\n\t\tconst bodyChildNodes = document.body.childNodes;\n\n\t\twhile ( bodyChildNodes.length > 0 ) {\n\t\t\tfragment.appendChild( bodyChildNodes[ 0 ] );\n\t\t}\n\n\t\treturn fragment;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/controller/datacontroller\n */\n\nimport {\n\tCKEditorError,\n\tEmitterMixin,\n\tObservableMixin,\n\tlogWarning\n} from '@ckeditor/ckeditor5-utils';\n\nimport Mapper from '../conversion/mapper';\n\nimport DowncastDispatcher, { type DowncastInsertEvent } from '../conversion/downcastdispatcher';\nimport { insertAttributesAndChildren, insertText } from '../conversion/downcasthelpers';\n\nimport UpcastDispatcher, {\n\ttype UpcastDocumentFragmentEvent,\n\ttype UpcastElementEvent,\n\ttype UpcastTextEvent\n} from '../conversion/upcastdispatcher';\nimport { convertText, convertToModelFragment } from '../conversion/upcasthelpers';\n\nimport ViewDocumentFragment from '../view/documentfragment';\nimport ViewDocument from '../view/document';\nimport ViewDowncastWriter from '../view/downcastwriter';\nimport type ViewElement from '../view/element';\nimport type { StylesProcessor } from '../view/stylesmap';\nimport type { MatcherPattern } from '../view/matcher';\n\nimport ModelRange from '../model/range';\nimport type Model from '../model/model';\nimport type ModelText from '../model/text';\nimport type ModelElement from '../model/element';\nimport type ModelTextProxy from '../model/textproxy';\nimport type ModelDocumentFragment from '../model/documentfragment';\nimport type { SchemaContextDefinition } from '../model/schema';\nimport type { BatchType } from '../model/batch';\nimport { autoParagraphEmptyRoots } from '../model/utils/autoparagraphing';\n\nimport HtmlDataProcessor from '../dataprocessor/htmldataprocessor';\nimport type DataProcessor from '../dataprocessor/dataprocessor';\n\n/**\n * Controller for the data pipeline. The data pipeline controls how data is retrieved from the document\n * and set inside it. Hence, the controller features two methods which allow to {@link ~DataController#get get}\n * and {@link ~DataController#set set} data of the {@link ~DataController#model model}\n * using the given:\n *\n * * {@link module:engine/dataprocessor/dataprocessor~DataProcessor data processor},\n * * downcast converters,\n * * upcast converters.\n *\n * An instance of the data controller is always available in the {@link module:core/editor/editor~Editor#data `editor.data`}\n * property:\n *\n * ```ts\n * editor.data.get( { rootName: 'customRoot' } ); // -> '<p>Hello!</p>'\n * ```\n */\nexport default class DataController extends EmitterMixin() {\n\t/**\n\t * Data model.\n\t */\n\tpublic readonly model: Model;\n\n\t/**\n\t * Mapper used for the conversion. It has no permanent bindings, because these are created while getting data and\n\t * ae cleared directly after the data are converted. However, the mapper is defined as a class property, because\n\t * it needs to be passed to the `DowncastDispatcher` as a conversion API.\n\t */\n\tpublic readonly mapper: Mapper;\n\n\t/**\n\t * Downcast dispatcher used by the {@link #get get method}. Downcast converters should be attached to it.\n\t */\n\tpublic readonly downcastDispatcher: DowncastDispatcher;\n\n\t/**\n\t * Upcast dispatcher used by the {@link #set set method}. Upcast converters should be attached to it.\n\t */\n\tpublic readonly upcastDispatcher: UpcastDispatcher;\n\n\t/**\n\t * The view document used by the data controller.\n\t */\n\tpublic readonly viewDocument: ViewDocument;\n\n\t/**\n\t * Styles processor used during the conversion.\n\t */\n\tpublic readonly stylesProcessor: StylesProcessor;\n\n\t/**\n\t * Data processor used specifically for HTML conversion.\n\t */\n\tpublic readonly htmlProcessor: HtmlDataProcessor;\n\n\t/**\n\t * Data processor used during the conversion.\n\t * Same instance as {@link #htmlProcessor} by default. Can be replaced at run time to handle different format, e.g. XML or Markdown.\n\t */\n\tpublic processor: DataProcessor;\n\n\t/**\n\t * The view downcast writer just for data conversion purposes, i.e. to modify\n\t * the {@link #viewDocument}.\n\t */\n\tprivate readonly _viewWriter: ViewDowncastWriter;\n\n\t/**\n\t * Creates a data controller instance.\n\t *\n\t * @param model Data model.\n\t * @param stylesProcessor The styles processor instance.\n\t */\n\tconstructor( model: Model, stylesProcessor: StylesProcessor ) {\n\t\tsuper();\n\n\t\tthis.model = model;\n\t\tthis.mapper = new Mapper();\n\n\t\tthis.downcastDispatcher = new DowncastDispatcher( {\n\t\t\tmapper: this.mapper,\n\t\t\tschema: model.schema\n\t\t} );\n\t\tthis.downcastDispatcher.on<DowncastInsertEvent<ModelText | ModelTextProxy>>( 'insert:$text', insertText(), { priority: 'lowest' } );\n\t\tthis.downcastDispatcher.on<DowncastInsertEvent>( 'insert', insertAttributesAndChildren(), { priority: 'lowest' } );\n\n\t\tthis.upcastDispatcher = new UpcastDispatcher( {\n\t\t\tschema: model.schema\n\t\t} );\n\n\t\tthis.viewDocument = new ViewDocument( stylesProcessor );\n\t\tthis.stylesProcessor = stylesProcessor;\n\t\tthis.htmlProcessor = new HtmlDataProcessor( this.viewDocument );\n\t\tthis.processor = this.htmlProcessor;\n\t\tthis._viewWriter = new ViewDowncastWriter( this.viewDocument );\n\n\t\t// Define default converters for text and elements.\n\t\t//\n\t\t// Note that if there is no default converter for the element it will be skipped, for instance `<b>foo</b>` will be\n\t\t// converted to nothing. We therefore add `convertToModelFragment` as a last converter so it converts children of that\n\t\t// element to the document fragment so `<b>foo</b>` will still be converted to `foo` even if there is no converter for `<b>`.\n\t\tthis.upcastDispatcher.on<UpcastTextEvent>( 'text', convertText(), { priority: 'lowest' } );\n\t\tthis.upcastDispatcher.on<UpcastElementEvent>( 'element', convertToModelFragment(), { priority: 'lowest' } );\n\t\tthis.upcastDispatcher.on<UpcastDocumentFragmentEvent>( 'documentFragment', convertToModelFragment(), { priority: 'lowest' } );\n\n\t\tObservableMixin().prototype.decorate.call( this, 'init' as any );\n\t\tObservableMixin().prototype.decorate.call( this, 'set' as any );\n\t\tObservableMixin().prototype.decorate.call( this, 'get' as any );\n\t\tObservableMixin().prototype.decorate.call( this, 'toView' as any );\n\t\tObservableMixin().prototype.decorate.call( this, 'toModel' as any );\n\n\t\t// Fire the `ready` event when the initialization has completed. Such low-level listener offers the possibility\n\t\t// to plug into the initialization pipeline without interrupting the initialization flow.\n\t\tthis.on<DataControllerInitEvent>( 'init', () => {\n\t\t\tthis.fire<DataControllerReadyEvent>( 'ready' );\n\t\t}, { priority: 'lowest' } );\n\n\t\t// Fix empty roots after DataController is 'ready' (note that the init method could be decorated and stopped).\n\t\t// We need to handle this event because initial data could be empty and the post-fixer would not get triggered.\n\t\tthis.on<DataControllerReadyEvent>( 'ready', () => {\n\t\t\tthis.model.enqueueChange( { isUndoable: false }, autoParagraphEmptyRoots );\n\t\t}, { priority: 'lowest' } );\n\t}\n\n\t/**\n\t * Returns the model's data converted by downcast dispatchers attached to {@link #downcastDispatcher} and\n\t * formatted by the {@link #processor data processor}.\n\t *\n\t * A warning is logged when you try to retrieve data for a detached root, as most probably this is a mistake. A detached root should\n\t * be treated like it is removed, and you should not save its data. Note, that the detached root data is always an empty string.\n\t *\n\t * @fires get\n\t * @param options Additional configuration for the retrieved data. `DataController` provides two optional\n\t * properties: `rootName` and `trim`. Other properties of this object are specified by various editor features.\n\t * @param options.rootName Root name. Default 'main'.\n\t * @param options.trim Whether returned data should be trimmed. This option is set to `empty` by default,\n\t * which means whenever editor content is considered empty, an empty string will be returned. To turn off trimming completely\n\t * use `'none'`. In such cases the exact content will be returned (for example a `<p>&nbsp;</p>` for an empty editor).\n\t * @returns Output data.\n\t */\n\tpublic get(\n\t\toptions: {\n\t\t\trootName?: string;\n\t\t\ttrim?: 'empty' | 'none';\n\t\t\t[ key: string ]: unknown;\n\t\t} = {}\n\t): string {\n\t\tconst { rootName = 'main', trim = 'empty' } = options;\n\n\t\tif ( !this._checkIfRootsExists( [ rootName ] ) ) {\n\t\t\t/**\n\t\t\t * Cannot get data from a non-existing root. This error is thrown when\n\t\t\t * {@link module:engine/controller/datacontroller~DataController#get `DataController#get()` method}\n\t\t\t * is called with a non-existent root name. For example, if there is an editor instance with only `main` root,\n\t\t\t * calling {@link module:engine/controller/datacontroller~DataController#get} like:\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * data.get( { rootName: 'root2' } );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * will throw this error.\n\t\t\t *\n\t\t\t * @error datacontroller-get-non-existent-root\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'datacontroller-get-non-existent-root', this );\n\t\t}\n\n\t\tconst root = this.model.document.getRoot( rootName )!;\n\n\t\tif ( !root.isAttached() ) {\n\t\t\t/**\n\t\t\t * Retrieving document data for a detached root.\n\t\t\t *\n\t\t\t * This usually indicates an error as a detached root should be considered \"removed\" and should not be included in the\n\t\t\t * document data.\n\t\t\t *\n\t\t\t * @error datacontroller-get-detached-root\n\t\t\t */\n\t\t\tlogWarning( 'datacontroller-get-detached-root', this );\n\t\t}\n\n\t\tif ( trim === 'empty' && !this.model.hasContent( root, { ignoreWhitespaces: true } ) ) {\n\t\t\treturn '';\n\t\t}\n\n\t\treturn this.stringify( root, options );\n\t}\n\n\t/**\n\t * Returns the content of the given {@link module:engine/model/element~Element model's element} or\n\t * {@link module:engine/model/documentfragment~DocumentFragment model document fragment} converted by the downcast converters\n\t * attached to the {@link #downcastDispatcher} and formatted by the {@link #processor data processor}.\n\t *\n\t * @param modelElementOrFragment The element whose content will be stringified.\n\t * @param options Additional configuration passed to the conversion process.\n\t * @returns Output data.\n\t */\n\tpublic stringify(\n\t\tmodelElementOrFragment: ModelElement | ModelDocumentFragment,\n\t\toptions: Record<string, unknown> = {}\n\t): string {\n\t\t// Model -> view.\n\t\tconst viewDocumentFragment = this.toView( modelElementOrFragment, options );\n\n\t\t// View -> data.\n\t\treturn this.processor.toData( viewDocumentFragment );\n\t}\n\n\t/**\n\t * Returns the content of the given {@link module:engine/model/element~Element model element} or\n\t * {@link module:engine/model/documentfragment~DocumentFragment model document fragment} converted by the downcast\n\t * converters attached to {@link #downcastDispatcher} into a\n\t * {@link module:engine/view/documentfragment~DocumentFragment view document fragment}.\n\t *\n\t * @fires toView\n\t * @param modelElementOrFragment Element or document fragment whose content will be converted.\n\t * @param options Additional configuration that will be available through the\n\t * {@link module:engine/conversion/downcastdispatcher~DowncastConversionApi#options} during the conversion process.\n\t * @returns Output view DocumentFragment.\n\t */\n\tpublic toView(\n\t\tmodelElementOrFragment: ModelElement | ModelDocumentFragment,\n\t\toptions: Record<string, unknown> = {}\n\t): ViewDocumentFragment {\n\t\tconst viewDocument = this.viewDocument;\n\t\tconst viewWriter = this._viewWriter;\n\n\t\t// Clear bindings so the call to this method returns correct results.\n\t\tthis.mapper.clearBindings();\n\n\t\t// First, convert elements.\n\t\tconst modelRange = ModelRange._createIn( modelElementOrFragment );\n\t\tconst viewDocumentFragment = new ViewDocumentFragment( viewDocument );\n\n\t\tthis.mapper.bindElements( modelElementOrFragment, viewDocumentFragment );\n\n\t\t// Prepare list of markers.\n\t\t// For document fragment, simply take the markers assigned to this document fragment.\n\t\t// For model root, all markers in that root will be taken.\n\t\t// For model element, we need to check which markers are intersecting with this element and relatively modify the markers' ranges.\n\t\t// Collapsed markers at element boundary, although considered as not intersecting with the element, will also be returned.\n\t\tconst markers = modelElementOrFragment.is( 'documentFragment' ) ?\n\t\t\tmodelElementOrFragment.markers :\n\t\t\t_getMarkersRelativeToElement( modelElementOrFragment );\n\n\t\tthis.downcastDispatcher.convert( modelRange, markers, viewWriter, options );\n\n\t\treturn viewDocumentFragment;\n\t}\n\n\t/**\n\t * Sets the initial input data parsed by the {@link #processor data processor} and\n\t * converted by the {@link #upcastDispatcher view-to-model converters}.\n\t * Initial data can be only set to a document whose {@link module:engine/model/document~Document#version} is equal 0.\n\t *\n\t * **Note** This method is {@link module:utils/observablemixin~Observable#decorate decorated} which is\n\t * used by e.g. collaborative editing plugin that syncs remote data on init.\n\t *\n\t * When data is passed as a string, it is initialized on the default `main` root:\n\t *\n\t * ```ts\n\t * dataController.init( '<p>Foo</p>' ); // Initializes data on the `main` root only, as no other is specified.\n\t * ```\n\t *\n\t * To initialize data on a different root or multiple roots at once, an object containing `rootName` - `data` pairs should be passed:\n\t *\n\t * ```ts\n\t * dataController.init( { main: '<p>Foo</p>', title: '<h1>Bar</h1>' } ); // Initializes data on both the `main` and `title` roots.\n\t * ```\n\t *\n\t * @fires init\n\t * @param data Input data as a string or an object containing the `rootName` - `data`\n\t * pairs to initialize data on multiple roots at once.\n\t * @returns Promise that is resolved after the data is set on the editor.\n\t */\n\tpublic init( data: string | Record<string, string> ): Promise<void> {\n\t\tif ( this.model.document.version ) {\n\t\t\t/**\n\t\t\t * Cannot set initial data to a non-empty {@link module:engine/model/document~Document}.\n\t\t\t * Initial data should be set once, during the {@link module:core/editor/editor~Editor} initialization,\n\t\t\t * when the {@link module:engine/model/document~Document#version} is equal 0.\n\t\t\t *\n\t\t\t * @error datacontroller-init-document-not-empty\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'datacontroller-init-document-not-empty', this );\n\t\t}\n\n\t\tlet initialData: Record<string, string> = {};\n\n\t\tif ( typeof data === 'string' ) {\n\t\t\tinitialData.main = data; // Default root is 'main'. To initiate data on a different root, object should be passed.\n\t\t} else {\n\t\t\tinitialData = data;\n\t\t}\n\n\t\tif ( !this._checkIfRootsExists( Object.keys( initialData ) ) ) {\n\t\t\t/**\n\t\t\t * Cannot init data on a non-existent root. This error is thrown when\n\t\t\t * {@link module:engine/controller/datacontroller~DataController#init DataController#init() method}\n\t\t\t * is called with non-existent root name. For example, if there is an editor instance with only `main` root,\n\t\t\t * calling {@link module:engine/controller/datacontroller~DataController#init} like:\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * data.init( { main: '<p>Foo</p>', root2: '<p>Bar</p>' } );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * will throw this error.\n\t\t\t *\n\t\t\t * @error datacontroller-init-non-existent-root\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'datacontroller-init-non-existent-root', this );\n\t\t}\n\n\t\tthis.model.enqueueChange( { isUndoable: false }, writer => {\n\t\t\tfor ( const rootName of Object.keys( initialData ) ) {\n\t\t\t\tconst modelRoot = this.model.document.getRoot( rootName )!;\n\n\t\t\t\twriter.insert( this.parse( initialData[ rootName ], modelRoot ), modelRoot, 0 );\n\t\t\t}\n\t\t} );\n\n\t\treturn Promise.resolve();\n\t}\n\n\t/**\n\t * Sets the input data parsed by the {@link #processor data processor} and\n\t * converted by the {@link #upcastDispatcher view-to-model converters}.\n\t * This method can be used any time to replace existing editor data with the new one without clearing the\n\t * {@link module:engine/model/document~Document#history document history}.\n\t *\n\t * This method also creates a batch with all the changes applied. If all you need is to parse data, use\n\t * the {@link #parse} method.\n\t *\n\t * When data is passed as a string it is set on the default `main` root:\n\t *\n\t * ```ts\n\t * dataController.set( '<p>Foo</p>' ); // Sets data on the `main` root, as no other is specified.\n\t * ```\n\t *\n\t * To set data on a different root or multiple roots at once, an object containing `rootName` - `data` pairs should be passed:\n\t *\n\t * ```ts\n\t * dataController.set( { main: '<p>Foo</p>', title: '<h1>Bar</h1>' } ); // Sets data on the `main` and `title` roots as specified.\n\t * ```\n\t *\n\t * To set the data with a preserved undo stack and add the change to the undo stack, set `{ isUndoable: true }` as a `batchType` option.\n\t *\n\t * ```ts\n\t * dataController.set( '<p>Foo</p>', { batchType: { isUndoable: true } } );\n\t * ```\n\t *\n\t * @fires set\n\t * @param data Input data as a string or an object containing the `rootName` - `data`\n\t * pairs to set data on multiple roots at once.\n\t * @param options Options for setting data.\n\t * @param options.batchType The batch type that will be used to create a batch for the changes applied by this method.\n\t * By default, the batch will be set as {@link module:engine/model/batch~Batch#isUndoable not undoable} and the undo stack will be\n\t * cleared after the new data is applied (all undo steps will be removed). If the batch type `isUndoable` flag is be set to `true`,\n\t * the undo stack will be preserved instead and not cleared when new data is applied.\n\t */\n\tpublic set( data: string | Record<string, string>, options: { batchType?: BatchType } = {} ): void {\n\t\tlet newData: Record<string, string> = {};\n\n\t\tif ( typeof data === 'string' ) {\n\t\t\tnewData.main = data; // The default root is 'main'. To set data on a different root, an object should be passed.\n\t\t} else {\n\t\t\tnewData = data;\n\t\t}\n\n\t\tif ( !this._checkIfRootsExists( Object.keys( newData ) ) ) {\n\t\t\t/**\n\t\t\t * Cannot set data on a non-existent root. This error is thrown when the\n\t\t\t * {@link module:engine/controller/datacontroller~DataController#set DataController#set() method}\n\t\t\t * is called with non-existent root name. For example, if there is an editor instance with only the default `main` root,\n\t\t\t * calling {@link module:engine/controller/datacontroller~DataController#set} like:\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * data.set( { main: '<p>Foo</p>', root2: '<p>Bar</p>' } );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * will throw this error.\n\t\t\t *\n\t\t\t * @error datacontroller-set-non-existent-root\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'datacontroller-set-non-existent-root', this );\n\t\t}\n\n\t\tthis.model.enqueueChange( options.batchType || {}, writer => {\n\t\t\twriter.setSelection( null );\n\t\t\twriter.removeSelectionAttribute( this.model.document.selection.getAttributeKeys() );\n\n\t\t\tfor ( const rootName of Object.keys( newData ) ) {\n\t\t\t\t// Save to model.\n\t\t\t\tconst modelRoot = this.model.document.getRoot( rootName )!;\n\n\t\t\t\twriter.remove( writer.createRangeIn( modelRoot ) );\n\t\t\t\twriter.insert( this.parse( newData[ rootName ], modelRoot ), modelRoot, 0 );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Returns the data parsed by the {@link #processor data processor} and then converted by upcast converters\n\t * attached to the {@link #upcastDispatcher}.\n\t *\n\t * @see #set\n\t * @param data Data to parse.\n\t * @param context Base context in which the view will be converted to the model.\n\t * See: {@link module:engine/conversion/upcastdispatcher~UpcastDispatcher#convert}.\n\t * @returns Parsed data.\n\t */\n\tpublic parse( data: string, context: SchemaContextDefinition = '$root' ): ModelDocumentFragment {\n\t\t// data -> view\n\t\tconst viewDocumentFragment = this.processor.toView( data );\n\n\t\t// view -> model\n\t\treturn this.toModel( viewDocumentFragment, context );\n\t}\n\n\t/**\n\t * Returns the result of the given {@link module:engine/view/element~Element view element} or\n\t * {@link module:engine/view/documentfragment~DocumentFragment view document fragment} converted by the\n\t * {@link #upcastDispatcher view-to-model converters}, wrapped by {@link module:engine/model/documentfragment~DocumentFragment}.\n\t *\n\t * When marker elements were converted during the conversion process, it will be set as a document fragment's\n\t * {@link module:engine/model/documentfragment~DocumentFragment#markers static markers map}.\n\t *\n\t * @fires toModel\n\t * @param viewElementOrFragment The element or document fragment whose content will be converted.\n\t * @param context Base context in which the view will be converted to the model.\n\t * See: {@link module:engine/conversion/upcastdispatcher~UpcastDispatcher#convert}.\n\t * @returns Output document fragment.\n\t */\n\tpublic toModel(\n\t\tviewElementOrFragment: ViewElement | ViewDocumentFragment,\n\t\tcontext: SchemaContextDefinition = '$root'\n\t): ModelDocumentFragment {\n\t\treturn this.model.change( writer => {\n\t\t\treturn this.upcastDispatcher.convert( viewElementOrFragment, writer, context );\n\t\t} );\n\t}\n\n\t/**\n\t * Adds the style processor normalization rules.\n\t *\n\t * You can implement your own rules as well as use one of the available processor rules:\n\t *\n\t * * background: {@link module:engine/view/styles/background~addBackgroundRules}\n\t * * border: {@link module:engine/view/styles/border~addBorderRules}\n\t * * margin: {@link module:engine/view/styles/margin~addMarginRules}\n\t * * padding: {@link module:engine/view/styles/padding~addPaddingRules}\n\t */\n\tpublic addStyleProcessorRules( callback: ( stylesProcessor: StylesProcessor ) => void ): void {\n\t\tcallback( this.stylesProcessor );\n\t}\n\n\t/**\n\t * Registers a {@link module:engine/view/matcher~MatcherPattern} on an {@link #htmlProcessor htmlProcessor}\n\t * and a {@link #processor processor} for view elements whose content should be treated as raw data\n\t * and not processed during the conversion from DOM to view elements.\n\t *\n\t * The raw data can be later accessed by the {@link module:engine/view/element~Element#getCustomProperty view element custom property}\n\t * `\"$rawContent\"`.\n\t *\n\t * @param pattern Pattern matching all view elements whose content should be treated as a raw data.\n\t */\n\tpublic registerRawContentMatcher( pattern: MatcherPattern ): void {\n\t\t// No need to register the pattern if both the `htmlProcessor` and `processor` are the same instances.\n\t\tif ( this.processor && this.processor !== this.htmlProcessor ) {\n\t\t\tthis.processor.registerRawContentMatcher( pattern );\n\t\t}\n\n\t\tthis.htmlProcessor.registerRawContentMatcher( pattern );\n\t}\n\n\t/**\n\t * Removes all event listeners set by the DataController.\n\t */\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Checks whether all provided root names are actually existing editor roots.\n\t *\n\t * @param rootNames Root names to check.\n\t * @returns Whether all provided root names are existing editor roots.\n\t */\n\tprivate _checkIfRootsExists( rootNames: Array<string> ): boolean {\n\t\tfor ( const rootName of rootNames ) {\n\t\t\tif ( !this.model.document.getRoot( rootName ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n}\n\n/**\n * Event fired once the data initialization has finished.\n *\n * @eventName ~DataController#ready\n */\nexport type DataControllerReadyEvent = {\n\tname: 'ready';\n\targs: [];\n};\n\n/**\n * An event fired after the {@link ~DataController#init `init()` method} was run. It can be {@link ~DataController#listenTo listened to} in\n * order to adjust or modify the initialization flow. However, if the `init` event is stopped or prevented,\n * the {@link ~DataController#event:ready `ready` event} should be fired manually.\n *\n * The `init` event is fired by the decorated {@link ~DataController#init} method.\n * See {@link module:utils/observablemixin~Observable#decorate} for more information and samples.\n *\n * @eventName ~DataController#init\n */\nexport type DataControllerInitEvent = {\n\tname: 'init';\n\targs: [ Parameters<DataController[ 'init' ]> ];\n\treturn: ReturnType<DataController[ 'init' ]>;\n};\n\n/**\n * An event fired after {@link ~DataController#set set() method} has been run.\n *\n * The `set` event is fired by the decorated {@link ~DataController#set} method.\n * See {@link module:utils/observablemixin~Observable#decorate} for more information and samples.\n *\n * @eventName ~DataController#set\n */\nexport type DataControllerSetEvent = {\n\tname: 'set';\n\targs: [ Parameters<DataController[ 'set' ]> ];\n\treturn: ReturnType<DataController[ 'set' ]>;\n};\n\n/**\n * Event fired after the {@link ~DataController#get get() method} has been run.\n *\n * The `get` event is fired by the decorated {@link ~DataController#get} method.\n * See {@link module:utils/observablemixin~Observable#decorate} for more information and samples.\n *\n * @eventName ~DataController#get\n */\nexport type DataControllerGetEvent = {\n\tname: 'get';\n\targs: [ Parameters<DataController[ 'get' ]> ];\n\treturn: ReturnType<DataController[ 'get' ]>;\n};\n\n/**\n * Event fired after the {@link ~DataController#toView toView() method} has been run.\n *\n * The `toView` event is fired by the decorated {@link ~DataController#toView} method.\n * See {@link module:utils/observablemixin~Observable#decorate} for more information and samples.\n *\n * @eventName ~DataController#toView\n */\nexport type DataControllerToViewEvent = {\n\tname: 'toView';\n\targs: [ Parameters<DataController[ 'toView' ]> ];\n\treturn: ReturnType<DataController[ 'toView' ]>;\n};\n\n/**\n * Event fired after the {@link ~DataController#toModel toModel() method} has been run.\n *\n * The `toModel` event is fired by the decorated {@link ~DataController#toModel} method.\n * See {@link module:utils/observablemixin~Observable#decorate} for more information and samples.\n *\n * @eventName ~DataController#toModel\n */\nexport type DataControllerToModelEvent = {\n\tname: 'toModel';\n\targs: [ Parameters<DataController[ 'toModel' ]> ];\n\treturn: ReturnType<DataController[ 'toModel' ]>;\n};\n\n/**\n * Helper function for downcast conversion.\n *\n * Takes a document element (element that is added to a model document) and checks which markers are inside it. If the marker is collapsed\n * at element boundary, it is considered as contained inside the element and marker range is returned. Otherwise, if the marker is\n * intersecting with the element, the intersection is returned.\n */\nfunction _getMarkersRelativeToElement( element: ModelElement ): Map<string, ModelRange> {\n\tconst result: Array<[ string, ModelRange ]> = [];\n\tconst doc = element.root.document;\n\n\tif ( !doc ) {\n\t\treturn new Map();\n\t}\n\n\tconst elementRange = ModelRange._createIn( element );\n\n\tfor ( const marker of doc.model.markers ) {\n\t\tconst markerRange = marker.getRange();\n\n\t\tconst isMarkerCollapsed = markerRange.isCollapsed;\n\t\tconst isMarkerAtElementBoundary = markerRange.start.isEqual( elementRange.start ) || markerRange.end.isEqual( elementRange.end );\n\n\t\tif ( isMarkerCollapsed && isMarkerAtElementBoundary ) {\n\t\t\tresult.push( [ marker.name, markerRange ] );\n\t\t} else {\n\t\t\tconst updatedMarkerRange = elementRange.getIntersection( markerRange );\n\n\t\t\tif ( updatedMarkerRange ) {\n\t\t\t\tresult.push( [ marker.name, updatedMarkerRange ] );\n\t\t\t}\n\t\t}\n\t}\n\n\t// Sort the markers in a stable fashion to ensure that the order in which they are\n\t// added to the model's marker collection does not affect how they are\n\t// downcast. One particular use case that we are targeting here, is one where\n\t// two markers are adjacent but not overlapping, such as an insertion/deletion\n\t// suggestion pair representing the replacement of a range of text. In this\n\t// case, putting the markers in DOM order causes the first marker's end to be\n\t// serialized right after the second marker's start, while putting the markers\n\t// in reverse DOM order causes it to be right before the second marker's\n\t// start. So, we sort these in a way that ensures non-intersecting ranges are in\n\t// reverse DOM order, and intersecting ranges are in something approximating\n\t// reverse DOM order (since reverse DOM order doesn't have a precise meaning\n\t// when working with intersecting ranges).\n\tresult.sort( ( [ n1, r1 ], [ n2, r2 ] ) => {\n\t\tif ( r1.end.compareWith( r2.start ) !== 'after' ) {\n\t\t\t// m1.end <= m2.start -- m1 is entirely <= m2\n\t\t\treturn 1;\n\t\t} else if ( r1.start.compareWith( r2.end ) !== 'before' ) {\n\t\t\t// m1.start >= m2.end -- m1 is entirely >= m2\n\t\t\treturn -1;\n\t\t} else {\n\t\t\t// they overlap, so use their start positions as the primary sort key and\n\t\t\t// end positions as the secondary sort key\n\t\t\tswitch ( r1.start.compareWith( r2.start ) ) {\n\t\t\t\tcase 'before':\n\t\t\t\t\treturn 1;\n\t\t\t\tcase 'after':\n\t\t\t\t\treturn -1;\n\t\t\t\tdefault:\n\t\t\t\t\tswitch ( r1.end.compareWith( r2.end ) ) {\n\t\t\t\t\t\tcase 'before':\n\t\t\t\t\t\t\treturn 1;\n\t\t\t\t\t\tcase 'after':\n\t\t\t\t\t\t\treturn -1;\n\t\t\t\t\t\tdefault:\n\t\t\t\t\t\t\treturn n2.localeCompare( n1 );\n\t\t\t\t\t}\n\t\t\t}\n\t\t}\n\t} );\n\n\treturn new Map( result );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/conversion/conversion\n */\n\nimport {\n\tCKEditorError,\n\ttoArray,\n\ttype ArrayOrItem,\n\ttype PriorityString\n} from '@ckeditor/ckeditor5-utils';\n\nimport UpcastHelpers from './upcasthelpers';\nimport DowncastHelpers, {\n\ttype AttributeCreatorFunction,\n\ttype AttributeDescriptor\n} from './downcasthelpers';\n\nimport type DowncastDispatcher from './downcastdispatcher';\nimport type UpcastDispatcher from './upcastdispatcher';\nimport type ElementDefinition from '../view/elementdefinition';\nimport type { MatcherPattern } from '../view/matcher';\n\n/**\n * A utility class that helps add converters to upcast and downcast dispatchers.\n *\n * We recommend reading the {@glink framework/deep-dive/conversion/intro editor conversion} guide first to\n * understand the core concepts of the conversion mechanisms.\n *\n * An instance of the conversion manager is available in the\n * {@link module:core/editor/editor~Editor#conversion `editor.conversion`} property\n * and by default has the following groups of dispatchers (i.e. directions of conversion):\n *\n * * `downcast` (editing and data downcasts)\n * * `editingDowncast`\n * * `dataDowncast`\n * * `upcast`\n *\n * # One-way converters\n *\n * To add a converter to a specific group, use the {@link module:engine/conversion/conversion~Conversion#for `for()`}\n * method:\n *\n * ```ts\n * // Add a converter to editing downcast and data downcast.\n * editor.conversion.for( 'downcast' ).elementToElement( config ) );\n *\n * // Add a converter to the data pipepline only:\n * editor.conversion.for( 'dataDowncast' ).elementToElement( dataConversionConfig ) );\n *\n * // And a slightly different one for the editing pipeline:\n * editor.conversion.for( 'editingDowncast' ).elementToElement( editingConversionConfig ) );\n * ```\n *\n * See {@link module:engine/conversion/conversion~Conversion#for `for()`} method documentation to learn more about\n * available conversion helpers and how to use your custom ones.\n *\n * # Two-way converters\n *\n * Besides using one-way converters via the `for()` method, you can also use other methods available in this\n * class to add two-way converters (upcast and downcast):\n *\n * * {@link module:engine/conversion/conversion~Conversion#elementToElement `elementToElement()`} &ndash;\n * Model element to view element and vice versa.\n * * {@link module:engine/conversion/conversion~Conversion#attributeToElement `attributeToElement()`} &ndash;\n * Model attribute to view element and vice versa.\n * * {@link module:engine/conversion/conversion~Conversion#attributeToAttribute `attributeToAttribute()`} &ndash;\n * Model attribute to view attribute and vice versa.\n */\nexport default class Conversion {\n\t/**\n\t * Maps dispatchers group name to ConversionHelpers instances.\n\t */\n\tprivate readonly _helpers = new Map<string, DowncastHelpers | UpcastHelpers>();\n\n\tprivate readonly _downcast: Array<DowncastDispatcher>;\n\tprivate readonly _upcast: Array<UpcastDispatcher>;\n\n\t/**\n\t * Creates a new conversion instance.\n\t */\n\tconstructor(\n\t\tdowncastDispatchers: ArrayOrItem<DowncastDispatcher>,\n\t\tupcastDispatchers: ArrayOrItem<UpcastDispatcher>\n\t) {\n\t\t// Define default 'downcast' & 'upcast' dispatchers groups. Those groups are always available as two-way converters needs them.\n\t\tthis._downcast = toArray( downcastDispatchers );\n\t\tthis._createConversionHelpers( { name: 'downcast', dispatchers: this._downcast, isDowncast: true } );\n\n\t\tthis._upcast = toArray( upcastDispatchers );\n\t\tthis._createConversionHelpers( { name: 'upcast', dispatchers: this._upcast, isDowncast: false } );\n\t}\n\n\tpublic addAlias(\n\t\talias: `${ string }Downcast`,\n\t\tdispatcher: DowncastDispatcher\n\t): void;\n\tpublic addAlias(\n\t\talias: `${ string }Upcast`,\n\t\tdispatcher: UpcastDispatcher\n\t): void;\n\tpublic addAlias(\n\t\talias: string,\n\t\tdispatcher: DowncastDispatcher | UpcastDispatcher\n\t): void;\n\n\t/**\n\t * Define an alias for registered dispatcher.\n\t *\n\t * ```ts\n\t * const conversion = new Conversion(\n\t * \t[ dataDowncastDispatcher, editingDowncastDispatcher ],\n\t * \tupcastDispatcher\n\t * );\n\t *\n\t * conversion.addAlias( 'dataDowncast', dataDowncastDispatcher );\n\t * ```\n\t *\n\t * @param alias An alias of a dispatcher.\n\t * @param dispatcher Dispatcher which should have an alias.\n\t */\n\tpublic addAlias(\n\t\talias: string,\n\t\tdispatcher: DowncastDispatcher | UpcastDispatcher\n\t): void {\n\t\tconst isDowncast = this._downcast.includes( dispatcher as any );\n\t\tconst isUpcast = this._upcast.includes( dispatcher as any );\n\n\t\tif ( !isUpcast && !isDowncast ) {\n\t\t\t/**\n\t\t\t * Trying to register an alias for a dispatcher that nas not been registered.\n\t\t\t *\n\t\t\t * @error conversion-add-alias-dispatcher-not-registered\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'conversion-add-alias-dispatcher-not-registered',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\tthis._createConversionHelpers( { name: alias, dispatchers: [ dispatcher ], isDowncast } );\n\t}\n\n\tpublic for( groupName: 'downcast' | 'dataDowncast' | 'editingDowncast' ): DowncastHelpers;\n\tpublic for( groupName: 'upcast' ): UpcastHelpers;\n\tpublic for<T extends string>( groupName: T ): ConversionType<T>;\n\n\t/**\n\t * Provides a chainable API to assign converters to a conversion dispatchers group.\n\t *\n\t * If the given group name has not been registered, the\n\t * {@link module:utils/ckeditorerror~CKEditorError `conversion-for-unknown-group` error} is thrown.\n\t *\n\t * You can use conversion helpers available directly in the `for()` chain or your custom ones via\n\t * the {@link module:engine/conversion/conversionhelpers~ConversionHelpers#add `add()`} method.\n\t *\n\t * # Using built-in conversion helpers\n\t *\n\t * The `for()` chain comes with a set of conversion helpers which you can use like this:\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'downcast' )\n\t * \t.elementToElement( config1 ) // Adds an element-to-element downcast converter.\n\t * \t.attributeToElement( config2 ); // Adds an attribute-to-element downcast converter.\n\t *\n\t * editor.conversion.for( 'upcast' )\n\t * \t.elementToAttribute( config3 ); // Adds an element-to-attribute upcast converter.\n\t * ```\n\t *\n\t * Refer to the documentation of built-in conversion helpers to learn about their configuration options.\n\t *\n\t * * downcast (model-to-view) conversion helpers:\n\t *\n\t *\t* {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToElement `elementToElement()`},\n\t *\t* {@link module:engine/conversion/downcasthelpers~DowncastHelpers#attributeToElement `attributeToElement()`},\n\t *\t* {@link module:engine/conversion/downcasthelpers~DowncastHelpers#attributeToAttribute `attributeToAttribute()`}.\n\t *\t* {@link module:engine/conversion/downcasthelpers~DowncastHelpers#markerToElement `markerToElement()`}.\n\t *\t* {@link module:engine/conversion/downcasthelpers~DowncastHelpers#markerToHighlight `markerToHighlight()`}.\n\t *\n\t * * upcast (view-to-model) conversion helpers:\n\t *\n\t *\t* {@link module:engine/conversion/upcasthelpers~UpcastHelpers#elementToElement `elementToElement()`},\n\t *\t* {@link module:engine/conversion/upcasthelpers~UpcastHelpers#elementToAttribute `elementToAttribute()`},\n\t *\t* {@link module:engine/conversion/upcasthelpers~UpcastHelpers#attributeToAttribute `attributeToAttribute()`}.\n\t *\t* {@link module:engine/conversion/upcasthelpers~UpcastHelpers#elementToMarker `elementToMarker()`}.\n\t *\n\t * # Using custom conversion helpers\n\t *\n\t * If you need to implement an atypical converter, you can do so by calling:\n\t *\n\t * ```ts\n\t * editor.conversion.for( direction ).add( customHelper );\n\t * ```\n\t *\n\t * The `.add()` method takes exactly one parameter, which is a function. This function should accept one parameter that\n\t * is a dispatcher instance. The function should add an actual converter to the passed dispatcher instance.\n\t *\n\t * Example:\n\t *\n\t * ```ts\n\t * editor.conversion.for( 'upcast' ).add( dispatcher => {\n\t * \tdispatcher.on( 'element:a', ( evt, data, conversionApi ) => {\n\t * \t\t// Do something with a view <a> element.\n\t * \t} );\n\t * } );\n\t * ```\n\t *\n\t * Refer to the documentation of {@link module:engine/conversion/upcastdispatcher~UpcastDispatcher}\n\t * and {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher} to learn how to write\n\t * custom converters.\n\t *\n\t * @param groupName The name of dispatchers group to add the converters to.\n\t */\n\tpublic for( groupName: string ): DowncastHelpers | UpcastHelpers {\n\t\tif ( !this._helpers.has( groupName ) ) {\n\t\t\t/**\n\t\t\t * Trying to add a converter to an unknown dispatchers group.\n\t\t\t *\n\t\t\t * @error conversion-for-unknown-group\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'conversion-for-unknown-group', this );\n\t\t}\n\n\t\treturn this._helpers.get( groupName )!;\n\t}\n\n\t/**\n\t * Sets up converters between the model and the view that convert a model element to a view element (and vice versa).\n\t * For example, the model `<paragraph>Foo</paragraph>` is turned into `<p>Foo</p>` in the view.\n\t *\n\t * ```ts\n\t * // A simple conversion from the `paragraph` model element to the `<p>` view element (and vice versa).\n\t * editor.conversion.elementToElement( { model: 'paragraph', view: 'p' } );\n\t *\n\t * // Override other converters by specifying a converter definition with a higher priority.\n\t * editor.conversion.elementToElement( { model: 'paragraph', view: 'div', converterPriority: 'high' } );\n\t *\n\t * // View specified as an object instead of a string.\n\t * editor.conversion.elementToElement( {\n\t * \tmodel: 'fancyParagraph',\n\t * \tview: {\n\t * \t\tname: 'p',\n\t * \t\tclasses: 'fancy'\n\t * \t}\n\t * } );\n\t *\n\t * // Use `upcastAlso` to define other view elements that should also be converted to a `paragraph` element.\n\t * editor.conversion.elementToElement( {\n\t * \tmodel: 'paragraph',\n\t * \tview: 'p',\n\t * \tupcastAlso: [\n\t * \t\t'div',\n\t * \t\t{\n\t * \t\t\t// Any element with the `display: block` style.\n\t * \t\t\tstyles: {\n\t * \t\t\t\tdisplay: 'block'\n\t * \t\t\t}\n\t * \t\t}\n\t * \t]\n\t * } );\n\t *\n\t * // `upcastAlso` set as callback enables a conversion of a wide range of different view elements.\n\t * editor.conversion.elementToElement( {\n\t * \tmodel: 'heading',\n\t * \tview: 'h2',\n\t * \t// Convert \"heading-like\" paragraphs to headings.\n\t * \tupcastAlso: viewElement => {\n\t * \t\tconst fontSize = viewElement.getStyle( 'font-size' );\n\t *\n\t * \t\tif ( !fontSize ) {\n\t * \t\t\treturn null;\n\t * \t\t}\n\t *\n\t * \t\tconst match = fontSize.match( /(\\d+)\\s*px/ );\n\t *\n\t * \t\tif ( !match ) {\n\t * \t\t\treturn null;\n\t * \t\t}\n\t *\n\t * \t\tconst size = Number( match[ 1 ] );\n\t *\n\t * \t\tif ( size > 26 ) {\n\t * \t\t\t// Returned value can be an object with the matched properties.\n\t * \t\t\t// These properties will be \"consumed\" during the conversion.\n\t * \t\t\t// See `engine.view.Matcher~MatcherPattern` and `engine.view.Matcher#match` for more details.\n\t *\n\t * \t\t\treturn { name: true, styles: [ 'font-size' ] };\n\t * \t\t}\n\t *\n\t * \t\treturn null;\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * `definition.model` is a `String` with a model element name to convert from or to.\n\t *\n\t * @param definition The converter definition.\n\t */\n\tpublic elementToElement( definition: {\n\t\tmodel: string;\n\t\tview: ElementDefinition;\n\t\tupcastAlso?: ArrayOrItem<ElementDefinition | MatcherPattern>;\n\t\tconverterPriority?: PriorityString;\n\t} ): void {\n\t\t// Set up downcast converter.\n\t\tthis.for( 'downcast' ).elementToElement( definition );\n\n\t\t// Set up upcast converter.\n\t\tfor ( const { model, view } of _getAllUpcastDefinitions( definition ) ) {\n\t\t\tthis.for( 'upcast' )\n\t\t\t\t.elementToElement( {\n\t\t\t\t\tmodel,\n\t\t\t\t\tview,\n\t\t\t\t\tconverterPriority: definition.converterPriority\n\t\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Sets up converters between the model and the view that convert a model attribute to a view element (and vice versa).\n\t * For example, a model text node with `\"Foo\"` as data and the `bold` attribute will be turned to `<strong>Foo</strong>` in the view.\n\t *\n\t * ```ts\n\t * // A simple conversion from the `bold=true` attribute to the `<strong>` view element (and vice versa).\n\t * editor.conversion.attributeToElement( { model: 'bold', view: 'strong' } );\n\t *\n\t * // Override other converters by specifying a converter definition with a higher priority.\n\t * editor.conversion.attributeToElement( { model: 'bold', view: 'b', converterPriority: 'high' } );\n\t *\n\t * // View specified as an object instead of a string.\n\t * editor.conversion.attributeToElement( {\n\t * \tmodel: 'bold',\n\t * \tview: {\n\t * \t\tname: 'span',\n\t * \t\tclasses: 'bold'\n\t * \t}\n\t * } );\n\t *\n\t * // Use `config.model.name` to define the conversion only from a given node type, `$text` in this case.\n\t * // The same attribute on different elements may then be handled by a different converter.\n\t * editor.conversion.attributeToElement( {\n\t * \tmodel: {\n\t * \t\tkey: 'textDecoration',\n\t * \t\tvalues: [ 'underline', 'lineThrough' ],\n\t * \t\tname: '$text'\n\t * \t},\n\t * \tview: {\n\t * \t\tunderline: {\n\t * \t\t\tname: 'span',\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'text-decoration': 'underline'\n\t * \t\t\t}\n\t * \t\t},\n\t * \t\tlineThrough: {\n\t * \t\t\tname: 'span',\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'text-decoration': 'line-through'\n\t * \t\t\t}\n\t * \t\t}\n\t * \t}\n\t * } );\n\t *\n\t * // Use `upcastAlso` to define other view elements that should also be converted to the `bold` attribute.\n\t * editor.conversion.attributeToElement( {\n\t * \tmodel: 'bold',\n\t * \tview: 'strong',\n\t * \tupcastAlso: [\n\t * \t\t'b',\n\t * \t\t{\n\t * \t\t\tname: 'span',\n\t * \t\t\tclasses: 'bold'\n\t * \t\t},\n\t * \t\t{\n\t * \t\t\tname: 'span',\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'font-weight': 'bold'\n\t * \t\t\t}\n\t * \t\t},\n\t * \t\tviewElement => {\n\t * \t\t\tconst fontWeight = viewElement.getStyle( 'font-weight' );\n\t *\n\t * \t\t\tif ( viewElement.is( 'element', 'span' ) && fontWeight && /\\d+/.test() && Number( fontWeight ) > 500 ) {\n\t * \t\t\t\t// Returned value can be an object with the matched properties.\n\t * \t\t\t\t// These properties will be \"consumed\" during the conversion.\n\t * \t\t\t\t// See `engine.view.Matcher~MatcherPattern` and `engine.view.Matcher#match` for more details.\n\t *\n\t * \t\t\t\treturn {\n\t * \t\t\t\t\tname: true,\n\t * \t\t\t\t\tstyles: [ 'font-weight' ]\n\t * \t\t\t\t};\n\t * \t\t\t}\n\t * \t\t}\n\t * \t]\n\t * } );\n\t *\n\t * // Conversion from and to a model attribute key whose value is an enum (`fontSize=big|small`).\n\t * // `upcastAlso` set as callback enables a conversion of a wide range of different view elements.\n\t * editor.conversion.attributeToElement( {\n\t * \tmodel: {\n\t * \t\tkey: 'fontSize',\n\t * \t\tvalues: [ 'big', 'small' ]\n\t * \t},\n\t * \tview: {\n\t * \t\tbig: {\n\t * \t\t\tname: 'span',\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'font-size': '1.2em'\n\t * \t\t\t}\n\t * \t\t},\n\t * \t\tsmall: {\n\t * \t\t\tname: 'span',\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'font-size': '0.8em'\n\t * \t\t\t}\n\t * \t\t}\n\t * \t},\n\t * \tupcastAlso: {\n\t * \t\tbig: viewElement => {\n\t * \t\t\tconst fontSize = viewElement.getStyle( 'font-size' );\n\t *\n\t * \t\t\tif ( !fontSize ) {\n\t * \t\t\t\treturn null;\n\t * \t\t\t}\n\t *\n\t * \t\t\tconst match = fontSize.match( /(\\d+)\\s*px/ );\n\t *\n\t * \t\t\tif ( !match ) {\n\t * \t\t\t\treturn null;\n\t * \t\t\t}\n\t *\n\t * \t\t\tconst size = Number( match[ 1 ] );\n\t *\n\t * \t\t\tif ( viewElement.is( 'element', 'span' ) && size > 10 ) {\n\t * \t\t\t\t// Returned value can be an object with the matched properties.\n\t * \t\t\t\t// These properties will be \"consumed\" during the conversion.\n\t * \t\t\t\t// See `engine.view.Matcher~MatcherPattern` and `engine.view.Matcher#match` for more details.\n\t *\n\t * \t\t\t\treturn { name: true, styles: [ 'font-size' ] };\n\t * \t\t\t}\n\t *\n\t * \t\t\treturn null;\n\t * \t\t},\n\t * \t\tsmall: viewElement => {\n\t * \t\t\tconst fontSize = viewElement.getStyle( 'font-size' );\n\t *\n\t * \t\t\tif ( !fontSize ) {\n\t * \t\t\t\treturn null;\n\t * \t\t\t}\n\t *\n\t * \t\t\tconst match = fontSize.match( /(\\d+)\\s*px/ );\n\t *\n\t * \t\t\tif ( !match ) {\n\t * \t\t\t\treturn null;\n\t * \t\t\t}\n\t *\n\t * \t\t\tconst size = Number( match[ 1 ] );\n\t *\n\t * \t\t\tif ( viewElement.is( 'element', 'span' ) && size < 10 ) {\n\t * \t\t\t\t// Returned value can be an object with the matched properties.\n\t * \t\t\t\t// These properties will be \"consumed\" during the conversion.\n\t * \t\t\t\t// See `engine.view.Matcher~MatcherPattern` and `engine.view.Matcher#match` for more details.\n\t *\n\t * \t\t\t\treturn { name: true, styles: [ 'font-size' ] };\n\t * \t\t\t}\n\t *\n\t * \t\t\treturn null;\n\t * \t\t}\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * The `definition.model` parameter specifies which model attribute should be converted from or to. It can be a `{ key, value }` object\n\t * describing the attribute key and value to convert or a `String` specifying just the attribute key (in such a case\n\t * `value` is set to `true`).\n\t *\n\t * @param definition The converter definition.\n\t */\n\tpublic attributeToElement<TValues extends string>(\n\t\tdefinition: {\n\t\t\tmodel: string | {\n\t\t\t\tkey: string;\n\t\t\t\tname?: string;\n\t\t\t};\n\t\t\tview: ElementDefinition;\n\t\t\tupcastAlso?: ArrayOrItem<MatcherPattern>;\n\t\t\tconverterPriority?: PriorityString;\n\t\t} | {\n\t\t\tmodel: {\n\t\t\t\tkey: string;\n\t\t\t\tname?: string;\n\t\t\t\tvalues: Array<TValues>;\n\t\t\t};\n\t\t\tview: Record<TValues, ElementDefinition>;\n\t\t\tupcastAlso?: Record<TValues, ArrayOrItem<MatcherPattern>>;\n\t\t\tconverterPriority?: PriorityString;\n\t\t}\n\t): void {\n\t\t// Set up downcast converter.\n\t\tthis.for( 'downcast' ).attributeToElement( definition );\n\n\t\t// Set up upcast converter.\n\t\tfor ( const { model, view } of _getAllUpcastDefinitions( definition ) ) {\n\t\t\tthis.for( 'upcast' )\n\t\t\t\t.elementToAttribute( {\n\t\t\t\t\tview,\n\t\t\t\t\tmodel,\n\t\t\t\t\tconverterPriority: definition.converterPriority\n\t\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Sets up converters between the model and the view that convert a model attribute to a view attribute (and vice versa). For example,\n\t * `<imageBlock src='foo.jpg'></imageBlock>` is converted to `<img src='foo.jpg'></img>` (the same attribute key and value).\n\t * This type of converters is intended to be used with {@link module:engine/model/element~Element model element} nodes.\n\t * To convert the text attributes,\n\t * the {@link module:engine/conversion/conversion~Conversion#attributeToElement `attributeToElement converter`}should be set up.\n\t *\n\t * ```ts\n\t * // A simple conversion from the `source` model attribute to the `src` view attribute (and vice versa).\n\t * editor.conversion.attributeToAttribute( { model: 'source', view: 'src' } );\n\t *\n\t * // Attribute values are strictly specified.\n\t * editor.conversion.attributeToAttribute( {\n\t * \tmodel: {\n\t * \t\tname: 'imageInline',\n\t * \t\tkey: 'aside',\n\t * \t\tvalues: [ 'aside' ]\n\t * \t},\n\t * \tview: {\n\t * \t\taside: {\n\t * \t\t\tname: 'img',\n\t * \t\t\tkey: 'class',\n\t * \t\t\tvalue: [ 'aside', 'half-size' ]\n\t * \t\t}\n\t * \t}\n\t * } );\n\t *\n\t * // Set the style attribute.\n\t * editor.conversion.attributeToAttribute( {\n\t * \tmodel: {\n\t * \t\tname: 'imageInline',\n\t * \t\tkey: 'aside',\n\t * \t\tvalues: [ 'aside' ]\n\t * \t},\n\t * \tview: {\n\t * \t\taside: {\n\t * \t\t\tname: 'img',\n\t * \t\t\tkey: 'style',\n\t * \t\t\tvalue: {\n\t * \t\t\t\tfloat: 'right',\n\t * \t\t\t\twidth: '50%',\n\t * \t\t\t\tmargin: '5px'\n\t * \t\t\t}\n\t * \t\t}\n\t * \t}\n\t * } );\n\t *\n\t * // Conversion from and to a model attribute key whose value is an enum (`align=right|center`).\n\t * // Use `upcastAlso` to define other view elements that should also be converted to the `align=right` attribute.\n\t * editor.conversion.attributeToAttribute( {\n\t * \tmodel: {\n\t * \t\tkey: 'align',\n\t * \t\tvalues: [ 'right', 'center' ]\n\t * \t},\n\t * \tview: {\n\t * \t\tright: {\n\t * \t\t\tkey: 'class',\n\t * \t\t\tvalue: 'align-right'\n\t * \t\t},\n\t * \t\tcenter: {\n\t * \t\t\tkey: 'class',\n\t * \t\t\tvalue: 'align-center'\n\t * \t\t}\n\t * \t},\n\t * \tupcastAlso: {\n\t * \t\tright: {\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'text-align': 'right'\n\t * \t\t\t}\n\t * \t\t},\n\t * \t\tcenter: {\n\t * \t\t\tstyles: {\n\t * \t\t\t\t'text-align': 'center'\n\t * \t\t\t}\n\t * \t\t}\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * The `definition.model` parameter specifies which model attribute should be converted from and to.\n\t * It can be a `{ key, [ values ], [ name ] }` object or a `String`, which will be treated like `{ key: definition.model }`.\n\t * The `key` property is the model attribute key to convert from and to.\n\t * The `values` are the possible model attribute values. If the `values` parameter is not set, the model attribute value\n\t * will be the same as the view attribute value.\n\t * If `name` is set, the conversion will be set up only for model elements with the given name.\n\t *\n\t * The `definition.view` parameter specifies which view attribute should be converted from and to.\n\t * It can be a `{ key, value, [ name ] }` object or a `String`, which will be treated like `{ key: definition.view }`.\n\t * The `key` property is the view attribute key to convert from and to.\n\t * The `value` is the view attribute value to convert from and to. If `definition.value` is not set, the view attribute value will be\n\t * the same as the model attribute value.\n\t * If `key` is `'class'`, `value` can be a `String` or an array of `String`s.\n\t * If `key` is `'style'`, `value` is an object with key-value pairs.\n\t * In other cases, `value` is a `String`.\n\t * If `name` is set, the conversion will be set up only for model elements with the given name.\n\t * If `definition.model.values` is set, `definition.view` is an object that assigns values from `definition.model.values`\n\t * to `{ key, value, [ name ] }` objects.\n\t *\n\t * `definition.upcastAlso` specifies which other matching view elements should also be upcast to the given model configuration.\n\t * If `definition.model.values` is set, `definition.upcastAlso` should be an object assigning values from `definition.model.values`\n\t * to {@link module:engine/view/matcher~MatcherPattern}s or arrays of {@link module:engine/view/matcher~MatcherPattern}s.\n\t *\n\t * **Note:** `definition.model` and `definition.view` form should be mirrored, so the same types of parameters should\n\t * be given in both parameters.\n\t *\n\t * @param definition The converter definition.\n\t * @param definition.model The model attribute to convert from and to.\n\t * @param definition.view The view attribute to convert from and to.\n\t * @param definition.upcastAlso Any view element matching `definition.upcastAlso` will also be converted to the given model attribute.\n\t * `definition.upcastAlso` is used only if `config.model.values` is specified.\n\t */\n\tpublic attributeToAttribute<TValues extends string>(\n\t\tdefinition: {\n\t\t\tmodel: string | {\n\t\t\t\tkey: string;\n\t\t\t\tname?: string;\n\t\t\t};\n\t\t\tview: string | ( AttributeDescriptor & { name?: string } );\n\t\t\tupcastAlso?: ArrayOrItem<string | ( AttributeDescriptor & { name?: string } ) | AttributeCreatorFunction>;\n\t\t\tconverterPriority?: PriorityString;\n\t\t} | {\n\t\t\tmodel: {\n\t\t\t\tkey: string;\n\t\t\t\tname?: string;\n\t\t\t\tvalues: Array<TValues>;\n\t\t\t};\n\t\t\tview: Record<TValues, ( AttributeDescriptor & { name?: string } )>;\n\t\t\tupcastAlso?: Record<TValues, ( AttributeDescriptor & { name?: string } ) | AttributeCreatorFunction>;\n\t\t\tconverterPriority?: PriorityString;\n\t\t} ): void {\n\t\t// Set up downcast converter.\n\t\tthis.for( 'downcast' ).attributeToAttribute( definition );\n\n\t\t// Set up upcast converter.\n\t\tfor ( const { model, view } of _getAllUpcastDefinitions( definition ) ) {\n\t\t\tthis.for( 'upcast' )\n\t\t\t\t.attributeToAttribute( {\n\t\t\t\t\tview,\n\t\t\t\t\tmodel\n\t\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Creates and caches conversion helpers for given dispatchers group.\n\t *\n\t * @param options.name Group name.\n\t */\n\tprivate _createConversionHelpers(\n\t\t{ name, dispatchers, isDowncast }: {\n\t\t\tname: string;\n\t\t\tdispatchers: Array<DowncastDispatcher | UpcastDispatcher>;\n\t\t\tisDowncast: boolean;\n\t\t}\n\t): void {\n\t\tif ( this._helpers.has( name ) ) {\n\t\t\t/**\n\t\t\t * Trying to register a group name that has already been registered.\n\t\t\t *\n\t\t\t * @error conversion-group-exists\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'conversion-group-exists', this );\n\t\t}\n\n\t\tconst helpers = isDowncast ?\n\t\t\tnew DowncastHelpers( dispatchers as Array<DowncastDispatcher> ) :\n\t\t\tnew UpcastHelpers( dispatchers as Array<UpcastDispatcher> );\n\n\t\tthis._helpers.set( name, helpers );\n\t}\n}\n\n/**\n * Helper function that creates a joint array out of an item passed in `definition.view` and items passed in\n * `definition.upcastAlso`.\n */\nfunction* _getAllUpcastDefinitions( definition: any ): IterableIterator<{ model: any; view: any }> {\n\tif ( definition.model.values ) {\n\t\tfor ( const value of definition.model.values ) {\n\t\t\tconst model = { key: definition.model.key, value };\n\t\t\tconst view = definition.view[ value ];\n\t\t\tconst upcastAlso = definition.upcastAlso ? definition.upcastAlso[ value ] : undefined;\n\n\t\t\tyield* _getUpcastDefinition( model, view, upcastAlso );\n\t\t}\n\t} else {\n\t\tyield* _getUpcastDefinition( definition.model, definition.view, definition.upcastAlso );\n\t}\n}\n\nfunction* _getUpcastDefinition( model: unknown, view: unknown, upcastAlso?: unknown ): any {\n\tyield { model, view };\n\n\tif ( upcastAlso ) {\n\t\tfor ( const upcastAlsoItem of toArray( upcastAlso ) ) {\n\t\t\tyield { model, view: upcastAlsoItem };\n\t\t}\n\t}\n}\n\ntype ConversionType<T extends string> = T extends `${ string }Downcast`\n\t? DowncastHelpers\n\t: T extends `${ string }Upcast`\n\t\t? UpcastHelpers\n\t\t: DowncastHelpers | UpcastHelpers;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* eslint-disable @typescript-eslint/no-unused-vars */\n\nimport type Batch from '../batch';\nimport type Document from '../document';\nimport type { Selectable } from '../selection';\n\n/**\n * @module engine/model/operation/operation\n */\n\n/**\n * Abstract base operation class.\n */\nexport default abstract class Operation {\n\t/**\n\t * {@link module:engine/model/document~Document#version} on which operation can be applied. If you try to\n\t * {@link module:engine/model/model~Model#applyOperation apply} operation with different base version than the\n\t * {@link module:engine/model/document~Document#version document version} the\n\t * {@link module:utils/ckeditorerror~CKEditorError model-document-applyOperation-wrong-version} error is thrown.\n\t */\n\tpublic baseVersion: number | null;\n\n\t/**\n\t * Defines whether operation is executed on attached or detached {@link module:engine/model/item~Item items}.\n\t */\n\tpublic readonly isDocumentOperation: boolean;\n\n\t/**\n\t * {@link module:engine/model/batch~Batch Batch} to which the operation is added or `null` if the operation is not\n\t * added to any batch yet.\n\t */\n\tpublic batch: Batch | null;\n\n\t/**\n\t * Operation type.\n\t */\n\tpublic readonly abstract type: string;\n\n\t/**\n\t * Base operation constructor.\n\t *\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor( baseVersion: number | null ) {\n\t\tthis.baseVersion = baseVersion;\n\t\tthis.isDocumentOperation = this.baseVersion !== null;\n\t\tthis.batch = null;\n\t}\n\n\t/**\n\t * A selectable that will be affected by the operation after it is executed.\n\t *\n\t * The exact returned parameter differs between operation types.\n\t */\n\tpublic abstract get affectedSelectable(): Selectable;\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t *\n\t * @returns Clone of this operation.\n\t */\n\tpublic abstract clone(): Operation;\n\n\t/**\n\t * Creates and returns a reverse operation. Reverse operation when executed right after\n\t * the original operation will bring back tree model state to the point before the original\n\t * operation execution. In other words, it reverses changes done by the original operation.\n\t *\n\t * Keep in mind that tree model state may change since executing the original operation,\n\t * so reverse operation will be \"outdated\". In that case you will need to transform it by\n\t * all operations that were executed after the original operation.\n\t *\n\t * @returns Reversed operation.\n\t */\n\tpublic abstract getReversed(): Operation;\n\n\t/**\n\t * Executes the operation - modifications described by the operation properties will be applied to the model tree.\n\t *\n\t * @internal\n\t */\n\tpublic abstract _execute(): void;\n\n\t/**\n\t * Checks whether the operation's parameters are correct and the operation can be correctly executed. Throws\n\t * an error if operation is not valid.\n\t *\n\t * @internal\n\t */\n\tpublic _validate(): void {\n\t}\n\n\t/**\n\t * Custom toJSON method to solve child-parent circular dependencies.\n\t *\n\t * @returns Clone of this object with the operation property replaced with string.\n\t */\n\tpublic toJSON(): unknown {\n\t\t// This method creates only a shallow copy, all nested objects should be defined separately.\n\t\t// See https://github.com/ckeditor/ckeditor5-engine/issues/1477.\n\t\tconst json: any = Object.assign( {}, this );\n\n\t\tjson.__className = ( this.constructor as any ).className;\n\n\t\t// Remove reference to the parent `Batch` to avoid circular dependencies.\n\t\tdelete json.batch;\n\n\t\t// Only document operations are shared with other clients so it is not necessary to keep this information.\n\t\tdelete json.isDocumentOperation;\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * Name of the operation class used for serialization.\n\t */\n\tpublic static get className(): string {\n\t\treturn 'Operation';\n\t}\n\n\t/**\n\t * Creates `Operation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param doc Document on which this operation will be applied.\n\t */\n\tpublic static fromJSON( json: any, document: Document ): Operation {\n\t\treturn new ( this as any )( json.baseVersion );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // log() {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( this.toString() );\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/utils\n */\n\nimport Node from '../node';\nimport Range from '../range';\nimport Text from '../text';\nimport TextProxy from '../textproxy';\n\nimport type DocumentFragment from '../documentfragment';\nimport type Element from '../element';\nimport type Item from '../item';\nimport type NodeList from '../nodelist';\nimport type Position from '../position';\n\nimport { CKEditorError, isIterable } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Inserts given nodes at given position.\n *\n * @internal\n * @param position Position at which nodes should be inserted.\n * @param normalizedNodes Nodes to insert.\n * @returns Range spanning over inserted elements.\n */\nexport function _insert( position: Position, nodes: NodeSet ): Range {\n\tconst normalizedNodes = _normalizeNodes( nodes );\n\n\t// We have to count offset before inserting nodes because they can get merged and we would get wrong offsets.\n\tconst offset = normalizedNodes.reduce( ( sum, node ) => sum + node.offsetSize, 0 );\n\tconst parent = position.parent;\n\n\t// Insertion might be in a text node, we should split it if that's the case.\n\t_splitNodeAtPosition( position );\n\tconst index = position.index;\n\n\t// Insert nodes at given index. After splitting we have a proper index and insertion is between nodes,\n\t// using basic `Element` API.\n\tparent._insertChild( index, normalizedNodes );\n\n\t// Merge text nodes, if possible. Merging is needed only at points where inserted nodes \"touch\" \"old\" nodes.\n\t_mergeNodesAtIndex( parent, index + normalizedNodes.length );\n\t_mergeNodesAtIndex( parent, index );\n\n\treturn new Range( position, position.getShiftedBy( offset ) );\n}\n\n/**\n * Removed nodes in given range. Only {@link module:engine/model/range~Range#isFlat flat} ranges are accepted.\n *\n * @internal\n * @param range Range containing nodes to remove.\n */\nexport function _remove( this: any, range: Range ): Array<Node> {\n\tif ( !range.isFlat ) {\n\t\t/**\n\t\t * Trying to remove a range which starts and ends in different element.\n\t\t *\n\t\t * @error operation-utils-remove-range-not-flat\n\t\t */\n\t\tthrow new CKEditorError(\n\t\t\t'operation-utils-remove-range-not-flat',\n\t\t\tthis\n\t\t);\n\t}\n\n\tconst parent = range.start.parent;\n\n\t// Range may be inside text nodes, we have to split them if that's the case.\n\t_splitNodeAtPosition( range.start );\n\t_splitNodeAtPosition( range.end );\n\n\t// Remove the text nodes using basic `Element` API.\n\tconst removed = parent._removeChildren( range.start.index, range.end.index - range.start.index );\n\n\t// Merge text nodes, if possible. After some nodes were removed, node before and after removed range will be\n\t// touching at the position equal to the removed range beginning. We check merging possibility there.\n\t_mergeNodesAtIndex( parent, range.start.index );\n\n\treturn removed;\n}\n\n/**\n * Moves nodes in given range to given target position. Only {@link module:engine/model/range~Range#isFlat flat} ranges are accepted.\n *\n * @internal\n * @param sourceRange Range containing nodes to move.\n * @param targetPosition Position to which nodes should be moved.\n * @returns Range containing moved nodes.\n */\nexport function _move( this: any, sourceRange: Range, targetPosition: Position ): Range {\n\tif ( !sourceRange.isFlat ) {\n\t\t/**\n\t\t * Trying to move a range which starts and ends in different element.\n\t\t *\n\t\t * @error operation-utils-move-range-not-flat\n\t\t */\n\t\tthrow new CKEditorError(\n\t\t\t'operation-utils-move-range-not-flat',\n\t\t\tthis\n\t\t);\n\t}\n\n\tconst nodes = _remove( sourceRange );\n\n\t// We have to fix `targetPosition` because model changed after nodes from `sourceRange` got removed and\n\t// that change might have an impact on `targetPosition`.\n\ttargetPosition = targetPosition._getTransformedByDeletion( sourceRange.start, sourceRange.end.offset - sourceRange.start.offset )!;\n\n\treturn _insert( targetPosition, nodes );\n}\n\n/**\n * Sets given attribute on nodes in given range. The attributes are only set on top-level nodes of the range, not on its children.\n *\n * @internal\n * @param range Range containing nodes that should have the attribute set. Must be a flat range.\n * @param key Key of attribute to set.\n * @param value Attribute value.\n */\nexport function _setAttribute( range: Range, key: string, value: unknown ): void {\n\t// Range might start or end in text nodes, so we have to split them.\n\t_splitNodeAtPosition( range.start );\n\t_splitNodeAtPosition( range.end );\n\n\t// Iterate over all items in the range.\n\tfor ( const item of range.getItems( { shallow: true } ) ) {\n\t\t// Iterator will return `TextProxy` instances but we know that those text proxies will\n\t\t// always represent full text nodes (this is guaranteed thanks to splitting we did before).\n\t\t// So, we can operate on those text proxies' text nodes.\n\t\tconst node = item.is( '$textProxy' ) ? item.textNode : item;\n\n\t\tif ( value !== null ) {\n\t\t\tnode._setAttribute( key, value );\n\t\t} else {\n\t\t\tnode._removeAttribute( key );\n\t\t}\n\n\t\t// After attributes changing it may happen that some text nodes can be merged. Try to merge with previous node.\n\t\t_mergeNodesAtIndex( node.parent!, node.index! );\n\t}\n\n\t// Try to merge last changed node with it's previous sibling (not covered by the loop above).\n\t_mergeNodesAtIndex( range.end.parent, range.end.index );\n}\n\n/**\n * Normalizes given object or an array of objects to an array of {@link module:engine/model/node~Node nodes}. See\n * {@link ~NodeSet NodeSet} for details on how normalization is performed.\n *\n * @internal\n * @param nodes Objects to normalize.\n * @returns Normalized nodes.\n */\nexport function _normalizeNodes( nodes: NodeSet ): Array<Node> {\n\tconst normalized: Array<Node> = [];\n\n\tfunction convert( nodes: NodeSet ) {\n\t\tif ( typeof nodes == 'string' ) {\n\t\t\tnormalized.push( new Text( nodes ) );\n\t\t} else if ( nodes instanceof TextProxy ) {\n\t\t\tnormalized.push( new Text( nodes.data, nodes.getAttributes() ) );\n\t\t} else if ( nodes instanceof Node ) {\n\t\t\tnormalized.push( nodes );\n\t\t} else if ( isIterable( nodes ) ) {\n\t\t\tfor ( const node of nodes ) {\n\t\t\t\tconvert( node );\n\t\t\t}\n\t\t}\n\t\t// Skip unrecognized type.\n\t}\n\n\tconvert( nodes );\n\n\t// Merge text nodes.\n\tfor ( let i = 1; i < normalized.length; i++ ) {\n\t\tconst node = normalized[ i ];\n\t\tconst prev = normalized[ i - 1 ];\n\n\t\tif ( node instanceof Text && prev instanceof Text && _haveSameAttributes( node, prev ) ) {\n\t\t\t// Doing this instead changing `prev.data` because `data` is readonly.\n\t\t\tnormalized.splice( i - 1, 2, new Text( prev.data + node.data, prev.getAttributes() ) );\n\t\t\ti--;\n\t\t}\n\t}\n\n\treturn normalized;\n}\n\n/**\n * Checks if nodes before and after given index in given element are {@link module:engine/model/text~Text text nodes} and\n * merges them into one node if they have same attributes.\n *\n * Merging is done by removing two text nodes and inserting a new text node containing data from both merged text nodes.\n *\n * @param element Parent element of nodes to merge.\n * @param index Index between nodes to merge.\n */\nfunction _mergeNodesAtIndex( element: Element | DocumentFragment, index: number ) {\n\tconst nodeBefore = element.getChild( index - 1 );\n\tconst nodeAfter = element.getChild( index );\n\n\t// Check if both of those nodes are text objects with same attributes.\n\tif ( nodeBefore && nodeAfter && nodeBefore.is( '$text' ) && nodeAfter.is( '$text' ) && _haveSameAttributes( nodeBefore, nodeAfter ) ) {\n\t\t// Append text of text node after index to the before one.\n\t\tconst mergedNode = new Text( nodeBefore.data + nodeAfter.data, nodeBefore.getAttributes() );\n\n\t\t// Remove separate text nodes.\n\t\telement._removeChildren( index - 1, 2 );\n\n\t\t// Insert merged text node.\n\t\telement._insertChild( index - 1, mergedNode );\n\t}\n}\n\n/**\n * Checks if given position is in a text node, and if so, splits the text node in two text nodes, each of them\n * containing a part of original text node.\n *\n * @param position Position at which node should be split.\n */\nfunction _splitNodeAtPosition( position: Position ): void {\n\tconst textNode = position.textNode;\n\tconst element = position.parent;\n\n\tif ( textNode ) {\n\t\tconst offsetDiff = position.offset - textNode.startOffset!;\n\t\tconst index = textNode.index!;\n\n\t\telement._removeChildren( index, 1 );\n\n\t\tconst firstPart = new Text( textNode.data.substr( 0, offsetDiff ), textNode.getAttributes() );\n\t\tconst secondPart = new Text( textNode.data.substr( offsetDiff ), textNode.getAttributes() );\n\n\t\telement._insertChild( index, [ firstPart, secondPart ] );\n\t}\n}\n\n/**\n * Checks whether two given nodes have same attributes.\n *\n * @param nodeA Node to check.\n * @param nodeB Node to check.\n * @returns `true` if nodes have same attributes, `false` otherwise.\n */\nfunction _haveSameAttributes( nodeA: Node, nodeB: Node ): boolean | undefined {\n\tconst iteratorA = nodeA.getAttributes();\n\tconst iteratorB = nodeB.getAttributes();\n\n\tfor ( const attr of iteratorA ) {\n\t\tif ( attr[ 1 ] !== nodeB.getAttribute( attr[ 0 ] ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\titeratorB.next();\n\t}\n\n\treturn iteratorB.next().done;\n}\n\n/**\n * Value that can be normalized to an array of {@link module:engine/model/node~Node nodes}.\n *\n * Non-arrays are normalized as follows:\n * * {@link module:engine/model/node~Node Node} is left as is,\n * * {@link module:engine/model/textproxy~TextProxy TextProxy} and `string` are normalized to {@link module:engine/model/text~Text Text},\n * * {@link module:engine/model/nodelist~NodeList NodeList} is normalized to an array containing all nodes that are in that node list,\n * * {@link module:engine/model/documentfragment~DocumentFragment DocumentFragment} is normalized to an array containing all of it's\n * * children.\n *\n * Arrays are processed item by item like non-array values and flattened to one array. Normalization always results in\n * a flat array of {@link module:engine/model/node~Node nodes}. Consecutive text nodes (or items normalized to text nodes) will be\n * merged if they have same attributes.\n */\nexport type NodeSet = Item | string | NodeList | DocumentFragment | Iterable<Item | string | NodeList | DocumentFragment>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/moveoperation\n */\n\nimport Operation from './operation';\nimport Position from '../position';\nimport Range from '../range';\nimport { _move } from './utils';\nimport type { Selectable } from '../selection';\n\nimport { CKEditorError, compareArrays } from '@ckeditor/ckeditor5-utils';\n\nimport type Document from '../document';\n\n// @if CK_DEBUG_ENGINE // const ModelRange = require( '../range' ).default;\n\n/**\n * Operation to move a range of {@link module:engine/model/item~Item model items}\n * to given {@link module:engine/model/position~Position target position}.\n */\nexport default class MoveOperation extends Operation {\n\t/**\n\t * Position before the first {@link module:engine/model/item~Item model item} to move.\n\t */\n\tpublic sourcePosition: Position;\n\n\t/**\n\t * Offset size of moved range.\n\t */\n\tpublic howMany: number;\n\n\t/**\n\t * Position at which moved nodes will be inserted.\n\t */\n\tpublic targetPosition: Position;\n\n\t/**\n\t * Creates a move operation.\n\t *\n\t * @param sourcePosition Position before the first {@link module:engine/model/item~Item model item} to move.\n\t * @param howMany Offset size of moved range. Moved range will start from `sourcePosition` and end at\n\t * `sourcePosition` with offset shifted by `howMany`.\n\t * @param targetPosition Position at which moved nodes will be inserted.\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor( sourcePosition: Position, howMany: number, targetPosition: Position, baseVersion: number | null ) {\n\t\tsuper( baseVersion );\n\n\t\tthis.sourcePosition = sourcePosition.clone();\n\t\t// `'toNext'` because `sourcePosition` is a bit like a start of the moved range.\n\t\tthis.sourcePosition.stickiness = 'toNext';\n\n\t\tthis.howMany = howMany;\n\n\t\tthis.targetPosition = targetPosition.clone();\n\t\tthis.targetPosition.stickiness = 'toNone';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'move' | 'remove' | 'reinsert' {\n\t\tif ( this.targetPosition.root.rootName == '$graveyard' ) {\n\t\t\treturn 'remove';\n\t\t} else if ( this.sourcePosition.root.rootName == '$graveyard' ) {\n\t\t\treturn 'reinsert';\n\t\t}\n\n\t\treturn 'move';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\treturn [\n\t\t\tRange._createFromPositionAndShift( this.sourcePosition, this.howMany ),\n\t\t\tRange._createFromPositionAndShift( this.targetPosition, 0 )\n\t\t];\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t */\n\tpublic clone(): MoveOperation {\n\t\treturn new MoveOperation( this.sourcePosition, this.howMany, this.targetPosition, this.baseVersion );\n\t}\n\n\t/**\n\t * Returns the start position of the moved range after it got moved. This may be different than\n\t * {@link module:engine/model/operation/moveoperation~MoveOperation#targetPosition} in some cases, i.e. when a range is moved\n\t * inside the same parent but {@link module:engine/model/operation/moveoperation~MoveOperation#targetPosition targetPosition}\n\t * is after {@link module:engine/model/operation/moveoperation~MoveOperation#sourcePosition sourcePosition}.\n\t *\n\t * ```\n\t * vv vv\n\t * abcdefg ===> adefbcg\n\t * ^ ^\n\t * targetPos movedRangeStart\n\t * offset 6 offset 4\n\t *```\n\t */\n\tpublic getMovedRangeStart(): Position {\n\t\treturn this.targetPosition._getTransformedByDeletion( this.sourcePosition, this.howMany )!;\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\tconst newTargetPosition = this.sourcePosition._getTransformedByInsertion( this.targetPosition, this.howMany );\n\n\t\treturn new MoveOperation( this.getMovedRangeStart(), this.howMany, newTargetPosition, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _validate(): void {\n\t\tconst sourceElement = this.sourcePosition.parent;\n\t\tconst targetElement = this.targetPosition.parent;\n\t\tconst sourceOffset = this.sourcePosition.offset;\n\t\tconst targetOffset = this.targetPosition.offset;\n\n\t\t// Validate whether move operation has correct parameters.\n\t\t// Validation is pretty complex but move operation is one of the core ways to manipulate the document state.\n\t\t// We expect that many errors might be connected with one of scenarios described below.\n\t\tif ( sourceOffset + this.howMany > sourceElement.maxOffset ) {\n\t\t\t/**\n\t\t\t * The nodes which should be moved do not exist.\n\t\t\t *\n\t\t\t * @error move-operation-nodes-do-not-exist\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'move-operation-nodes-do-not-exist', this\n\t\t\t);\n\t\t} else if ( sourceElement === targetElement && sourceOffset < targetOffset && targetOffset < sourceOffset + this.howMany ) {\n\t\t\t/**\n\t\t\t * Trying to move a range of nodes into the middle of that range.\n\t\t\t *\n\t\t\t * @error move-operation-range-into-itself\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'move-operation-range-into-itself', this\n\t\t\t);\n\t\t} else if ( this.sourcePosition.root == this.targetPosition.root ) {\n\t\t\tif ( compareArrays( this.sourcePosition.getParentPath(), this.targetPosition.getParentPath() ) == 'prefix' ) {\n\t\t\t\tconst i = this.sourcePosition.path.length - 1;\n\n\t\t\t\tif ( this.targetPosition.path[ i ] >= sourceOffset && this.targetPosition.path[ i ] < sourceOffset + this.howMany ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * Trying to move a range of nodes into one of nodes from that range.\n\t\t\t\t\t *\n\t\t\t\t\t * @error move-operation-node-into-itself\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t\t'move-operation-node-into-itself', this\n\t\t\t\t\t);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic _execute(): void {\n\t\t_move( Range._createFromPositionAndShift( this.sourcePosition, this.howMany ), this.targetPosition );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.sourcePosition = this.sourcePosition.toJSON();\n\t\tjson.targetPosition = this.targetPosition.toJSON();\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'MoveOperation';\n\t}\n\n\t/**\n\t * Creates `MoveOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): MoveOperation {\n\t\tconst sourcePosition = Position.fromJSON( json.sourcePosition, document );\n\t\tconst targetPosition = Position.fromJSON( json.targetPosition, document );\n\n\t\treturn new this( sourcePosition, json.howMany, targetPosition, json.baseVersion );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \tconst range = ModelRange._createFromPositionAndShift( this.sourcePosition, this.howMany );\n\n\t// @if CK_DEBUG_ENGINE // \treturn `MoveOperation( ${ this.baseVersion } ): ${ range } -> ${ this.targetPosition }`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/insertoperation\n */\n\nimport Operation from './operation';\nimport Position from '../position';\nimport NodeList from '../nodelist';\nimport MoveOperation from './moveoperation';\nimport { _insert, _normalizeNodes, type NodeSet } from './utils';\nimport Text from '../text';\nimport Element from '../element';\nimport type { Selectable } from '../selection';\n\nimport type Document from '../document';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Operation to insert one or more nodes at given position in the model.\n */\nexport default class InsertOperation extends Operation {\n\t/**\n\t * Position of insertion.\n\t *\n\t * @readonly\n\t */\n\tpublic position: Position;\n\n\t/**\n\t * List of nodes to insert.\n\t *\n\t * @readonly\n\t */\n\tpublic nodes: NodeList;\n\n\t/**\n\t * Flag deciding how the operation should be transformed. If set to `true`, nodes might get additional attributes\n\t * during operational transformation. This happens when the operation insertion position is inside of a range\n\t * where attributes have changed.\n\t */\n\tpublic shouldReceiveAttributes: boolean;\n\n\t/**\n\t * Creates an insert operation.\n\t *\n\t * @param position Position of insertion.\n\t * @param nodes The list of nodes to be inserted.\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor( position: Position, nodes: NodeSet, baseVersion: number | null ) {\n\t\tsuper( baseVersion );\n\n\t\tthis.position = position.clone();\n\t\tthis.position.stickiness = 'toNone';\n\t\tthis.nodes = new NodeList( _normalizeNodes( nodes ) );\n\t\tthis.shouldReceiveAttributes = false;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'insert' {\n\t\treturn 'insert';\n\t}\n\n\t/**\n\t * Total offset size of inserted nodes.\n\t */\n\tpublic get howMany(): number {\n\t\treturn this.nodes.maxOffset;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\treturn this.position.clone();\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t */\n\tpublic clone(): InsertOperation {\n\t\tconst nodes = new NodeList( [ ...this.nodes ].map( node => node._clone( true ) ) );\n\t\tconst insert = new InsertOperation( this.position, nodes, this.baseVersion );\n\n\t\tinsert.shouldReceiveAttributes = this.shouldReceiveAttributes;\n\n\t\treturn insert;\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\tconst graveyard = this.position.root.document!.graveyard;\n\t\tconst gyPosition = new Position( graveyard, [ 0 ] );\n\n\t\treturn new MoveOperation( this.position, this.nodes.maxOffset, gyPosition, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _validate(): void {\n\t\tconst targetElement = this.position.parent;\n\n\t\tif ( !targetElement || targetElement.maxOffset < this.position.offset ) {\n\t\t\t/**\n\t\t\t * Insertion position is invalid.\n\t\t\t *\n\t\t\t * @error insert-operation-position-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'insert-operation-position-invalid',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic _execute(): void {\n\t\t// What happens here is that we want original nodes be passed to writer because we want original nodes\n\t\t// to be inserted to the model. But in InsertOperation, we want to keep those nodes as they were added\n\t\t// to the operation, not modified. For example, text nodes can get merged or cropped while Elements can\n\t\t// get children. It is important that InsertOperation has the copy of original nodes in intact state.\n\t\tconst originalNodes = this.nodes;\n\t\tthis.nodes = new NodeList( [ ...originalNodes ].map( node => node._clone( true ) ) );\n\n\t\t_insert( this.position, originalNodes );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.position = this.position.toJSON();\n\t\tjson.nodes = this.nodes.toJSON();\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'InsertOperation';\n\t}\n\n\t/**\n\t * Creates `InsertOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): InsertOperation {\n\t\tconst children = [];\n\n\t\tfor ( const child of json.nodes ) {\n\t\t\tif ( child.name ) {\n\t\t\t\t// If child has name property, it is an Element.\n\t\t\t\tchildren.push( Element.fromJSON( child ) );\n\t\t\t} else {\n\t\t\t\t// Otherwise, it is a Text node.\n\t\t\t\tchildren.push( Text.fromJSON( child ) );\n\t\t\t}\n\t\t}\n\n\t\tconst insert = new InsertOperation( Position.fromJSON( json.position, document ), children, json.baseVersion );\n\t\tinsert.shouldReceiveAttributes = json.shouldReceiveAttributes;\n\n\t\treturn insert;\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \tconst nodeString = this.nodes.length > 1 ? `[ ${ this.nodes.length } ]` : this.nodes.getNode( 0 );\n\n\t// @if CK_DEBUG_ENGINE // \treturn `InsertOperation( ${ this.baseVersion } ): ${ nodeString } -> ${ this.position }`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/splitoperation\n */\n\nimport Operation from './operation';\nimport MergeOperation from './mergeoperation';\nimport Position from '../position';\nimport Range from '../range';\nimport { _insert, _move } from './utils';\n\nimport type Document from '../document';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\nimport type { Selectable } from '../selection';\n\n/**\n * Operation to split {@link module:engine/model/element~Element an element} at given\n * {@link module:engine/model/operation/splitoperation~SplitOperation#splitPosition split position} into two elements,\n * both containing a part of the element's original content.\n */\nexport default class SplitOperation extends Operation {\n\t/**\n\t * Position at which an element should be split.\n\t */\n\tpublic splitPosition: Position;\n\n\t/**\n\t * Total offset size of elements that are in the split element after `position`.\n\t */\n\tpublic howMany: number;\n\n\t/**\n\t * Position at which the clone of split element (or element from graveyard) will be inserted.\n\t */\n\tpublic insertionPosition: Position;\n\n\t/**\n\t * Position in the graveyard root before the element which should be used as a parent of the nodes after `position`.\n\t * If it is not set, a copy of the the `position` parent will be used.\n\t *\n\t * The default behavior is to clone the split element. Element from graveyard is used during undo.\n\t */\n\tpublic graveyardPosition: Position | null;\n\n\t/**\n\t * Creates a split operation.\n\t *\n\t * @param splitPosition Position at which an element should be split.\n\t * @param howMany Total offset size of elements that are in the split element after `position`.\n\t * @param insertionPosition Position at which the clone of split element (or element from graveyard) will be inserted.\n\t * @param graveyardPosition Position in the graveyard root before the element which\n\t * should be used as a parent of the nodes after `position`. If it is not set, a copy of the the `position` parent will be used.\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor(\n\t\tsplitPosition: Position,\n\t\thowMany: number,\n\t\tinsertionPosition: Position,\n\t\tgraveyardPosition: Position | null,\n\t\tbaseVersion: number | null\n\t) {\n\t\tsuper( baseVersion );\n\n\t\tthis.splitPosition = splitPosition.clone();\n\t\t// Keep position sticking to the next node. This way any new content added at the place where the element is split\n\t\t// will be left in the original element.\n\t\tthis.splitPosition.stickiness = 'toNext';\n\n\t\tthis.howMany = howMany;\n\t\tthis.insertionPosition = insertionPosition;\n\t\tthis.graveyardPosition = graveyardPosition ? graveyardPosition.clone() : null;\n\n\t\tif ( this.graveyardPosition ) {\n\t\t\tthis.graveyardPosition.stickiness = 'toNext';\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'split' {\n\t\treturn 'split';\n\t}\n\n\t/**\n\t * Position inside the new clone of a split element.\n\t *\n\t * This is a position where nodes that are after the split position will be moved to.\n\t */\n\tpublic get moveTargetPosition(): Position {\n\t\tconst path = this.insertionPosition.path.slice();\n\t\tpath.push( 0 );\n\n\t\treturn new Position( this.insertionPosition.root, path );\n\t}\n\n\t/**\n\t * Artificial range that contains all the nodes from the split element that will be moved to the new element.\n\t * The range starts at {@link #splitPosition} and ends in the same parent, at `POSITIVE_INFINITY` offset.\n\t */\n\tpublic get movedRange(): Range {\n\t\tconst end = this.splitPosition.getShiftedBy( Number.POSITIVE_INFINITY );\n\n\t\treturn new Range( this.splitPosition, end );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\t// These could be positions but `Selectable` type only supports `Iterable<Range>`.\n\t\tconst ranges = [\n\t\t\tRange._createFromPositionAndShift( this.splitPosition, 0 ),\n\t\t\tRange._createFromPositionAndShift( this.insertionPosition, 0 )\n\t\t];\n\n\t\tif ( this.graveyardPosition ) {\n\t\t\tranges.push( Range._createFromPositionAndShift( this.graveyardPosition, 0 ) );\n\t\t}\n\n\t\treturn ranges;\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t *\n\t * @returns Clone of this operation.\n\t */\n\tpublic clone(): SplitOperation {\n\t\treturn new SplitOperation( this.splitPosition, this.howMany, this.insertionPosition, this.graveyardPosition, this.baseVersion );\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\tconst graveyard = this.splitPosition.root.document!.graveyard;\n\t\tconst graveyardPosition = new Position( graveyard, [ 0 ] );\n\n\t\treturn new MergeOperation( this.moveTargetPosition, this.howMany, this.splitPosition, graveyardPosition, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _validate(): void {\n\t\tconst element = this.splitPosition.parent;\n\t\tconst offset = this.splitPosition.offset;\n\n\t\t// Validate whether split operation has correct parameters.\n\t\tif ( !element || element.maxOffset < offset ) {\n\t\t\t/**\n\t\t\t * Split position is invalid.\n\t\t\t *\n\t\t\t * @error split-operation-position-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'split-operation-position-invalid', this );\n\t\t} else if ( !element.parent ) {\n\t\t\t/**\n\t\t\t * Cannot split root element.\n\t\t\t *\n\t\t\t * @error split-operation-split-in-root\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'split-operation-split-in-root', this );\n\t\t} else if ( this.howMany != element.maxOffset - this.splitPosition.offset ) {\n\t\t\t/**\n\t\t\t * Split operation specifies wrong number of nodes to move.\n\t\t\t *\n\t\t\t * @error split-operation-how-many-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'split-operation-how-many-invalid', this );\n\t\t} else if ( this.graveyardPosition && !this.graveyardPosition.nodeAfter ) {\n\t\t\t/**\n\t\t\t * Graveyard position invalid.\n\t\t\t *\n\t\t\t * @error split-operation-graveyard-position-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'split-operation-graveyard-position-invalid', this );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic _execute(): void {\n\t\tconst splitElement = this.splitPosition.parent;\n\n\t\tif ( this.graveyardPosition ) {\n\t\t\t_move( Range._createFromPositionAndShift( this.graveyardPosition, 1 ), this.insertionPosition );\n\t\t} else {\n\t\t\tconst newElement = ( splitElement as any )._clone();\n\n\t\t\t_insert( this.insertionPosition, newElement );\n\t\t}\n\n\t\tconst sourceRange = new Range(\n\t\t\tPosition._createAt( splitElement, this.splitPosition.offset ),\n\t\t\tPosition._createAt( splitElement, splitElement.maxOffset )\n\t\t);\n\n\t\t_move( sourceRange, this.moveTargetPosition );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.splitPosition = this.splitPosition.toJSON();\n\t\tjson.insertionPosition = this.insertionPosition.toJSON();\n\n\t\tif ( this.graveyardPosition ) {\n\t\t\tjson.graveyardPosition = this.graveyardPosition.toJSON();\n\t\t}\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'SplitOperation';\n\t}\n\n\t/**\n\t * Helper function that returns a default insertion position basing on given `splitPosition`. The default insertion\n\t * position is after the split element.\n\t */\n\tpublic static getInsertionPosition( splitPosition: Position ): Position {\n\t\tconst path = splitPosition.path.slice( 0, -1 );\n\t\tpath[ path.length - 1 ]++;\n\n\t\treturn new Position( splitPosition.root, path, 'toPrevious' );\n\t}\n\n\t/**\n\t * Creates `SplitOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): SplitOperation {\n\t\tconst splitPosition = Position.fromJSON( json.splitPosition, document );\n\t\tconst insertionPosition = Position.fromJSON( json.insertionPosition, document );\n\t\tconst graveyardPosition = json.graveyardPosition ? Position.fromJSON( json.graveyardPosition, document ) : null;\n\n\t\treturn new this( splitPosition, json.howMany, insertionPosition, graveyardPosition, json.baseVersion );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `SplitOperation( ${ this.baseVersion } ): ${ this.splitPosition } ` +\n\t// @if CK_DEBUG_ENGINE // \t`( ${ this.howMany } ) -> ${ this.insertionPosition }` +\n\t// @if CK_DEBUG_ENGINE // \t`${ this.graveyardPosition ? ' with ' + this.graveyardPosition : '' }`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/mergeoperation\n */\n\nimport Operation from './operation';\nimport SplitOperation from './splitoperation';\nimport Position from '../position';\nimport Range from '../range';\nimport { _move } from './utils';\n\nimport type Document from '../document';\nimport type Element from '../element';\nimport type { Selectable } from '../selection';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Operation to merge two {@link module:engine/model/element~Element elements}.\n *\n * The merged element is the parent of {@link ~MergeOperation#sourcePosition} and it is merged into the parent of\n * {@link ~MergeOperation#targetPosition}. All nodes from the merged element are moved to {@link ~MergeOperation#targetPosition}.\n *\n * The merged element is moved to the graveyard at {@link ~MergeOperation#graveyardPosition}.\n */\nexport default class MergeOperation extends Operation {\n\t/**\n\t * Position inside the merged element. All nodes from that element after that position will be moved to {@link #targetPosition}.\n\t */\n\tpublic sourcePosition: Position;\n\n\t/**\n\t * Summary offset size of nodes which will be moved from the merged element to the new parent.\n\t */\n\tpublic howMany: number;\n\n\t/**\n\t * Position which the nodes from the merged elements will be moved to.\n\t */\n\tpublic targetPosition: Position;\n\n\t/**\n\t * Position in graveyard to which the merged element will be moved.\n\t */\n\tpublic graveyardPosition: Position;\n\n\t/**\n\t * Creates a merge operation.\n\t *\n\t * @param sourcePosition Position inside the merged element. All nodes from that\n\t * element after that position will be moved to {@link #targetPosition}.\n\t * @param howMany Summary offset size of nodes which will be moved from the merged element to the new parent.\n\t * @param targetPosition Position which the nodes from the merged elements will be moved to.\n\t * @param graveyardPosition Position in graveyard to which the merged element will be moved.\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor(\n\t\tsourcePosition: Position,\n\t\thowMany: number,\n\t\ttargetPosition: Position,\n\t\tgraveyardPosition: Position,\n\t\tbaseVersion: number | null\n\t) {\n\t\tsuper( baseVersion );\n\n\t\tthis.sourcePosition = sourcePosition.clone();\n\t\t// This is, and should always remain, the first position in its parent.\n\t\tthis.sourcePosition.stickiness = 'toPrevious';\n\n\t\tthis.howMany = howMany;\n\t\tthis.targetPosition = targetPosition.clone();\n\n\t\t// Except of a rare scenario in `MergeOperation` x `MergeOperation` transformation,\n\t\t// this is, and should always remain, the last position in its parent.\n\t\tthis.targetPosition.stickiness = 'toNext';\n\t\tthis.graveyardPosition = graveyardPosition.clone();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'merge' {\n\t\treturn 'merge';\n\t}\n\n\t/**\n\t * Position before the merged element (which will be deleted).\n\t */\n\tpublic get deletionPosition(): Position {\n\t\treturn new Position( this.sourcePosition.root, this.sourcePosition.path.slice( 0, -1 ) );\n\t}\n\n\t/**\n\t * Artificial range that contains all the nodes from the merged element that will be moved to {@link ~MergeOperation#sourcePosition}.\n\t * The range starts at {@link ~MergeOperation#sourcePosition} and ends in the same parent, at `POSITIVE_INFINITY` offset.\n\t */\n\tpublic get movedRange(): Range {\n\t\tconst end = this.sourcePosition.getShiftedBy( Number.POSITIVE_INFINITY );\n\n\t\treturn new Range( this.sourcePosition, end );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\tconst mergedElement = this.sourcePosition.parent as Element;\n\n\t\treturn [\n\t\t\tRange._createOn( mergedElement ),\n\n\t\t\t// These could be positions but `Selectable` type only supports `Iterable<Range>`.\n\t\t\tRange._createFromPositionAndShift( this.targetPosition, 0 ),\n\t\t\tRange._createFromPositionAndShift( this.graveyardPosition, 0 )\n\t\t];\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t */\n\tpublic clone(): MergeOperation {\n\t\treturn new MergeOperation( this.sourcePosition, this.howMany, this.targetPosition, this.graveyardPosition, this.baseVersion );\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\t// Positions in this method are transformed by this merge operation because the split operation bases on\n\t\t// the context after this merge operation happened (because split operation reverses it).\n\t\t// So we need to acknowledge that the merge operation happened and those positions changed a little.\n\t\tconst targetPosition = this.targetPosition._getTransformedByMergeOperation( this );\n\n\t\tconst path = this.sourcePosition.path.slice( 0, -1 );\n\t\tconst insertionPosition = new Position( this.sourcePosition.root, path )._getTransformedByMergeOperation( this );\n\n\t\treturn new SplitOperation( targetPosition, this.howMany, insertionPosition, this.graveyardPosition, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _validate(): void {\n\t\tconst sourceElement = this.sourcePosition.parent;\n\t\tconst targetElement = this.targetPosition.parent;\n\n\t\t// Validate whether merge operation has correct parameters.\n\t\tif ( !sourceElement.parent ) {\n\t\t\t/**\n\t\t\t * Merge source position is invalid. The element to be merged must have a parent node.\n\t\t\t *\n\t\t\t * @error merge-operation-source-position-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'merge-operation-source-position-invalid', this );\n\t\t} else if ( !targetElement.parent ) {\n\t\t\t/**\n\t\t\t * Merge target position is invalid. The element to be merged must have a parent node.\n\t\t\t *\n\t\t\t * @error merge-operation-target-position-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'merge-operation-target-position-invalid', this );\n\t\t} else if ( this.howMany != sourceElement.maxOffset ) {\n\t\t\t/**\n\t\t\t * Merge operation specifies wrong number of nodes to move.\n\t\t\t *\n\t\t\t * @error merge-operation-how-many-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'merge-operation-how-many-invalid', this );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic _execute(): void {\n\t\tconst mergedElement = this.sourcePosition.parent as Element;\n\t\tconst sourceRange = Range._createIn( mergedElement );\n\n\t\t_move( sourceRange, this.targetPosition );\n\t\t_move( Range._createOn( mergedElement ), this.graveyardPosition );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.sourcePosition = json.sourcePosition.toJSON();\n\t\tjson.targetPosition = json.targetPosition.toJSON();\n\t\tjson.graveyardPosition = json.graveyardPosition.toJSON();\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'MergeOperation';\n\t}\n\n\t/**\n\t * Creates `MergeOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): MergeOperation {\n\t\tconst sourcePosition = Position.fromJSON( json.sourcePosition, document );\n\t\tconst targetPosition = Position.fromJSON( json.targetPosition, document );\n\t\tconst graveyardPosition = Position.fromJSON( json.graveyardPosition, document );\n\n\t\treturn new this( sourcePosition, json.howMany, targetPosition, graveyardPosition, json.baseVersion );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `MergeOperation( ${ this.baseVersion } ): ` +\n\t// @if CK_DEBUG_ENGINE // \t\t`${ this.sourcePosition } -> ${ this.targetPosition }` +\n\t// @if CK_DEBUG_ENGINE // \t\t` ( ${ this.howMany } ), ${ this.graveyardPosition }`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/markeroperation\n */\n\nimport Operation from './operation';\nimport Range from '../range';\n\nimport type Document from '../document';\nimport type MarkerCollection from '../markercollection';\nimport type { Selectable } from '../selection';\n\nexport default class MarkerOperation extends Operation {\n\t/**\n\t * Marker name.\n\t *\n\t * @readonly\n\t */\n\tpublic name: string;\n\n\t/**\n\t * Marker range before the change.\n\t *\n\t * @readonly\n\t */\n\tpublic oldRange: Range | null;\n\n\t/**\n\t * Marker range after the change.\n\t *\n\t * @readonly\n\t */\n\tpublic newRange: Range | null;\n\n\t/**\n\t * Specifies whether the marker operation affects the data produced by the data pipeline\n\t * (is persisted in the editor's data).\n\t *\n\t * @readonly\n\t */\n\tpublic affectsData: boolean;\n\n\t/**\n\t * Marker collection on which change should be executed.\n\t */\n\tprivate readonly _markers: MarkerCollection;\n\n\t/**\n\t * @param name Marker name.\n\t * @param oldRange Marker range before the change.\n\t * @param newRange Marker range after the change.\n\t * @param markers Marker collection on which change should be executed.\n\t * @param affectsData Specifies whether the marker operation affects the data produced by the data pipeline\n\t * (is persisted in the editor's data).\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor(\n\t\tname: string,\n\t\toldRange: Range | null,\n\t\tnewRange: Range | null,\n\t\tmarkers: MarkerCollection,\n\t\taffectsData: boolean,\n\t\tbaseVersion: number | null\n\t) {\n\t\tsuper( baseVersion );\n\n\t\tthis.name = name;\n\t\tthis.oldRange = oldRange ? oldRange.clone() : null;\n\t\tthis.newRange = newRange ? newRange.clone() : null;\n\t\tthis.affectsData = affectsData;\n\n\t\tthis._markers = markers;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'marker' {\n\t\treturn 'marker';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\tconst ranges = [];\n\n\t\tif ( this.oldRange ) {\n\t\t\tranges.push( this.oldRange.clone() );\n\t\t}\n\n\t\tif ( this.newRange ) {\n\t\t\tif ( this.oldRange ) {\n\t\t\t\tranges.push( ...this.newRange.getDifference( this.oldRange ) );\n\t\t\t} else {\n\t\t\t\tranges.push( this.newRange.clone() );\n\t\t\t}\n\t\t}\n\n\t\treturn ranges;\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t */\n\tpublic clone(): MarkerOperation {\n\t\treturn new MarkerOperation( this.name, this.oldRange, this.newRange, this._markers, this.affectsData, this.baseVersion );\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\treturn new MarkerOperation( this.name, this.newRange, this.oldRange, this._markers, this.affectsData, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic _execute(): void {\n\t\tif ( this.newRange ) {\n\t\t\tthis._markers._set( this.name, this.newRange, true, this.affectsData );\n\t\t} else {\n\t\t\tthis._markers._remove( this.name );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tif ( this.oldRange ) {\n\t\t\tjson.oldRange = this.oldRange.toJSON();\n\t\t}\n\n\t\tif ( this.newRange ) {\n\t\t\tjson.newRange = this.newRange.toJSON();\n\t\t}\n\n\t\tdelete json._markers;\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'MarkerOperation';\n\t}\n\n\t/**\n\t * Creates `MarkerOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): MarkerOperation {\n\t\treturn new MarkerOperation(\n\t\t\tjson.name,\n\t\t\tjson.oldRange ? Range.fromJSON( json.oldRange, document ) : null,\n\t\t\tjson.newRange ? Range.fromJSON( json.newRange, document ) : null,\n\t\t\tdocument.model.markers,\n\t\t\tjson.affectsData,\n\t\t\tjson.baseVersion\n\t\t);\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `MarkerOperation( ${ this.baseVersion } ): ` +\n\t// @if CK_DEBUG_ENGINE //\t\t`\"${ this.name }\": ${ this.oldRange } -> ${ this.newRange }`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","import baseIsEqual from './_baseIsEqual.js';\n\n/**\n * Performs a deep comparison between two values to determine if they are\n * equivalent.\n *\n * **Note:** This method supports comparing arrays, array buffers, booleans,\n * date objects, error objects, maps, numbers, `Object` objects, regexes,\n * sets, strings, symbols, and typed arrays. `Object` objects are compared\n * by their own, not inherited, enumerable properties. Functions and DOM\n * nodes are compared by strict equality, i.e. `===`.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Lang\n * @param {*} value The value to compare.\n * @param {*} other The other value to compare.\n * @returns {boolean} Returns `true` if the values are equivalent, else `false`.\n * @example\n *\n * var object = { 'a': 1 };\n * var other = { 'a': 1 };\n *\n * _.isEqual(object, other);\n * // => true\n *\n * object === other;\n * // => false\n */\nfunction isEqual(value, other) {\n return baseIsEqual(value, other);\n}\n\nexport default isEqual;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/attributeoperation\n */\n\nimport Operation from './operation';\nimport { _setAttribute } from './utils';\nimport Range from '../range';\n\nimport type Document from '../document';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\nimport { isEqual } from 'lodash-es';\nimport type { Selectable } from '../selection';\n\n/**\n * Operation to change nodes' attribute.\n *\n * Using this class you can add, remove or change value of the attribute.\n */\nexport default class AttributeOperation extends Operation {\n\t/**\n\t * Range on which operation should be applied.\n\t *\n\t * @readonly\n\t */\n\tpublic range: Range;\n\n\t/**\n\t * Key of an attribute to change or remove.\n\t *\n\t * @readonly\n\t */\n\tpublic key: string;\n\n\t/**\n\t * Old value of the attribute with given key or `null`, if attribute was not set before.\n\t *\n\t * @readonly\n\t */\n\tpublic oldValue: unknown;\n\n\t/**\n\t * New value of the attribute with given key or `null`, if operation should remove attribute.\n\t *\n\t * @readonly\n\t */\n\tpublic newValue: unknown;\n\n\t/**\n\t * Creates an operation that changes, removes or adds attributes.\n\t *\n\t * If only `newValue` is set, attribute will be added on a node. Note that all nodes in operation's range must not\n\t * have an attribute with the same key as the added attribute.\n\t *\n\t * If only `oldValue` is set, then attribute with given key will be removed. Note that all nodes in operation's range\n\t * must have an attribute with that key added.\n\t *\n\t * If both `newValue` and `oldValue` are set, then the operation will change the attribute value. Note that all nodes in\n\t * operation's ranges must already have an attribute with given key and `oldValue` as value\n\t *\n\t * @param range Range on which the operation should be applied. Must be a flat range.\n\t * @param key Key of an attribute to change or remove.\n\t * @param oldValue Old value of the attribute with given key or `null`, if attribute was not set before.\n\t * @param newValue New value of the attribute with given key or `null`, if operation should remove attribute.\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor( range: Range, key: string, oldValue: unknown, newValue: unknown, baseVersion: number | null ) {\n\t\tsuper( baseVersion );\n\n\t\tthis.range = range.clone();\n\t\tthis.key = key;\n\t\tthis.oldValue = oldValue === undefined ? null : oldValue;\n\t\tthis.newValue = newValue === undefined ? null : newValue;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'addAttribute' | 'removeAttribute' | 'changeAttribute' {\n\t\tif ( this.oldValue === null ) {\n\t\t\treturn 'addAttribute';\n\t\t} else if ( this.newValue === null ) {\n\t\t\treturn 'removeAttribute';\n\t\t} else {\n\t\t\treturn 'changeAttribute';\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\treturn this.range.clone();\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t */\n\tpublic clone(): AttributeOperation {\n\t\treturn new AttributeOperation( this.range, this.key, this.oldValue, this.newValue, this.baseVersion );\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\treturn new AttributeOperation( this.range, this.key, this.newValue, this.oldValue, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.range = this.range.toJSON();\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _validate(): void {\n\t\tif ( !this.range.isFlat ) {\n\t\t\t/**\n\t\t\t * The range to change is not flat.\n\t\t\t *\n\t\t\t * @error attribute-operation-range-not-flat\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'attribute-operation-range-not-flat', this );\n\t\t}\n\n\t\tfor ( const item of this.range.getItems( { shallow: true } ) ) {\n\t\t\tif ( this.oldValue !== null && !isEqual( item.getAttribute( this.key ), this.oldValue ) ) {\n\t\t\t\t/**\n\t\t\t\t * Changed node has different attribute value than operation's old attribute value.\n\t\t\t\t *\n\t\t\t\t * @error attribute-operation-wrong-old-value\n\t\t\t\t * @param item\n\t\t\t\t * @param key\n\t\t\t\t * @param value\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'attribute-operation-wrong-old-value',\n\t\t\t\t\tthis,\n\t\t\t\t\t{ item, key: this.key, value: this.oldValue }\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tif ( this.oldValue === null && this.newValue !== null && item.hasAttribute( this.key ) ) {\n\t\t\t\t/**\n\t\t\t\t * The attribute with given key already exists for the given node.\n\t\t\t\t *\n\t\t\t\t * @error attribute-operation-attribute-exists\n\t\t\t\t * @param node\n\t\t\t\t * @param key\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'attribute-operation-attribute-exists',\n\t\t\t\t\tthis,\n\t\t\t\t\t{ node: item, key: this.key }\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic _execute(): void {\n\t\t// If value to set is same as old value, don't do anything.\n\t\tif ( !isEqual( this.oldValue, this.newValue ) ) {\n\t\t\t// Execution.\n\t\t\t_setAttribute( this.range, this.key, this.newValue );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'AttributeOperation';\n\t}\n\n\t/**\n\t * Creates `AttributeOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): AttributeOperation {\n\t\treturn new AttributeOperation( Range.fromJSON( json.range, document ), json.key, json.oldValue, json.newValue, json.baseVersion );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `AttributeOperation( ${ this.baseVersion } ): ` +\n\t// @if CK_DEBUG_ENGINE // \t\t`\"${ this.key }\": ${ JSON.stringify( this.oldValue ) }` +\n\t// @if CK_DEBUG_ENGINE // \t\t` -> ${ JSON.stringify( this.newValue ) }, ${ this.range }`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/nooperation\n */\n\nimport Operation from './operation';\nimport type { Selectable } from '../selection';\n\n/**\n * Operation which is doing nothing (\"empty operation\", \"do-nothing operation\", \"noop\"). This is an operation,\n * which when executed does not change the tree model. It still has some parameters defined for transformation purposes.\n *\n * In most cases this operation is a result of transforming operations. When transformation returns\n * {@link module:engine/model/operation/nooperation~NoOperation} it means that changes done by the transformed operation\n * have already been applied.\n */\nexport default class NoOperation extends Operation {\n\tpublic get type(): 'noop' {\n\t\treturn 'noop';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\treturn null;\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t */\n\tpublic clone(): NoOperation {\n\t\treturn new NoOperation( this.baseVersion );\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\treturn new NoOperation( this.baseVersion! + 1 );\n\t}\n\n\t/** @internal */\n\tpublic _execute(): void {\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'NoOperation';\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `NoOperation( ${ this.baseVersion } )`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/renameoperation\n */\n\nimport Operation from './operation';\nimport Element from '../element';\nimport Position from '../position';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type Document from '../document';\nimport type { Selectable } from '../selection';\n\n/**\n * Operation to change element's name.\n *\n * Using this class you can change element's name.\n */\nexport default class RenameOperation extends Operation {\n\t/**\n\t * Position before an element to change.\n\t */\n\tpublic position: Position;\n\n\t/**\n\t * Current name of the element.\n\t */\n\tpublic oldName: string;\n\n\t/**\n\t * New name for the element.\n\t */\n\tpublic newName: string;\n\n\t/**\n\t * Creates an operation that changes element's name.\n\t *\n\t * @param position Position before an element to change.\n\t * @param oldName Current name of the element.\n\t * @param newName New name for the element.\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor( position: Position, oldName: string, newName: string, baseVersion: number | null ) {\n\t\tsuper( baseVersion );\n\n\t\tthis.position = position;\n\t\t// This position sticks to the next node because it is a position before the node that we want to change.\n\t\tthis.position.stickiness = 'toNext';\n\n\t\tthis.oldName = oldName;\n\t\tthis.newName = newName;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'rename' {\n\t\treturn 'rename';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\treturn this.position.nodeAfter;\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t *\n\t * @returns Clone of this operation.\n\t */\n\tpublic clone(): RenameOperation {\n\t\treturn new RenameOperation( this.position.clone(), this.oldName, this.newName, this.baseVersion );\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\treturn new RenameOperation( this.position.clone(), this.newName, this.oldName, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _validate(): void {\n\t\tconst element = this.position.nodeAfter;\n\n\t\tif ( !( element instanceof Element ) ) {\n\t\t\t/**\n\t\t\t * Given position is invalid or node after it is not instance of Element.\n\t\t\t *\n\t\t\t * @error rename-operation-wrong-position\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'rename-operation-wrong-position',\n\t\t\t\tthis\n\t\t\t);\n\t\t} else if ( element.name !== this.oldName ) {\n\t\t\t/**\n\t\t\t * Element to change has different name than operation's old name.\n\t\t\t *\n\t\t\t * @error rename-operation-wrong-name\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'rename-operation-wrong-name',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic _execute(): void {\n\t\tconst element = this.position.nodeAfter;\n\n\t\t( element as any ).name = this.newName;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.position = this.position.toJSON();\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'RenameOperation';\n\t}\n\n\t/**\n\t * Creates `RenameOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): RenameOperation {\n\t\treturn new RenameOperation( Position.fromJSON( json.position, document ), json.oldName, json.newName, json.baseVersion );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `RenameOperation( ${ this.baseVersion } ): ` +\n\t// @if CK_DEBUG_ENGINE // \t\t`${ this.position }: \"${ this.oldName }\" -> \"${ this.newName }\"`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/rootattributeoperation\n */\n\nimport Operation from './operation';\n\nimport type Document from '../document';\nimport type RootElement from '../rootelement';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\nimport type { Selectable } from '../selection';\n\n/**\n * Operation to change root element's attribute. Using this class you can add, remove or change value of the attribute.\n *\n * This operation is needed, because root elements can't be changed through\n * {@link module:engine/model/operation/attributeoperation~AttributeOperation}.\n * It is because {@link module:engine/model/operation/attributeoperation~AttributeOperation}\n * requires a range to change and root element can't\n * be a part of range because every {@link module:engine/model/position~Position} has to be inside a root.\n * {@link module:engine/model/position~Position} can't be created before a root element.\n */\nexport default class RootAttributeOperation extends Operation {\n\t/**\n\t * Root element to change.\n\t */\n\tpublic readonly root: RootElement;\n\n\t/**\n\t * Key of an attribute to change or remove.\n\t */\n\tpublic readonly key: string;\n\n\t/**\n\t * Old value of the attribute with given key or `null`, if attribute was not set before.\n\t *\n\t * @readonly\n\t */\n\tpublic oldValue: unknown;\n\n\t/**\n\t * New value of the attribute with given key or `null`, if operation should remove attribute.\n\t *\n\t * @readonly\n\t */\n\tpublic newValue: unknown;\n\n\t/**\n\t * Creates an operation that changes, removes or adds attributes on root element.\n\t *\n\t * @see module:engine/model/operation/attributeoperation~AttributeOperation\n\t * @param root Root element to change.\n\t * @param key Key of an attribute to change or remove.\n\t * @param oldValue Old value of the attribute with given key or `null`, if attribute was not set before.\n\t * @param newValue New value of the attribute with given key or `null`, if operation should remove attribute.\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation\n\t * can be applied or `null` if the operation operates on detached (non-document) tree.\n\t */\n\tconstructor(\n\t\troot: RootElement,\n\t\tkey: string,\n\t\toldValue: unknown,\n\t\tnewValue: unknown,\n\t\tbaseVersion: number | null\n\t) {\n\t\tsuper( baseVersion );\n\n\t\tthis.root = root;\n\t\tthis.key = key;\n\t\tthis.oldValue = oldValue === undefined ? null : oldValue;\n\t\tthis.newValue = newValue === undefined ? null : newValue;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'addRootAttribute' | 'removeRootAttribute' | 'changeRootAttribute' {\n\t\tif ( this.oldValue === null ) {\n\t\t\treturn 'addRootAttribute';\n\t\t} else if ( this.newValue === null ) {\n\t\t\treturn 'removeRootAttribute';\n\t\t} else {\n\t\t\treturn 'changeRootAttribute';\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\treturn this.root;\n\t}\n\n\t/**\n\t * Creates and returns an operation that has the same parameters as this operation.\n\t *\n\t * @returns Clone of this operation.\n\t */\n\tpublic clone(): RootAttributeOperation {\n\t\treturn new RootAttributeOperation( this.root, this.key, this.oldValue, this.newValue, this.baseVersion );\n\t}\n\n\t/**\n\t * See {@link module:engine/model/operation/operation~Operation#getReversed `Operation#getReversed()`}.\n\t */\n\tpublic getReversed(): Operation {\n\t\treturn new RootAttributeOperation( this.root, this.key, this.newValue, this.oldValue, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _validate(): void {\n\t\tif ( this.root != this.root.root || this.root.is( 'documentFragment' ) ) {\n\t\t\t/**\n\t\t\t * The element to change is not a root element.\n\t\t\t *\n\t\t\t * @error rootattribute-operation-not-a-root\n\t\t\t * @param root\n\t\t\t * @param key\n\t\t\t * @param value\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'rootattribute-operation-not-a-root',\n\t\t\t\tthis,\n\t\t\t\t{ root: this.root, key: this.key }\n\t\t\t);\n\t\t}\n\n\t\tif ( this.oldValue !== null && this.root.getAttribute( this.key ) !== this.oldValue ) {\n\t\t\t/**\n\t\t\t * The attribute which should be removed does not exist for the given node.\n\t\t\t *\n\t\t\t * @error rootattribute-operation-wrong-old-value\n\t\t\t * @param root\n\t\t\t * @param key\n\t\t\t * @param value\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'rootattribute-operation-wrong-old-value',\n\t\t\t\tthis,\n\t\t\t\t{ root: this.root, key: this.key }\n\t\t\t);\n\t\t}\n\n\t\tif ( this.oldValue === null && this.newValue !== null && this.root.hasAttribute( this.key ) ) {\n\t\t\t/**\n\t\t\t * The attribute with given key already exists for the given node.\n\t\t\t *\n\t\t\t * @error rootattribute-operation-attribute-exists\n\t\t\t * @param root\n\t\t\t * @param key\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'rootattribute-operation-attribute-exists',\n\t\t\t\tthis,\n\t\t\t\t{ root: this.root, key: this.key }\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _execute(): void {\n\t\tif ( this.newValue !== null ) {\n\t\t\tthis.root._setAttribute( this.key, this.newValue );\n\t\t} else {\n\t\t\tthis.root._removeAttribute( this.key );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.root = this.root.toJSON();\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'RootAttributeOperation';\n\t}\n\n\t/**\n\t * Creates `RootAttributeOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): RootAttributeOperation {\n\t\tif ( !document.getRoot( json.root ) ) {\n\t\t\t/**\n\t\t\t * Cannot create RootAttributeOperation for document. Root with specified name does not exist.\n\t\t\t *\n\t\t\t * @error rootattribute-operation-fromjson-no-root\n\t\t\t * @param rootName\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'rootattribute-operation-fromjson-no-root', this, { rootName: json.root } );\n\t\t}\n\n\t\treturn new RootAttributeOperation( document.getRoot( json.root )!, json.key, json.oldValue, json.newValue, json.baseVersion );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `RootAttributeOperation( ${ this.baseVersion } ): ` +\n\t// @if CK_DEBUG_ENGINE //\t\t`\"${ this.key }\": ${ JSON.stringify( this.oldValue ) }` +\n\t// @if CK_DEBUG_ENGINE //\t\t` -> ${ JSON.stringify( this.newValue ) }, ${ this.root.rootName }`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/rootoperation\n */\n\nimport Operation from './operation';\n\nimport type Document from '../document';\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\nimport type { Selectable } from '../selection';\n\n/**\n * Operation that creates (or attaches) or detaches a root element.\n */\nexport default class RootOperation extends Operation {\n\t/**\n\t * Root name to create or detach.\n\t */\n\tpublic readonly rootName: string;\n\n\t/**\n\t * Root element name.\n\t */\n\tpublic readonly elementName: string;\n\n\t/**\n\t * Specifies whether the operation adds (`true`) or detaches the root (`false`).\n\t */\n\tpublic readonly isAdd: boolean;\n\n\t/**\n\t * Document which owns the root.\n\t */\n\tprivate readonly _document: Document;\n\n\t/**\n\t * Creates an operation that creates or removes a root element.\n\t *\n\t * @param rootName Root name to create or detach.\n\t * @param elementName Root element name.\n\t * @param isAdd Specifies whether the operation adds (`true`) or detaches the root (`false`).\n\t * @param document Document which owns the root.\n\t * @param baseVersion Document {@link module:engine/model/document~Document#version} on which operation can be applied.\n\t */\n\tconstructor(\n\t\trootName: string,\n\t\telementName: string,\n\t\tisAdd: boolean,\n\t\tdocument: Document,\n\t\tbaseVersion: number\n\t) {\n\t\tsuper( baseVersion );\n\n\t\tthis.rootName = rootName;\n\t\tthis.elementName = elementName;\n\t\tthis.isAdd = isAdd;\n\t\tthis._document = document;\n\n\t\t// Make sure that the root exists ASAP, this is important for RTC.\n\t\t// If the root was dynamically added, there will be more operations that operate on/in this root.\n\t\t// These operations will require root element instance (in operation property or in position instance).\n\t\t// If the root is not created ahead of time, instantiating such operations may fail.\n\t\tif ( !this._document.getRoot( this.rootName ) ) {\n\t\t\tconst root = this._document.createRoot( this.elementName, this.rootName );\n\n\t\t\troot._isAttached = false;\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override get type(): 'addRoot' | 'detachRoot' {\n\t\treturn this.isAdd ? 'addRoot' : 'detachRoot';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\treturn this._document.getRoot( this.rootName );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override clone(): RootOperation {\n\t\treturn new RootOperation( this.rootName, this.elementName, this.isAdd, this._document, this.baseVersion! );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override getReversed(): RootOperation {\n\t\treturn new RootOperation( this.rootName, this.elementName, !this.isAdd, this._document, this.baseVersion! + 1 );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override _execute(): void {\n\t\tthis._document.getRoot( this.rootName )!._isAttached = this.isAdd;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tdelete json._document;\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'RootOperation';\n\t}\n\n\t/**\n\t * Creates `RootOperation` object from deserialized object, i.e. from parsed JSON string.\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static override fromJSON( json: any, document: Document ): RootOperation {\n\t\treturn new RootOperation( json.rootName, json.elementName, json.isAdd, document, json.baseVersion );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn `RootOperation( ${ this.baseVersion } ): ` +\n\t// @if CK_DEBUG_ENGINE //\t\t`${ this.type } -> \"${ this.rootName }\" (${ this.elementName })`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/operationfactory\n */\n\nimport AttributeOperation from './attributeoperation';\nimport InsertOperation from './insertoperation';\nimport MarkerOperation from './markeroperation';\nimport MoveOperation from './moveoperation';\nimport NoOperation from './nooperation';\nimport Operation from './operation';\nimport RenameOperation from './renameoperation';\nimport RootAttributeOperation from './rootattributeoperation';\nimport RootOperation from './rootoperation';\nimport SplitOperation from './splitoperation';\nimport MergeOperation from './mergeoperation';\n\nimport type Document from '../document';\n\nconst operations: {\n\t[ className: string ]: {\n\t\tfromJSON( json: any, document: Document ): Operation;\n\t};\n} = {};\n\noperations[ AttributeOperation.className ] = AttributeOperation;\noperations[ InsertOperation.className ] = InsertOperation;\noperations[ MarkerOperation.className ] = MarkerOperation;\noperations[ MoveOperation.className ] = MoveOperation;\noperations[ NoOperation.className ] = NoOperation;\noperations[ Operation.className ] = Operation;\noperations[ RenameOperation.className ] = RenameOperation;\noperations[ RootAttributeOperation.className ] = RootAttributeOperation;\noperations[ RootOperation.className ] = RootOperation;\noperations[ SplitOperation.className ] = SplitOperation;\noperations[ MergeOperation.className ] = MergeOperation;\n\n/**\n * A factory class for creating operations.\n */\nexport default abstract class OperationFactory {\n\t/**\n\t * Creates an operation instance from a JSON object (parsed JSON string).\n\t *\n\t * @param json Deserialized JSON object.\n\t * @param document Document on which this operation will be applied.\n\t */\n\tpublic static fromJSON( json: any, document: Document ): Operation {\n\t\treturn operations[ json.__className ].fromJSON( json, document );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport InsertOperation from './insertoperation';\nimport AttributeOperation from './attributeoperation';\nimport RenameOperation from './renameoperation';\nimport MarkerOperation from './markeroperation';\nimport MoveOperation from './moveoperation';\nimport RootAttributeOperation from './rootattributeoperation';\nimport RootOperation from './rootoperation';\nimport MergeOperation from './mergeoperation';\nimport SplitOperation from './splitoperation';\nimport NoOperation from './nooperation';\nimport Range from '../range';\nimport Position from '../position';\n\nimport type Operation from './operation';\nimport type Document from '../document';\nimport type History from '../history';\n\nimport { compareArrays } from '@ckeditor/ckeditor5-utils';\n\ntype TransformationFunction = ( a: Operation, b: Operation, context: TransformationContext ) => Array<Operation>;\n\nconst transformations = new Map<Function, Map<Function, TransformationFunction>>();\n\n/**\n * @module engine/model/operation/transform\n */\n\n/**\n * Sets a transformation function to be be used to transform instances of class `OperationA` by instances of class `OperationB`.\n *\n * The `transformationFunction` is passed three parameters:\n *\n * * `a` - operation to be transformed, an instance of `OperationA`,\n * * `b` - operation to be transformed by, an instance of `OperationB`,\n * * {@link module:engine/model/operation/transform~TransformationContext `context`} - object with additional information about\n * transformation context.\n *\n * The `transformationFunction` should return transformation result, which is an array with one or multiple\n * {@link module:engine/model/operation/operation~Operation operation} instances.\n *\n * @param transformationFunction Function to use for transforming.\n */\nfunction setTransformation<\n\tClassA extends new ( ...args: Array<any> ) => Operation,\n\tClassB extends new ( ...args: Array<any> ) => Operation\n>(\n\tOperationA: ClassA,\n\tOperationB: ClassB,\n\ttransformationFunction: ( a: InstanceType<ClassA>, b: InstanceType<ClassB>, context: TransformationContext ) => Array<Operation>\n) {\n\tlet aGroup = transformations.get( OperationA );\n\n\tif ( !aGroup ) {\n\t\taGroup = new Map();\n\t\ttransformations.set( OperationA, aGroup );\n\t}\n\n\taGroup.set( OperationB, transformationFunction as TransformationFunction );\n}\n\n/**\n * Returns a previously set transformation function for transforming an instance of `OperationA` by an instance of `OperationB`.\n *\n * If no transformation was set for given pair of operations, {@link module:engine/model/operation/transform~noUpdateTransformation}\n * is returned. This means that if no transformation was set, the `OperationA` instance will not change when transformed\n * by the `OperationB` instance.\n *\n * @returns Function set to transform an instance of `OperationA` by an instance of `OperationB`.\n */\nfunction getTransformation( OperationA: Function, OperationB: Function ): TransformationFunction {\n\tconst aGroup = transformations.get( OperationA );\n\n\tif ( aGroup && aGroup.has( OperationB ) ) {\n\t\treturn aGroup.get( OperationB )!;\n\t}\n\n\treturn noUpdateTransformation;\n}\n\n/**\n * A transformation function that only clones operation to transform, without changing it.\n */\nfunction noUpdateTransformation( a: Operation ): Array<Operation> {\n\treturn [ a ];\n}\n\n/**\n * Transforms operation `a` by operation `b`.\n *\n * @param a Operation to be transformed.\n * @param b Operation to transform by.\n * @param context Transformation context for this transformation.\n * @returns Transformation result.\n */\nexport function transform( a: Operation, b: Operation, context: TransformationContext = {} ): Array<Operation> {\n\tconst transformationFunction = getTransformation( a.constructor, b.constructor );\n\n\t/* eslint-disable no-useless-catch */\n\ttry {\n\t\ta = a.clone();\n\n\t\treturn transformationFunction( a, b, context );\n\t} catch ( e: any ) {\n\t\t// @if CK_DEBUG // console.warn( 'Error during operation transformation!', e.message );\n\t\t// @if CK_DEBUG // console.warn( 'Transformed operation', a );\n\t\t// @if CK_DEBUG // console.warn( 'Operation transformed by', b );\n\t\t// @if CK_DEBUG // console.warn( 'context.aIsStrong', context.aIsStrong );\n\t\t// @if CK_DEBUG // console.warn( 'context.aWasUndone', context.aWasUndone );\n\t\t// @if CK_DEBUG // console.warn( 'context.bWasUndone', context.bWasUndone );\n\t\t// @if CK_DEBUG // console.warn( 'context.abRelation', context.abRelation );\n\t\t// @if CK_DEBUG // console.warn( 'context.baRelation', context.baRelation );\n\n\t\tthrow e;\n\t}\n\t/* eslint-enable no-useless-catch */\n}\n\n/**\n * Performs a transformation of two sets of operations - `operationsA` and `operationsB`. The transformation is two-way -\n * both transformed `operationsA` and transformed `operationsB` are returned.\n *\n * Note, that the first operation in each set should base on the same document state (\n * {@link module:engine/model/document~Document#version document version}).\n *\n * It is assumed that `operationsA` are \"more important\" during conflict resolution between two operations.\n *\n * New copies of both passed arrays and operations inside them are returned. Passed arguments are not altered.\n *\n * Base versions of the transformed operations sets are updated accordingly. For example, assume that base versions are `4`\n * and there are `3` operations in `operationsA` and `5` operations in `operationsB`. Then:\n *\n * * transformed `operationsA` will start from base version `9` (`4` base version + `5` operations B),\n * * transformed `operationsB` will start from base version `7` (`4` base version + `3` operations A).\n *\n * If no operation was broken into two during transformation, then both sets will end up with an operation that bases on version `11`:\n *\n * * transformed `operationsA` start from `9` and there are `3` of them, so the last will have `baseVersion` equal to `11`,\n * * transformed `operationsB` start from `7` and there are `5` of them, so the last will have `baseVersion` equal to `11`.\n *\n * @param operationsA\n * @param operationsB\n * @param options Additional transformation options.\n * @param options.document Document which the operations change.\n * @param options.useRelations Whether during transformation relations should be used (used during undo for better conflict resolution).\n * @param options.padWithNoOps Whether additional {@link module:engine/model/operation/nooperation~NoOperation}s\n * should be added to the transformation results to force the same last base version for both transformed sets (in case\n * if some operations got broken into multiple operations during transformation).\n * @param options.forceWeakRemove If set to `false`, remove operation will be always stronger than move operation,\n * so the removed nodes won't end up back in the document root. When set to `true`, context data will be used.\n * @returns Transformation result.\n */\nexport function transformSets(\n\toperationsA: Array<Operation>,\n\toperationsB: Array<Operation>,\n\toptions: {\n\t\tdocument: Document;\n\t\tuseRelations?: boolean;\n\t\tpadWithNoOps?: boolean;\n\t\tforceWeakRemove?: boolean;\n\t}\n): TransformSetsResult {\n\t// Create new arrays so the originally passed arguments are not changed.\n\t// No need to clone operations, they are cloned as they are transformed.\n\toperationsA = operationsA.slice();\n\toperationsB = operationsB.slice();\n\n\tconst contextFactory = new ContextFactory( options.document, options.useRelations, options.forceWeakRemove );\n\tcontextFactory.setOriginalOperations( operationsA );\n\tcontextFactory.setOriginalOperations( operationsB );\n\n\tconst originalOperations = contextFactory.originalOperations;\n\n\t// If one of sets is empty there is simply nothing to transform, so return sets as they are.\n\tif ( operationsA.length == 0 || operationsB.length == 0 ) {\n\t\treturn { operationsA, operationsB, originalOperations };\n\t}\n\t//\n\t// Following is a description of transformation process:\n\t//\n\t// There are `operationsA` and `operationsB` to be transformed, both by both.\n\t//\n\t// So, suppose we have sets of two operations each: `operationsA` = `[ a1, a2 ]`, `operationsB` = `[ b1, b2 ]`.\n\t//\n\t// Remember, that we can only transform operations that base on the same context. We assert that `a1` and `b1` base on\n\t// the same context and we transform them. Then, we get `a1'` and `b1'`. `a2` bases on a context with `a1` -- `a2`\n\t// is an operation that followed `a1`. Similarly, `b2` bases on a context with `b1`.\n\t//\n\t// However, since `a1'` is a result of transformation by `b1`, `a1'` now also has a context with `b1`. This means that\n\t// we can safely transform `a1'` by `b2`. As we finish transforming `a1`, we also transformed all `operationsB`.\n\t// All `operationsB` also have context including `a1`. Now, we can properly transform `a2` by those operations.\n\t//\n\t// The transformation process can be visualized on a transformation diagram (\"diamond diagram\"):\n\t//\n\t// [the initial state]\n\t// [common for a1 and b1]\n\t//\n\t// *\n\t// / \\\n\t// / \\\n\t// b1 a1\n\t// / \\\n\t// / \\\n\t// * *\n\t// / \\ / \\\n\t// / \\ / \\\n\t// b2 a1' b1' a2\n\t// / \\ / \\\n\t// / \\ / \\\n\t// * * *\n\t// \\ / \\ /\n\t// \\ / \\ /\n\t// a1'' b2' a2' b1''\n\t// \\ / \\ /\n\t// \\ / \\ /\n\t// * *\n\t// \\ /\n\t// \\ /\n\t// a2'' b2''\n\t// \\ /\n\t// \\ /\n\t// *\n\t//\n\t// [the final state]\n\t//\n\t// The final state can be reached from the initial state by applying `a1`, `a2`, `b1''` and `b2''`, as well as by\n\t// applying `b1`, `b2`, `a1''`, `a2''`. Note how the operations get to a proper common state before each pair is\n\t// transformed.\n\t//\n\t// Another thing to consider is that an operation during transformation can be broken into multiple operations.\n\t// Suppose that `a1` * `b1` = `[ a11', a12' ]` (instead of `a1'` that we considered previously).\n\t//\n\t// In that case, we leave `a12'` for later and we continue transforming `a11'` until it is transformed by all `operationsB`\n\t// (in our case it is just `b2`). At this point, `b1` is transformed by \"whole\" `a1`, while `b2` is only transformed\n\t// by `a11'`. Similarly, `a12'` is only transformed by `b1`. This leads to a conclusion that we need to start transforming `a12'`\n\t// from the moment just after it was broken. So, `a12'` is transformed by `b2`. Now, \"the whole\" `a1` is transformed\n\t// by `operationsB`, while all `operationsB` are transformed by \"the whole\" `a1`. This means that we can continue with\n\t// following `operationsA` (in our case it is just `a2`).\n\t//\n\t// Of course, also `operationsB` can be broken. However, since we focus on transforming operation `a` to the end,\n\t// the only thing to do is to store both pieces of operation `b`, so that the next transformed operation `a` will\n\t// be transformed by both of them.\n\t//\n\t// *\n\t// / \\\n\t// / \\\n\t// / \\\n\t// b1 a1\n\t// / \\\n\t// / \\\n\t// / \\\n\t// * *\n\t// / \\ / \\\n\t// / a11' / \\\n\t// / \\ / \\\n\t// b2 * b1' a2\n\t// / / \\ / \\\n\t// / / a12' / \\\n\t// / / \\ / \\\n\t// * b2' * *\n\t// \\ / / \\ /\n\t// a11'' / b21'' \\ /\n\t// \\ / / \\ /\n\t// * * a2' b1''\n\t// \\ / \\ \\ /\n\t// a12'' b22''\\ \\ /\n\t// \\ / \\ \\ /\n\t// * a2'' *\n\t// \\ \\ /\n\t// \\ \\ b21'''\n\t// \\ \\ /\n\t// a2''' *\n\t// \\ /\n\t// \\ b22'''\n\t// \\ /\n\t// *\n\t//\n\t// Note, how `a1` is broken and transformed into `a11'` and `a12'`, while `b2'` got broken and transformed into `b21''` and `b22''`.\n\t//\n\t// Having all that on mind, here is an outline for the transformation process algorithm:\n\t//\n\t// 1. We have `operationsA` and `operationsB` array, which we dynamically update as the transformation process goes.\n\t//\n\t// 2. We take next (or first) operation from `operationsA` and check from which operation `b` we need to start transforming it.\n\t// All original `operationsA` are set to be transformed starting from the first operation `b`.\n\t//\n\t// 3. We take operations from `operationsB`, one by one, starting from the correct one, and transform operation `a`\n\t// by operation `b` (and vice versa). We update `operationsA` and `operationsB` by replacing the original operations\n\t// with the transformation results.\n\t//\n\t// 4. If operation is broken into multiple operations, we save all the new operations in the place of the\n\t// original operation.\n\t//\n\t// 5. Additionally, if operation `a` was broken, for the \"new\" operation, we remember from which operation `b` it should\n\t// be transformed by.\n\t//\n\t// 6. We continue transforming \"current\" operation `a` until it is transformed by all `operationsB`. Then, go to 2.\n\t// unless the last operation `a` was transformed.\n\t//\n\t// The actual implementation of the above algorithm is slightly different, as only one loop (while) is used.\n\t// The difference is that we have \"current\" `a` operation to transform and we store the index of the next `b` operation\n\t// to transform by. Each loop operates on two indexes then: index pointing to currently processed `a` operation and\n\t// index pointing to next `b` operation. Each loop is just one `a * b` + `b * a` transformation. After each loop\n\t// operation `b` index is updated. If all `b` operations were visited for the current `a` operation, we change\n\t// current `a` operation index to the next one.\n\t//\n\n\t// For each operation `a`, keeps information what is the index in `operationsB` from which the transformation should start.\n\tconst nextTransformIndex = new WeakMap<Operation, number>();\n\n\t// For all the original `operationsA`, set that they should be transformed starting from the first of `operationsB`.\n\tfor ( const op of operationsA ) {\n\t\tnextTransformIndex.set( op, 0 );\n\t}\n\n\t// Additional data that is used for some postprocessing after the main transformation process is done.\n\tconst data = {\n\t\tnextBaseVersionA: operationsA[ operationsA.length - 1 ].baseVersion! + 1,\n\t\tnextBaseVersionB: operationsB[ operationsB.length - 1 ].baseVersion! + 1,\n\t\toriginalOperationsACount: operationsA.length,\n\t\toriginalOperationsBCount: operationsB.length\n\t};\n\n\t// Index of currently transformed operation `a`.\n\tlet i = 0;\n\n\t// While not all `operationsA` are transformed...\n\twhile ( i < operationsA.length ) {\n\t\t// Get \"current\" operation `a`.\n\t\tconst opA = operationsA[ i ];\n\n\t\t// For the \"current\" operation `a`, get the index of the next operation `b` to transform by.\n\t\tconst indexB = nextTransformIndex.get( opA )!;\n\n\t\t// If operation `a` was already transformed by every operation `b`, change \"current\" operation `a` to the next one.\n\t\tif ( indexB == operationsB.length ) {\n\t\t\ti++;\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst opB = operationsB[ indexB ];\n\n\t\t// Transform `a` by `b` and `b` by `a`.\n\t\tconst newOpsA = transform( opA, opB, contextFactory.getContext( opA, opB, true ) );\n\t\tconst newOpsB = transform( opB, opA, contextFactory.getContext( opB, opA, false ) );\n\t\t// As a result we get one or more `newOpsA` and one or more `newOpsB` operations.\n\n\t\t// Update contextual information about operations.\n\t\tcontextFactory.updateRelation( opA, opB );\n\n\t\tcontextFactory.setOriginalOperations( newOpsA, opA );\n\t\tcontextFactory.setOriginalOperations( newOpsB, opB );\n\n\t\t// For new `a` operations, update their index of the next operation `b` to transform them by.\n\t\t//\n\t\t// This is needed even if there was only one result (`a` was not broken) because that information is used\n\t\t// at the beginning of this loop every time.\n\t\tfor ( const newOpA of newOpsA ) {\n\t\t\t// Acknowledge, that operation `b` also might be broken into multiple operations.\n\t\t\t//\n\t\t\t// This is why we raise `indexB` not just by 1. If `newOpsB` are multiple operations, they will be\n\t\t\t// spliced in the place of `opB`. So we need to change `transformBy` accordingly, so that an operation won't\n\t\t\t// be transformed by the same operation (part of it) again.\n\t\t\tnextTransformIndex.set( newOpA, indexB + newOpsB.length );\n\t\t}\n\n\t\t// Update `operationsA` and `operationsB` with the transformed versions.\n\t\toperationsA.splice( i, 1, ...newOpsA );\n\t\toperationsB.splice( indexB, 1, ...newOpsB );\n\t}\n\n\tif ( options.padWithNoOps ) {\n\t\t// If no-operations padding is enabled, count how many extra `a` and `b` operations were generated.\n\t\tconst brokenOperationsACount = operationsA.length - data.originalOperationsACount;\n\t\tconst brokenOperationsBCount = operationsB.length - data.originalOperationsBCount;\n\n\t\t// Then, if that number is not the same, pad `operationsA` or `operationsB` with correct number of no-ops so\n\t\t// that the base versions are equalled.\n\t\t//\n\t\t// Note that only one array will be updated, as only one of those subtractions can be greater than zero.\n\t\tpadWithNoOps( operationsA, brokenOperationsBCount - brokenOperationsACount );\n\t\tpadWithNoOps( operationsB, brokenOperationsACount - brokenOperationsBCount );\n\t}\n\n\t// Finally, update base versions of transformed operations.\n\tupdateBaseVersions( operationsA, data.nextBaseVersionB );\n\tupdateBaseVersions( operationsB, data.nextBaseVersionA );\n\n\treturn { operationsA, operationsB, originalOperations };\n}\n\n/**\n * The result of {@link module:engine/model/operation/transform~transformSets}.\n */\nexport interface TransformSetsResult {\n\n\t/**\n\t * Transformed `operationsA`.\n\t */\n\toperationsA: Array<Operation>;\n\n\t/**\n\t * Transformed `operationsB`.\n\t */\n\toperationsB: Array<Operation>;\n\n\t/**\n\t * A map that links transformed operations to original operations. The keys are the transformed\n\t * operations and the values are the original operations from the input (`operationsA` and `operationsB`).\n\t */\n\toriginalOperations: Map<Operation, Operation>;\n}\n\n/**\n * Gathers additional data about operations processed during transformation. Can be used to obtain contextual information\n * about two operations that are about to be transformed. This contextual information can be used for better conflict resolution.\n */\nclass ContextFactory {\n\tpublic readonly originalOperations: Map<Operation, Operation>;\n\n\tprivate readonly _history: History;\n\tprivate readonly _useRelations?: boolean;\n\tprivate readonly _forceWeakRemove: boolean;\n\tprivate readonly _relations: Map<Operation, Map<Operation, any>>;\n\n\t/**\n\t * Creates `ContextFactory` instance.\n\t *\n\t * @param document Document which the operations change.\n\t * @param useRelations Whether during transformation relations should be used (used during undo for\n\t * better conflict resolution).\n\t * @param forceWeakRemove If set to `false`, remove operation will be always stronger than move operation,\n\t * so the removed nodes won't end up back in the document root. When set to `true`, context data will be used.\n\t */\n\tconstructor( document: Document, useRelations: boolean | undefined, forceWeakRemove = false ) {\n\t\t// For each operation that is created during transformation process, we keep a reference to the original operation\n\t\t// which it comes from. The original operation works as a kind of \"identifier\". Every contextual information\n\t\t// gathered during transformation that we want to save for given operation, is actually saved for the original operation.\n\t\t// This way no matter if operation `a` is cloned, then transformed, even breaks, we still have access to the previously\n\t\t// gathered data through original operation reference.\n\t\tthis.originalOperations = new Map();\n\n\t\t// `model.History` instance which information about undone operations will be taken from.\n\t\tthis._history = document.history;\n\n\t\t// Whether additional context should be used.\n\t\tthis._useRelations = useRelations;\n\n\t\tthis._forceWeakRemove = !!forceWeakRemove;\n\n\t\t// Relations is a double-map structure (maps in map) where for two operations we store how those operations were related\n\t\t// to each other. Those relations are evaluated during transformation process. For every transformated pair of operations\n\t\t// we keep relations between them.\n\t\tthis._relations = new Map();\n\t}\n\n\t/**\n\t * Sets \"original operation\" for given operations.\n\t *\n\t * During transformation process, operations are cloned, then changed, then processed again, sometimes broken into two\n\t * or multiple operations. When gathering additional data it is important that all operations can be somehow linked\n\t * so a cloned and transformed \"version\" still kept track of the data assigned earlier to it.\n\t *\n\t * The original operation object will be used as such an universal linking id. Throughout the transformation process\n\t * all cloned operations will refer to \"the original operation\" when storing and reading additional data.\n\t *\n\t * If `takeFrom` is not set, each operation from `operations` array will be assigned itself as \"the original operation\".\n\t * This should be used as an initialization step.\n\t *\n\t * If `takeFrom` is set, each operation from `operations` will be assigned the same original operation as assigned\n\t * for `takeFrom` operation. This should be used to update original operations. It should be used in a way that\n\t * `operations` are the result of `takeFrom` transformation to ensure proper \"original operation propagation\".\n\t */\n\tpublic setOriginalOperations( operations: Array<Operation>, takeFrom: Operation | null = null ): void {\n\t\tconst originalOperation = takeFrom ? this.originalOperations.get( takeFrom ) : null;\n\n\t\tfor ( const operation of operations ) {\n\t\t\tthis.originalOperations.set( operation, originalOperation || operation );\n\t\t}\n\t}\n\n\t/**\n\t * Saves a relation between operations `opA` and `opB`.\n\t *\n\t * Relations are then later used to help solve conflicts when operations are transformed.\n\t */\n\tpublic updateRelation( opA: Operation, opB: Operation ): void {\n\t\t// The use of relations is described in a bigger detail in transformation functions.\n\t\t//\n\t\t// In brief, this function, for specified pairs of operation types, checks how positions defined in those operations relate.\n\t\t// Then those relations are saved. For example, for two move operations, it is saved if one of those operations target\n\t\t// position is before the other operation source position. This kind of information gives contextual information when\n\t\t// transformation is used during undo. Similar checks are done for other pairs of operations.\n\t\t//\n\t\tif ( opA instanceof MoveOperation ) {\n\t\t\tif ( opB instanceof MergeOperation ) {\n\t\t\t\tif ( opA.targetPosition.isEqual( opB.sourcePosition ) || opB.movedRange.containsPosition( opA.targetPosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'insertAtSource' );\n\t\t\t\t} else if ( opA.targetPosition.isEqual( opB.deletionPosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'insertBetween' );\n\t\t\t\t} else if ( opA.targetPosition.isAfter( opB.sourcePosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'moveTargetAfter' );\n\t\t\t\t}\n\t\t\t} else if ( opB instanceof MoveOperation ) {\n\t\t\t\tif ( opA.targetPosition.isEqual( opB.sourcePosition ) || opA.targetPosition.isBefore( opB.sourcePosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'insertBefore' );\n\t\t\t\t} else {\n\t\t\t\t\tthis._setRelation( opA, opB, 'insertAfter' );\n\t\t\t\t}\n\t\t\t}\n\t\t} else if ( opA instanceof SplitOperation ) {\n\t\t\tif ( opB instanceof MergeOperation ) {\n\t\t\t\tif ( opA.splitPosition.isBefore( opB.sourcePosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'splitBefore' );\n\t\t\t\t}\n\t\t\t} else if ( opB instanceof MoveOperation ) {\n\t\t\t\tif ( opA.splitPosition.isEqual( opB.sourcePosition ) || opA.splitPosition.isBefore( opB.sourcePosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'splitBefore' );\n\t\t\t\t} else {\n\t\t\t\t\tconst range = Range._createFromPositionAndShift( opB.sourcePosition, opB.howMany );\n\n\t\t\t\t\tif ( opA.splitPosition.hasSameParentAs( opB.sourcePosition ) && range.containsPosition( opA.splitPosition ) ) {\n\t\t\t\t\t\tconst howMany = range.end.offset - opA.splitPosition.offset;\n\t\t\t\t\t\tconst offset = opA.splitPosition.offset - range.start.offset;\n\n\t\t\t\t\t\tthis._setRelation( opA, opB, { howMany, offset } );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} else if ( opA instanceof MergeOperation ) {\n\t\t\tif ( opB instanceof MergeOperation ) {\n\t\t\t\tif ( !opA.targetPosition.isEqual( opB.sourcePosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'mergeTargetNotMoved' );\n\t\t\t\t}\n\n\t\t\t\tif ( opA.sourcePosition.isEqual( opB.targetPosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'mergeSourceNotMoved' );\n\t\t\t\t}\n\n\t\t\t\tif ( opA.sourcePosition.isEqual( opB.sourcePosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'mergeSameElement' );\n\t\t\t\t}\n\t\t\t} else if ( opB instanceof SplitOperation ) {\n\t\t\t\tif ( opA.sourcePosition.isEqual( opB.splitPosition ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, 'splitAtSource' );\n\t\t\t\t}\n\t\t\t}\n\t\t} else if ( opA instanceof MarkerOperation ) {\n\t\t\tconst markerRange = opA.newRange;\n\n\t\t\tif ( !markerRange ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( opB instanceof MoveOperation ) {\n\t\t\t\tconst movedRange = Range._createFromPositionAndShift( opB.sourcePosition, opB.howMany );\n\n\t\t\t\tconst affectedLeft = movedRange.containsPosition( markerRange.start ) ||\n\t\t\t\t\tmovedRange.start.isEqual( markerRange.start );\n\n\t\t\t\tconst affectedRight = movedRange.containsPosition( markerRange.end ) ||\n\t\t\t\t\tmovedRange.end.isEqual( markerRange.end );\n\n\t\t\t\tif ( ( affectedLeft || affectedRight ) && !movedRange.containsRange( markerRange ) ) {\n\t\t\t\t\tthis._setRelation( opA, opB, {\n\t\t\t\t\t\tside: affectedLeft ? 'left' : 'right',\n\t\t\t\t\t\tpath: affectedLeft ? markerRange.start.path.slice() : markerRange.end.path.slice()\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t} else if ( opB instanceof MergeOperation ) {\n\t\t\t\tconst wasInLeftElement = markerRange.start.isEqual( opB.targetPosition );\n\t\t\t\tconst wasStartBeforeMergedElement = markerRange.start.isEqual( opB.deletionPosition );\n\t\t\t\tconst wasEndBeforeMergedElement = markerRange.end.isEqual( opB.deletionPosition );\n\t\t\t\tconst wasInRightElement = markerRange.end.isEqual( opB.sourcePosition );\n\n\t\t\t\tif ( wasInLeftElement || wasStartBeforeMergedElement || wasEndBeforeMergedElement || wasInRightElement ) {\n\t\t\t\t\tthis._setRelation( opA, opB, {\n\t\t\t\t\t\twasInLeftElement,\n\t\t\t\t\t\twasStartBeforeMergedElement,\n\t\t\t\t\t\twasEndBeforeMergedElement,\n\t\t\t\t\t\twasInRightElement\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Evaluates and returns contextual information about two given operations `opA` and `opB` which are about to be transformed.\n\t */\n\tpublic getContext( opA: Operation, opB: Operation, aIsStrong: boolean ): TransformationContext {\n\t\treturn {\n\t\t\taIsStrong,\n\t\t\taWasUndone: this._wasUndone( opA ),\n\t\t\tbWasUndone: this._wasUndone( opB ),\n\t\t\tabRelation: this._useRelations ? this._getRelation( opA, opB ) : null,\n\t\t\tbaRelation: this._useRelations ? this._getRelation( opB, opA ) : null,\n\t\t\tforceWeakRemove: this._forceWeakRemove\n\t\t};\n\t}\n\n\t/**\n\t * Returns whether given operation `op` has already been undone.\n\t *\n\t * Information whether an operation was undone gives more context when making a decision when two operations are in conflict.\n\t */\n\tpublic _wasUndone( op: Operation ): boolean {\n\t\t// For `op`, get its original operation. After all, if `op` is a clone (or even transformed clone) of another\n\t\t// operation, literally `op` couldn't be undone. It was just generated. If anything, it was the operation it origins\n\t\t// from which was undone. So get that original operation.\n\t\tconst originalOp = this.originalOperations.get( op )!;\n\n\t\t// And check with the document if the original operation was undone.\n\t\treturn ( originalOp as any ).wasUndone || this._history.isUndoneOperation( originalOp );\n\t}\n\n\t/**\n\t * Returns a relation between `opA` and an operation which is undone by `opB`. This can be `String` value if a relation\n\t * was set earlier or `null` if there was no relation between those operations.\n\t *\n\t * This is a little tricky to understand, so let's compare it to `ContextFactory#_wasUndone`.\n\t *\n\t * When `wasUndone( opB )` is used, we check if the `opB` has already been undone. It is obvious, that the\n\t * undoing operation must happen after the undone operation. So, essentially, we have `opB`, we take document history,\n\t * we look forward in the future and ask if in that future `opB` was undone.\n\t *\n\t * Relations is a backward process to `wasUndone()`.\n\t *\n\t * Long story short - using relations is asking what happened in the past. Looking back. This time we have an undoing\n\t * operation `opB` which has undone some other operation. When there is a transformation `opA` x `opB` and there is\n\t * a conflict to solve and `opB` is an undoing operation, we can look back in the history and see what was a relation\n\t * between `opA` and the operation which `opB` undone. Basing on that relation from the past, we can now make\n\t * a better decision when resolving a conflict between two operations, because we know more about the context of\n\t * those two operations.\n\t *\n\t * This is why this function does not return a relation directly between `opA` and `opB` because we need to look\n\t * back to search for a meaningful contextual information.\n\t */\n\tpublic _getRelation( opA: Operation, opB: Operation ): any {\n\t\t// Get the original operation. Similarly as in `wasUndone()` it is used as an universal identifier for stored data.\n\t\tconst origB = this.originalOperations.get( opB )!;\n\t\tconst undoneB = this._history.getUndoneOperation( origB );\n\n\t\t// If `opB` is not undoing any operation, there is no relation.\n\t\tif ( !undoneB ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst origA = this.originalOperations.get( opA )!;\n\t\tconst relationsA = this._relations.get( origA );\n\n\t\t// Get all relations for `opA`, and check if there is a relation with `opB`-undone-counterpart. If so, return it.\n\t\tif ( relationsA ) {\n\t\t\treturn relationsA.get( undoneB ) || null;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Helper function for `ContextFactory#updateRelations`.\n\t */\n\tprivate _setRelation( opA: Operation, opB: Operation, relation: any ): void {\n\t\t// As always, setting is for original operations, not the clones/transformed operations.\n\t\tconst origA = this.originalOperations.get( opA )!;\n\t\tconst origB = this.originalOperations.get( opB )!;\n\n\t\tlet relationsA = this._relations.get( origA );\n\n\t\tif ( !relationsA ) {\n\t\t\trelationsA = new Map();\n\t\t\tthis._relations.set( origA, relationsA );\n\t\t}\n\n\t\trelationsA.set( origB, relation );\n\t}\n}\n\n/**\n * Holds additional contextual information about a transformed pair of operations (`a` and `b`). Those information\n * can be used for better conflict resolving.\n */\n\nexport type TransformationContext = {\n\n\t/**\n\t * Whether `a` is strong operation in this transformation, or weak.\n\t */\n\taIsStrong?: boolean;\n\n\t/**\n\t * Whether `a` operation was undone.\n\t */\n\taWasUndone?: boolean;\n\n\t/**\n\t * Whether `b` operation was undone.\n\t */\n\tbWasUndone?: boolean;\n\n\t/**\n\t * The relation between `a` operation and an operation undone by `b` operation.\n\t */\n\tabRelation?: any;\n\n\t/**\n\t * The relation between `b` operation and an operation undone by `a` operation.\n\t */\n\tbaRelation?: any;\n\n\tforceWeakRemove?: boolean;\n};\n\n/**\n * An utility function that updates {@link module:engine/model/operation/operation~Operation#baseVersion base versions}\n * of passed operations.\n *\n * The function simply sets `baseVersion` as a base version of the first passed operation and then increments it for\n * each following operation in `operations`.\n *\n * @param operations Operations to update.\n * @param baseVersion Base version to set for the first operation in `operations`.\n */\nfunction updateBaseVersions( operations: ReadonlyArray<Operation>, baseVersion: number ) {\n\tfor ( const operation of operations ) {\n\t\toperation.baseVersion = baseVersion++;\n\t}\n}\n\n/**\n * Adds `howMany` instances of {@link module:engine/model/operation/nooperation~NoOperation} to `operations` set.\n */\nfunction padWithNoOps( operations: Array<Operation>, howMany: number ) {\n\tfor ( let i = 0; i < howMany; i++ ) {\n\t\toperations.push( new NoOperation( 0 ) );\n\t}\n}\n\n// -----------------------\n\nsetTransformation( AttributeOperation, AttributeOperation, ( a, b, context ) => {\n\t// If operations in conflict, check if their ranges intersect and manage them properly.\n\t//\n\t// Operations can be in conflict only if:\n\t//\n\t// * their key is the same (they change the same attribute), and\n\t// * they are in the same parent (operations for ranges [ 1 ] - [ 3 ] and [ 2, 0 ] - [ 2, 5 ] change different\n\t// elements and can't be in conflict).\n\tif ( a.key === b.key && a.range.start.hasSameParentAs( b.range.start ) ) {\n\t\t// First, we want to apply change to the part of a range that has not been changed by the other operation.\n\t\tconst operations = a.range.getDifference( b.range ).map( range => {\n\t\t\treturn new AttributeOperation( range, a.key, a.oldValue, a.newValue, 0 );\n\t\t} );\n\n\t\t// Then we take care of the common part of ranges.\n\t\tconst common = a.range.getIntersection( b.range );\n\n\t\tif ( common ) {\n\t\t\t// If this operation is more important, we also want to apply change to the part of the\n\t\t\t// original range that has already been changed by the other operation. Since that range\n\t\t\t// got changed we also have to update `oldValue`.\n\t\t\tif ( context.aIsStrong ) {\n\t\t\t\toperations.push( new AttributeOperation( common, b.key, b.newValue, a.newValue, 0 ) );\n\t\t\t}\n\t\t}\n\n\t\tif ( operations.length == 0 ) {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t}\n\n\t\treturn operations;\n\t} else {\n\t\t// If operations don't conflict, simply return an array containing just a clone of this operation.\n\t\treturn [ a ];\n\t}\n} );\n\nsetTransformation( AttributeOperation, InsertOperation, ( a, b ) => {\n\t// Case 1:\n\t//\n\t// The attribute operation range includes the position where nodes were inserted.\n\t// There are two possible scenarios: the inserted nodes were text and they should receive attributes or\n\t// the inserted nodes were elements and they should not receive attributes.\n\t//\n\tif ( a.range.start.hasSameParentAs( b.position ) && a.range.containsPosition( b.position ) ) {\n\t\t// If new nodes should not receive attributes, two separated ranges will be returned.\n\t\t// Otherwise, one expanded range will be returned.\n\t\tconst range = a.range._getTransformedByInsertion( b.position, b.howMany, !b.shouldReceiveAttributes );\n\t\tconst result = range.map( r => {\n\t\t\treturn new AttributeOperation( r, a.key, a.oldValue, a.newValue, a.baseVersion );\n\t\t} );\n\n\t\tif ( b.shouldReceiveAttributes ) {\n\t\t\t// `AttributeOperation#range` includes some newly inserted text.\n\t\t\t// The operation should also change the attribute of that text. An example:\n\t\t\t//\n\t\t\t// Bold should be applied on the following range:\n\t\t\t// <p>Fo[zb]ar</p>\n\t\t\t//\n\t\t\t// In meantime, new text is typed:\n\t\t\t// <p>Fozxxbar</p>\n\t\t\t//\n\t\t\t// Bold should be applied also on the new text:\n\t\t\t// <p>Fo[zxxb]ar</p>\n\t\t\t// <p>Fo<$text bold=\"true\">zxxb</$text>ar</p>\n\t\t\t//\n\t\t\t// There is a special case to consider here to consider.\n\t\t\t//\n\t\t\t// Consider setting an attribute with multiple possible values, for example `highlight`. The inserted text might\n\t\t\t// have already an attribute value applied and the `oldValue` property of the attribute operation might be wrong:\n\t\t\t//\n\t\t\t// Attribute `highlight=\"yellow\"` should be applied on the following range:\n\t\t\t// <p>Fo[zb]ar<p>\n\t\t\t//\n\t\t\t// In meantime, character `x` with `highlight=\"red\"` is typed:\n\t\t\t// <p>Fo[z<$text highlight=\"red\">x</$text>b]ar</p>\n\t\t\t//\n\t\t\t// In this case we cannot simply apply operation changing the attribute value from `null` to `\"yellow\"` for the whole range\n\t\t\t// because that would lead to an exception (`oldValue` is incorrect for `x`).\n\t\t\t//\n\t\t\t// We also cannot break the original range as this would mess up a scenario when there are multiple following\n\t\t\t// insert operations, because then only the first inserted character is included in those ranges:\n\t\t\t// <p>Fo[z][x][b]ar</p> --> <p>Fo[z][x]x[b]ar</p> --> <p>Fo[z][x]xx[b]ar</p>\n\t\t\t//\n\t\t\t// So, the attribute range needs be expanded, no matter what attributes are set on the inserted nodes:\n\t\t\t//\n\t\t\t// <p>Fo[z<$text highlight=\"red\">x</$text>b]ar</p> <--- Change from `null` to `yellow`, throwing an exception.\n\t\t\t//\n\t\t\t// But before that operation would be applied, we will add an additional attribute operation that will change\n\t\t\t// attributes on the inserted nodes in a way which would make the original operation correct:\n\t\t\t//\n\t\t\t// <p>Fo[z{<$text highlight=\"red\">}x</$text>b]ar</p> <--- Change range `{}` from `red` to `null`.\n\t\t\t// <p>Fo[zxb]ar</p> <--- Now change from `null` to `yellow` is completely fine.\n\t\t\t//\n\n\t\t\t// Generate complementary attribute operation. Be sure to add it before the original operation.\n\t\t\tconst op = _getComplementaryAttributeOperations( b, a.key, a.oldValue );\n\n\t\t\tif ( op ) {\n\t\t\t\tresult.unshift( op );\n\t\t\t}\n\t\t}\n\n\t\t// If nodes should not receive new attribute, we are done here.\n\t\treturn result;\n\t}\n\n\t// If insert operation is not expanding the attribute operation range, simply transform the range.\n\ta.range = a.range._getTransformedByInsertion( b.position, b.howMany, false )[ 0 ];\n\n\treturn [ a ];\n} );\n\n/**\n * Helper function for `AttributeOperation` x `InsertOperation` (and reverse) transformation.\n *\n * For given `insertOperation` it checks the inserted node if it has an attribute `key` set to a value different\n * than `newValue`. If so, it generates an `AttributeOperation` which changes the value of `key` attribute to `newValue`.\n */\nfunction _getComplementaryAttributeOperations( insertOperation: InsertOperation, key: string, newValue: unknown ) {\n\tconst nodes = insertOperation.nodes;\n\n\t// At the beginning we store the attribute value from the first node.\n\tconst insertValue = nodes.getNode( 0 )!.getAttribute( key );\n\n\tif ( insertValue == newValue ) {\n\t\treturn null;\n\t}\n\n\tconst range = new Range( insertOperation.position, insertOperation.position.getShiftedBy( insertOperation.howMany ) );\n\n\treturn new AttributeOperation( range, key, insertValue, newValue, 0 );\n}\n\nsetTransformation( AttributeOperation, MergeOperation, ( a, b ) => {\n\tconst ranges = [];\n\n\t// Case 1:\n\t//\n\t// Attribute change on the merged element. In this case, the merged element was moved to the graveyard.\n\t// An additional attribute operation that will change the (re)moved element needs to be generated.\n\t//\n\tif ( a.range.start.hasSameParentAs( b.deletionPosition ) ) {\n\t\tif ( a.range.containsPosition( b.deletionPosition ) || a.range.start.isEqual( b.deletionPosition ) ) {\n\t\t\tranges.push( Range._createFromPositionAndShift( b.graveyardPosition, 1 ) );\n\t\t}\n\t}\n\n\tconst range = a.range._getTransformedByMergeOperation( b );\n\n\t// Do not add empty (collapsed) ranges to the result. `range` may be collapsed if it contained only the merged element.\n\tif ( !range.isCollapsed ) {\n\t\tranges.push( range );\n\t}\n\n\t// Create `AttributeOperation`s out of the ranges.\n\treturn ranges.map( range => {\n\t\treturn new AttributeOperation( range, a.key, a.oldValue, a.newValue, a.baseVersion );\n\t} );\n} );\n\nsetTransformation( AttributeOperation, MoveOperation, ( a, b ) => {\n\tconst ranges = _breakRangeByMoveOperation( a.range, b );\n\n\t// Create `AttributeOperation`s out of the ranges.\n\treturn ranges.map( range => new AttributeOperation( range, a.key, a.oldValue, a.newValue, a.baseVersion ) );\n} );\n\n/**\n * Helper function for `AttributeOperation` x `MoveOperation` transformation.\n *\n * Takes the passed `range` and transforms it by move operation `moveOp` in a specific way. Only top-level nodes of `range`\n * are considered to be in the range. If move operation moves nodes deep from inside of the range, those nodes won't\n * be included in the result. In other words, top-level nodes of the ranges from the result are exactly the same as\n * top-level nodes of the original `range`.\n *\n * This is important for `AttributeOperation` because, for its range, it changes only the top-level nodes. So we need to\n * track only how those nodes have been affected by `MoveOperation`.\n */\nfunction _breakRangeByMoveOperation( range: Range, moveOp: MoveOperation ) {\n\tconst moveRange = Range._createFromPositionAndShift( moveOp.sourcePosition, moveOp.howMany );\n\n\t// We are transforming `range` (original range) by `moveRange` (range moved by move operation). As usual when it comes to\n\t// transforming a ranges, we may have a common part of the ranges and we may have a difference part (zero to two ranges).\n\tlet common = null;\n\tlet difference: Array<Range> = [];\n\n\t// Let's compare the ranges.\n\tif ( moveRange.containsRange( range, true ) ) {\n\t\t// If the whole original range is moved, treat it whole as a common part. There's also no difference part.\n\t\tcommon = range;\n\t} else if ( range.start.hasSameParentAs( moveRange.start ) ) {\n\t\t// If the ranges are \"on the same level\" (in the same parent) then move operation may move exactly those nodes\n\t\t// that are changed by the attribute operation. In this case we get common part and difference part in the usual way.\n\t\tdifference = range.getDifference( moveRange );\n\t\tcommon = range.getIntersection( moveRange );\n\t} else {\n\t\t// In any other situation we assume that original range is different than move range, that is that move operation\n\t\t// moves other nodes that attribute operation change. Even if the moved range is deep inside in the original range.\n\t\t//\n\t\t// Note that this is different than in `.getIntersection` (we would get a common part in that case) and different\n\t\t// than `.getDifference` (we would get two ranges).\n\t\tdifference = [ range ];\n\t}\n\n\tconst result: Array<Range> = [];\n\n\t// The default behaviour of `_getTransformedByMove` might get wrong results for difference part, though, so\n\t// we do it by hand.\n\tfor ( let diff of difference ) {\n\t\t// First, transform the range by removing moved nodes. Since this is a difference, this is safe, `null` won't be returned\n\t\t// as the range is different than the moved range.\n\t\tdiff = diff._getTransformedByDeletion( moveOp.sourcePosition, moveOp.howMany )!;\n\n\t\t// Transform also `targetPosition`.\n\t\tconst targetPosition = moveOp.getMovedRangeStart();\n\n\t\t// Spread the range only if moved nodes are inserted only between the top-level nodes of the `diff` range.\n\t\tconst spread = diff.start.hasSameParentAs( targetPosition );\n\n\t\t// Transform by insertion of moved nodes.\n\t\tconst diffs = diff._getTransformedByInsertion( targetPosition, moveOp.howMany, spread );\n\n\t\tresult.push( ...diffs );\n\t}\n\n\t// Common part can be simply transformed by the move operation. This is because move operation will not target to\n\t// that common part (the operation would have to target inside its own moved range).\n\tif ( common ) {\n\t\tresult.push(\n\t\t\tcommon._getTransformedByMove( moveOp.sourcePosition, moveOp.targetPosition, moveOp.howMany, false )[ 0 ]\n\t\t);\n\t}\n\n\treturn result;\n}\n\nsetTransformation( AttributeOperation, SplitOperation, ( a, b ) => {\n\t// Case 1:\n\t//\n\t// Split node is the last node in `AttributeOperation#range`.\n\t// `AttributeOperation#range` needs to be expanded to include the new (split) node.\n\t//\n\t// Attribute `type` to be changed to `numbered` but the `listItem` is split.\n\t// <listItem type=\"bulleted\">foobar</listItem>\n\t//\n\t// After split:\n\t// <listItem type=\"bulleted\">foo</listItem><listItem type=\"bulleted\">bar</listItem>\n\t//\n\t// After attribute change:\n\t// <listItem type=\"numbered\">foo</listItem><listItem type=\"numbered\">foo</listItem>\n\t//\n\tif ( a.range.end.isEqual( b.insertionPosition ) ) {\n\t\tif ( !b.graveyardPosition ) {\n\t\t\ta.range.end.offset++;\n\t\t}\n\n\t\treturn [ a ];\n\t}\n\n\t// Case 2:\n\t//\n\t// Split position is inside `AttributeOperation#range`, at the same level, so the nodes to change are\n\t// not going to make a flat range.\n\t//\n\t// Content with range-to-change and split position:\n\t// <p>Fo[zb^a]r</p>\n\t//\n\t// After split:\n\t// <p>Fozb</p><p>ar</p>\n\t//\n\t// Make two separate ranges containing all nodes to change:\n\t// <p>Fo[zb]</p><p>[a]r</p>\n\t//\n\tif ( a.range.start.hasSameParentAs( b.splitPosition ) && a.range.containsPosition( b.splitPosition ) ) {\n\t\tconst secondPart = a.clone();\n\n\t\tsecondPart.range = new Range(\n\t\t\tb.moveTargetPosition.clone(),\n\t\t\ta.range.end._getCombined( b.splitPosition, b.moveTargetPosition )\n\t\t);\n\n\t\t( a.range as any ).end = b.splitPosition.clone();\n\t\ta.range.end.stickiness = 'toPrevious';\n\n\t\treturn [ a, secondPart ];\n\t}\n\n\t// The default case.\n\t//\n\ta.range = a.range._getTransformedBySplitOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( InsertOperation, AttributeOperation, ( a, b ) => {\n\tconst result: Array<Operation> = [ a ];\n\n\t// Case 1:\n\t//\n\t// The attribute operation range includes the position where nodes were inserted.\n\t// There are two possible scenarios: the inserted nodes were text and they should receive attributes or\n\t// the inserted nodes were elements and they should not receive attributes.\n\t//\n\t// This is a mirror scenario to the one described in `AttributeOperation` x `InsertOperation` transformation,\n\t// although this case is a little less complicated. In this case we simply need to change attributes of the\n\t// inserted nodes and that's it.\n\t//\n\tif ( a.shouldReceiveAttributes && a.position.hasSameParentAs( b.range.start ) && b.range.containsPosition( a.position ) ) {\n\t\tconst op = _getComplementaryAttributeOperations( a, b.key, b.newValue );\n\n\t\tif ( op ) {\n\t\t\tresult.push( op );\n\t\t}\n\t}\n\n\t// The default case is: do nothing.\n\t// `AttributeOperation` does not change the model tree structure so `InsertOperation` does not need to be changed.\n\t//\n\treturn result;\n} );\n\nsetTransformation( InsertOperation, InsertOperation, ( a, b, context ) => {\n\t// Case 1:\n\t//\n\t// Two insert operations insert nodes at the same position. Since they are the same, it needs to be decided\n\t// what will be the order of inserted nodes. However, there is no additional information to help in that\n\t// decision. Also, when `b` will be transformed by `a`, the same order must be maintained.\n\t//\n\t// To achieve that, we will check if the operation is strong.\n\t// If it is, it won't get transformed. If it is not, it will be moved.\n\t//\n\tif ( a.position.isEqual( b.position ) && context.aIsStrong ) {\n\t\treturn [ a ];\n\t}\n\n\t// The default case.\n\t//\n\ta.position = a.position._getTransformedByInsertOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( InsertOperation, MoveOperation, ( a, b ) => {\n\t// The default case.\n\t//\n\ta.position = a.position._getTransformedByMoveOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( InsertOperation, SplitOperation, ( a, b ) => {\n\t// The default case.\n\t//\n\ta.position = a.position._getTransformedBySplitOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( InsertOperation, MergeOperation, ( a, b ) => {\n\ta.position = a.position._getTransformedByMergeOperation( b );\n\n\treturn [ a ];\n} );\n\n// -----------------------\n\nsetTransformation( MarkerOperation, InsertOperation, ( a, b ) => {\n\tif ( a.oldRange ) {\n\t\ta.oldRange = a.oldRange._getTransformedByInsertOperation( b )[ 0 ];\n\t}\n\n\tif ( a.newRange ) {\n\t\ta.newRange = a.newRange._getTransformedByInsertOperation( b )[ 0 ];\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( MarkerOperation, MarkerOperation, ( a, b, context ) => {\n\tif ( a.name == b.name ) {\n\t\tif ( context.aIsStrong ) {\n\t\t\ta.oldRange = b.newRange ? b.newRange.clone() : null;\n\t\t} else {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t}\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( MarkerOperation, MergeOperation, ( a, b ) => {\n\tif ( a.oldRange ) {\n\t\ta.oldRange = a.oldRange._getTransformedByMergeOperation( b );\n\t}\n\n\tif ( a.newRange ) {\n\t\ta.newRange = a.newRange._getTransformedByMergeOperation( b );\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( MarkerOperation, MoveOperation, ( a, b, context ) => {\n\tif ( a.oldRange ) {\n\t\ta.oldRange = Range._createFromRanges( a.oldRange._getTransformedByMoveOperation( b ) );\n\t}\n\n\tif ( a.newRange ) {\n\t\tif ( context.abRelation ) {\n\t\t\tconst aNewRange = Range._createFromRanges( a.newRange._getTransformedByMoveOperation( b ) );\n\n\t\t\tif ( context.abRelation.side == 'left' && b.targetPosition.isEqual( a.newRange.start ) ) {\n\t\t\t\t( a.newRange as any ).end = aNewRange.end;\n\t\t\t\t( a.newRange.start as any ).path = context.abRelation.path;\n\n\t\t\t\treturn [ a ];\n\t\t\t} else if ( context.abRelation.side == 'right' && b.targetPosition.isEqual( a.newRange.end ) ) {\n\t\t\t\t( a.newRange as any ).start = aNewRange.start;\n\t\t\t\t( a.newRange.end as any ).path = context.abRelation.path;\n\n\t\t\t\treturn [ a ];\n\t\t\t}\n\t\t}\n\n\t\ta.newRange = Range._createFromRanges( a.newRange._getTransformedByMoveOperation( b ) );\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( MarkerOperation, SplitOperation, ( a, b, context ) => {\n\tif ( a.oldRange ) {\n\t\ta.oldRange = a.oldRange._getTransformedBySplitOperation( b );\n\t}\n\n\tif ( a.newRange ) {\n\t\tif ( context.abRelation ) {\n\t\t\tconst aNewRange = a.newRange._getTransformedBySplitOperation( b );\n\n\t\t\tif ( a.newRange.start.isEqual( b.splitPosition ) && context.abRelation.wasStartBeforeMergedElement ) {\n\t\t\t\t( a.newRange as any ).start = Position._createAt( b.insertionPosition );\n\t\t\t} else if ( a.newRange.start.isEqual( b.splitPosition ) && !context.abRelation.wasInLeftElement ) {\n\t\t\t\t( a.newRange as any ).start = Position._createAt( b.moveTargetPosition );\n\t\t\t}\n\n\t\t\tif ( a.newRange.end.isEqual( b.splitPosition ) && context.abRelation.wasInRightElement ) {\n\t\t\t\t( a.newRange as any ).end = Position._createAt( b.moveTargetPosition );\n\t\t\t} else if ( a.newRange.end.isEqual( b.splitPosition ) && context.abRelation.wasEndBeforeMergedElement ) {\n\t\t\t\t( a.newRange as any ).end = Position._createAt( b.insertionPosition );\n\t\t\t} else {\n\t\t\t\t( a.newRange as any ).end = aNewRange.end;\n\t\t\t}\n\n\t\t\treturn [ a ];\n\t\t}\n\n\t\ta.newRange = a.newRange._getTransformedBySplitOperation( b );\n\t}\n\n\treturn [ a ];\n} );\n\n// -----------------------\n\nsetTransformation( MergeOperation, InsertOperation, ( a, b ) => {\n\tif ( a.sourcePosition.hasSameParentAs( b.position ) ) {\n\t\ta.howMany += b.howMany;\n\t}\n\n\ta.sourcePosition = a.sourcePosition._getTransformedByInsertOperation( b );\n\ta.targetPosition = a.targetPosition._getTransformedByInsertOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( MergeOperation, MergeOperation, ( a, b, context ) => {\n\t// Case 1:\n\t//\n\t// Same merge operations.\n\t//\n\t// Both operations have same source and target positions. So the element already got merged and there is\n\t// theoretically nothing to do.\n\t//\n\tif ( a.sourcePosition.isEqual( b.sourcePosition ) && a.targetPosition.isEqual( b.targetPosition ) ) {\n\t\t// There are two ways that we can provide a do-nothing operation.\n\t\t//\n\t\t// First is simply a NoOperation instance. We will use it if `b` operation was not undone.\n\t\t//\n\t\t// Second is a merge operation that has the source operation in the merged element - in the graveyard -\n\t\t// same target position and `howMany` equal to `0`. So it is basically merging an empty element from graveyard\n\t\t// which is almost the same as NoOperation.\n\t\t//\n\t\t// This way the merge operation can be later transformed by split operation\n\t\t// to provide correct undo. This will be used if `b` operation was undone (only then it is correct).\n\t\t//\n\t\tif ( !context.bWasUndone ) {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t} else {\n\t\t\tconst path = b.graveyardPosition.path.slice();\n\t\t\tpath.push( 0 );\n\n\t\t\ta.sourcePosition = new Position( b.graveyardPosition.root, path );\n\t\t\ta.howMany = 0;\n\n\t\t\treturn [ a ];\n\t\t}\n\t}\n\n\t// Case 2:\n\t//\n\t// Same merge source position but different target position.\n\t//\n\t// This can happen during collaboration. For example, if one client merged a paragraph to the previous paragraph\n\t// and the other person removed that paragraph and merged the same paragraph to something before:\n\t//\n\t// Client A:\n\t// <p>Foo</p><p>Bar</p><p>[]Xyz</p>\n\t// <p>Foo</p><p>BarXyz</p>\n\t//\n\t// Client B:\n\t// <p>Foo</p>[<p>Bar</p>]<p>Xyz</p>\n\t// <p>Foo</p><p>[]Xyz</p>\n\t// <p>FooXyz</p>\n\t//\n\t// In this case we need to decide where finally \"Xyz\" will land:\n\t//\n\t// <p>FooXyz</p> graveyard: <p>Bar</p>\n\t// <p>Foo</p> graveyard: <p>BarXyz</p>\n\t//\n\t// Let's move it in a way so that a merge operation that does not target to graveyard is more important so that\n\t// nodes does not end up in the graveyard. It makes sense. Both for Client A and for Client B \"Xyz\" finally did not\n\t// end up in the graveyard (see above).\n\t//\n\t// If neither or both operations point to graveyard, then let `aIsStrong` decide.\n\t//\n\tif (\n\t\ta.sourcePosition.isEqual( b.sourcePosition ) && !a.targetPosition.isEqual( b.targetPosition ) &&\n\t\t!context.bWasUndone && context.abRelation != 'splitAtSource'\n\t) {\n\t\tconst aToGraveyard = a.targetPosition.root.rootName == '$graveyard';\n\t\tconst bToGraveyard = b.targetPosition.root.rootName == '$graveyard';\n\n\t\t// If `aIsWeak` it means that `a` points to graveyard while `b` doesn't. Don't move nodes then.\n\t\tconst aIsWeak = aToGraveyard && !bToGraveyard;\n\n\t\t// If `bIsWeak` it means that `b` points to graveyard while `a` doesn't. Force moving nodes then.\n\t\tconst bIsWeak = bToGraveyard && !aToGraveyard;\n\n\t\t// Force move if `b` is weak or neither operation is weak but `a` is stronger through `context.aIsStrong`.\n\t\tconst forceMove = bIsWeak || ( !aIsWeak && context.aIsStrong );\n\n\t\tif ( forceMove ) {\n\t\t\tconst sourcePosition = b.targetPosition._getTransformedByMergeOperation( b );\n\t\t\tconst targetPosition = a.targetPosition._getTransformedByMergeOperation( b );\n\n\t\t\treturn [ new MoveOperation( sourcePosition, a.howMany, targetPosition, 0 ) ];\n\t\t} else {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t}\n\t}\n\n\t// The default case.\n\t//\n\tif ( a.sourcePosition.hasSameParentAs( b.targetPosition ) ) {\n\t\ta.howMany += b.howMany;\n\t}\n\n\ta.sourcePosition = a.sourcePosition._getTransformedByMergeOperation( b );\n\ta.targetPosition = a.targetPosition._getTransformedByMergeOperation( b );\n\n\t// Handle positions in graveyard.\n\t// If graveyard positions are same and `a` operation is strong - do not transform.\n\tif ( !a.graveyardPosition.isEqual( b.graveyardPosition ) || !context.aIsStrong ) {\n\t\ta.graveyardPosition = a.graveyardPosition._getTransformedByMergeOperation( b );\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( MergeOperation, MoveOperation, ( a, b, context ) => {\n\t// Case 1:\n\t//\n\t// The element to merge got removed.\n\t//\n\t// Merge operation does support merging elements which are not siblings. So it would not be a problem\n\t// from technical point of view. However, if the element was removed, the intention of the user deleting it\n\t// was to have it all deleted, together with its children. From user experience point of view, moving back the\n\t// removed nodes might be unexpected. This means that in this scenario we will block the merging.\n\t//\n\t// The exception of this rule would be if the remove operation was later undone.\n\t//\n\tconst removedRange = Range._createFromPositionAndShift( b.sourcePosition, b.howMany );\n\n\tif ( b.type == 'remove' && !context.bWasUndone && !context.forceWeakRemove ) {\n\t\tif ( a.deletionPosition.hasSameParentAs( b.sourcePosition ) && removedRange.containsPosition( a.sourcePosition ) ) {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t}\n\t}\n\n\t// The default case.\n\t//\n\tif ( a.sourcePosition.hasSameParentAs( b.targetPosition ) ) {\n\t\ta.howMany += b.howMany;\n\t}\n\n\tif ( a.sourcePosition.hasSameParentAs( b.sourcePosition ) ) {\n\t\ta.howMany -= b.howMany;\n\t}\n\n\ta.sourcePosition = a.sourcePosition._getTransformedByMoveOperation( b );\n\ta.targetPosition = a.targetPosition._getTransformedByMoveOperation( b );\n\n\t// `MergeOperation` graveyard position is like `MoveOperation` target position. It is a position where element(s) will\n\t// be moved. Like in other similar cases, we need to consider the scenario when those positions are same.\n\t// Here, we will treat `MergeOperation` like it is always strong (see `InsertOperation` x `InsertOperation` for comparison).\n\t// This means that we won't transform graveyard position if it is equal to move operation target position.\n\tif ( !a.graveyardPosition.isEqual( b.targetPosition ) ) {\n\t\ta.graveyardPosition = a.graveyardPosition._getTransformedByMoveOperation( b );\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( MergeOperation, SplitOperation, ( a, b, context ) => {\n\tif ( b.graveyardPosition ) {\n\t\t// If `b` operation defines graveyard position, a node from graveyard will be moved. This means that we need to\n\t\t// transform `a.graveyardPosition` accordingly.\n\t\ta.graveyardPosition = a.graveyardPosition._getTransformedByDeletion( b.graveyardPosition, 1 )!;\n\n\t\t// This is a scenario foreseen in `MergeOperation` x `MergeOperation`, with two identical merge operations.\n\t\t//\n\t\t// So, there was `MergeOperation` x `MergeOperation` transformation earlier. Now, `a` is a merge operation which\n\t\t// source position is in graveyard. Interestingly, split operation wants to use the node to be merged by `a`. This\n\t\t// means that `b` is undoing that merge operation from earlier, which caused `a` to be in graveyard.\n\t\t//\n\t\t// If that's the case, at this point, we will only \"fix\" `a.howMany`. It was earlier set to `0` in\n\t\t// `MergeOperation` x `MergeOperation` transformation. Later transformations in this function will change other\n\t\t// properties.\n\t\t//\n\t\tif ( a.deletionPosition.isEqual( b.graveyardPosition ) ) {\n\t\t\ta.howMany = b.howMany;\n\t\t}\n\t}\n\n\t// Case 1:\n\t//\n\t// Merge operation moves nodes to the place where split happens.\n\t// This is a classic situation when there are two paragraphs, and there is a split (enter) after the first\n\t// paragraph and there is a merge (delete) at the beginning of the second paragraph:\n\t//\n\t// <p>Foo{}</p><p>[]Bar</p>.\n\t//\n\t// Split is after `Foo`, while merge is from `Bar` to the end of `Foo`.\n\t//\n\t// State after split:\n\t// <p>Foo</p><p></p><p>Bar</p>\n\t//\n\t// Now, `Bar` should be merged to the new paragraph:\n\t// <p>Foo</p><p>Bar</p>\n\t//\n\t// Instead of merging it to the original paragraph:\n\t// <p>FooBar</p><p></p>\n\t//\n\t// This means that `targetPosition` needs to be transformed. This is the default case though.\n\t// For example, if the split would be after `F`, `targetPosition` should also be transformed.\n\t//\n\t// There are three exceptions, though, when we want to keep `targetPosition` as it was.\n\t//\n\t// First exception is when the merge target position is inside an element (not at the end, as usual). This\n\t// happens when the merge operation earlier was transformed by \"the same\" merge operation. If merge operation\n\t// targets inside the element we want to keep the original target position (and not transform it) because\n\t// we have additional context telling us that we want to merge to the original element. We can check if the\n\t// merge operation points inside element by checking what is `SplitOperation#howMany`. Since merge target position\n\t// is same as split position, if `howMany` is non-zero, it means that the merge target position is inside an element.\n\t//\n\t// Second exception is when the element to merge is in the graveyard and split operation uses it. In that case\n\t// if target position would be transformed, the merge operation would target at the source position:\n\t//\n\t// root: <p>Foo</p>\t\t\t\tgraveyard: <p></p>\n\t//\n\t// SplitOperation: root [ 0, 3 ] using graveyard [ 0 ] (howMany = 0)\n\t// MergeOperation: graveyard [ 0, 0 ] -> root [ 0, 3 ] (howMany = 0)\n\t//\n\t// Since split operation moves the graveyard node back to the root, the merge operation source position changes.\n\t// We would like to merge from the empty <p> to the \"Foo\" <p>:\n\t//\n\t// root: <p>Foo</p><p></p>\t\t\tgraveyard:\n\t//\n\t// MergeOperation#sourcePosition = root [ 1, 0 ]\n\t//\n\t// If `targetPosition` is transformed, it would become root [ 1, 0 ] as well. It has to be kept as it was.\n\t//\n\t// Third exception is connected with relations. If this happens during undo and we have explicit information\n\t// that target position has not been affected by the operation which is undone by this split then this split should\n\t// not move the target position either.\n\t//\n\tif ( a.targetPosition.isEqual( b.splitPosition ) ) {\n\t\tconst mergeInside = b.howMany != 0;\n\t\tconst mergeSplittingElement = b.graveyardPosition && a.deletionPosition.isEqual( b.graveyardPosition );\n\n\t\tif ( mergeInside || mergeSplittingElement || context.abRelation == 'mergeTargetNotMoved' ) {\n\t\t\ta.sourcePosition = a.sourcePosition._getTransformedBySplitOperation( b );\n\n\t\t\treturn [ a ];\n\t\t}\n\t}\n\n\t// Case 2:\n\t//\n\t// Merge source is at the same position as split position. This sometimes happen, mostly during undo.\n\t// The decision here is mostly to choose whether merge source position should stay where it is (so it will be at the end of the\n\t// split element) or should be move to the beginning of the new element.\n\t//\n\tif ( a.sourcePosition.isEqual( b.splitPosition ) ) {\n\t\t// Use context to check if `SplitOperation` is not undoing a merge operation, that didn't change the `a` operation.\n\t\t// This scenario happens the undone merge operation moved nodes at the source position of `a` operation.\n\t\t// In that case `a` operation source position should stay where it is.\n\t\tif ( context.abRelation == 'mergeSourceNotMoved' ) {\n\t\t\ta.howMany = 0;\n\t\t\ta.targetPosition = a.targetPosition._getTransformedBySplitOperation( b );\n\n\t\t\treturn [ a ];\n\t\t}\n\n\t\t// This merge operation might have been earlier transformed by a merge operation which both merged the same element.\n\t\t// See that case in `MergeOperation` x `MergeOperation` transformation. In that scenario, if the merge operation has been undone,\n\t\t// the special case is not applied.\n\t\t//\n\t\t// Now, the merge operation is transformed by the split which has undone that previous merge operation.\n\t\t// So now we are fixing situation which was skipped in `MergeOperation` x `MergeOperation` case.\n\t\t//\n\t\tif ( context.abRelation == 'mergeSameElement' || a.sourcePosition.offset > 0 ) {\n\t\t\ta.sourcePosition = b.moveTargetPosition.clone();\n\t\t\ta.targetPosition = a.targetPosition._getTransformedBySplitOperation( b );\n\n\t\t\treturn [ a ];\n\t\t}\n\t}\n\n\t// The default case.\n\t//\n\tif ( a.sourcePosition.hasSameParentAs( b.splitPosition ) ) {\n\t\ta.howMany = b.splitPosition.offset;\n\t}\n\n\ta.sourcePosition = a.sourcePosition._getTransformedBySplitOperation( b );\n\ta.targetPosition = a.targetPosition._getTransformedBySplitOperation( b );\n\n\treturn [ a ];\n} );\n\n// -----------------------\n\nsetTransformation( MoveOperation, InsertOperation, ( a, b ) => {\n\tconst moveRange = Range._createFromPositionAndShift( a.sourcePosition, a.howMany );\n\tconst transformed = moveRange._getTransformedByInsertOperation( b, false )[ 0 ];\n\n\ta.sourcePosition = transformed.start;\n\ta.howMany = transformed.end.offset - transformed.start.offset;\n\n\t// See `InsertOperation` x `MoveOperation` transformation for details on this case.\n\t//\n\t// In summary, both operations point to the same place, so the order of nodes needs to be decided.\n\t// `MoveOperation` is considered weaker, so it is always transformed, unless there was a certain relation\n\t// between operations.\n\t//\n\tif ( !a.targetPosition.isEqual( b.position ) ) {\n\t\ta.targetPosition = a.targetPosition._getTransformedByInsertOperation( b );\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( MoveOperation, MoveOperation, ( a, b, context ) => {\n\t//\n\t// Setting and evaluating some variables that will be used in special cases and default algorithm.\n\t//\n\t// Create ranges from `MoveOperations` properties.\n\tconst rangeA = Range._createFromPositionAndShift( a.sourcePosition, a.howMany );\n\tconst rangeB = Range._createFromPositionAndShift( b.sourcePosition, b.howMany );\n\n\t// Assign `context.aIsStrong` to a different variable, because the value may change during execution of\n\t// this algorithm and we do not want to override original `context.aIsStrong` that will be used in later transformations.\n\tlet aIsStrong = context.aIsStrong;\n\n\t// This will be used to decide the order of nodes if both operations target at the same position.\n\t// By default, use strong/weak operation mechanism.\n\tlet insertBefore = !context.aIsStrong;\n\n\t// If the relation is set, then use it to decide nodes order.\n\tif ( context.abRelation == 'insertBefore' || context.baRelation == 'insertAfter' ) {\n\t\tinsertBefore = true;\n\t} else if ( context.abRelation == 'insertAfter' || context.baRelation == 'insertBefore' ) {\n\t\tinsertBefore = false;\n\t}\n\n\t// `a.targetPosition` could be affected by the `b` operation. We will transform it.\n\tlet newTargetPosition;\n\n\tif ( a.targetPosition.isEqual( b.targetPosition ) && insertBefore ) {\n\t\tnewTargetPosition = a.targetPosition._getTransformedByDeletion(\n\t\t\tb.sourcePosition,\n\t\t\tb.howMany\n\t\t);\n\t} else {\n\t\tnewTargetPosition = a.targetPosition._getTransformedByMove(\n\t\t\tb.sourcePosition,\n\t\t\tb.targetPosition,\n\t\t\tb.howMany\n\t\t);\n\t}\n\n\t//\n\t// Special case #1 + mirror.\n\t//\n\t// Special case when both move operations' target positions are inside nodes that are\n\t// being moved by the other move operation. So in other words, we move ranges into inside of each other.\n\t// This case can't be solved reasonably (on the other hand, it should not happen often).\n\tif ( _moveTargetIntoMovedRange( a, b ) && _moveTargetIntoMovedRange( b, a ) ) {\n\t\t// Instead of transforming operation, we return a reverse of the operation that we transform by.\n\t\t// So when the results of this \"transformation\" will be applied, `b` MoveOperation will get reversed.\n\t\treturn [ b.getReversed() ];\n\t}\n\t//\n\t// End of special case #1.\n\t//\n\n\t//\n\t// Special case #2.\n\t//\n\t// Check if `b` operation targets inside `rangeA`.\n\tconst bTargetsToA = rangeA.containsPosition( b.targetPosition );\n\n\t// If `b` targets to `rangeA` and `rangeA` contains `rangeB`, `b` operation has no influence on `a` operation.\n\t// You might say that operation `b` is captured inside operation `a`.\n\tif ( bTargetsToA && rangeA.containsRange( rangeB, true ) ) {\n\t\t// There is a mini-special case here, where `rangeB` is on other level than `rangeA`. That's why\n\t\t// we need to transform `a` operation anyway.\n\t\t( rangeA as any ).start = rangeA.start._getTransformedByMove( b.sourcePosition, b.targetPosition, b.howMany );\n\t\t( rangeA as any ).end = rangeA.end._getTransformedByMove( b.sourcePosition, b.targetPosition, b.howMany );\n\n\t\treturn _makeMoveOperationsFromRanges( [ rangeA ], newTargetPosition! );\n\t}\n\n\t//\n\t// Special case #2 mirror.\n\t//\n\tconst aTargetsToB = rangeB.containsPosition( a.targetPosition );\n\n\tif ( aTargetsToB && rangeB.containsRange( rangeA, true ) ) {\n\t\t// `a` operation is \"moved together\" with `b` operation.\n\t\t// Here, just move `rangeA` \"inside\" `rangeB`.\n\t\t( rangeA as any ).start = rangeA.start._getCombined( b.sourcePosition, b.getMovedRangeStart() );\n\t\t( rangeA as any ).end = rangeA.end._getCombined( b.sourcePosition, b.getMovedRangeStart() );\n\n\t\treturn _makeMoveOperationsFromRanges( [ rangeA ], newTargetPosition! );\n\t}\n\t//\n\t// End of special case #2.\n\t//\n\n\t//\n\t// Special case #3 + mirror.\n\t//\n\t// `rangeA` has a node which is an ancestor of `rangeB`. In other words, `rangeB` is inside `rangeA`\n\t// but not on the same tree level. In such case ranges have common part but we have to treat it\n\t// differently, because in such case those ranges are not really conflicting and should be treated like\n\t// two separate ranges. Also we have to discard two difference parts.\n\tconst aCompB = compareArrays( a.sourcePosition.getParentPath(), b.sourcePosition.getParentPath() );\n\n\tif ( aCompB == 'prefix' || aCompB == 'extension' ) {\n\t\t// Transform `rangeA` by `b` operation and make operation out of it, and that's all.\n\t\t// Note that this is a simplified version of default case, but here we treat the common part (whole `rangeA`)\n\t\t// like a one difference part.\n\t\t( rangeA as any ).start = rangeA.start._getTransformedByMove( b.sourcePosition, b.targetPosition, b.howMany );\n\t\t( rangeA as any ).end = rangeA.end._getTransformedByMove( b.sourcePosition, b.targetPosition, b.howMany );\n\n\t\treturn _makeMoveOperationsFromRanges( [ rangeA ], newTargetPosition! );\n\t}\n\t//\n\t// End of special case #3.\n\t//\n\n\t//\n\t// Default case - ranges are on the same level or are not connected with each other.\n\t//\n\t// Modifier for default case.\n\t// Modifies `aIsStrong` flag in certain conditions.\n\t//\n\t// If only one of operations is a remove operation, we force remove operation to be the \"stronger\" one\n\t// to provide more expected results.\n\tif ( a.type == 'remove' && b.type != 'remove' && !context.aWasUndone && !context.forceWeakRemove ) {\n\t\taIsStrong = true;\n\t} else if ( a.type != 'remove' && b.type == 'remove' && !context.bWasUndone && !context.forceWeakRemove ) {\n\t\taIsStrong = false;\n\t}\n\n\t// Handle operation's source ranges - check how `rangeA` is affected by `b` operation.\n\t// This will aggregate transformed ranges.\n\tconst ranges = [];\n\n\t// Get the \"difference part\" of `a` operation source range.\n\t// This is an array with one or two ranges. Two ranges if `rangeB` is inside `rangeA`.\n\tconst difference = rangeA.getDifference( rangeB );\n\n\tfor ( const range of difference ) {\n\t\t// Transform those ranges by `b` operation. For example if `b` moved range from before those ranges, fix those ranges.\n\t\t( range as any ).start = range.start._getTransformedByDeletion( b.sourcePosition, b.howMany );\n\t\t( range as any ).end = range.end._getTransformedByDeletion( b.sourcePosition, b.howMany );\n\n\t\t// If `b` operation targets into `rangeA` on the same level, spread `rangeA` into two ranges.\n\t\tconst shouldSpread = compareArrays( range.start.getParentPath(), b.getMovedRangeStart().getParentPath() ) == 'same';\n\t\tconst newRanges = range._getTransformedByInsertion( b.getMovedRangeStart(), b.howMany, shouldSpread );\n\n\t\tranges.push( ...newRanges );\n\t}\n\n\t// Then, we have to manage the \"common part\" of both move ranges.\n\tconst common = rangeA.getIntersection( rangeB );\n\n\tif ( common !== null && aIsStrong ) {\n\t\t// Calculate the new position of that part of original range.\n\t\t( common as any ).start = common.start._getCombined( b.sourcePosition, b.getMovedRangeStart() );\n\t\t( common as any ).end = common.end._getCombined( b.sourcePosition, b.getMovedRangeStart() );\n\n\t\t// Take care of proper range order.\n\t\t//\n\t\t// Put `common` at appropriate place. Keep in mind that we are interested in original order.\n\t\t// Basically there are only three cases: there is zero, one or two difference ranges.\n\t\t//\n\t\t// If there is zero difference ranges, just push `common` in the array.\n\t\tif ( ranges.length === 0 ) {\n\t\t\tranges.push( common );\n\t\t}\n\t\t// If there is one difference range, we need to check whether common part was before it or after it.\n\t\telse if ( ranges.length == 1 ) {\n\t\t\tif ( rangeB.start.isBefore( rangeA.start ) || rangeB.start.isEqual( rangeA.start ) ) {\n\t\t\t\tranges.unshift( common );\n\t\t\t} else {\n\t\t\t\tranges.push( common );\n\t\t\t}\n\t\t}\n\t\t// If there are more ranges (which means two), put common part between them. This is the only scenario\n\t\t// where there could be two difference ranges so we don't have to make any comparisons.\n\t\telse {\n\t\t\tranges.splice( 1, 0, common );\n\t\t}\n\t}\n\n\tif ( ranges.length === 0 ) {\n\t\t// If there are no \"source ranges\", nothing should be changed.\n\t\t// Note that this can happen only if `aIsStrong == false` and `rangeA.isEqual( rangeB )`.\n\t\treturn [ new NoOperation( a.baseVersion ) ];\n\t}\n\n\treturn _makeMoveOperationsFromRanges( ranges, newTargetPosition! );\n} );\n\nsetTransformation( MoveOperation, SplitOperation, ( a, b, context ) => {\n\tlet newTargetPosition = a.targetPosition.clone();\n\n\t// Do not transform if target position is same as split insertion position and this split comes from undo.\n\t// This should be done on relations but it is too much work for now as it would require relations working in collaboration.\n\t// We need to make a decision how we will resolve such conflict and this is less harmful way.\n\tif ( !a.targetPosition.isEqual( b.insertionPosition ) || !b.graveyardPosition || context.abRelation == 'moveTargetAfter' ) {\n\t\tnewTargetPosition = a.targetPosition._getTransformedBySplitOperation( b );\n\t}\n\n\t// Case 1:\n\t//\n\t// Last element in the moved range got split.\n\t//\n\t// In this case the default range transformation will not work correctly as the element created by\n\t// split operation would be outside the range. The range to move needs to be fixed manually.\n\t//\n\tconst moveRange = Range._createFromPositionAndShift( a.sourcePosition, a.howMany );\n\n\tif ( moveRange.end.isEqual( b.insertionPosition ) ) {\n\t\t// Do it only if this is a \"natural\" split, not a one that comes from undo.\n\t\t// If this is undo split, only `targetPosition` needs to be changed (if the move is a remove).\n\t\tif ( !b.graveyardPosition ) {\n\t\t\ta.howMany++;\n\t\t}\n\n\t\ta.targetPosition = newTargetPosition;\n\n\t\treturn [ a ];\n\t}\n\n\t// Case 2:\n\t//\n\t// Split happened between the moved nodes. In this case two ranges to move need to be generated.\n\t//\n\t// Characters `ozba` are moved to the end of paragraph `Xyz` but split happened.\n\t// <p>F[oz|ba]r</p><p>Xyz</p>\n\t//\n\t// After split:\n\t// <p>F[oz</p><p>ba]r</p><p>Xyz</p>\n\t//\n\t// Correct ranges:\n\t// <p>F[oz]</p><p>[ba]r</p><p>Xyz</p>\n\t//\n\t// After move:\n\t// <p>F</p><p>r</p><p>Xyzozba</p>\n\t//\n\tif ( moveRange.start.hasSameParentAs( b.splitPosition ) && moveRange.containsPosition( b.splitPosition ) ) {\n\t\tlet rightRange = new Range( b.splitPosition, moveRange.end );\n\t\trightRange = rightRange._getTransformedBySplitOperation( b );\n\n\t\tconst ranges = [\n\t\t\tnew Range( moveRange.start, b.splitPosition ),\n\t\t\trightRange\n\t\t];\n\n\t\treturn _makeMoveOperationsFromRanges( ranges, newTargetPosition );\n\t}\n\n\t// Case 3:\n\t//\n\t// Move operation targets at the split position. We need to decide if the nodes should be inserted\n\t// at the end of the split element or at the beginning of the new element.\n\t//\n\tif ( a.targetPosition.isEqual( b.splitPosition ) && context.abRelation == 'insertAtSource' ) {\n\t\tnewTargetPosition = b.moveTargetPosition;\n\t}\n\n\t// Case 4:\n\t//\n\t// Move operation targets just after the split element. We need to decide if the nodes should be inserted\n\t// between two parts of split element, or after the new element.\n\t//\n\t// Split at `|`, while move operation moves `<p>Xyz</p>` and targets at `^`:\n\t// <p>Foo|bar</p>^<p>baz</p>\n\t// <p>Foo</p>^<p>bar</p><p>baz</p> or <p>Foo</p><p>bar</p>^<p>baz</p>?\n\t//\n\t// If there is no contextual information between operations (for example, they come from collaborative\n\t// editing), we don't want to put some unrelated content (move) between parts of related content (split parts).\n\t// However, if the split is from undo, in the past, the moved content might be targeting between the\n\t// split parts, meaning that was exactly user's intention:\n\t//\n\t// <p>Foo</p>^<p>bar</p>\t\t<--- original situation, in \"past\".\n\t// <p>Foobar</p>^\t\t\t\t<--- after merge target position is transformed.\n\t// <p>Foo|bar</p>^\t\t\t\t<--- then the merge is undone, and split happens, which leads us to current situation.\n\t//\n\t// In this case it is pretty clear that the intention was to put new paragraph between those nodes,\n\t// so we need to transform accordingly. We can detect this scenario thanks to relations.\n\t//\n\tif ( a.targetPosition.isEqual( b.insertionPosition ) && context.abRelation == 'insertBetween' ) {\n\t\tnewTargetPosition = a.targetPosition;\n\t}\n\n\t// The default case.\n\t//\n\tconst transformed = moveRange._getTransformedBySplitOperation( b );\n\tconst ranges = [ transformed ];\n\n\t// Case 5:\n\t//\n\t// Moved range contains graveyard element used by split operation. Add extra move operation to the result.\n\t//\n\tif ( b.graveyardPosition ) {\n\t\tconst movesGraveyardElement = moveRange.start.isEqual( b.graveyardPosition ) || moveRange.containsPosition( b.graveyardPosition );\n\n\t\tif ( a.howMany > 1 && movesGraveyardElement && !context.aWasUndone ) {\n\t\t\tranges.push( Range._createFromPositionAndShift( b.insertionPosition, 1 ) );\n\t\t}\n\t}\n\n\treturn _makeMoveOperationsFromRanges( ranges, newTargetPosition );\n} );\n\nsetTransformation( MoveOperation, MergeOperation, ( a, b, context ) => {\n\tconst movedRange = Range._createFromPositionAndShift( a.sourcePosition, a.howMany );\n\n\tif ( b.deletionPosition.hasSameParentAs( a.sourcePosition ) && movedRange.containsPosition( b.sourcePosition ) ) {\n\t\tif ( a.type == 'remove' && !context.forceWeakRemove ) {\n\t\t\t// Case 1:\n\t\t\t//\n\t\t\t// The element to remove got merged.\n\t\t\t//\n\t\t\t// Merge operation does support merging elements which are not siblings. So it would not be a problem\n\t\t\t// from technical point of view. However, if the element was removed, the intention of the user\n\t\t\t// deleting it was to have it all deleted. From user experience point of view, moving back the\n\t\t\t// removed nodes might be unexpected. This means that in this scenario we will reverse merging and remove the element.\n\t\t\t//\n\t\t\tif ( !context.aWasUndone ) {\n\t\t\t\tconst results = [];\n\n\t\t\t\tlet gyMoveSource = b.graveyardPosition.clone();\n\t\t\t\tlet splitNodesMoveSource = b.targetPosition._getTransformedByMergeOperation( b );\n\n\t\t\t\tif ( a.howMany > 1 ) {\n\t\t\t\t\tresults.push( new MoveOperation( a.sourcePosition, a.howMany - 1, a.targetPosition, 0 ) );\n\n\t\t\t\t\tgyMoveSource = gyMoveSource._getTransformedByMove( a.sourcePosition, a.targetPosition, a.howMany - 1 );\n\t\t\t\t\tsplitNodesMoveSource = splitNodesMoveSource._getTransformedByMove( a.sourcePosition, a.targetPosition, a.howMany - 1 );\n\t\t\t\t}\n\n\t\t\t\tconst gyMoveTarget = b.deletionPosition._getCombined( a.sourcePosition, a.targetPosition );\n\t\t\t\tconst gyMove = new MoveOperation( gyMoveSource, 1, gyMoveTarget, 0 );\n\n\t\t\t\tconst splitNodesMoveTargetPath = gyMove.getMovedRangeStart().path.slice();\n\t\t\t\tsplitNodesMoveTargetPath.push( 0 );\n\n\t\t\t\tconst splitNodesMoveTarget = new Position( gyMove.targetPosition.root, splitNodesMoveTargetPath );\n\t\t\t\tsplitNodesMoveSource = splitNodesMoveSource._getTransformedByMove( gyMoveSource, gyMoveTarget, 1 );\n\t\t\t\tconst splitNodesMove = new MoveOperation( splitNodesMoveSource, b.howMany, splitNodesMoveTarget, 0 );\n\n\t\t\t\tresults.push( gyMove );\n\t\t\t\tresults.push( splitNodesMove );\n\n\t\t\t\treturn results;\n\t\t\t}\n\t\t} else {\n\t\t\t// Case 2:\n\t\t\t//\n\t\t\t// The element to move got merged and it was the only element to move.\n\t\t\t// In this case just don't do anything, leave the node in the graveyard. Without special case\n\t\t\t// it would be a move operation that moves 0 nodes, so maybe it is better just to return no-op.\n\t\t\t//\n\t\t\tif ( a.howMany == 1 ) {\n\t\t\t\tif ( !context.bWasUndone ) {\n\t\t\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t\t\t} else {\n\t\t\t\t\ta.sourcePosition = b.graveyardPosition.clone();\n\t\t\t\t\ta.targetPosition = a.targetPosition._getTransformedByMergeOperation( b );\n\n\t\t\t\t\treturn [ a ];\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t// The default case.\n\t//\n\tconst moveRange = Range._createFromPositionAndShift( a.sourcePosition, a.howMany );\n\tconst transformed = moveRange._getTransformedByMergeOperation( b );\n\n\ta.sourcePosition = transformed.start;\n\ta.howMany = transformed.end.offset - transformed.start.offset;\n\ta.targetPosition = a.targetPosition._getTransformedByMergeOperation( b );\n\n\treturn [ a ];\n} );\n\n// -----------------------\n\nsetTransformation( RenameOperation, InsertOperation, ( a, b ) => {\n\ta.position = a.position._getTransformedByInsertOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( RenameOperation, MergeOperation, ( a, b ) => {\n\t// Case 1:\n\t//\n\t// Element to rename got merged, so it was moved to `b.graveyardPosition`.\n\t//\n\tif ( a.position.isEqual( b.deletionPosition ) ) {\n\t\ta.position = b.graveyardPosition.clone();\n\t\ta.position.stickiness = 'toNext';\n\n\t\treturn [ a ];\n\t}\n\n\ta.position = a.position._getTransformedByMergeOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( RenameOperation, MoveOperation, ( a, b ) => {\n\ta.position = a.position._getTransformedByMoveOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( RenameOperation, RenameOperation, ( a, b, context ) => {\n\tif ( a.position.isEqual( b.position ) ) {\n\t\tif ( context.aIsStrong ) {\n\t\t\ta.oldName = b.newName;\n\t\t} else {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t}\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( RenameOperation, SplitOperation, ( a, b ) => {\n\t// Case 1:\n\t//\n\t// The element to rename has been split. In this case, the new element should be also renamed.\n\t//\n\t// User decides to change the paragraph to a list item:\n\t// <paragraph>Foobar</paragraph>\n\t//\n\t// However, in meantime, split happens:\n\t// <paragraph>Foo</paragraph><paragraph>bar</paragraph>\n\t//\n\t// As a result, rename both elements:\n\t// <listItem>Foo</listItem><listItem>bar</listItem>\n\t//\n\tconst renamePath = a.position.path;\n\tconst splitPath = b.splitPosition.getParentPath();\n\n\tif ( compareArrays( renamePath, splitPath ) == 'same' && !b.graveyardPosition ) {\n\t\tconst extraRename = new RenameOperation( a.position.getShiftedBy( 1 ), a.oldName, a.newName, 0 );\n\n\t\treturn [ a, extraRename ];\n\t}\n\n\t// The default case.\n\t//\n\ta.position = a.position._getTransformedBySplitOperation( b );\n\n\treturn [ a ];\n} );\n\n// -----------------------\n\nsetTransformation( RootAttributeOperation, RootAttributeOperation, ( a, b, context ) => {\n\tif ( a.root === b.root && a.key === b.key ) {\n\t\tif ( !context.aIsStrong || a.newValue === b.newValue ) {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t} else {\n\t\t\ta.oldValue = b.newValue;\n\t\t}\n\t}\n\n\treturn [ a ];\n} );\n\n// -----------------------\n\nsetTransformation( RootOperation, RootOperation, ( a, b ) => {\n\tif ( a.rootName === b.rootName && a.isAdd === b.isAdd ) {\n\t\treturn [ new NoOperation( 0 ) ];\n\t}\n\n\treturn [ a ];\n} );\n\n// -----------------------\n\nsetTransformation( SplitOperation, InsertOperation, ( a, b ) => {\n\t// The default case.\n\t//\n\tif ( a.splitPosition.hasSameParentAs( b.position ) && a.splitPosition.offset < b.position.offset ) {\n\t\ta.howMany += b.howMany;\n\t}\n\n\ta.splitPosition = a.splitPosition._getTransformedByInsertOperation( b );\n\ta.insertionPosition = a.insertionPosition._getTransformedByInsertOperation( b );\n\n\treturn [ a ];\n} );\n\nsetTransformation( SplitOperation, MergeOperation, ( a, b, context ) => {\n\t// Case 1:\n\t//\n\t// Split element got merged. If two different elements were merged, clients will have different content.\n\t//\n\t// Example. Merge at `{}`, split at `[]`:\n\t// <heading>Foo</heading>{}<paragraph>B[]ar</paragraph>\n\t//\n\t// On merge side it will look like this:\n\t// <heading>FooB[]ar</heading>\n\t// <heading>FooB</heading><heading>ar</heading>\n\t//\n\t// On split side it will look like this:\n\t// <heading>Foo</heading>{}<paragraph>B</paragraph><paragraph>ar</paragraph>\n\t// <heading>FooB</heading><paragraph>ar</paragraph>\n\t//\n\t// Clearly, the second element is different for both clients.\n\t//\n\t// We could use the removed merge element from graveyard as a split element but then clients would have a different\n\t// model state (in graveyard), because the split side client would still have an element in graveyard (removed by merge).\n\t//\n\t// To overcome this, in `SplitOperation` x `MergeOperation` transformation we will add additional `SplitOperation`\n\t// in the graveyard, which will actually clone the merged-and-deleted element. Then, that cloned element will be\n\t// used for splitting. Example below.\n\t//\n\t// Original state:\n\t// <heading>Foo</heading>{}<paragraph>B[]ar</paragraph>\n\t//\n\t// Merge side client:\n\t//\n\t// After merge:\n\t// <heading>FooB[]ar</heading> graveyard: <paragraph></paragraph>\n\t//\n\t// Extra split:\n\t// <heading>FooB[]ar</heading> graveyard: <paragraph></paragraph><paragraph></paragraph>\n\t//\n\t// Use the \"cloned\" element from graveyard:\n\t// <heading>FooB</heading><paragraph>ar</paragraph> graveyard: <paragraph></paragraph>\n\t//\n\t// Split side client:\n\t//\n\t// After split:\n\t// <heading>Foo</heading>{}<paragraph>B</paragraph><paragraph>ar</paragraph>\n\t//\n\t// After merge:\n\t// <heading>FooB</heading><paragraph>ar</paragraph> graveyard: <paragraph></paragraph>\n\t//\n\t// This special case scenario only applies if the original split operation clones the split element.\n\t// If the original split operation has `graveyardPosition` set, it all doesn't have sense because split operation\n\t// knows exactly which element it should use. So there would be no original problem with different contents.\n\t//\n\t// Additionally, the special case applies only if the merge wasn't already undone.\n\t//\n\tif ( !a.graveyardPosition && !context.bWasUndone && a.splitPosition.hasSameParentAs( b.sourcePosition ) ) {\n\t\tconst splitPath = b.graveyardPosition.path.slice();\n\t\tsplitPath.push( 0 );\n\n\t\tconst splitPosition = new Position( b.graveyardPosition.root, splitPath );\n\t\tconst insertionPosition = SplitOperation.getInsertionPosition( new Position( b.graveyardPosition.root, splitPath ) );\n\n\t\tconst additionalSplit = new SplitOperation( splitPosition, 0, insertionPosition, null, 0 );\n\n\t\ta.splitPosition = a.splitPosition._getTransformedByMergeOperation( b );\n\t\ta.insertionPosition = SplitOperation.getInsertionPosition( a.splitPosition );\n\t\ta.graveyardPosition = additionalSplit.insertionPosition.clone();\n\t\ta.graveyardPosition.stickiness = 'toNext';\n\n\t\treturn [ additionalSplit, a ];\n\t}\n\n\t// The default case.\n\t//\n\tif ( a.splitPosition.hasSameParentAs( b.deletionPosition ) && !a.splitPosition.isAfter( b.deletionPosition ) ) {\n\t\ta.howMany--;\n\t}\n\n\tif ( a.splitPosition.hasSameParentAs( b.targetPosition ) ) {\n\t\ta.howMany += b.howMany;\n\t}\n\n\ta.splitPosition = a.splitPosition._getTransformedByMergeOperation( b );\n\ta.insertionPosition = SplitOperation.getInsertionPosition( a.splitPosition );\n\n\tif ( a.graveyardPosition ) {\n\t\ta.graveyardPosition = a.graveyardPosition._getTransformedByMergeOperation( b );\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( SplitOperation, MoveOperation, ( a, b, context ) => {\n\tconst rangeToMove = Range._createFromPositionAndShift( b.sourcePosition, b.howMany );\n\n\tif ( a.graveyardPosition ) {\n\t\t// Case 1:\n\t\t//\n\t\t// Split operation graveyard node was moved. In this case move operation is stronger. Since graveyard element\n\t\t// is already moved to the correct position, we need to only move the nodes after the split position.\n\t\t// This will be done by `MoveOperation` instead of `SplitOperation`.\n\t\t//\n\t\tconst gyElementMoved = rangeToMove.start.isEqual( a.graveyardPosition ) || rangeToMove.containsPosition( a.graveyardPosition );\n\n\t\tif ( !context.bWasUndone && gyElementMoved ) {\n\t\t\tconst sourcePosition = a.splitPosition._getTransformedByMoveOperation( b );\n\n\t\t\tconst newParentPosition = a.graveyardPosition._getTransformedByMoveOperation( b );\n\t\t\tconst newTargetPath = newParentPosition.path.slice();\n\t\t\tnewTargetPath.push( 0 );\n\n\t\t\tconst newTargetPosition = new Position( newParentPosition.root, newTargetPath );\n\t\t\tconst moveOp = new MoveOperation( sourcePosition, a.howMany, newTargetPosition, 0 );\n\n\t\t\treturn [ moveOp ];\n\t\t}\n\n\t\ta.graveyardPosition = a.graveyardPosition._getTransformedByMoveOperation( b );\n\t}\n\n\t// Case 2:\n\t//\n\t// Split is at a position where nodes were moved.\n\t//\n\t// This is a scenario described in `MoveOperation` x `SplitOperation` transformation but from the\n\t// \"split operation point of view\".\n\t//\n\tconst splitAtTarget = a.splitPosition.isEqual( b.targetPosition );\n\n\tif ( splitAtTarget && ( context.baRelation == 'insertAtSource' || context.abRelation == 'splitBefore' ) ) {\n\t\ta.howMany += b.howMany;\n\t\ta.splitPosition = a.splitPosition._getTransformedByDeletion( b.sourcePosition, b.howMany )!;\n\t\ta.insertionPosition = SplitOperation.getInsertionPosition( a.splitPosition );\n\n\t\treturn [ a ];\n\t}\n\n\tif ( splitAtTarget && context.abRelation && context.abRelation.howMany ) {\n\t\tconst { howMany, offset } = context.abRelation;\n\n\t\ta.howMany += howMany;\n\t\ta.splitPosition = a.splitPosition.getShiftedBy( offset );\n\n\t\treturn [ a ];\n\t}\n\n\t// Case 3:\n\t//\n\t// If the split position is inside the moved range, we need to shift the split position to a proper place.\n\t// The position cannot be moved together with moved range because that would result in splitting of an incorrect element.\n\t//\n\t// Characters `bc` should be moved to the second paragraph while split position is between them:\n\t// <paragraph>A[b|c]d</paragraph><paragraph>Xyz</paragraph>\n\t//\n\t// After move, new split position is incorrect:\n\t// <paragraph>Ad</paragraph><paragraph>Xb|cyz</paragraph>\n\t//\n\t// Correct split position:\n\t// <paragraph>A|d</paragraph><paragraph>Xbcyz</paragraph>\n\t//\n\t// After split:\n\t// <paragraph>A</paragraph><paragraph>d</paragraph><paragraph>Xbcyz</paragraph>\n\t//\n\tif ( a.splitPosition.hasSameParentAs( b.sourcePosition ) && rangeToMove.containsPosition( a.splitPosition ) ) {\n\t\tconst howManyRemoved = b.howMany - ( a.splitPosition.offset - b.sourcePosition.offset );\n\t\ta.howMany -= howManyRemoved;\n\n\t\tif ( a.splitPosition.hasSameParentAs( b.targetPosition ) && a.splitPosition.offset < b.targetPosition.offset ) {\n\t\t\ta.howMany += b.howMany;\n\t\t}\n\n\t\ta.splitPosition = b.sourcePosition.clone();\n\t\ta.insertionPosition = SplitOperation.getInsertionPosition( a.splitPosition );\n\n\t\treturn [ a ];\n\t}\n\n\t// The default case.\n\t// Don't change `howMany` if move operation does not really move anything.\n\t//\n\tif ( !b.sourcePosition.isEqual( b.targetPosition ) ) {\n\t\tif ( a.splitPosition.hasSameParentAs( b.sourcePosition ) && a.splitPosition.offset <= b.sourcePosition.offset ) {\n\t\t\ta.howMany -= b.howMany;\n\t\t}\n\n\t\tif ( a.splitPosition.hasSameParentAs( b.targetPosition ) && a.splitPosition.offset < b.targetPosition.offset ) {\n\t\t\ta.howMany += b.howMany;\n\t\t}\n\t}\n\n\t// Change position stickiness to force a correct transformation.\n\ta.splitPosition.stickiness = 'toNone';\n\ta.splitPosition = a.splitPosition._getTransformedByMoveOperation( b );\n\ta.splitPosition.stickiness = 'toNext';\n\n\tif ( a.graveyardPosition ) {\n\t\ta.insertionPosition = a.insertionPosition._getTransformedByMoveOperation( b );\n\t} else {\n\t\ta.insertionPosition = SplitOperation.getInsertionPosition( a.splitPosition );\n\t}\n\n\treturn [ a ];\n} );\n\nsetTransformation( SplitOperation, SplitOperation, ( a, b, context ) => {\n\t// Case 1:\n\t//\n\t// Split at the same position.\n\t//\n\t// If there already was a split at the same position as in `a` operation, it means that the intention\n\t// conveyed by `a` operation has already been fulfilled and `a` should not do anything (to avoid double split).\n\t//\n\t// However, there is a difference if these are new splits or splits created by undo. These have different\n\t// intentions. Also splits moving back different elements from graveyard have different intentions. They\n\t// are just different operations.\n\t//\n\t// So we cancel split operation only if it was really identical.\n\t//\n\t// Also, there is additional case, where split operations aren't identical and should not be cancelled, however the\n\t// default transformation is incorrect too.\n\t//\n\tif ( a.splitPosition.isEqual( b.splitPosition ) ) {\n\t\tif ( !a.graveyardPosition && !b.graveyardPosition ) {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t}\n\n\t\tif ( a.graveyardPosition && b.graveyardPosition && a.graveyardPosition.isEqual( b.graveyardPosition ) ) {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t}\n\n\t\t// Use context to know that the `a.splitPosition` should stay where it is.\n\t\t// This happens during undo when first a merge operation moved nodes to `a.splitPosition` and now `b` operation undoes that merge.\n\t\tif ( context.abRelation == 'splitBefore' ) {\n\t\t\t// Since split is at the same position, there are no nodes left to split.\n\t\t\ta.howMany = 0;\n\n\t\t\t// Note: there was `if ( a.graveyardPosition )` here but it was uncovered in tests and I couldn't find any scenarios for now.\n\t\t\t// That would have to be a `SplitOperation` that didn't come from undo but is transformed by operations that were undone.\n\t\t\t// It could happen if `context` is enabled in collaboration.\n\t\t\ta.graveyardPosition = a.graveyardPosition!._getTransformedBySplitOperation( b );\n\n\t\t\treturn [ a ];\n\t\t}\n\t}\n\n\t// Case 2:\n\t//\n\t// Same node is using to split different elements. This happens in undo when previously same element was merged to\n\t// two different elements. This is described in `MergeOperation` x `MergeOperation` transformation.\n\t//\n\t// In this case we will follow the same logic. We will assume that `insertionPosition` is same for both\n\t// split operations. This might not always be true but in the real cases that were experienced it was. After all,\n\t// if these splits are reverses of merge operations that were merging the same element, then the `insertionPosition`\n\t// should be same for both of those splits.\n\t//\n\t// Again, we will decide which operation is stronger by checking if split happens in graveyard or in non-graveyard root.\n\t//\n\tif ( a.graveyardPosition && b.graveyardPosition && a.graveyardPosition.isEqual( b.graveyardPosition ) ) {\n\t\tconst aInGraveyard = a.splitPosition.root.rootName == '$graveyard';\n\t\tconst bInGraveyard = b.splitPosition.root.rootName == '$graveyard';\n\n\t\t// If `aIsWeak` it means that `a` points to graveyard while `b` doesn't. Don't move nodes then.\n\t\tconst aIsWeak = aInGraveyard && !bInGraveyard;\n\n\t\t// If `bIsWeak` it means that `b` points to graveyard while `a` doesn't. Force moving nodes then.\n\t\tconst bIsWeak = bInGraveyard && !aInGraveyard;\n\n\t\t// Force move if `b` is weak or neither operation is weak but `a` is stronger through `context.aIsStrong`.\n\t\tconst forceMove = bIsWeak || ( !aIsWeak && context.aIsStrong );\n\n\t\tif ( forceMove ) {\n\t\t\tconst result = [];\n\n\t\t\t// First we need to move any nodes split by `b` back to where they were.\n\t\t\t// Do it only if `b` actually moved something.\n\t\t\tif ( b.howMany ) {\n\t\t\t\tresult.push( new MoveOperation( b.moveTargetPosition, b.howMany, b.splitPosition, 0 ) );\n\t\t\t}\n\n\t\t\t// Then we need to move nodes from `a` split position to their new element.\n\t\t\t// Do it only if `a` actually should move something.\n\t\t\tif ( a.howMany ) {\n\t\t\t\tresult.push( new MoveOperation( a.splitPosition, a.howMany, a.moveTargetPosition, 0 ) );\n\t\t\t}\n\n\t\t\treturn result;\n\t\t} else {\n\t\t\treturn [ new NoOperation( 0 ) ];\n\t\t}\n\t}\n\n\tif ( a.graveyardPosition ) {\n\t\ta.graveyardPosition = a.graveyardPosition._getTransformedBySplitOperation( b );\n\t}\n\n\t// Case 3:\n\t//\n\t// Position where operation `b` inserted a new node after split is the same as the operation `a` split position.\n\t// As in similar cases, there is ambiguity if the split should be before the new node (created by `b`) or after.\n\t//\n\tif ( a.splitPosition.isEqual( b.insertionPosition ) && context.abRelation == 'splitBefore' ) {\n\t\ta.howMany++;\n\n\t\treturn [ a ];\n\t}\n\n\t// Case 4:\n\t//\n\t// This is a mirror to the case 2. above.\n\t//\n\tif ( b.splitPosition.isEqual( a.insertionPosition ) && context.baRelation == 'splitBefore' ) {\n\t\tconst newPositionPath = b.insertionPosition.path.slice();\n\t\tnewPositionPath.push( 0 );\n\n\t\tconst newPosition = new Position( b.insertionPosition.root, newPositionPath );\n\t\tconst moveOp = new MoveOperation( a.insertionPosition, 1, newPosition, 0 );\n\n\t\treturn [ a, moveOp ];\n\t}\n\n\t// The default case.\n\t//\n\tif ( a.splitPosition.hasSameParentAs( b.splitPosition ) && a.splitPosition.offset < b.splitPosition.offset ) {\n\t\ta.howMany -= b.howMany;\n\t}\n\n\ta.splitPosition = a.splitPosition._getTransformedBySplitOperation( b );\n\ta.insertionPosition = SplitOperation.getInsertionPosition( a.splitPosition );\n\n\treturn [ a ];\n} );\n\n/**\n * Checks whether `MoveOperation` `targetPosition` is inside a node from the moved range of the other `MoveOperation`.\n */\nfunction _moveTargetIntoMovedRange( a: MoveOperation, b: MoveOperation ) {\n\treturn a.targetPosition._getTransformedByDeletion( b.sourcePosition, b.howMany ) === null;\n}\n\n/**\n * Helper function for `MoveOperation` x `MoveOperation` transformation. Converts given ranges and target position to\n * move operations and returns them.\n *\n * Ranges and target position will be transformed on-the-fly when generating operations.\n *\n * Given `ranges` should be in the order of how they were in the original transformed operation.\n *\n * Given `targetPosition` is the target position of the first range from `ranges`.\n */\nfunction _makeMoveOperationsFromRanges( ranges: Array<Range>, targetPosition: Position ) {\n\t// At this moment we have some ranges and a target position, to which those ranges should be moved.\n\t// Order in `ranges` array is the go-to order of after transformation.\n\t//\n\t// We are almost done. We have `ranges` and `targetPosition` to make operations from.\n\t// Unfortunately, those operations may affect each other. Precisely, first operation after move\n\t// may affect source range and target position of second and third operation. Same with second\n\t// operation affecting third.\n\t//\n\t// We need to fix those source ranges and target positions once again, before converting `ranges` to operations.\n\tconst operations = [];\n\n\t// Keep in mind that nothing will be transformed if there is just one range in `ranges`.\n\tfor ( let i = 0; i < ranges.length; i++ ) {\n\t\t// Create new operation out of a range and target position.\n\t\tconst range = ranges[ i ];\n\t\tconst op = new MoveOperation(\n\t\t\trange.start,\n\t\t\trange.end.offset - range.start.offset,\n\t\t\ttargetPosition,\n\t\t\t0\n\t\t);\n\n\t\toperations.push( op );\n\n\t\t// Transform other ranges by the generated operation.\n\t\tfor ( let j = i + 1; j < ranges.length; j++ ) {\n\t\t\t// All ranges in `ranges` array should be:\n\t\t\t//\n\t\t\t// * non-intersecting (these are part of original operation source range), and\n\t\t\t// * `targetPosition` does not target into them (opposite would mean that transformed operation targets \"inside itself\").\n\t\t\t//\n\t\t\t// This means that the transformation will be \"clean\" and always return one result.\n\t\t\tranges[ j ] = ranges[ j ]._getTransformedByMove( op.sourcePosition, op.targetPosition, op.howMany )[ 0 ];\n\t\t}\n\n\t\ttargetPosition = targetPosition._getTransformedByMove( op.sourcePosition, op.targetPosition, op.howMany );\n\t}\n\n\treturn operations;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/liveposition\n */\n\nimport Position, { type PositionOffset, type PositionStickiness } from './position';\n\nimport type { ModelApplyOperationEvent } from './model';\nimport type DocumentFragment from './documentfragment';\nimport type Item from './item';\nimport type Operation from './operation/operation';\nimport type RootElement from './rootelement';\n\nimport { CKEditorError, EmitterMixin } from '@ckeditor/ckeditor5-utils';\n\n/**\n * `LivePosition` is a type of {@link module:engine/model/position~Position Position}\n * that updates itself as {@link module:engine/model/document~Document document}\n * is changed through operations. It may be used as a bookmark.\n *\n * **Note:** Contrary to {@link module:engine/model/position~Position}, `LivePosition` works only in roots that are\n * {@link module:engine/model/rootelement~RootElement}.\n * If {@link module:engine/model/documentfragment~DocumentFragment} is passed, error will be thrown.\n *\n * **Note:** Be very careful when dealing with `LivePosition`. Each `LivePosition` instance bind events that might\n * have to be unbound.\n * Use {@link module:engine/model/liveposition~LivePosition#detach} whenever you don't need `LivePosition` anymore.\n */\nexport default class LivePosition extends EmitterMixin( Position ) {\n\t/**\n\t * Root of the position path.\n\t */\n\tdeclare public readonly root: RootElement;\n\n\t/**\n\t * Creates a live position.\n\t *\n\t * @see module:engine/model/position~Position\n\t */\n\tconstructor( root: RootElement, path: Array<number>, stickiness: PositionStickiness = 'toNone' ) {\n\t\tsuper( root, path, stickiness );\n\n\t\tif ( !this.root.is( 'rootElement' ) ) {\n\t\t\t/**\n\t\t\t * LivePosition's root has to be an instance of RootElement.\n\t\t\t *\n\t\t\t * @error model-liveposition-root-not-rootelement\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-liveposition-root-not-rootelement', root );\n\t\t}\n\n\t\tbindWithDocument.call( this );\n\t}\n\n\t/**\n\t * Unbinds all events previously bound by `LivePosition`. Use it whenever you don't need `LivePosition` instance\n\t * anymore (i.e. when leaving scope in which it was declared or before re-assigning variable that was\n\t * referring to it).\n\t */\n\tpublic detach(): void {\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Creates a {@link module:engine/model/position~Position position instance}, which is equal to this live position.\n\t */\n\tpublic toPosition(): Position {\n\t\treturn new Position( this.root, this.path.slice(), this.stickiness );\n\t}\n\n\t/**\n\t * Creates a `LivePosition` instance that is equal to position.\n\t */\n\tpublic static fromPosition( position: Position, stickiness?: PositionStickiness ): LivePosition {\n\t\treturn new this( position.root as RootElement, position.path.slice(), stickiness ? stickiness : position.stickiness );\n\t}\n\n\t/**\n\t * @internal\n\t * @see module:engine/model/position~Position._createAfter\n\t */\n\tdeclare public static readonly _createAfter: ( item: Item | DocumentFragment, stickiness?: PositionStickiness ) => LivePosition;\n\n\t/**\n\t * @internal\n\t * @see module:engine/model/position~Position._createBefore\n\t */\n\tdeclare public static readonly _createBefore: ( item: Item | DocumentFragment, stickiness?: PositionStickiness ) => LivePosition;\n\n\t/**\n\t * @internal\n\t * @see module:engine/model/position~Position._createAt\n\t */\n\tdeclare public static readonly _createAt: (\n\t\titemOrPosition: Item | Position | DocumentFragment,\n\t\toffset?: PositionOffset,\n\t\tstickiness?: PositionStickiness\n\t) => LivePosition;\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nLivePosition.prototype.is = function( type: string ): boolean {\n\treturn type === 'livePosition' || type === 'model:livePosition' ||\n\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\ttype == 'position' || type === 'model:position';\n};\n\n/**\n * Binds this `LivePosition` to the {@link module:engine/model/document~Document document} that owns\n * this position's {@link module:engine/model/position~Position#root root}.\n */\nfunction bindWithDocument( this: LivePosition ) {\n\tthis.listenTo<ModelApplyOperationEvent>(\n\t\tthis.root.document!.model,\n\t\t'applyOperation',\n\t\t( event, args ) => {\n\t\t\tconst operation = args[ 0 ];\n\n\t\t\tif ( !operation.isDocumentOperation ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\ttransform.call( this, operation );\n\t\t},\n\t\t{ priority: 'low' }\n\t);\n}\n\n/**\n * Updates this position accordingly to the updates applied to the model. Bases on change events.\n */\nfunction transform( this: LivePosition, operation: Operation ) {\n\tconst result = this.getTransformedByOperation( operation );\n\n\tif ( !this.isEqual( result ) ) {\n\t\tconst oldPosition = this.toPosition();\n\n\t\t( this as any ).path = result.path;\n\t\t( this as any ).root = result.root;\n\n\t\tthis.fire<LivePositionChangeEvent>( 'change', oldPosition );\n\t}\n}\n\n/**\n * Fired when `LivePosition` instance is changed due to changes on {@link module:engine/model/document~Document}.\n *\n * @eventName ~LivePosition#change\n * @param oldPosition Position equal to this live position before it got changed.\n */\nexport type LivePositionChangeEvent = {\n\tname: 'change';\n\targs: [ oldPosition: Position ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/batch\n */\n\nimport { logWarning } from '@ckeditor/ckeditor5-utils';\nimport type Operation from './operation/operation';\n\n/**\n * A batch instance groups model changes ({@link module:engine/model/operation/operation~Operation operations}). All operations\n * grouped in a single batch can be reverted together, so you can also think about a batch as of a single undo step. If you want\n * to extend a given undo step, you can add more changes to the batch using {@link module:engine/model/model~Model#enqueueChange}:\n *\n * ```ts\n * model.enqueueChange( batch, writer => {\n * \twriter.insertText( 'foo', paragraph, 'end' );\n * } );\n * ```\n *\n * @see module:engine/model/model~Model#enqueueChange\n * @see module:engine/model/model~Model#change\n */\nexport default class Batch {\n\t/**\n\t * An array of operations that compose this batch.\n\t */\n\tpublic readonly operations: Array<Operation>;\n\n\t/**\n\t * Whether the batch can be undone through the undo feature.\n\t */\n\tpublic readonly isUndoable: boolean;\n\n\t/**\n\t * Whether the batch includes operations created locally (`true`) or operations created on other, remote editors (`false`).\n\t */\n\tpublic readonly isLocal: boolean;\n\n\t/**\n\t * Whether the batch was created by the undo feature and undoes other operations.\n\t */\n\tpublic readonly isUndo: boolean;\n\n\t/**\n\t * Whether the batch includes operations connected with typing.\n\t */\n\tpublic readonly isTyping: boolean;\n\n\t/**\n\t * Creates a batch instance.\n\t *\n\t * @see module:engine/model/model~Model#enqueueChange\n\t * @see module:engine/model/model~Model#change\n\t * @param type A set of flags that specify the type of the batch. Batch type can alter how some of the features work\n\t * when encountering a given `Batch` instance (for example, when a feature listens to applied operations).\n\t */\n\tconstructor( type: BatchType = {} ) {\n\t\tif ( typeof type === 'string' ) {\n\t\t\ttype = type === 'transparent' ? { isUndoable: false } : {};\n\n\t\t\t/**\n\t\t\t * The string value for a `type` property of the `Batch` constructor has been deprecated and will be removed in the near future.\n\t\t\t * Please refer to the {@link module:engine/model/batch~Batch#constructor `Batch` constructor API documentation} for more\n\t\t\t * information.\n\t\t\t *\n\t\t\t * @error batch-constructor-deprecated-string-type\n\t\t\t */\n\t\t\tlogWarning( 'batch-constructor-deprecated-string-type' );\n\t\t}\n\n\t\tconst { isUndoable = true, isLocal = true, isUndo = false, isTyping = false } = type;\n\n\t\tthis.operations = [];\n\t\tthis.isUndoable = isUndoable;\n\t\tthis.isLocal = isLocal;\n\t\tthis.isUndo = isUndo;\n\t\tthis.isTyping = isTyping;\n\t}\n\n\t/**\n\t * The type of the batch.\n\t *\n\t * **This property has been deprecated and is always set to the `'default'` value.**\n\t *\n\t * It can be one of the following values:\n\t * * `'default'` &ndash; All \"normal\" batches. This is the most commonly used type.\n\t * * `'transparent'` &ndash; A batch that should be ignored by other features, i.e. an initial batch or collaborative editing\n\t * changes.\n\t *\n\t * @deprecated\n\t */\n\tpublic get type(): 'default' {\n\t\t/**\n\t\t * The {@link module:engine/model/batch~Batch#type `Batch#type` } property has been deprecated and will be removed in the near\n\t\t * future. Use `Batch#isLocal`, `Batch#isUndoable`, `Batch#isUndo` and `Batch#isTyping` instead.\n\t\t *\n\t\t * @error batch-type-deprecated\n\t\t */\n\t\tlogWarning( 'batch-type-deprecated' );\n\n\t\treturn 'default';\n\t}\n\n\t/**\n\t * Returns the base version of this batch, which is equal to the base version of the first operation in the batch.\n\t * If there are no operations in the batch or neither operation has the base version set, it returns `null`.\n\t */\n\tpublic get baseVersion(): number | null {\n\t\tfor ( const op of this.operations ) {\n\t\t\tif ( op.baseVersion !== null ) {\n\t\t\t\treturn op.baseVersion;\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Adds an operation to the batch instance.\n\t *\n\t * @param operation An operation to add.\n\t * @returns The added operation.\n\t */\n\tpublic addOperation( operation: Operation ): Operation {\n\t\toperation.batch = this;\n\t\tthis.operations.push( operation );\n\n\t\treturn operation;\n\t}\n}\n\n/**\n * A set of flags that specify the type of the batch. Batch type can alter how some of the features work\n * when encountering a given `Batch` instance (for example, when a feature listens to applied operations).\n */\nexport interface BatchType {\n\n\t/**\n\t * Whether a batch can be undone through undo feature.\n\t *\n\t * @default true\n\t */\n\tisUndoable?: boolean;\n\n\t/**\n\t * Whether a batch includes operations created locally (`true`) or operations created on\n\t * other, remote editors (`false`).\n\t *\n\t * @default true\n\t */\n\tisLocal?: boolean;\n\n\t/**\n\t * Whether a batch was created by the undo feature and undoes other operations.\n\t *\n\t * @default false\n\t */\n\tisUndo?: boolean;\n\n\t/**\n\t * Whether a batch includes operations connected with a typing action.\n\t *\n\t * @default false\n\t */\n\tisTyping?: boolean;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/differ\n */\n\nimport Position from './position';\nimport Range from './range';\n\nimport type { default as MarkerCollection, MarkerData } from './markercollection';\nimport type AttributeOperation from './operation/attributeoperation';\nimport type DocumentFragment from './documentfragment';\nimport type Element from './element';\nimport type InsertOperation from './operation/insertoperation';\nimport type Item from './item';\nimport type MergeOperation from './operation/mergeoperation';\nimport type MoveOperation from './operation/moveoperation';\nimport type Node from './node';\nimport type RootElement from './rootelement';\nimport type Operation from './operation/operation';\nimport type RenameOperation from './operation/renameoperation';\nimport type SplitOperation from './operation/splitoperation';\nimport type RootOperation from './operation/rootoperation';\nimport type RootAttributeOperation from './operation/rootattributeoperation';\n\n/**\n * Calculates the difference between two model states.\n *\n * Receives operations that are to be applied on the model document. Marks parts of the model document tree which\n * are changed and saves the state of these elements before the change. Then, it compares saved elements with the\n * changed elements, after all changes are applied on the model document. Calculates the diff between saved\n * elements and new ones and returns a change set.\n */\nexport default class Differ {\n\t/**\n\t * Reference to the model's marker collection.\n\t */\n\tprivate readonly _markerCollection: MarkerCollection;\n\n\t/**\n\t * A map that stores changes that happened in a given element.\n\t *\n\t * The keys of the map are references to the model elements.\n\t * The values of the map are arrays with changes that were done on this element.\n\t */\n\tprivate readonly _changesInElement: Map<Element | DocumentFragment, Array<ChangeItem>> = new Map();\n\n\t/**\n\t * A map that stores \"element's children snapshots\". A snapshot is representing children of a given element before\n\t * the first change was applied on that element. Snapshot items are objects with two properties: `name`,\n\t * containing the element name (or `'$text'` for a text node) and `attributes` which is a map of the node's attributes.\n\t */\n\tprivate readonly _elementSnapshots: Map<Element | DocumentFragment, Array<{ name: string; attributes: Map<string, unknown> }>>\n\t\t= new Map();\n\n\t/**\n\t * A map that stores all changed markers.\n\t *\n\t * The keys of the map are marker names.\n\t * The values of the map are objects with the following properties:\n\t * - `oldMarkerData`,\n\t * - `newMarkerData`.\n\t */\n\tprivate readonly _changedMarkers: Map<string, { newMarkerData: MarkerData; oldMarkerData: MarkerData }> = new Map();\n\n\t/**\n\t * A map that stores all roots that have been changed.\n\t *\n\t * The keys are the names of the roots while value represents the changes.\n\t */\n\tprivate readonly _changedRoots: Map<string, DiffItemRoot> = new Map();\n\n\t/**\n\t * Stores the number of changes that were processed. Used to order the changes chronologically. It is important\n\t * when changes are sorted.\n\t */\n\tprivate _changeCount: number = 0;\n\n\t/**\n\t * For efficiency purposes, `Differ` stores the change set returned by the differ after {@link #getChanges} call.\n\t * Cache is reset each time a new operation is buffered. If the cache has not been reset, {@link #getChanges} will\n\t * return the cached value instead of calculating it again.\n\t *\n\t * This property stores those changes that did not take place in graveyard root.\n\t */\n\tprivate _cachedChanges: Array<DiffItem> | null = null;\n\n\t/**\n\t * For efficiency purposes, `Differ` stores the change set returned by the differ after the {@link #getChanges} call.\n\t * The cache is reset each time a new operation is buffered. If the cache has not been reset, {@link #getChanges} will\n\t * return the cached value instead of calculating it again.\n\t *\n\t * This property stores all changes evaluated by `Differ`, including those that took place in the graveyard.\n\t */\n\tprivate _cachedChangesWithGraveyard: Array<DiffItem> | null = null;\n\n\t/**\n\t * Set of model items that were marked to get refreshed in {@link #_refreshItem}.\n\t */\n\tprivate _refreshedItems: Set<Item> = new Set();\n\n\t/**\n\t * Creates a `Differ` instance.\n\t *\n\t * @param markerCollection Model's marker collection.\n\t */\n\tconstructor( markerCollection: MarkerCollection ) {\n\t\tthis._markerCollection = markerCollection;\n\t}\n\n\t/**\n\t * Informs whether there are any changes buffered in `Differ`.\n\t */\n\tpublic get isEmpty(): boolean {\n\t\treturn this._changesInElement.size == 0 && this._changedMarkers.size == 0 && this._changedRoots.size == 0;\n\t}\n\n\t/**\n\t * Buffers the given operation. An operation has to be buffered before it is executed.\n\t *\n\t * @param operationToBuffer An operation to buffer.\n\t */\n\tpublic bufferOperation( operationToBuffer: Operation ): void {\n\t\t// Below we take an operation, check its type, then use its parameters in marking (private) methods.\n\t\t// The general rule is to not mark elements inside inserted element. All inserted elements are re-rendered.\n\t\t// Marking changes in them would cause a \"double\" changing then.\n\t\t//\n\t\tconst operation = operationToBuffer as (\n\t\t\tInsertOperation |\n\t\t\tAttributeOperation |\n\t\t\tMoveOperation |\n\t\t\tRenameOperation |\n\t\t\tSplitOperation |\n\t\t\tMergeOperation |\n\t\t\tRootOperation |\n\t\t\tRootAttributeOperation\n\t\t);\n\n\t\t// Note: an operation that happens inside a non-loaded root will be ignored. If the operation happens partially inside\n\t\t// a non-loaded root, that part will be ignored (this may happen for move or marker operations).\n\t\t//\n\t\tswitch ( operation.type ) {\n\t\t\tcase 'insert': {\n\t\t\t\tif ( this._isInInsertedElement( operation.position.parent ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tthis._markInsert( operation.position.parent, operation.position.offset, operation.nodes.maxOffset );\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t\tcase 'addAttribute':\n\t\t\tcase 'removeAttribute':\n\t\t\tcase 'changeAttribute': {\n\t\t\t\tfor ( const item of operation.range.getItems( { shallow: true } ) ) {\n\t\t\t\t\tif ( this._isInInsertedElement( item.parent! ) ) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\tthis._markAttribute( item );\n\t\t\t\t}\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t\tcase 'remove':\n\t\t\tcase 'move':\n\t\t\tcase 'reinsert': {\n\t\t\t\t// When range is moved to the same position then not mark it as a change.\n\t\t\t\t// See: https://github.com/ckeditor/ckeditor5-engine/issues/1664.\n\t\t\t\tif (\n\t\t\t\t\toperation.sourcePosition.isEqual( operation.targetPosition ) ||\n\t\t\t\t\toperation.sourcePosition.getShiftedBy( operation.howMany ).isEqual( operation.targetPosition )\n\t\t\t\t) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst sourceParentInserted = this._isInInsertedElement( operation.sourcePosition.parent );\n\t\t\t\tconst targetParentInserted = this._isInInsertedElement( operation.targetPosition.parent );\n\n\t\t\t\tif ( !sourceParentInserted ) {\n\t\t\t\t\tthis._markRemove( operation.sourcePosition.parent, operation.sourcePosition.offset, operation.howMany );\n\t\t\t\t}\n\n\t\t\t\tif ( !targetParentInserted ) {\n\t\t\t\t\tthis._markInsert( operation.targetPosition.parent, operation.getMovedRangeStart().offset, operation.howMany );\n\t\t\t\t}\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t\tcase 'rename': {\n\t\t\t\tif ( this._isInInsertedElement( operation.position.parent ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tthis._markRemove( operation.position.parent, operation.position.offset, 1 );\n\t\t\t\tthis._markInsert( operation.position.parent, operation.position.offset, 1 );\n\n\t\t\t\tconst range = Range._createFromPositionAndShift( operation.position, 1 );\n\n\t\t\t\tfor ( const marker of this._markerCollection.getMarkersIntersectingRange( range ) ) {\n\t\t\t\t\tconst markerData = marker.getData();\n\n\t\t\t\t\tthis.bufferMarkerChange( marker.name, markerData, markerData );\n\t\t\t\t}\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t\tcase 'split': {\n\t\t\t\tconst splitElement = operation.splitPosition.parent;\n\n\t\t\t\t// Mark that children of the split element were removed.\n\t\t\t\tif ( !this._isInInsertedElement( splitElement ) ) {\n\t\t\t\t\tthis._markRemove( splitElement, operation.splitPosition.offset, operation.howMany );\n\t\t\t\t}\n\n\t\t\t\t// Mark that the new element (split copy) was inserted.\n\t\t\t\tif ( !this._isInInsertedElement( operation.insertionPosition.parent ) ) {\n\t\t\t\t\tthis._markInsert( operation.insertionPosition.parent, operation.insertionPosition.offset, 1 );\n\t\t\t\t}\n\n\t\t\t\t// If the split took the element from the graveyard, mark that the element from the graveyard was removed.\n\t\t\t\tif ( operation.graveyardPosition ) {\n\t\t\t\t\tthis._markRemove( operation.graveyardPosition.parent, operation.graveyardPosition.offset, 1 );\n\t\t\t\t}\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t\tcase 'merge': {\n\t\t\t\t// Mark that the merged element was removed.\n\t\t\t\tconst mergedElement = operation.sourcePosition.parent;\n\n\t\t\t\tif ( !this._isInInsertedElement( mergedElement.parent! ) ) {\n\t\t\t\t\tthis._markRemove( mergedElement.parent!, ( mergedElement as any ).startOffset, 1 );\n\t\t\t\t}\n\n\t\t\t\t// Mark that the merged element was inserted into graveyard.\n\t\t\t\tconst graveyardParent = operation.graveyardPosition.parent;\n\n\t\t\t\tthis._markInsert( graveyardParent, operation.graveyardPosition.offset, 1 );\n\n\t\t\t\t// Mark that children of merged element were inserted at new parent.\n\t\t\t\tconst mergedIntoElement = operation.targetPosition.parent;\n\n\t\t\t\tif ( !this._isInInsertedElement( mergedIntoElement ) ) {\n\t\t\t\t\tthis._markInsert( mergedIntoElement, operation.targetPosition.offset, mergedElement.maxOffset );\n\t\t\t\t}\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t\tcase 'detachRoot':\n\t\t\tcase 'addRoot': {\n\t\t\t\tconst root = operation.affectedSelectable as RootElement;\n\n\t\t\t\tif ( !root._isLoaded ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// Don't buffer if the root state does not change.\n\t\t\t\tif ( root.isAttached() == operation.isAdd ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tthis._bufferRootStateChange( operation.rootName, operation.isAdd );\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t\tcase 'addRootAttribute':\n\t\t\tcase 'removeRootAttribute':\n\t\t\tcase 'changeRootAttribute': {\n\t\t\t\tif ( !operation.root._isLoaded ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst rootName = operation.root.rootName;\n\n\t\t\t\tthis._bufferRootAttributeChange( rootName, operation.key, operation.oldValue, operation.newValue );\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\t// Clear cache after each buffered operation as it is no longer valid.\n\t\tthis._cachedChanges = null;\n\t}\n\n\t/**\n\t * Buffers a marker change.\n\t *\n\t * @param markerName The name of the marker that changed.\n\t * @param oldMarkerData Marker data before the change.\n\t * @param newMarkerData Marker data after the change.\n\t */\n\tpublic bufferMarkerChange( markerName: string, oldMarkerData: MarkerData, newMarkerData: MarkerData ): void {\n\t\tif ( oldMarkerData.range && oldMarkerData.range.root.is( 'rootElement' ) && !oldMarkerData.range.root._isLoaded ) {\n\t\t\toldMarkerData.range = null;\n\t\t}\n\n\t\tif ( newMarkerData.range && newMarkerData.range.root.is( 'rootElement' ) && !newMarkerData.range.root._isLoaded ) {\n\t\t\tnewMarkerData.range = null;\n\t\t}\n\n\t\tlet buffered = this._changedMarkers.get( markerName );\n\n\t\tif ( !buffered ) {\n\t\t\tbuffered = { newMarkerData, oldMarkerData };\n\n\t\t\tthis._changedMarkers.set( markerName, buffered );\n\t\t} else {\n\t\t\tbuffered.newMarkerData = newMarkerData;\n\t\t}\n\n\t\tif ( buffered.oldMarkerData.range == null && newMarkerData.range == null ) {\n\t\t\t// The marker is going to be removed (`newMarkerData.range == null`) but it did not exist before the first buffered change\n\t\t\t// (`buffered.oldMarkerData.range == null`). In this case, do not keep the marker in buffer at all.\n\t\t\tthis._changedMarkers.delete( markerName );\n\t\t}\n\t}\n\n\t/**\n\t * Returns all markers that should be removed as a result of buffered changes.\n\t *\n\t * @returns Markers to remove. Each array item is an object containing the `name` and `range` properties.\n\t */\n\tpublic getMarkersToRemove(): Array<{ name: string; range: Range }> {\n\t\tconst result = [];\n\n\t\tfor ( const [ name, change ] of this._changedMarkers ) {\n\t\t\tif ( change.oldMarkerData.range != null ) {\n\t\t\t\tresult.push( { name, range: change.oldMarkerData.range } );\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t}\n\n\t/**\n\t * Returns all markers which should be added as a result of buffered changes.\n\t *\n\t * @returns Markers to add. Each array item is an object containing the `name` and `range` properties.\n\t */\n\tpublic getMarkersToAdd(): Array<{ name: string; range: Range }> {\n\t\tconst result = [];\n\n\t\tfor ( const [ name, change ] of this._changedMarkers ) {\n\t\t\tif ( change.newMarkerData.range != null ) {\n\t\t\t\tresult.push( { name, range: change.newMarkerData.range } );\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t}\n\n\t/**\n\t * Returns all markers which changed.\n\t */\n\tpublic getChangedMarkers(): Array<{\n\t\tname: string;\n\t\tdata: {\n\t\t\toldRange: Range | null;\n\t\t\tnewRange: Range | null;\n\t\t};\n\t}> {\n\t\treturn Array.from( this._changedMarkers ).map( ( [ name, change ] ) => (\n\t\t\t{\n\t\t\t\tname,\n\t\t\t\tdata: {\n\t\t\t\t\toldRange: change.oldMarkerData.range,\n\t\t\t\t\tnewRange: change.newMarkerData.range\n\t\t\t\t}\n\t\t\t}\n\t\t) );\n\t}\n\n\t/**\n\t * Checks whether some of the buffered changes affect the editor data.\n\t *\n\t * Types of changes which affect the editor data:\n\t *\n\t * * model structure changes,\n\t * * attribute changes,\n\t * * a root is added or detached,\n\t * * changes of markers which were defined as `affectsData`,\n\t * * changes of markers' `affectsData` property.\n\t */\n\tpublic hasDataChanges(): boolean {\n\t\tif ( this._changesInElement.size > 0 ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tif ( this._changedRoots.size > 0 ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tfor ( const { newMarkerData, oldMarkerData } of this._changedMarkers.values() ) {\n\t\t\tif ( newMarkerData.affectsData !== oldMarkerData.affectsData ) {\n\t\t\t\treturn true;\n\t\t\t}\n\n\t\t\tif ( newMarkerData.affectsData ) {\n\t\t\t\tconst markerAdded = newMarkerData.range && !oldMarkerData.range;\n\t\t\t\tconst markerRemoved = !newMarkerData.range && oldMarkerData.range;\n\t\t\t\tconst markerChanged = newMarkerData.range && oldMarkerData.range && !newMarkerData.range.isEqual( oldMarkerData.range );\n\n\t\t\t\tif ( markerAdded || markerRemoved || markerChanged ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Calculates the diff between the old model tree state (the state before the first buffered operations since the last {@link #reset}\n\t * call) and the new model tree state (actual one). It should be called after all buffered operations are executed.\n\t *\n\t * The diff set is returned as an array of {@link module:engine/model/differ~DiffItem diff items}, each describing a change done\n\t * on the model. The items are sorted by the position on which the change happened. If a position\n\t * {@link module:engine/model/position~Position#isBefore is before} another one, it will be on an earlier index in the diff set.\n\t *\n\t * **Note**: Elements inside inserted element will not have a separate diff item, only the top most element change will be reported.\n\t *\n\t * Because calculating the diff is a costly operation, the result is cached. If no new operation was buffered since the\n\t * previous {@link #getChanges} call, the next call will return the cached value.\n\t *\n\t * @param options Additional options.\n\t * @param options.includeChangesInGraveyard If set to `true`, also changes that happened\n\t * in the graveyard root will be returned. By default, changes in the graveyard root are not returned.\n\t * @returns Diff between the old and the new model tree state.\n\t */\n\tpublic getChanges( options: { includeChangesInGraveyard?: boolean } = {} ): Array<DiffItem> {\n\t\t// If there are cached changes, just return them instead of calculating changes again.\n\t\tif ( this._cachedChanges ) {\n\t\t\tif ( options.includeChangesInGraveyard ) {\n\t\t\t\treturn this._cachedChangesWithGraveyard!.slice();\n\t\t\t} else {\n\t\t\t\treturn this._cachedChanges.slice();\n\t\t\t}\n\t\t}\n\n\t\t// Will contain returned results.\n\t\tlet diffSet: Array<DiffItem & DiffItemInternal> = [];\n\n\t\t// Check all changed elements.\n\t\tfor ( const element of this._changesInElement.keys() ) {\n\t\t\t// Get changes for this element and sort them.\n\t\t\tconst changes = this._changesInElement.get( element )!.sort( ( a, b ) => {\n\t\t\t\tif ( a.offset === b.offset ) {\n\t\t\t\t\tif ( a.type != b.type ) {\n\t\t\t\t\t\t// If there are multiple changes at the same position, \"remove\" change should be first.\n\t\t\t\t\t\t// If the order is different, for example, we would first add some nodes and then removed them\n\t\t\t\t\t\t// (instead of the nodes that we should remove).\n\t\t\t\t\t\treturn a.type == 'remove' ? -1 : 1;\n\t\t\t\t\t}\n\n\t\t\t\t\treturn 0;\n\t\t\t\t}\n\n\t\t\t\treturn a.offset < b.offset ? -1 : 1;\n\t\t\t} );\n\n\t\t\t// Get children of this element before any change was applied on it.\n\t\t\tconst snapshotChildren = this._elementSnapshots.get( element )!;\n\t\t\t// Get snapshot of current element's children.\n\t\t\tconst elementChildren = _getChildrenSnapshot( element.getChildren() );\n\n\t\t\t// Generate actions basing on changes done on element.\n\t\t\tconst actions = _generateActionsFromChanges( snapshotChildren.length, changes );\n\n\t\t\tlet i = 0; // Iterator in `elementChildren` array -- iterates through current children of element.\n\t\t\tlet j = 0; // Iterator in `snapshotChildren` array -- iterates through old children of element.\n\n\t\t\t// Process every action.\n\t\t\tfor ( const action of actions ) {\n\t\t\t\tif ( action === 'i' ) {\n\t\t\t\t\t// Generate diff item for this element and insert it into the diff set.\n\t\t\t\t\tdiffSet.push( this._getInsertDiff( element, i, elementChildren[ i ] ) );\n\n\t\t\t\t\ti++;\n\t\t\t\t} else if ( action === 'r' ) {\n\t\t\t\t\t// Generate diff item for this element and insert it into the diff set.\n\t\t\t\t\tdiffSet.push( this._getRemoveDiff( element, i, snapshotChildren[ j ] ) );\n\n\t\t\t\t\tj++;\n\t\t\t\t} else if ( action === 'a' ) {\n\t\t\t\t\t// Take attributes from saved and current children.\n\t\t\t\t\tconst elementAttributes = elementChildren[ i ].attributes;\n\t\t\t\t\tconst snapshotAttributes = snapshotChildren[ j ].attributes;\n\t\t\t\t\tlet range;\n\n\t\t\t\t\tif ( elementChildren[ i ].name == '$text' ) {\n\t\t\t\t\t\trange = new Range( Position._createAt( element, i ), Position._createAt( element, i + 1 ) );\n\t\t\t\t\t} else {\n\t\t\t\t\t\tconst index = element.offsetToIndex( i );\n\t\t\t\t\t\trange = new Range( Position._createAt( element, i ), Position._createAt( element.getChild( index )!, 0 ) );\n\t\t\t\t\t}\n\n\t\t\t\t\t// Generate diff items for this change (there might be multiple attributes changed and\n\t\t\t\t\t// there is a single diff for each of them) and insert them into the diff set.\n\t\t\t\t\tdiffSet.push( ...this._getAttributesDiff( range, snapshotAttributes, elementAttributes ) );\n\n\t\t\t\t\ti++;\n\t\t\t\t\tj++;\n\t\t\t\t} else {\n\t\t\t\t\t// `action` is 'equal'. Child not changed.\n\t\t\t\t\ti++;\n\t\t\t\t\tj++;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// Then, sort the changes by the position (change at position before other changes is first).\n\t\tdiffSet.sort( ( a, b ) => {\n\t\t\t// If the change is in different root, we don't care much, but we'd like to have all changes in given\n\t\t\t// root \"together\" in the array. So let's just sort them by the root name. It does not matter which root\n\t\t\t// will be processed first.\n\t\t\tif ( a.position!.root != b.position!.root ) {\n\t\t\t\treturn a.position!.root.rootName! < b.position!.root.rootName! ? -1 : 1;\n\t\t\t}\n\n\t\t\t// If change happens at the same position...\n\t\t\tif ( a.position!.isEqual( b.position! ) ) {\n\t\t\t\t// Keep chronological order of operations.\n\t\t\t\treturn a.changeCount! - b.changeCount!;\n\t\t\t}\n\n\t\t\t// If positions differ, position \"on the left\" should be earlier in the result.\n\t\t\treturn a.position!.isBefore( b.position! ) ? -1 : 1;\n\t\t} );\n\n\t\t// Glue together multiple changes (mostly on text nodes).\n\t\tfor ( let i = 1, prevIndex = 0; i < diffSet.length; i++ ) {\n\t\t\tconst prevDiff = diffSet[ prevIndex ];\n\t\t\tconst thisDiff = diffSet[ i ];\n\n\t\t\t// Glue remove changes if they happen on text on same position.\n\t\t\tconst isConsecutiveTextRemove =\n\t\t\t\tprevDiff.type == 'remove' && thisDiff.type == 'remove' &&\n\t\t\t\tprevDiff.name == '$text' && thisDiff.name == '$text' &&\n\t\t\t\tprevDiff.position.isEqual( thisDiff.position );\n\n\t\t\t// Glue insert changes if they happen on text on consecutive fragments.\n\t\t\tconst isConsecutiveTextAdd =\n\t\t\t\tprevDiff.type == 'insert' && thisDiff.type == 'insert' &&\n\t\t\t\tprevDiff.name == '$text' && thisDiff.name == '$text' &&\n\t\t\t\tprevDiff.position.parent == thisDiff.position.parent &&\n\t\t\t\tprevDiff.position.offset + prevDiff.length == thisDiff.position.offset;\n\n\t\t\t// Glue attribute changes if they happen on consecutive fragments and have same key, old value and new value.\n\t\t\tconst isConsecutiveAttributeChange =\n\t\t\t\tprevDiff.type == 'attribute' && thisDiff.type == 'attribute' &&\n\t\t\t\tprevDiff.position!.parent == thisDiff.position!.parent &&\n\t\t\t\tprevDiff.range.isFlat && thisDiff.range.isFlat &&\n\t\t\t\t( prevDiff.position!.offset + prevDiff.length! ) == thisDiff.position!.offset &&\n\t\t\t\tprevDiff.attributeKey == thisDiff.attributeKey &&\n\t\t\t\tprevDiff.attributeOldValue == thisDiff.attributeOldValue &&\n\t\t\t\tprevDiff.attributeNewValue == thisDiff.attributeNewValue;\n\n\t\t\tif ( isConsecutiveTextRemove || isConsecutiveTextAdd || isConsecutiveAttributeChange ) {\n\t\t\t\tprevDiff.length!++;\n\n\t\t\t\tif ( isConsecutiveAttributeChange ) {\n\t\t\t\t\t( prevDiff.range as any ).end = prevDiff.range.end.getShiftedBy( 1 );\n\t\t\t\t}\n\n\t\t\t\tdiffSet[ i ] = null as any;\n\t\t\t} else {\n\t\t\t\tprevIndex = i;\n\t\t\t}\n\t\t}\n\n\t\tdiffSet = diffSet.filter( v => v );\n\n\t\t// Remove `changeCount` property from diff items. It is used only for sorting and is internal thing.\n\t\tfor ( const item of diffSet ) {\n\t\t\tdelete item.changeCount;\n\n\t\t\tif ( item.type == 'attribute' ) {\n\t\t\t\tdelete item.position;\n\t\t\t\tdelete item.length;\n\t\t\t}\n\t\t}\n\n\t\tthis._changeCount = 0;\n\n\t\t// Cache changes.\n\t\tthis._cachedChangesWithGraveyard = diffSet;\n\t\tthis._cachedChanges = diffSet.filter( _changesInGraveyardFilter );\n\n\t\tif ( options.includeChangesInGraveyard ) {\n\t\t\treturn this._cachedChangesWithGraveyard.slice();\n\t\t} else {\n\t\t\treturn this._cachedChanges.slice();\n\t\t}\n\t}\n\n\t/**\n\t * Returns all roots that have changed (either were attached, or detached, or their attributes changed).\n\t *\n\t * @returns Diff between the old and the new roots state.\n\t */\n\tpublic getChangedRoots(): Array<DiffItemRoot> {\n\t\treturn Array.from( this._changedRoots.values() ).map( diffItem => {\n\t\t\tconst entry = { ...diffItem };\n\n\t\t\tif ( entry.state !== undefined ) {\n\t\t\t\t// The root was attached or detached -- do not return its attributes changes.\n\t\t\t\t// If the root was attached, it should be handled as a whole, together with its attributes, the same way as model nodes.\n\t\t\t\t// If the root was detached, its attributes should be discarded anyway.\n\t\t\t\t//\n\t\t\t\t// Keep in mind that filtering must happen on this stage (when retrieving changes). If filtering happens on-the-fly as\n\t\t\t\t// the attributes change, it may lead to incorrect situation, e.g.: detach root, change attribute, re-attach root.\n\t\t\t\t// In this case, attribute change cannot be filtered. After the root is re-attached, the attribute change must be kept.\n\t\t\t\tdelete entry.attributes;\n\t\t\t}\n\n\t\t\treturn entry;\n\t\t} );\n\t}\n\n\t/**\n\t * Returns a set of model items that were marked to get refreshed.\n\t */\n\tpublic getRefreshedItems(): Set<Item> {\n\t\treturn new Set( this._refreshedItems );\n\t}\n\n\t/**\n\t * Resets `Differ`. Removes all buffered changes.\n\t */\n\tpublic reset(): void {\n\t\tthis._changesInElement.clear();\n\t\tthis._elementSnapshots.clear();\n\t\tthis._changedMarkers.clear();\n\t\tthis._changedRoots.clear();\n\t\tthis._refreshedItems = new Set();\n\t\tthis._cachedChanges = null;\n\t}\n\n\t/**\n\t * Buffers the root state change after the root was attached or detached\n\t */\n\tprivate _bufferRootStateChange( rootName: string, isAttached: boolean ): void {\n\t\tif ( !this._changedRoots.has( rootName ) ) {\n\t\t\tthis._changedRoots.set( rootName, { name: rootName, state: isAttached ? 'attached' : 'detached' } );\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst diffItem = this._changedRoots.get( rootName )!;\n\n\t\tif ( diffItem.state !== undefined ) {\n\t\t\t// Root `state` can only toggle between one of the values and no value. It cannot be any other way,\n\t\t\t// because if the root was originally attached it can only become detached. Then, if it is re-attached in the same batch of\n\t\t\t// changes, it gets back to \"no change\" (which means no value). Same if the root was originally detached.\n\t\t\tdelete diffItem.state;\n\n\t\t\tif ( diffItem.attributes === undefined ) {\n\t\t\t\t// If there is no `state` change and no `attributes` change, remove the entry.\n\t\t\t\tthis._changedRoots.delete( rootName );\n\t\t\t}\n\t\t} else {\n\t\t\tdiffItem.state = isAttached ? 'attached' : 'detached';\n\t\t}\n\t}\n\n\t/**\n\t * Buffers a root attribute change.\n\t */\n\tprivate _bufferRootAttributeChange( rootName: string, key: string, oldValue: unknown, newValue: unknown ): void {\n\t\tconst diffItem: DiffItemRoot = this._changedRoots.get( rootName ) || { name: rootName };\n\t\tconst attrs: Record<string, { oldValue: unknown; newValue: unknown }> = diffItem.attributes || {};\n\n\t\tif ( attrs[ key ] ) {\n\t\t\t// If this attribute or metadata was already changed earlier and is changed again, check to what value it is changed.\n\t\t\tconst attrEntry = attrs[ key ];\n\n\t\t\tif ( newValue === attrEntry.oldValue ) {\n\t\t\t\t// If it was changed back to the old value, remove the entry.\n\t\t\t\tdelete attrs[ key ];\n\t\t\t} else {\n\t\t\t\t// If it was changed to a different value, update the entry.\n\t\t\t\tattrEntry.newValue = newValue;\n\t\t\t}\n\t\t} else {\n\t\t\t// If this attribute or metadata was not set earlier, add an entry.\n\t\t\tattrs[ key ] = { oldValue, newValue };\n\t\t}\n\n\t\tif ( Object.entries( attrs ).length === 0 ) {\n\t\t\t// If attributes or metadata changes set became empty, remove it from the diff item.\n\t\t\tdelete diffItem.attributes;\n\n\t\t\tif ( diffItem.state === undefined ) {\n\t\t\t\t// If there is no `state` change and no `attributes` change, remove the entry.\n\t\t\t\tthis._changedRoots.delete( rootName );\n\t\t\t}\n\t\t} else {\n\t\t\t// Make sure that, if a new object in the structure was created, it gets set.\n\t\t\tdiffItem.attributes = attrs;\n\n\t\t\tthis._changedRoots.set( rootName, diffItem );\n\t\t}\n\t}\n\n\t/**\n\t * Marks the given `item` in differ to be \"refreshed\". It means that the item will be marked as removed and inserted\n\t * in the differ changes set, so it will be effectively re-converted when the differ changes are handled by a dispatcher.\n\t *\n\t * @internal\n\t * @param item Item to refresh.\n\t */\n\tpublic _refreshItem( item: Item ): void {\n\t\tif ( this._isInInsertedElement( item.parent! ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._markRemove( item.parent!, item.startOffset!, item.offsetSize );\n\t\tthis._markInsert( item.parent!, item.startOffset!, item.offsetSize );\n\n\t\tthis._refreshedItems.add( item );\n\n\t\tconst range = Range._createOn( item );\n\n\t\tfor ( const marker of this._markerCollection.getMarkersIntersectingRange( range ) ) {\n\t\t\tconst markerData = marker.getData();\n\n\t\t\tthis.bufferMarkerChange( marker.name, markerData, markerData );\n\t\t}\n\n\t\t// Clear cache after each buffered operation as it is no longer valid.\n\t\tthis._cachedChanges = null;\n\t}\n\n\t/**\n\t * Buffers all the data related to given root like it was all just added to the editor.\n\t *\n\t * Following changes are buffered:\n\t *\n\t * * root is attached,\n\t * * all root content is inserted,\n\t * * all root attributes are added,\n\t * * all markers inside the root are added.\n\t *\n\t * @internal\n\t */\n\tpublic _bufferRootLoad( root: RootElement ): void {\n\t\tif ( !root.isAttached() ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._bufferRootStateChange( root.rootName, true );\n\t\tthis._markInsert( root, 0, root.maxOffset );\n\n\t\t// Buffering root attribute changes makes sense and is actually needed, even though we buffer root state change above.\n\t\t// Because the root state change is buffered, the root attributes changes are not returned by the differ.\n\t\t// But, if the root attribute is removed in the same change block, or the root is detached, then the differ results would be wrong.\n\t\t//\n\t\tfor ( const key of root.getAttributeKeys() ) {\n\t\t\tthis._bufferRootAttributeChange( root.rootName, key, null, root.getAttribute( key ) );\n\t\t}\n\n\t\tfor ( const marker of this._markerCollection ) {\n\t\t\tif ( marker.getRange().root == root ) {\n\t\t\t\tconst markerData = marker.getData();\n\n\t\t\t\tthis.bufferMarkerChange( marker.name, { ...markerData, range: null }, markerData );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Saves and handles an insert change.\n\t */\n\tprivate _markInsert( parent: Element | DocumentFragment, offset: number, howMany: number ) {\n\t\tif ( parent.root.is( 'rootElement' ) && !parent.root._isLoaded ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst changeItem = { type: 'insert', offset, howMany, count: this._changeCount++ } as const;\n\n\t\tthis._markChange( parent, changeItem );\n\t}\n\n\t/**\n\t * Saves and handles a remove change.\n\t */\n\tprivate _markRemove( parent: Element | DocumentFragment, offset: number, howMany: number ) {\n\t\tif ( parent.root.is( 'rootElement' ) && !parent.root._isLoaded ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst changeItem = { type: 'remove', offset, howMany, count: this._changeCount++ } as const;\n\n\t\tthis._markChange( parent, changeItem );\n\n\t\tthis._removeAllNestedChanges( parent, offset, howMany );\n\t}\n\n\t/**\n\t * Saves and handles an attribute change.\n\t */\n\tprivate _markAttribute( item: Item ): void {\n\t\tif ( item.root.is( 'rootElement' ) && !item.root._isLoaded ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst changeItem = { type: 'attribute', offset: item.startOffset!, howMany: item.offsetSize, count: this._changeCount++ } as const;\n\n\t\tthis._markChange( item.parent as Element, changeItem );\n\t}\n\n\t/**\n\t * Saves and handles a model change.\n\t */\n\tprivate _markChange( parent: Element | DocumentFragment, changeItem: ChangeItem ): void {\n\t\t// First, make a snapshot of this parent's children (it will be made only if it was not made before).\n\t\tthis._makeSnapshot( parent );\n\n\t\t// Then, get all changes that already were done on the element (empty array if this is the first change).\n\t\tconst changes = this._getChangesForElement( parent );\n\n\t\t// Then, look through all the changes, and transform them or the new change.\n\t\tthis._handleChange( changeItem, changes );\n\n\t\t// Add the new change.\n\t\tchanges.push( changeItem );\n\n\t\t// Remove incorrect changes. During transformation some change might be, for example, included in another.\n\t\t// In that case, the change will have `howMany` property set to `0` or less. We need to remove those changes.\n\t\tfor ( let i = 0; i < changes.length; i++ ) {\n\t\t\tif ( changes[ i ].howMany < 1 ) {\n\t\t\t\tchanges.splice( i, 1 );\n\n\t\t\t\ti--;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Gets an array of changes that have already been saved for a given element.\n\t */\n\tprivate _getChangesForElement( element: Element | DocumentFragment ): Array<ChangeItem> {\n\t\tlet changes: Array<ChangeItem>;\n\n\t\tif ( this._changesInElement.has( element ) ) {\n\t\t\tchanges = this._changesInElement.get( element )!;\n\t\t} else {\n\t\t\tchanges = [];\n\n\t\t\tthis._changesInElement.set( element, changes );\n\t\t}\n\n\t\treturn changes;\n\t}\n\n\t/**\n\t * Saves a children snapshot for a given element.\n\t */\n\tprivate _makeSnapshot( element: Element | DocumentFragment ): void {\n\t\tif ( !this._elementSnapshots.has( element ) ) {\n\t\t\tthis._elementSnapshots.set( element, _getChildrenSnapshot( element.getChildren() ) );\n\t\t}\n\t}\n\n\t/**\n\t * For a given newly saved change, compares it with a change already done on the element and modifies the incoming\n\t * change and/or the old change.\n\t *\n\t * @param inc Incoming (new) change.\n\t * @param changes An array containing all the changes done on that element.\n\t */\n\tprivate _handleChange( inc: ChangeItem, changes: Array<ChangeItem> ): void {\n\t\t// We need a helper variable that will store how many nodes are to be still handled for this change item.\n\t\t// `nodesToHandle` (how many nodes still need to be handled) and `howMany` (how many nodes were affected)\n\t\t// needs to be differentiated.\n\t\t//\n\t\t// This comes up when there are multiple changes that are affected by `inc` change item.\n\t\t//\n\t\t// For example: assume two insert changes: `{ offset: 2, howMany: 1 }` and `{ offset: 5, howMany: 1 }`.\n\t\t// Assume that `inc` change is remove `{ offset: 2, howMany: 2, nodesToHandle: 2 }`.\n\t\t//\n\t\t// Then, we:\n\t\t// - \"forget\" about first insert change (it is \"eaten\" by remove),\n\t\t// - because of that, at the end we will want to remove only one node (`nodesToHandle = 1`),\n\t\t// - but still we have to change offset of the second insert change from `5` to `3`!\n\t\t//\n\t\t// So, `howMany` does not change throughout items transformation and keeps information about how many nodes were affected,\n\t\t// while `nodesToHandle` means how many nodes need to be handled after the change item is transformed by other changes.\n\t\tinc.nodesToHandle = inc.howMany;\n\n\t\tfor ( const old of changes ) {\n\t\t\tconst incEnd = inc.offset + inc.howMany;\n\t\t\tconst oldEnd = old.offset + old.howMany;\n\n\t\t\tif ( inc.type == 'insert' ) {\n\t\t\t\tif ( old.type == 'insert' ) {\n\t\t\t\t\tif ( inc.offset <= old.offset ) {\n\t\t\t\t\t\told.offset += inc.howMany;\n\t\t\t\t\t} else if ( inc.offset < oldEnd ) {\n\t\t\t\t\t\told.howMany += inc.nodesToHandle;\n\t\t\t\t\t\tinc.nodesToHandle = 0;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif ( old.type == 'remove' ) {\n\t\t\t\t\tif ( inc.offset < old.offset ) {\n\t\t\t\t\t\told.offset += inc.howMany;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif ( old.type == 'attribute' ) {\n\t\t\t\t\tif ( inc.offset <= old.offset ) {\n\t\t\t\t\t\told.offset += inc.howMany;\n\t\t\t\t\t} else if ( inc.offset < oldEnd ) {\n\t\t\t\t\t\t// This case is more complicated, because attribute change has to be split into two.\n\t\t\t\t\t\t// Example (assume that uppercase and lowercase letters mean different attributes):\n\t\t\t\t\t\t//\n\t\t\t\t\t\t// initial state:\t\tabcxyz\n\t\t\t\t\t\t// attribute change:\taBCXYz\n\t\t\t\t\t\t// incoming insert:\t\taBCfooXYz\n\t\t\t\t\t\t//\n\t\t\t\t\t\t// Change ranges cannot intersect because each item has to be described exactly (it was either\n\t\t\t\t\t\t// not changed, inserted, removed, or its attribute was changed). That's why old attribute\n\t\t\t\t\t\t// change has to be split and both parts has to be handled separately from now on.\n\t\t\t\t\t\tconst howMany = old.howMany;\n\n\t\t\t\t\t\told.howMany = inc.offset - old.offset;\n\n\t\t\t\t\t\t// Add the second part of attribute change to the beginning of processed array so it won't\n\t\t\t\t\t\t// be processed again in this loop.\n\t\t\t\t\t\tchanges.unshift( {\n\t\t\t\t\t\t\ttype: 'attribute',\n\t\t\t\t\t\t\toffset: incEnd,\n\t\t\t\t\t\t\thowMany: howMany - old.howMany,\n\t\t\t\t\t\t\tcount: this._changeCount++\n\t\t\t\t\t\t} );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( inc.type == 'remove' ) {\n\t\t\t\tif ( old.type == 'insert' ) {\n\t\t\t\t\tif ( incEnd <= old.offset ) {\n\t\t\t\t\t\told.offset -= inc.howMany;\n\t\t\t\t\t} else if ( incEnd <= oldEnd ) {\n\t\t\t\t\t\tif ( inc.offset < old.offset ) {\n\t\t\t\t\t\t\tconst intersectionLength = incEnd - old.offset;\n\n\t\t\t\t\t\t\told.offset = inc.offset;\n\n\t\t\t\t\t\t\told.howMany -= intersectionLength;\n\t\t\t\t\t\t\tinc.nodesToHandle -= intersectionLength;\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\told.howMany -= inc.nodesToHandle;\n\t\t\t\t\t\t\tinc.nodesToHandle = 0;\n\t\t\t\t\t\t}\n\t\t\t\t\t} else {\n\t\t\t\t\t\tif ( inc.offset <= old.offset ) {\n\t\t\t\t\t\t\tinc.nodesToHandle -= old.howMany;\n\t\t\t\t\t\t\told.howMany = 0;\n\t\t\t\t\t\t} else if ( inc.offset < oldEnd ) {\n\t\t\t\t\t\t\tconst intersectionLength = oldEnd - inc.offset;\n\n\t\t\t\t\t\t\told.howMany -= intersectionLength;\n\t\t\t\t\t\t\tinc.nodesToHandle -= intersectionLength;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif ( old.type == 'remove' ) {\n\t\t\t\t\tif ( incEnd <= old.offset ) {\n\t\t\t\t\t\told.offset -= inc.howMany;\n\t\t\t\t\t} else if ( inc.offset < old.offset ) {\n\t\t\t\t\t\tinc.nodesToHandle += old.howMany;\n\t\t\t\t\t\told.howMany = 0;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif ( old.type == 'attribute' ) {\n\t\t\t\t\tif ( incEnd <= old.offset ) {\n\t\t\t\t\t\told.offset -= inc.howMany;\n\t\t\t\t\t} else if ( inc.offset < old.offset ) {\n\t\t\t\t\t\tconst intersectionLength = incEnd - old.offset;\n\n\t\t\t\t\t\told.offset = inc.offset;\n\t\t\t\t\t\told.howMany -= intersectionLength;\n\t\t\t\t\t} else if ( inc.offset < oldEnd ) {\n\t\t\t\t\t\tif ( incEnd <= oldEnd ) {\n\t\t\t\t\t\t\t// On first sight in this case we don't need to split attribute operation into two.\n\t\t\t\t\t\t\t// However the changes set is later converted to actions (see `_generateActionsFromChanges`).\n\t\t\t\t\t\t\t// For that reason, no two changes may intersect.\n\t\t\t\t\t\t\t// So we cannot have an attribute change that \"contains\" remove change.\n\t\t\t\t\t\t\t// Attribute change needs to be split.\n\t\t\t\t\t\t\tconst howMany = old.howMany;\n\n\t\t\t\t\t\t\told.howMany = inc.offset - old.offset;\n\n\t\t\t\t\t\t\tconst howManyAfter = howMany - old.howMany - inc.nodesToHandle;\n\n\t\t\t\t\t\t\t// Add the second part of attribute change to the beginning of processed array so it won't\n\t\t\t\t\t\t\t// be processed again in this loop.\n\t\t\t\t\t\t\tchanges.unshift( {\n\t\t\t\t\t\t\t\ttype: 'attribute',\n\t\t\t\t\t\t\t\toffset: inc.offset,\n\t\t\t\t\t\t\t\thowMany: howManyAfter,\n\t\t\t\t\t\t\t\tcount: this._changeCount++\n\t\t\t\t\t\t\t} );\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\told.howMany -= oldEnd - inc.offset;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( inc.type == 'attribute' ) {\n\t\t\t\t// In case of attribute change, `howMany` should be kept same as `nodesToHandle`. It's not an error.\n\t\t\t\tif ( old.type == 'insert' ) {\n\t\t\t\t\tif ( inc.offset < old.offset && incEnd > old.offset ) {\n\t\t\t\t\t\tif ( incEnd > oldEnd ) {\n\t\t\t\t\t\t\t// This case is similar to a case described when incoming change was insert and old change was attribute.\n\t\t\t\t\t\t\t// See comment above.\n\t\t\t\t\t\t\t//\n\t\t\t\t\t\t\t// This time incoming change is attribute. We need to split incoming change in this case too.\n\t\t\t\t\t\t\t// However this time, the second part of the attribute change needs to be processed further\n\t\t\t\t\t\t\t// because there might be other changes that it collides with.\n\t\t\t\t\t\t\tconst attributePart = {\n\t\t\t\t\t\t\t\ttype: 'attribute',\n\t\t\t\t\t\t\t\toffset: oldEnd,\n\t\t\t\t\t\t\t\thowMany: incEnd - oldEnd,\n\t\t\t\t\t\t\t\tcount: this._changeCount++\n\t\t\t\t\t\t\t} as const;\n\n\t\t\t\t\t\t\tthis._handleChange( attributePart, changes );\n\n\t\t\t\t\t\t\tchanges.push( attributePart );\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tinc.nodesToHandle = old.offset - inc.offset;\n\t\t\t\t\t\tinc.howMany = inc.nodesToHandle;\n\t\t\t\t\t} else if ( inc.offset >= old.offset && inc.offset < oldEnd ) {\n\t\t\t\t\t\tif ( incEnd > oldEnd ) {\n\t\t\t\t\t\t\tinc.nodesToHandle = incEnd - oldEnd;\n\t\t\t\t\t\t\tinc.offset = oldEnd;\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tinc.nodesToHandle = 0;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif ( old.type == 'remove' ) {\n\t\t\t\t\t// This is a case when attribute change \"contains\" remove change.\n\t\t\t\t\t// The attribute change needs to be split into two because changes cannot intersect.\n\t\t\t\t\tif ( inc.offset < old.offset && incEnd > old.offset ) {\n\t\t\t\t\t\tconst attributePart = {\n\t\t\t\t\t\t\ttype: 'attribute',\n\t\t\t\t\t\t\toffset: old.offset,\n\t\t\t\t\t\t\thowMany: incEnd - old.offset,\n\t\t\t\t\t\t\tcount: this._changeCount++\n\t\t\t\t\t\t} as const;\n\n\t\t\t\t\t\tthis._handleChange( attributePart, changes );\n\n\t\t\t\t\t\tchanges.push( attributePart );\n\n\t\t\t\t\t\tinc.nodesToHandle = old.offset - inc.offset;\n\t\t\t\t\t\tinc.howMany = inc.nodesToHandle;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif ( old.type == 'attribute' ) {\n\t\t\t\t\t// There are only two conflicting scenarios possible here:\n\t\t\t\t\tif ( inc.offset >= old.offset && incEnd <= oldEnd ) {\n\t\t\t\t\t\t// `old` change includes `inc` change, or they are the same.\n\t\t\t\t\t\tinc.nodesToHandle = 0;\n\t\t\t\t\t\tinc.howMany = 0;\n\t\t\t\t\t\tinc.offset = 0;\n\t\t\t\t\t} else if ( inc.offset <= old.offset && incEnd >= oldEnd ) {\n\t\t\t\t\t\t// `inc` change includes `old` change.\n\t\t\t\t\t\told.howMany = 0;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tinc.howMany = inc.nodesToHandle;\n\t\tdelete inc.nodesToHandle;\n\t}\n\n\t/**\n\t * Returns an object with a single insert change description.\n\t *\n\t * @param parent The element in which the change happened.\n\t * @param offset The offset at which change happened.\n\t * @param elementSnapshot The snapshot of the removed element a character.\n\t * @returns The diff item.\n\t */\n\tprivate _getInsertDiff(\n\t\tparent: Element | DocumentFragment,\n\t\toffset: number,\n\t\telementSnapshot: { name: string; attributes: Map<string, unknown> }\n\t): DiffItemInsert & DiffItemInternal {\n\t\treturn {\n\t\t\ttype: 'insert',\n\t\t\tposition: Position._createAt( parent, offset ),\n\t\t\tname: elementSnapshot.name,\n\t\t\tattributes: new Map( elementSnapshot.attributes ),\n\t\t\tlength: 1,\n\t\t\tchangeCount: this._changeCount++\n\t\t};\n\t}\n\n\t/**\n\t * Returns an object with a single remove change description.\n\t *\n\t * @param parent The element in which change happened.\n\t * @param offset The offset at which change happened.\n\t * @param elementSnapshot The snapshot of the removed element a character.\n\t * @returns The diff item.\n\t */\n\tprivate _getRemoveDiff(\n\t\tparent: Element | DocumentFragment,\n\t\toffset: number,\n\t\telementSnapshot: { name: string; attributes: Map<string, unknown> }\n\t): DiffItemRemove & DiffItemInternal {\n\t\treturn {\n\t\t\ttype: 'remove',\n\t\t\tposition: Position._createAt( parent, offset ),\n\t\t\tname: elementSnapshot.name,\n\t\t\tattributes: new Map( elementSnapshot.attributes ),\n\t\t\tlength: 1,\n\t\t\tchangeCount: this._changeCount++\n\t\t};\n\t}\n\n\t/**\n\t * Returns an array of objects where each one is a single attribute change description.\n\t *\n\t * @param range The range where the change happened.\n\t * @param oldAttributes A map, map iterator or compatible object that contains attributes before the change.\n\t * @param newAttributes A map, map iterator or compatible object that contains attributes after the change.\n\t * @returns An array containing one or more diff items.\n\t */\n\tprivate _getAttributesDiff(\n\t\trange: Range,\n\t\toldAttributes: Map<string, unknown>,\n\t\tnewAttributes: Map<string, unknown>\n\t): Array<DiffItemAttribute & DiffItemInternal> {\n\t\t// Results holder.\n\t\tconst diffs: Array<DiffItemAttribute & DiffItemInternal> = [];\n\n\t\t// Clone new attributes as we will be performing changes on this object.\n\t\tnewAttributes = new Map( newAttributes );\n\n\t\t// Look through old attributes.\n\t\tfor ( const [ key, oldValue ] of oldAttributes ) {\n\t\t\t// Check what is the new value of the attribute (or if it was removed).\n\t\t\tconst newValue = newAttributes.has( key ) ? newAttributes.get( key ) : null;\n\n\t\t\t// If values are different (or attribute was removed)...\n\t\t\tif ( newValue !== oldValue ) {\n\t\t\t\t// Add diff item.\n\t\t\t\tdiffs.push( {\n\t\t\t\t\ttype: 'attribute',\n\t\t\t\t\tposition: range.start,\n\t\t\t\t\trange: range.clone(),\n\t\t\t\t\tlength: 1,\n\t\t\t\t\tattributeKey: key,\n\t\t\t\t\tattributeOldValue: oldValue,\n\t\t\t\t\tattributeNewValue: newValue,\n\t\t\t\t\tchangeCount: this._changeCount++\n\t\t\t\t} );\n\t\t\t}\n\n\t\t\t// Prevent returning two diff items for the same change.\n\t\t\tnewAttributes.delete( key );\n\t\t}\n\n\t\t// Look through new attributes that weren't handled above.\n\t\tfor ( const [ key, newValue ] of newAttributes ) {\n\t\t\t// Each of them is a new attribute. Add diff item.\n\t\t\tdiffs.push( {\n\t\t\t\ttype: 'attribute',\n\t\t\t\tposition: range.start,\n\t\t\t\trange: range.clone(),\n\t\t\t\tlength: 1,\n\t\t\t\tattributeKey: key,\n\t\t\t\tattributeOldValue: null,\n\t\t\t\tattributeNewValue: newValue,\n\t\t\t\tchangeCount: this._changeCount++\n\t\t\t} );\n\t\t}\n\n\t\treturn diffs;\n\t}\n\n\t/**\n\t * Checks whether given element or any of its parents is an element that is buffered as an inserted element.\n\t */\n\tprivate _isInInsertedElement( element: Element | DocumentFragment ): boolean {\n\t\tconst parent = element.parent;\n\n\t\tif ( !parent ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst changes = this._changesInElement.get( parent );\n\t\tconst offset = element.startOffset;\n\n\t\tif ( changes ) {\n\t\t\tfor ( const change of changes ) {\n\t\t\t\tif ( change.type == 'insert' && offset! >= change.offset && offset! < change.offset + change.howMany ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn this._isInInsertedElement( parent );\n\t}\n\n\t/**\n\t * Removes deeply all buffered changes that are registered in elements from range specified by `parent`, `offset`\n\t * and `howMany`.\n\t */\n\tprivate _removeAllNestedChanges( parent: Element | DocumentFragment, offset: number, howMany: number ) {\n\t\tconst range = new Range( Position._createAt( parent, offset ), Position._createAt( parent, offset + howMany ) );\n\n\t\tfor ( const item of range.getItems( { shallow: true } ) ) {\n\t\t\tif ( item.is( 'element' ) ) {\n\t\t\t\tthis._elementSnapshots.delete( item );\n\t\t\t\tthis._changesInElement.delete( item );\n\n\t\t\t\tthis._removeAllNestedChanges( item, 0, item.maxOffset );\n\t\t\t}\n\t\t}\n\t}\n}\n\ninterface ChangeItem {\n\ttype: 'insert' | 'remove' | 'attribute';\n\toffset: number;\n\thowMany: number;\n\tcount: number;\n\tnodesToHandle?: number;\n}\n\n/**\n * Returns an array that is a copy of passed child list with the exception that text nodes are split to one or more\n * objects, each representing one character and attributes set on that character.\n */\nfunction _getChildrenSnapshot( children: Iterable<Node> ) {\n\tconst snapshot = [];\n\n\tfor ( const child of children ) {\n\t\tif ( child.is( '$text' ) ) {\n\t\t\tfor ( let i = 0; i < child.data.length; i++ ) {\n\t\t\t\tsnapshot.push( {\n\t\t\t\t\tname: '$text',\n\t\t\t\t\tattributes: new Map( child.getAttributes() )\n\t\t\t\t} );\n\t\t\t}\n\t\t} else {\n\t\t\tsnapshot.push( {\n\t\t\t\tname: ( child as Element ).name,\n\t\t\t\tattributes: new Map( child.getAttributes() )\n\t\t\t} );\n\t\t}\n\t}\n\n\treturn snapshot;\n}\n\n/**\n * Generates array of actions for given changes set.\n * It simulates what `diff` function does.\n * Generated actions are:\n * - 'e' for 'equal' - when item at that position did not change,\n * - 'i' for 'insert' - when item at that position was inserted,\n * - 'r' for 'remove' - when item at that position was removed,\n * - 'a' for 'attribute' - when item at that position has it attributes changed.\n *\n * Example (assume that uppercase letters have bold attribute, compare with function code):\n *\n * children before:\tfooBAR\n * children after:\tfoxybAR\n *\n * changes: type: remove, offset: 1, howMany: 1\n *\t\t\ttype: insert, offset: 2, howMany: 2\n *\t\t\ttype: attribute, offset: 4, howMany: 1\n *\n * expected actions: equal (f), remove (o), equal (o), insert (x), insert (y), attribute (b), equal (A), equal (R)\n *\n * steps taken by th script:\n *\n * 1. change = \"type: remove, offset: 1, howMany: 1\"; offset = 0; oldChildrenHandled = 0\n * 1.1 between this change and the beginning is one not-changed node, fill with one equal action, one old child has been handled\n * 1.2 this change removes one node, add one remove action\n * 1.3 change last visited `offset` to 1\n * 1.4 since an old child has been removed, one more old child has been handled\n * 1.5 actions at this point are: equal, remove\n *\n * 2. change = \"type: insert, offset: 2, howMany: 2\"; offset = 1; oldChildrenHandled = 2\n * 2.1 between this change and previous change is one not-changed node, add equal action, another one old children has been handled\n * 2.2 this change inserts two nodes, add two insert actions\n * 2.3 change last visited offset to the end of the inserted range, that is 4\n * 2.4 actions at this point are: equal, remove, equal, insert, insert\n *\n * 3. change = \"type: attribute, offset: 4, howMany: 1\"; offset = 4, oldChildrenHandled = 3\n * 3.1 between this change and previous change are no not-changed nodes\n * 3.2 this change changes one node, add one attribute action\n * 3.3 change last visited `offset` to the end of change range, that is 5\n * 3.4 since an old child has been changed, one more old child has been handled\n * 3.5 actions at this point are: equal, remove, equal, insert, insert, attribute\n *\n * 4. after loop oldChildrenHandled = 4, oldChildrenLength = 6 (fooBAR is 6 characters)\n * 4.1 fill up with two equal actions\n *\n * The result actions are: equal, remove, equal, insert, insert, attribute, equal, equal.\n */\nfunction _generateActionsFromChanges( oldChildrenLength: number, changes: Array<ChangeItem> ) {\n\tconst actions: Array<string> = [];\n\n\tlet offset = 0;\n\tlet oldChildrenHandled = 0;\n\n\t// Go through all buffered changes.\n\tfor ( const change of changes ) {\n\t\t// First, fill \"holes\" between changes with \"equal\" actions.\n\t\tif ( change.offset > offset ) {\n\t\t\tfor ( let i = 0; i < change.offset - offset; i++ ) {\n\t\t\t\tactions.push( 'e' );\n\t\t\t}\n\n\t\t\toldChildrenHandled += change.offset - offset;\n\t\t}\n\n\t\t// Then, fill up actions accordingly to change type.\n\t\tif ( change.type == 'insert' ) {\n\t\t\tfor ( let i = 0; i < change.howMany; i++ ) {\n\t\t\t\tactions.push( 'i' );\n\t\t\t}\n\n\t\t\t// The last handled offset is after inserted range.\n\t\t\toffset = change.offset + change.howMany;\n\t\t} else if ( change.type == 'remove' ) {\n\t\t\tfor ( let i = 0; i < change.howMany; i++ ) {\n\t\t\t\tactions.push( 'r' );\n\t\t\t}\n\n\t\t\t// The last handled offset is at the position where the nodes were removed.\n\t\t\toffset = change.offset;\n\t\t\t// We removed `howMany` old nodes, update `oldChildrenHandled`.\n\t\t\toldChildrenHandled += change.howMany;\n\t\t} else {\n\t\t\tactions.push( ...'a'.repeat( change.howMany ).split( '' ) );\n\n\t\t\t// The last handled offset is at the position after the changed range.\n\t\t\toffset = change.offset + change.howMany;\n\t\t\t// We changed `howMany` old nodes, update `oldChildrenHandled`.\n\t\t\toldChildrenHandled += change.howMany;\n\t\t}\n\t}\n\n\t// Fill \"equal\" actions at the end of actions set. Use `oldChildrenHandled` to see how many children\n\t// has not been changed / removed at the end of their parent.\n\tif ( oldChildrenHandled < oldChildrenLength ) {\n\t\tfor ( let i = 0; i < oldChildrenLength - oldChildrenHandled - offset; i++ ) {\n\t\t\tactions.push( 'e' );\n\t\t}\n\t}\n\n\treturn actions;\n}\n\n/**\n * Filter callback for Array.filter that filters out change entries that are in graveyard.\n */\nfunction _changesInGraveyardFilter( entry: DiffItem ) {\n\tconst posInGy = 'position' in entry && entry.position!.root.rootName == '$graveyard';\n\tconst rangeInGy = 'range' in entry && entry.range.root.rootName == '$graveyard';\n\n\treturn !posInGy && !rangeInGy;\n}\n\n/**\n * The single diff item.\n *\n * Could be one of:\n *\n * * {@link module:engine/model/differ~DiffItemInsert `DiffItemInsert`},\n * * {@link module:engine/model/differ~DiffItemRemove `DiffItemRemove`},\n * * {@link module:engine/model/differ~DiffItemAttribute `DiffItemAttribute`}.\n */\nexport type DiffItem = DiffItemInsert | DiffItemRemove | DiffItemAttribute;\n\n/**\n * A single diff item for inserted nodes.\n */\nexport interface DiffItemInsert {\n\n\t/**\n\t * The type of diff item.\n\t */\n\ttype: 'insert';\n\n\t/**\n\t * The name of the inserted elements or `'$text'` for a text node.\n\t */\n\tname: string;\n\n\t/**\n\t * Map of attributes that were set on the item while it was inserted.\n\t */\n\tattributes: Map<string, unknown>;\n\n\t/**\n\t * The position where the node was inserted.\n\t */\n\tposition: Position;\n\n\t/**\n\t * The length of an inserted text node. For elements, it is always 1 as each inserted element is counted as a one.\n\t */\n\tlength: number;\n}\n\n/**\n * A single diff item for removed nodes.\n */\nexport interface DiffItemRemove {\n\n\t/**\n\t * The type of diff item.\n\t */\n\ttype: 'remove';\n\n\t/**\n\t * The name of the removed element or `'$text'` for a text node.\n\t */\n\tname: string;\n\n\t/**\n\t * Map of attributes that were set on the item while it was removed.\n\t */\n\tattributes: Map<string, unknown>;\n\n\t/**\n\t * The position where the node was removed.\n\t */\n\tposition: Position;\n\n\t/**\n\t * The length of a removed text node. For elements, it is always 1, as each removed element is counted as a one.\n\t */\n\tlength: number;\n}\n\n/**\n * A single diff item for attribute change.\n */\nexport interface DiffItemAttribute {\n\n\t/**\n\t * The type of diff item.\n\t */\n\ttype: 'attribute';\n\n\t/**\n\t * The name of the changed attribute.\n\t */\n\tattributeKey: string;\n\n\t/**\n\t * An attribute previous value (before change).\n\t */\n\tattributeOldValue: unknown;\n\n\t/**\n\t * An attribute new value (after change).\n\t */\n\tattributeNewValue: unknown;\n\n\t/**\n\t * The range where the change happened.\n\t */\n\trange: Range;\n}\n\n/**\n * A single diff item for a changed root.\n */\nexport interface DiffItemRoot {\n\n\t/**\n\t * Name of the changed root.\n\t */\n\tname: string;\n\n\t/**\n\t * Set accordingly if the root got attached or detached. Otherwise, not set.\n\t */\n\tstate?: 'attached' | 'detached';\n\n\t/**\n\t * Keeps all attribute changes that happened on the root.\n\t *\n\t * The keys are keys of the changed attributes. The values are objects containing the attribute value before the change\n\t * (`oldValue`) and after the change (`newValue`).\n\t *\n\t * Note, that if the root state changed (`state` is set), then `attributes` property will not be set. All attributes should be\n\t * handled together with the root being attached or detached.\n\t */\n\tattributes?: Record<string, { oldValue: unknown; newValue: unknown }>;\n}\n\ninterface DiffItemInternal {\n\tchangeCount?: number;\n\tposition?: Position;\n\tlength?: number;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport type Operation from './operation/operation';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\n/**\n * @module engine/model/history\n */\n\n/**\n * `History` keeps the track of all the operations applied to the {@link module:engine/model/document~Document document}.\n */\nexport default class History {\n\t/**\n\t * Operations added to the history.\n\t */\n\tprivate _operations: Array<Operation> = [];\n\n\t/**\n\t * Holds an information which {@link module:engine/model/operation/operation~Operation operation} undoes which\n\t * {@link module:engine/model/operation/operation~Operation operation}.\n\t *\n\t * Keys of the map are \"undoing operations\", that is operations that undone some other operations. For each key, the\n\t * value is an operation that has been undone by the \"undoing operation\".\n\t */\n\tprivate _undoPairs: Map<Operation, Operation> = new Map();\n\n\t/**\n\t * Holds all undone operations.\n\t */\n\tprivate _undoneOperations: Set<Operation> = new Set();\n\n\t/**\n\t * A map that allows retrieving the operations fast based on the given base version.\n\t */\n\tprivate _baseVersionToOperationIndex: Map<number, number> = new Map();\n\n\t/**\n\t * The history version.\n\t */\n\tprivate _version: number = 0;\n\n\t/**\n\t * The gap pairs kept in the <from,to> format.\n\t *\n\t * Anytime the `history.version` is set to a version larger than `history.version + 1`,\n\t * a new <lastHistoryVersion, newHistoryVersion> entry is added to the map.\n\t */\n\tprivate _gaps: Map<number, number> = new Map();\n\n\t/**\n\t * The version of the last operation in the history.\n\t *\n\t * The history version is incremented automatically when a new operation is added to the history.\n\t * Setting the version manually should be done only in rare circumstances when a gap is planned\n\t * between history versions. When doing so, a gap will be created and the history will accept adding\n\t * an operation with base version equal to the new history version.\n\t */\n\tpublic get version(): number {\n\t\treturn this._version;\n\t}\n\n\tpublic set version( version: number ) {\n\t\t// Store a gap if there are some operations already in the history and the\n\t\t// new version does not increment the latest one.\n\t\tif ( this._operations.length && version > this._version + 1 ) {\n\t\t\tthis._gaps.set( this._version, version );\n\t\t}\n\n\t\tthis._version = version;\n\t}\n\n\t/**\n\t * The last history operation.\n\t */\n\tpublic get lastOperation(): Operation | undefined {\n\t\treturn this._operations[ this._operations.length - 1 ];\n\t}\n\n\t/**\n\t * Adds an operation to the history and increments the history version.\n\t *\n\t * The operation's base version should be equal to the history version. Otherwise an error is thrown.\n\t */\n\tpublic addOperation( operation: Operation ): void {\n\t\tif ( operation.baseVersion !== this.version ) {\n\t\t\t/**\n\t\t\t * Only operations with matching versions can be added to the history.\n\t\t\t *\n\t\t\t * @error model-document-history-addoperation-incorrect-version\n\t\t\t * @param errorData The operation and the current document history version.\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-document-history-addoperation-incorrect-version', this, {\n\t\t\t\toperation,\n\t\t\t\thistoryVersion: this.version\n\t\t\t} );\n\t\t}\n\n\t\tthis._operations.push( operation );\n\t\tthis._version++;\n\n\t\tthis._baseVersionToOperationIndex.set( operation.baseVersion, this._operations.length - 1 );\n\t}\n\n\t/**\n\t * Returns operations from the given range of operation base versions that were added to the history.\n\t *\n\t * Note that there may be gaps in operations base versions.\n\t *\n\t * @param fromBaseVersion Base version from which operations should be returned (inclusive).\n\t * @param toBaseVersion Base version up to which operations should be returned (exclusive).\n * @returns History operations for the given range, in chronological order.\n\t */\n\tpublic getOperations( fromBaseVersion?: number, toBaseVersion: number = this.version ): Array<Operation> {\n\t\t// When there is no operation in the history, return an empty array.\n\t\t// After that we can be sure that `firstOperation`, `lastOperation` are not nullish.\n\t\tif ( !this._operations.length ) {\n\t\t\treturn [];\n\t\t}\n\n\t\tconst firstOperation = this._operations[ 0 ];\n\n\t\tif ( fromBaseVersion === undefined ) {\n\t\t\tfromBaseVersion = firstOperation.baseVersion!;\n\t\t}\n\n\t\t// Change exclusive `toBaseVersion` to inclusive, so it will refer to the actual index.\n\t\t// Thanks to that mapping from base versions to operation indexes are possible.\n\t\tlet inclusiveTo = toBaseVersion - 1;\n\n\t\t// Check if \"from\" or \"to\" point to a gap between versions.\n\t\t// If yes, then change the incorrect position to the proper side of the gap.\n\t\t// Thanks to it, it will be possible to get index of the operation.\n\t\tfor ( const [ gapFrom, gapTo ] of this._gaps ) {\n\t\t\tif ( fromBaseVersion > gapFrom && fromBaseVersion < gapTo ) {\n\t\t\t\tfromBaseVersion = gapTo;\n\t\t\t}\n\n\t\t\tif ( inclusiveTo > gapFrom && inclusiveTo < gapTo ) {\n\t\t\t\tinclusiveTo = gapFrom - 1;\n\t\t\t}\n\t\t}\n\n\t\t// If the whole range is outside of the operation versions, then return an empty array.\n\t\tif ( inclusiveTo < firstOperation.baseVersion! || fromBaseVersion > this.lastOperation!.baseVersion! ) {\n\t\t\treturn [];\n\t\t}\n\n\t\tlet fromIndex = this._baseVersionToOperationIndex.get( fromBaseVersion );\n\n\t\t// If the range starts before the first operation, then use the first operation as the range's start.\n\t\tif ( fromIndex === undefined ) {\n\t\t\tfromIndex = 0;\n\t\t}\n\n\t\tlet toIndex = this._baseVersionToOperationIndex.get( inclusiveTo );\n\n\t\t// If the range ends after the last operation, then use the last operation as the range's end.\n\t\tif ( toIndex === undefined ) {\n\t\t\ttoIndex = this._operations.length - 1;\n\t\t}\n\n\t\t// Return the part of the history operations based on the calculated start index and end index.\n\t\treturn this._operations.slice(\n\t\t\tfromIndex,\n\n\t\t\t// The `toIndex` should be included in the returned operations, so add `1`.\n\t\t\ttoIndex + 1\n\t\t);\n\t}\n\n\t/**\n\t * Returns operation from the history that bases on given `baseVersion`.\n\t *\n\t * @param baseVersion Base version of the operation to get.\n\t * @returns Operation with given base version or `undefined` if there is no such operation in history.\n\t */\n\tpublic getOperation( baseVersion: number ): Operation | undefined {\n\t\tconst operationIndex = this._baseVersionToOperationIndex.get( baseVersion );\n\n\t\tif ( operationIndex === undefined ) {\n\t\t\treturn;\n\t\t}\n\n\t\treturn this._operations[ operationIndex ];\n\t}\n\n\t/**\n\t * Marks in history that one operation is an operation that is undoing the other operation. By marking operation this way,\n\t * history is keeping more context information about operations, which helps in operational transformation.\n\t *\n\t * @param undoneOperation Operation which is undone by `undoingOperation`.\n\t * @param undoingOperation Operation which undoes `undoneOperation`.\n\t */\n\tpublic setOperationAsUndone( undoneOperation: Operation, undoingOperation: Operation ): void {\n\t\tthis._undoPairs.set( undoingOperation, undoneOperation );\n\t\tthis._undoneOperations.add( undoneOperation );\n\t}\n\n\t/**\n\t * Checks whether given `operation` is undoing any other operation.\n\t *\n\t * @param operation Operation to check.\n\t * @returns `true` if given `operation` is undoing any other operation, `false` otherwise.\n\t */\n\tpublic isUndoingOperation( operation: Operation ): boolean {\n\t\treturn this._undoPairs.has( operation );\n\t}\n\n\t/**\n\t * Checks whether given `operation` has been undone by any other operation.\n\t *\n\t * @param operation Operation to check.\n\t * @returns `true` if given `operation` has been undone any other operation, `false` otherwise.\n\t */\n\tpublic isUndoneOperation( operation: Operation ): boolean {\n\t\treturn this._undoneOperations.has( operation );\n\t}\n\n\t/**\n\t * For given `undoingOperation`, returns the operation which has been undone by it.\n\t *\n\t * @returns Operation that has been undone by given `undoingOperation` or `undefined`\n\t * if given `undoingOperation` is not undoing any other operation.\n\t */\n\tpublic getUndoneOperation( undoingOperation: Operation ): Operation | undefined {\n\t\treturn this._undoPairs.get( undoingOperation );\n\t}\n\n\t/**\n\t * Resets the history of operations.\n\t */\n\tpublic reset(): void {\n\t\tthis._version = 0;\n\t\tthis._undoPairs = new Map();\n\t\tthis._operations = [];\n\t\tthis._undoneOperations = new Set();\n\t\tthis._gaps = new Map();\n\t\tthis._baseVersionToOperationIndex = new Map();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/rootelement\n */\n\nimport Element from './element';\n\nimport type Document from './document';\n\n/**\n * Type of {@link module:engine/model/element~Element} that is a root of a model tree.\n */\nexport default class RootElement extends Element {\n\t/**\n\t * Unique root name used to identify this root element by {@link module:engine/model/document~Document}.\n\t */\n\tpublic override readonly rootName: string;\n\n\t/**\n\t * Document that is an owner of this root.\n\t */\n\tprivate readonly _document: Document;\n\n\t/**\n\t * @internal\n\t */\n\tpublic _isAttached = true;\n\n\t/**\n\t * Informs if the root element is loaded (default).\n\t *\n\t * @internal\n\t */\n\tpublic _isLoaded = true;\n\n\t/**\n\t * Creates root element.\n\t *\n\t * @param document Document that is an owner of this root.\n\t * @param name Node name.\n\t * @param rootName Unique root name used to identify this root element by {@link module:engine/model/document~Document}.\n\t */\n\tconstructor( document: Document, name: string, rootName: string = 'main' ) {\n\t\tsuper( name );\n\n\t\tthis._document = document;\n\t\tthis.rootName = rootName;\n\t}\n\n\t/**\n\t * {@link module:engine/model/document~Document Document} that owns this root element.\n\t */\n\tpublic override get document(): Document {\n\t\treturn this._document;\n\t}\n\n\t/**\n\t * Informs if the root element is currently attached to the document, or not.\n\t *\n\t * A detached root is equivalent to being removed and cannot contain any children or markers.\n\t *\n\t * By default, a newly added root is attached. It can be detached using\n\t * {@link module:engine/model/writer~Writer#detachRoot `Writer#detachRoot`}. A detached root can be re-attached again using\n\t * {@link module:engine/model/writer~Writer#addRoot `Writer#addRoot`}.\n\t */\n\tpublic override isAttached(): boolean {\n\t\treturn this._isAttached;\n\t}\n\n\t/**\n\t * Converts `RootElement` instance to `string` containing its name.\n\t *\n\t * @returns `RootElement` instance converted to `string`.\n\t */\n\tpublic override toJSON(): unknown {\n\t\treturn this.rootName;\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \treturn this.rootName;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public override log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ModelRootElement: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nRootElement.prototype.is = function( type: string, name?: string ): boolean {\n\tif ( !name ) {\n\t\treturn type === 'rootElement' || type === 'model:rootElement' ||\n\t\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\t\ttype === 'element' || type === 'model:element' ||\n\t\t\ttype === 'node' || type === 'model:node';\n\t}\n\n\treturn name === this.name && (\n\t\ttype === 'rootElement' || type === 'model:rootElement' ||\n\t\t// From super.is(). This is highly utilised method and cannot call super. See ckeditor/ckeditor5#6529.\n\t\ttype === 'element' || type === 'model:element'\n\t);\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/document\n */\n\nimport Differ from './differ';\nimport DocumentSelection from './documentselection';\nimport History from './history';\nimport RootElement from './rootelement';\n\nimport type { SelectionChangeEvent } from './selection';\nimport type { default as Model, ModelApplyOperationEvent } from './model';\nimport type { MarkerCollectionUpdateEvent, MarkerChangeEvent } from './markercollection';\nimport type Batch from './batch';\nimport type Position from './position';\nimport type Range from './range';\nimport type Writer from './writer';\n\nimport {\n\tCKEditorError,\n\tCollection,\n\tEmitterMixin,\n\tisInsideSurrogatePair,\n\tisInsideCombinedSymbol\n} from '@ckeditor/ckeditor5-utils';\n\nimport { clone } from 'lodash-es';\n\n// @if CK_DEBUG_ENGINE // const { logDocument } = require( '../dev-utils/utils' );\n\nconst graveyardName = '$graveyard';\n\n/**\n * Data model's document. It contains the model's structure, its selection and the history of changes.\n *\n * Read more about working with the model in\n * {@glink framework/architecture/editing-engine#model introduction to the the editing engine's architecture}.\n *\n * Usually, the document contains just one {@link module:engine/model/document~Document#roots root element}, so\n * you can retrieve it by just calling {@link module:engine/model/document~Document#getRoot} without specifying its name:\n *\n * ```ts\n * model.document.getRoot(); // -> returns the main root\n * ```\n *\n * However, the document may contain multiple roots e.g. when the editor has multiple editable areas\n * (e.g. a title and a body of a message).\n */\nexport default class Document extends EmitterMixin() {\n\t/**\n\t * The {@link module:engine/model/model~Model model} that the document is a part of.\n\t */\n\tpublic readonly model: Model;\n\n\t/**\n\t * The document's history.\n\t */\n\tpublic readonly history: History;\n\n\t/**\n\t * The selection in this document.\n\t */\n\tpublic readonly selection: DocumentSelection;\n\n\t/**\n\t * A list of roots that are owned and managed by this document. Use {@link #createRoot}, {@link #getRoot} and\n\t * {@link #getRootNames} to manipulate it.\n\t */\n\tpublic readonly roots: Collection<RootElement>;\n\n\t/**\n\t * The model differ object. Its role is to buffer changes done on the model document and then calculate a diff of those changes.\n\t */\n\tpublic readonly differ: Differ;\n\n\t/**\n\t * Defines whether the document is in a read-only mode.\n\t *\n\t * The user should not be able to change the data of a document that is read-only.\n\t *\n\t * @readonly\n\t */\n\tpublic isReadOnly: boolean;\n\n\t/**\n\t * Post-fixer callbacks registered to the model document.\n\t */\n\tprivate readonly _postFixers: Set<ModelPostFixer>;\n\n\t/**\n\t * A flag that indicates whether the selection has changed since last change block.\n\t */\n\tprivate _hasSelectionChangedFromTheLastChangeBlock: boolean;\n\n\t/**\n\t * Creates an empty document instance with no {@link #roots} (other than\n\t * the {@link #graveyard graveyard root}).\n\t */\n\tconstructor( model: Model ) {\n\t\tsuper();\n\n\t\tthis.model = model;\n\t\tthis.history = new History();\n\t\tthis.selection = new DocumentSelection( this );\n\t\tthis.roots = new Collection( { idProperty: 'rootName' } );\n\t\tthis.differ = new Differ( model.markers );\n\t\tthis.isReadOnly = false;\n\n\t\tthis._postFixers = new Set();\n\t\tthis._hasSelectionChangedFromTheLastChangeBlock = false;\n\n\t\t// Graveyard tree root. Document always have a graveyard root, which stores removed nodes.\n\t\tthis.createRoot( '$root', graveyardName );\n\n\t\t// Then, still before an operation is applied on model, buffer the change in differ.\n\t\tthis.listenTo<ModelApplyOperationEvent>( model, 'applyOperation', ( evt, args ) => {\n\t\t\tconst operation = args[ 0 ];\n\n\t\t\tif ( operation.isDocumentOperation ) {\n\t\t\t\tthis.differ.bufferOperation( operation );\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\n\t\t// After the operation is applied, bump document's version and add the operation to the history.\n\t\tthis.listenTo<ModelApplyOperationEvent>( model, 'applyOperation', ( evt, args ) => {\n\t\t\tconst operation = args[ 0 ];\n\n\t\t\tif ( operation.isDocumentOperation ) {\n\t\t\t\tthis.history.addOperation( operation );\n\t\t\t}\n\t\t}, { priority: 'low' } );\n\n\t\t// Listen to selection changes. If selection changed, mark it.\n\t\tthis.listenTo<SelectionChangeEvent>( this.selection, 'change', () => {\n\t\t\tthis._hasSelectionChangedFromTheLastChangeBlock = true;\n\t\t} );\n\n\t\t// Buffer marker changes.\n\t\t// This is not covered in buffering operations because markers may change outside of them (when they\n\t\t// are modified using `model.markers` collection, not through `MarkerOperation`).\n\t\tthis.listenTo<MarkerCollectionUpdateEvent>( model.markers, 'update', ( evt, marker, oldRange, newRange, oldMarkerData ) => {\n\t\t\t// Copy the `newRange` to the new marker data as during the marker removal the range is not updated.\n\t\t\tconst newMarkerData = { ...marker.getData(), range: newRange };\n\n\t\t\t// Whenever marker is updated, buffer that change.\n\t\t\tthis.differ.bufferMarkerChange( marker.name, oldMarkerData, newMarkerData );\n\n\t\t\tif ( oldRange === null ) {\n\t\t\t\t// If this is a new marker, add a listener that will buffer change whenever marker changes.\n\t\t\t\tmarker.on<MarkerChangeEvent>( 'change', ( evt, oldRange ) => {\n\t\t\t\t\tconst markerData = marker.getData();\n\n\t\t\t\t\tthis.differ.bufferMarkerChange(\n\t\t\t\t\t\tmarker.name,\n\t\t\t\t\t\t{ ...markerData, range: oldRange },\n\t\t\t\t\t\tmarkerData\n\t\t\t\t\t);\n\t\t\t\t} );\n\t\t\t}\n\t\t} );\n\n\t\t// This is a solution for a problem that may occur during real-time editing. If one client detached a root and another added\n\t\t// something there at the same moment, the OT does not solve this problem currently. In such situation, the added elements would\n\t\t// stay in the detached root.\n\t\t//\n\t\t// This is incorrect, a detached root should be empty and all elements from it should be removed. To solve this, the post-fixer will\n\t\t// remove any element that is left in a detached root.\n\t\t//\n\t\t// Similarly, markers that are created at the beginning or at the end of the detached root will not be removed as well.\n\t\t//\n\t\t// The drawback of this solution over the OT solution is that the elements removed by the post-fixer will never be brought back.\n\t\t// If the root detachment gets undone (and the root is brought back), the removed elements will not be there.\n\t\tthis.registerPostFixer( writer => {\n\t\t\tlet result = false;\n\n\t\t\tfor ( const root of this.roots ) {\n\t\t\t\tif ( !root.isAttached() && !root.isEmpty ) {\n\t\t\t\t\twriter.remove( writer.createRangeIn( root ) );\n\n\t\t\t\t\tresult = true;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfor ( const marker of this.model.markers ) {\n\t\t\t\tif ( !marker.getRange().root.isAttached() ) {\n\t\t\t\t\twriter.removeMarker( marker );\n\n\t\t\t\t\tresult = true;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\treturn result;\n\t\t} );\n\t}\n\n\t/**\n\t * The document version. Every applied operation increases the version number. It is used to\n\t * ensure that operations are applied on a proper document version.\n\t *\n\t * This property is equal to {@link module:engine/model/history~History#version `model.Document#history#version`}.\n\t *\n\t * If the {@link module:engine/model/operation/operation~Operation#baseVersion base version} does not match the document version,\n\t * a {@link module:utils/ckeditorerror~CKEditorError model-document-applyoperation-wrong-version} error is thrown.\n\t */\n\tpublic get version(): number {\n\t\treturn this.history.version;\n\t}\n\n\tpublic set version( version: number ) {\n\t\tthis.history.version = version;\n\t}\n\n\t/**\n\t * The graveyard tree root. A document always has a graveyard root that stores removed nodes.\n\t */\n\tpublic get graveyard(): RootElement {\n\t\treturn this.getRoot( graveyardName )!;\n\t}\n\n\t/**\n\t * Creates a new root.\n\t *\n\t * **Note:** do not use this method after the editor has been initialized! If you want to dynamically add a root, use\n\t * {@link module:engine/model/writer~Writer#addRoot `model.Writer#addRoot`} instead.\n\t *\n\t * @param elementName The element name. Defaults to `'$root'` which also has some basic schema defined\n\t * (e.g. `$block` elements are allowed inside the `$root`). Make sure to define a proper schema if you use a different name.\n\t * @param rootName A unique root name.\n\t * @returns The created root.\n\t */\n\tpublic createRoot( elementName: string = '$root', rootName: string = 'main' ): RootElement {\n\t\tif ( this.roots.get( rootName ) ) {\n\t\t\t/**\n\t\t\t * A root with the specified name already exists.\n\t\t\t *\n\t\t\t * @error model-document-createroot-name-exists\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'model-document-createroot-name-exists', this, { name: rootName } );\n\t\t}\n\n\t\tconst root = new RootElement( this, elementName, rootName );\n\t\tthis.roots.add( root );\n\n\t\treturn root;\n\t}\n\n\t/**\n\t * Removes all event listeners set by the document instance.\n\t */\n\tpublic destroy(): void {\n\t\tthis.selection.destroy();\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Returns a root by its name.\n\t *\n\t * Detached roots are returned by this method. This is to be able to operate on the detached root (for example, to be able to create\n\t * a position inside such a root for undo feature purposes).\n\t *\n\t * @param name The root name of the root to return.\n\t * @returns The root registered under a given name or `null` when there is no root with the given name.\n\t */\n\tpublic getRoot( name: string = 'main' ): RootElement | null {\n\t\treturn this.roots.get( name );\n\t}\n\n\t/**\n\t * Returns an array with names of all roots added to the document (except the {@link #graveyard graveyard root}).\n\t *\n\t * Detached roots **are not** returned by this method by default. This is to make sure that all features or algorithms that operate\n\t * on the document data know which roots are still a part of the document and should be processed.\n\t *\n\t * @param includeDetached Specified whether detached roots should be returned as well.\n\t */\n\tpublic getRootNames( includeDetached = false ): Array<string> {\n\t\treturn this.getRoots( includeDetached ).map( root => root.rootName );\n\t}\n\n\t/**\n\t * Returns an array with all roots added to the document (except the {@link #graveyard graveyard root}).\n\t *\n\t * Detached roots **are not** returned by this method by default. This is to make sure that all features or algorithms that operate\n\t * on the document data know which roots are still a part of the document and should be processed.\n\t *\n\t * @param includeDetached Specified whether detached roots should be returned as well.\n\t */\n\tpublic getRoots( includeDetached = false ): Array<RootElement> {\n\t\treturn Array.from( this.roots )\n\t\t\t.filter( root => root != this.graveyard && ( includeDetached || root.isAttached() ) && root._isLoaded );\n\t}\n\n\t/**\n\t * Used to register a post-fixer callback. A post-fixer mechanism guarantees that the features\n\t * will operate on a correct model state.\n\t *\n\t * An execution of a feature may lead to an incorrect document tree state. The callbacks are used to fix the document tree after\n\t * it has changed. Post-fixers are fired just after all changes from the outermost change block were applied but\n\t * before the {@link module:engine/model/document~Document#event:change change event} is fired. If a post-fixer callback made\n\t * a change, it should return `true`. When this happens, all post-fixers are fired again to check if something else should\n\t * not be fixed in the new document tree state.\n\t *\n\t * As a parameter, a post-fixer callback receives a {@link module:engine/model/writer~Writer writer} instance connected with the\n\t * executed changes block. Thanks to that, all changes done by the callback will be added to the same\n\t * {@link module:engine/model/batch~Batch batch} (and undo step) as the original changes. This makes post-fixer changes transparent\n\t * for the user.\n\t *\n\t * An example of a post-fixer is a callback that checks if all the data were removed from the editor. If so, the\n\t * callback should add an empty paragraph so that the editor is never empty:\n\t *\n\t * ```ts\n\t * document.registerPostFixer( writer => {\n\t * \tconst changes = document.differ.getChanges();\n\t *\n\t * \t// Check if the changes lead to an empty root in the editor.\n\t * \tfor ( const entry of changes ) {\n\t * \t\tif ( entry.type == 'remove' && entry.position.root.isEmpty ) {\n\t * \t\t\twriter.insertElement( 'paragraph', entry.position.root, 0 );\n\t *\n\t * \t\t\t// It is fine to return early, even if multiple roots would need to be fixed.\n\t * \t\t\t// All post-fixers will be fired again, so if there are more empty roots, those will be fixed, too.\n\t * \t\t\treturn true;\n\t * \t\t}\n\t * \t}\n\t *\n\t * \treturn false;\n\t * } );\n\t * ```\n\t */\n\tpublic registerPostFixer( postFixer: ModelPostFixer ): void {\n\t\tthis._postFixers.add( postFixer );\n\t}\n\n\t/**\n\t * A custom `toJSON()` method to solve child-parent circular dependencies.\n\t *\n\t * @returns A clone of this object with the document property changed to a string.\n\t */\n\tpublic toJSON(): unknown {\n\t\tconst json: any = clone( this );\n\n\t\t// Due to circular references we need to remove parent reference.\n\t\tjson.selection = '[engine.model.DocumentSelection]';\n\t\tjson.model = '[engine.model.Model]';\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * Check if there were any changes done on document, and if so, call post-fixers,\n\t * fire `change` event for features and conversion and then reset the differ.\n\t * Fire `change:data` event when at least one operation or buffered marker changes the data.\n\t *\n\t * @internal\n\t * @fires change\n\t * @fires change:data\n\t * @param writer The writer on which post-fixers will be called.\n\t */\n\tpublic _handleChangeBlock( writer: Writer ): void {\n\t\tif ( this._hasDocumentChangedFromTheLastChangeBlock() ) {\n\t\t\tthis._callPostFixers( writer );\n\n\t\t\t// Refresh selection attributes according to the final position in the model after the change.\n\t\t\tthis.selection.refresh();\n\n\t\t\tif ( this.differ.hasDataChanges() ) {\n\t\t\t\tthis.fire<DocumentChangeEvent>( 'change:data', writer.batch );\n\t\t\t} else {\n\t\t\t\tthis.fire<DocumentChangeEvent>( 'change', writer.batch );\n\t\t\t}\n\n\t\t\t// Theoretically, it is not necessary to refresh selection after change event because\n\t\t\t// post-fixers are the last who should change the model, but just in case...\n\t\t\tthis.selection.refresh();\n\n\t\t\tthis.differ.reset();\n\t\t}\n\n\t\tthis._hasSelectionChangedFromTheLastChangeBlock = false;\n\t}\n\n\t/**\n\t * Returns whether there is a buffered change or if the selection has changed from the last\n\t * {@link module:engine/model/model~Model#enqueueChange `enqueueChange()` block}\n\t * or {@link module:engine/model/model~Model#change `change()` block}.\n\t *\n\t * @returns Returns `true` if document has changed from the last `change()` or `enqueueChange()` block.\n\t */\n\tprotected _hasDocumentChangedFromTheLastChangeBlock(): boolean {\n\t\treturn !this.differ.isEmpty || this._hasSelectionChangedFromTheLastChangeBlock;\n\t}\n\n\t/**\n\t * Returns the default root for this document which is either the first root that was added to the document using\n\t * {@link #createRoot} or the {@link #graveyard graveyard root} if no other roots were created.\n\t *\n\t * @returns The default root for this document.\n\t */\n\tprotected _getDefaultRoot(): RootElement {\n\t\tconst roots = this.getRoots();\n\n\t\treturn roots.length ? roots[ 0 ] : this.graveyard;\n\t}\n\n\t/**\n\t * Returns the default range for this selection. The default range is a collapsed range that starts and ends\n\t * at the beginning of this selection's document {@link #_getDefaultRoot default root}.\n\t *\n\t * @internal\n\t */\n\tpublic _getDefaultRange(): Range {\n\t\tconst defaultRoot = this._getDefaultRoot();\n\t\tconst model = this.model;\n\t\tconst schema = model.schema;\n\n\t\t// Find the first position where the selection can be put.\n\t\tconst position = model.createPositionFromPath( defaultRoot, [ 0 ] );\n\t\tconst nearestRange = schema.getNearestSelectionRange( position );\n\n\t\t// If valid selection range is not found - return range collapsed at the beginning of the root.\n\t\treturn nearestRange || model.createRange( position );\n\t}\n\n\t/**\n\t * Checks whether a given {@link module:engine/model/range~Range range} is a valid range for\n\t * the {@link #selection document's selection}.\n\t *\n\t * @internal\n\t * @param range A range to check.\n\t * @returns `true` if `range` is valid, `false` otherwise.\n\t */\n\tpublic _validateSelectionRange( range: Range ): boolean {\n\t\treturn validateTextNodePosition( range.start ) && validateTextNodePosition( range.end );\n\t}\n\n\t/**\n\t * Performs post-fixer loops. Executes post-fixer callbacks as long as none of them has done any changes to the model.\n\t *\n\t * @param writer The writer on which post-fixer callbacks will be called.\n\t */\n\tprivate _callPostFixers( writer: Writer ) {\n\t\tlet wasFixed = false;\n\n\t\tdo {\n\t\t\tfor ( const callback of this._postFixers ) {\n\t\t\t\t// Ensure selection attributes are up to date before each post-fixer.\n\t\t\t\t// https://github.com/ckeditor/ckeditor5-engine/issues/1673.\n\t\t\t\t//\n\t\t\t\t// It might be good to refresh the selection after each operation but at the moment it leads\n\t\t\t\t// to losing attributes for composition or and spell checking\n\t\t\t\t// https://github.com/ckeditor/ckeditor5-typing/issues/188\n\t\t\t\tthis.selection.refresh();\n\n\t\t\t\twasFixed = callback( writer );\n\n\t\t\t\tif ( wasFixed ) {\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t} while ( wasFixed );\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public log( version: any = null ): void {\n\t// @if CK_DEBUG_ENGINE // \tversion = version === null ? this.version : version;\n\t// @if CK_DEBUG_ENGINE // \tlogDocument( this, version );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n/**\n * Fired after each {@link module:engine/model/model~Model#enqueueChange `enqueueChange()` block} or the outermost\n * {@link module:engine/model/model~Model#change `change()` block} was executed and the document was changed\n * during that block's execution.\n *\n * The changes which this event will cover include:\n *\n * * document structure changes,\n * * selection changes,\n * * marker changes.\n *\n * If you want to be notified about all these changes, then simply listen to this event like this:\n *\n * ```ts\n * model.document.on( 'change', () => {\n * \tconsole.log( 'The document has changed!' );\n * } );\n * ```\n *\n * If, however, you only want to be notified about the data changes, then use `change:data` event,\n * which is fired for document structure changes and marker changes (which affects the data).\n *\n * ```ts\n * model.document.on( 'change:data', () => {\n * \tconsole.log( 'The data has changed!' );\n * } );\n * ```\n *\n * @eventName ~Document#change\n * @eventName ~Document#change:data\n * @param batch The batch that was used in the executed changes block.\n */\nexport type DocumentChangeEvent = {\n\tname: 'change' | 'change:data';\n\targs: [ batch: Batch ];\n};\n\n/**\n * Callback passed as an argument to the {@link module:engine/model/document~Document#registerPostFixer} method.\n */\nexport type ModelPostFixer = ( writer: Writer ) => boolean;\n\n/**\n * Checks whether given range boundary position is valid for document selection, meaning that is not between\n * unicode surrogate pairs or base character and combining marks.\n */\nfunction validateTextNodePosition( rangeBoundary: Position ) {\n\tconst textNode = rangeBoundary.textNode;\n\n\tif ( textNode ) {\n\t\tconst data = textNode.data;\n\t\tconst offset = rangeBoundary.offset - textNode.startOffset!;\n\n\t\treturn !isInsideSurrogatePair( data, offset ) && !isInsideCombinedSymbol( data, offset );\n\t}\n\n\treturn true;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/markercollection\n */\n\nimport TypeCheckable from './typecheckable';\nimport LiveRange, {\n\ttype LiveRangeChangeContentEvent,\n\ttype LiveRangeChangeRangeEvent,\n\ttype LiveRangeChangeEvent\n} from './liverange';\n\nimport type Position from './position';\nimport type Range from './range';\n\nimport { CKEditorError, EmitterMixin } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The collection of all {@link module:engine/model/markercollection~Marker markers} attached to the document.\n * It lets you {@link module:engine/model/markercollection~MarkerCollection#get get} markers or track them using\n * {@link module:engine/model/markercollection~MarkerCollection#event:update} event.\n *\n * To create, change or remove makers use {@link module:engine/model/writer~Writer model writers'} methods:\n * {@link module:engine/model/writer~Writer#addMarker} or {@link module:engine/model/writer~Writer#removeMarker}. Since\n * the writer is the only proper way to change the data model it is not possible to change markers directly using this\n * collection. All markers created by the writer will be automatically added to this collection.\n *\n * By default there is one marker collection available as {@link module:engine/model/model~Model#markers model property}.\n *\n * @see module:engine/model/markercollection~Marker\n */\nexport default class MarkerCollection extends EmitterMixin() implements Iterable<Marker> {\n\t/**\n\t * Stores {@link ~Marker markers} added to the collection.\n\t */\n\tprivate _markers: Map<string, Marker> = new Map();\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * Iterates over all {@link ~Marker markers} added to the collection.\n\t */\n\tpublic [ Symbol.iterator ](): IterableIterator<Marker> {\n\t\treturn this._markers.values();\n\t}\n\n\t/**\n\t * Checks if given {@link ~Marker marker} or marker name is in the collection.\n\t *\n\t * @param markerOrName Name of marker or marker instance to check.\n\t * @returns `true` if marker is in the collection, `false` otherwise.\n\t */\n\tpublic has( markerOrName: string | Marker ): boolean {\n\t\tconst markerName = markerOrName instanceof Marker ? markerOrName.name : markerOrName;\n\n\t\treturn this._markers.has( markerName );\n\t}\n\n\t/**\n\t * Returns {@link ~Marker marker} with given `markerName`.\n\t *\n\t * @param markerName Name of marker to get.\n\t * @returns Marker with given name or `null` if such marker was\n\t * not added to the collection.\n\t */\n\tpublic get( markerName: string ): Marker | null {\n\t\treturn this._markers.get( markerName ) || null;\n\t}\n\n\t/**\n\t * Creates and adds a {@link ~Marker marker} to the `MarkerCollection` with given name on given\n\t * {@link module:engine/model/range~Range range}.\n\t *\n\t * If `MarkerCollection` already had a marker with given name (or {@link ~Marker marker} was passed), the marker in\n\t * collection is updated and {@link module:engine/model/markercollection~MarkerCollection#event:update} event is fired\n\t * but only if there was a change (marker range or {@link module:engine/model/markercollection~Marker#managedUsingOperations}\n\t * flag has changed.\n\t *\n\t * @internal\n\t * @fires update\n\t * @param markerOrName Name of marker to set or marker instance to update.\n\t * @param range Marker range.\n\t * @param managedUsingOperations Specifies whether the marker is managed using operations.\n\t * @param affectsData Specifies whether the marker affects the data produced by the data pipeline\n\t * (is persisted in the editor's data).\n\t * @returns `Marker` instance which was added or updated.\n\t */\n\tpublic _set(\n\t\tmarkerOrName: string | Marker,\n\t\trange: Range,\n\t\tmanagedUsingOperations: boolean = false,\n\t\taffectsData: boolean = false\n\t): Marker {\n\t\tconst markerName = markerOrName instanceof Marker ? markerOrName.name : markerOrName;\n\n\t\tif ( markerName.includes( ',' ) ) {\n\t\t\t/**\n\t\t\t * Marker name cannot contain the \",\" character.\n\t\t\t *\n\t\t\t * @error markercollection-incorrect-marker-name\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'markercollection-incorrect-marker-name', this );\n\t\t}\n\n\t\tconst oldMarker = this._markers.get( markerName );\n\n\t\tif ( oldMarker ) {\n\t\t\tconst oldMarkerData = oldMarker.getData();\n\n\t\t\tconst oldRange = oldMarker.getRange();\n\t\t\tlet hasChanged = false;\n\n\t\t\tif ( !oldRange.isEqual( range ) ) {\n\t\t\t\toldMarker._attachLiveRange( LiveRange.fromRange( range ) );\n\t\t\t\thasChanged = true;\n\t\t\t}\n\n\t\t\tif ( managedUsingOperations != oldMarker.managedUsingOperations ) {\n\t\t\t\toldMarker._managedUsingOperations = managedUsingOperations;\n\t\t\t\thasChanged = true;\n\t\t\t}\n\n\t\t\tif ( typeof affectsData === 'boolean' && affectsData != oldMarker.affectsData ) {\n\t\t\t\toldMarker._affectsData = affectsData;\n\t\t\t\thasChanged = true;\n\t\t\t}\n\n\t\t\tif ( hasChanged ) {\n\t\t\t\tthis.fire<MarkerCollectionUpdateEvent>( `update:${ markerName }`, oldMarker, oldRange, range, oldMarkerData );\n\t\t\t}\n\n\t\t\treturn oldMarker;\n\t\t}\n\n\t\tconst liveRange = LiveRange.fromRange( range );\n\t\tconst marker = new Marker( markerName, liveRange, managedUsingOperations, affectsData );\n\n\t\tthis._markers.set( markerName, marker );\n\t\tthis.fire<MarkerCollectionUpdateEvent>( `update:${ markerName }`, marker, null, range, { ...marker.getData(), range: null } );\n\n\t\treturn marker;\n\t}\n\n\t/**\n\t * Removes given {@link ~Marker marker} or a marker with given name from the `MarkerCollection`.\n\t *\n\t * @internal\n\t * @fires update\n\t * @param markerOrName Marker or name of a marker to remove.\n\t * @returns `true` if marker was found and removed, `false` otherwise.\n\t */\n\tpublic _remove( markerOrName: string | Marker ): boolean {\n\t\tconst markerName = markerOrName instanceof Marker ? markerOrName.name : markerOrName;\n\t\tconst oldMarker = this._markers.get( markerName );\n\n\t\tif ( oldMarker ) {\n\t\t\tthis._markers.delete( markerName );\n\t\t\tthis.fire<MarkerCollectionUpdateEvent>( `update:${ markerName }`, oldMarker, oldMarker.getRange(), null, oldMarker.getData() );\n\n\t\t\tthis._destroyMarker( oldMarker );\n\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Fires an {@link module:engine/model/markercollection~MarkerCollection#event:update} event for the given {@link ~Marker marker}\n\t * but does not change the marker. Useful to force {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher downcast\n\t * conversion} for the marker.\n\t *\n\t * @internal\n\t * @fires update\n\t * @param markerOrName Marker or name of a marker to refresh.\n\t */\n\tpublic _refresh( markerOrName: string | Marker ): void {\n\t\tconst markerName = markerOrName instanceof Marker ? markerOrName.name : markerOrName;\n\t\tconst marker = this._markers.get( markerName );\n\n\t\tif ( !marker ) {\n\t\t\t/**\n\t\t\t * Marker with provided name does not exists.\n\t\t\t *\n\t\t\t * @error markercollection-refresh-marker-not-exists\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'markercollection-refresh-marker-not-exists', this );\n\t\t}\n\n\t\tconst range = marker.getRange();\n\n\t\tthis.fire<MarkerCollectionUpdateEvent>( `update:${ markerName }`, marker, range, range, marker.getData() );\n\t}\n\n\t/**\n\t * Returns iterator that iterates over all markers, which ranges contain given {@link module:engine/model/position~Position position}.\n\t */\n\tpublic* getMarkersAtPosition( position: Position ): IterableIterator<Marker> {\n\t\tfor ( const marker of this ) {\n\t\t\tif ( marker.getRange().containsPosition( position ) ) {\n\t\t\t\tyield marker;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Returns iterator that iterates over all markers, which intersects with given {@link module:engine/model/range~Range range}.\n\t */\n\tpublic* getMarkersIntersectingRange( range: Range ): Iterable<Marker> {\n\t\tfor ( const marker of this ) {\n\t\t\tif ( marker.getRange().getIntersection( range ) !== null ) {\n\t\t\t\tyield marker;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Destroys marker collection and all markers inside it.\n\t */\n\tpublic destroy(): void {\n\t\tfor ( const marker of this._markers.values() ) {\n\t\t\tthis._destroyMarker( marker );\n\t\t}\n\n\t\tthis._markers = null as any;\n\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Iterates over all markers that starts with given `prefix`.\n\t *\n\t * ```ts\n\t * const markerFooA = markersCollection.set( 'foo:a', rangeFooA );\n\t * const markerFooB = markersCollection.set( 'foo:b', rangeFooB );\n\t * const markerBarA = markersCollection.set( 'bar:a', rangeBarA );\n\t * const markerFooBarA = markersCollection.set( 'foobar:a', rangeFooBarA );\n\t * Array.from( markersCollection.getMarkersGroup( 'foo' ) ); // [ markerFooA, markerFooB ]\n\t * Array.from( markersCollection.getMarkersGroup( 'a' ) ); // []\n\t * ```\n\t */\n\tpublic* getMarkersGroup( prefix: string ): IterableIterator<Marker> {\n\t\tfor ( const marker of this._markers.values() ) {\n\t\t\tif ( marker.name.startsWith( prefix + ':' ) ) {\n\t\t\t\tyield marker;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Destroys the marker.\n\t */\n\tprivate _destroyMarker( marker: Marker ): void {\n\t\tmarker.stopListening();\n\t\tmarker._detachLiveRange();\n\t}\n}\n\nexport interface MarkerData {\n\n\t/**\n\t * Marker range. `null` if the marker was removed.\n\t */\n\trange: Range | null;\n\n\t/**\n\t * A property defining if the marker affects data.\n\t */\n\taffectsData: boolean;\n\n\t/**\n\t * A property defining if the marker is managed using operations.\n\t */\n\tmanagedUsingOperations: boolean;\n}\n\n/**\n * `Marker` is a continuous part of the model (like a range), is named and represents some kind of information about the\n * marked part of the model document. In contrary to {@link module:engine/model/node~Node nodes}, which are building blocks of\n * the model document tree, markers are not stored directly in the document tree but in the\n * {@link module:engine/model/model~Model#markers model markers' collection}. Still, they are document data, by giving\n * additional meaning to the part of a model document between marker start and marker end.\n *\n * In this sense, markers are similar to adding and converting attributes on nodes. The difference is that attribute is\n * connected with a given node (e.g. a character is bold no matter if it gets moved or content around it changes).\n * Markers on the other hand are continuous ranges and are characterized by their start and end position. This means that\n * any character in the marker is marked by the marker. For example, if a character is moved outside of marker it stops being\n * \"special\" and the marker is shrunk. Similarly, when a character is moved into the marker from other place in document\n * model, it starts being \"special\" and the marker is enlarged.\n *\n * Another upside of markers is that finding marked part of document is fast and easy. Using attributes to mark some nodes\n * and then trying to find that part of document would require traversing whole document tree. Marker gives instant access\n * to the range which it is marking at the moment.\n *\n * Markers are built from a name and a range.\n *\n * Range of the marker is updated automatically when document changes, using\n * {@link module:engine/model/liverange~LiveRange live range} mechanism.\n *\n * Name is used to group and identify markers. Names have to be unique, but markers can be grouped by\n * using common prefixes, separated with `:`, for example: `user:john` or `search:3`. That's useful in term of creating\n * namespaces for custom elements (e.g. comments, highlights). You can use this prefixes in\n * {@link module:engine/model/markercollection~MarkerCollection#event:update} listeners to listen on changes in a group of markers.\n * For instance: `model.markers.on( 'update:user', callback );` will be called whenever any `user:*` markers changes.\n *\n * There are two types of markers.\n *\n * 1. Markers managed directly, without using operations. They are added directly by {@link module:engine/model/writer~Writer}\n * to the {@link module:engine/model/markercollection~MarkerCollection} without any additional mechanism. They can be used\n * as bookmarks or visual markers. They are great for showing results of the find, or select link when the focus is in the input.\n *\n * 1. Markers managed using operations. These markers are also stored in {@link module:engine/model/markercollection~MarkerCollection}\n * but changes in these markers is managed the same way all other changes in the model structure - using operations.\n * Therefore, they are handled in the undo stack and synchronized between clients if the collaboration plugin is enabled.\n * This type of markers is useful for solutions like spell checking or comments.\n *\n * Both type of them should be added / updated by {@link module:engine/model/writer~Writer#addMarker}\n * and removed by {@link module:engine/model/writer~Writer#removeMarker} methods.\n *\n * ```ts\n * model.change( ( writer ) => {\n * \tconst marker = writer.addMarker( name, { range, usingOperation: true } );\n *\n * \t// ...\n *\n * \twriter.removeMarker( marker );\n * } );\n * ```\n *\n * See {@link module:engine/model/writer~Writer} to find more examples.\n *\n * Since markers need to track change in the document, for efficiency reasons, it is best to create and keep as little\n * markers as possible and remove them as soon as they are not needed anymore.\n *\n * Markers can be downcasted and upcasted.\n *\n * Markers downcast happens on {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:addMarker} and\n * {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:removeMarker} events.\n * Use {@link module:engine/conversion/downcasthelpers downcast converters} or attach a custom converter to mentioned events.\n * For {@link module:engine/controller/datacontroller~DataController data pipeline}, marker should be downcasted to an element.\n * Then, it can be upcasted back to a marker. Again, use {@link module:engine/conversion/upcasthelpers upcast converters} or\n * attach a custom converter to {@link module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:element}.\n *\n * `Marker` instances are created and destroyed only by {@link ~MarkerCollection MarkerCollection}.\n */\nclass Marker extends EmitterMixin( TypeCheckable ) {\n\t/**\n\t * Marker's name.\n\t */\n\tpublic readonly name: string;\n\n\t/**\n\t * Flag indicates if the marker is managed using operations or not.\n\t *\n\t * @internal\n\t */\n\tpublic _managedUsingOperations: boolean;\n\n\t/**\n\t * Specifies whether the marker affects the data produced by the data pipeline\n\t * (is persisted in the editor's data).\n\t *\n\t * @internal\n\t */\n\tpublic _affectsData: boolean;\n\n\t/**\n\t * Range marked by the marker.\n\t */\n\tprivate _liveRange: LiveRange | null;\n\n\t/**\n\t * Creates a marker instance.\n\t *\n\t * @param name Marker name.\n\t * @param liveRange Range marked by the marker.\n\t * @param managedUsingOperations Specifies whether the marker is managed using operations.\n\t * @param affectsData Specifies whether the marker affects the data produced by the data pipeline (is persisted in the editor's data).\n\t */\n\tconstructor(\n\t\tname: string,\n\t\tliveRange: LiveRange,\n\t\tmanagedUsingOperations: boolean,\n\t\taffectsData: boolean\n\t) {\n\t\tsuper();\n\n\t\tthis.name = name;\n\t\tthis._liveRange = this._attachLiveRange( liveRange );\n\t\tthis._managedUsingOperations = managedUsingOperations;\n\t\tthis._affectsData = affectsData;\n\t}\n\n\t/**\n\t * A value indicating if the marker is managed using operations.\n\t * See {@link ~Marker marker class description} to learn more about marker types.\n\t * See {@link module:engine/model/writer~Writer#addMarker}.\n\t */\n\tpublic get managedUsingOperations(): boolean {\n\t\tif ( !this._liveRange ) {\n\t\t\tthrow new CKEditorError( 'marker-destroyed', this );\n\t\t}\n\n\t\treturn this._managedUsingOperations;\n\t}\n\n\t/**\n\t * A value indicating if the marker changes the data.\n\t */\n\tpublic get affectsData(): boolean {\n\t\tif ( !this._liveRange ) {\n\t\t\tthrow new CKEditorError( 'marker-destroyed', this );\n\t\t}\n\n\t\treturn this._affectsData;\n\t}\n\n\t/**\n\t * Returns the marker data (properties defining the marker).\n\t */\n\tpublic getData(): MarkerData {\n\t\treturn {\n\t\t\trange: this.getRange(),\n\t\t\taffectsData: this.affectsData,\n\t\t\tmanagedUsingOperations: this.managedUsingOperations\n\t\t};\n\t}\n\n\t/**\n\t * Returns current marker start position.\n\t */\n\tpublic getStart(): Position {\n\t\tif ( !this._liveRange ) {\n\t\t\tthrow new CKEditorError( 'marker-destroyed', this );\n\t\t}\n\n\t\treturn this._liveRange.start.clone();\n\t}\n\n\t/**\n\t * Returns current marker end position.\n\t */\n\tpublic getEnd(): Position {\n\t\tif ( !this._liveRange ) {\n\t\t\tthrow new CKEditorError( 'marker-destroyed', this );\n\t\t}\n\n\t\treturn this._liveRange.end.clone();\n\t}\n\n\t/**\n\t * Returns a range that represents the current state of the marker.\n\t *\n\t * Keep in mind that returned value is a {@link module:engine/model/range~Range Range}, not a\n\t * {@link module:engine/model/liverange~LiveRange LiveRange}. This means that it is up-to-date and relevant only\n\t * until next model document change. Do not store values returned by this method. Instead, store {@link ~Marker#name}\n\t * and get `Marker` instance from {@link module:engine/model/markercollection~MarkerCollection MarkerCollection} every\n\t * time there is a need to read marker properties. This will guarantee that the marker has not been removed and\n\t * that it's data is up-to-date.\n\t */\n\tpublic getRange(): Range {\n\t\tif ( !this._liveRange ) {\n\t\t\tthrow new CKEditorError( 'marker-destroyed', this );\n\t\t}\n\n\t\treturn this._liveRange.toRange();\n\t}\n\n\t/**\n\t * Binds new live range to the marker and detach the old one if is attached.\n\t *\n\t * @internal\n\t * @param liveRange Live range to attach\n\t * @returns Attached live range.\n\t */\n\tpublic _attachLiveRange( liveRange: LiveRange ): LiveRange {\n\t\tif ( this._liveRange ) {\n\t\t\tthis._detachLiveRange();\n\t\t}\n\n\t\t// Delegating does not work with namespaces. Alternatively, we could delegate all events (using `*`).\n\t\tliveRange.delegate( 'change:range' ).to( this );\n\t\tliveRange.delegate( 'change:content' ).to( this );\n\n\t\tthis._liveRange = liveRange;\n\n\t\treturn liveRange;\n\t}\n\n\t/**\n\t * Unbinds and destroys currently attached live range.\n\t *\n\t * @internal\n\t */\n\tpublic _detachLiveRange(): void {\n\t\tthis._liveRange!.stopDelegating( 'change:range', this );\n\t\tthis._liveRange!.stopDelegating( 'change:content', this );\n\t\tthis._liveRange!.detach();\n\t\tthis._liveRange = null;\n\t}\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nMarker.prototype.is = function( type: string ): boolean {\n\treturn type === 'marker' || type === 'model:marker';\n};\n\nexport type { Marker };\n\n/**\n * Fired whenever {@link ~Marker#_liveRange marker range} is changed due to changes on {@link module:engine/model/document~Document}.\n * This is a delegated {@link module:engine/model/liverange~LiveRange#event:change:range LiveRange change:range event}.\n *\n * When marker is removed from {@link module:engine/model/markercollection~MarkerCollection MarkerCollection},\n * all event listeners listening to it should be removed. It is best to do it on\n * {@link module:engine/model/markercollection~MarkerCollection#event:update MarkerCollection update event}.\n *\n * @see module:engine/model/liverange~LiveRange#event:change:range\n * @eventName ~Marker#change:range\n */\nexport type MarkerChangeRangeEvent = LiveRangeChangeRangeEvent;\n\n/**\n * Fired whenever change on {@link module:engine/model/document~Document} is done inside {@link ~Marker#_liveRange marker range}.\n * This is a delegated {@link module:engine/model/liverange~LiveRange#event:change:content LiveRange change:content event}.\n *\n * When marker is removed from {@link module:engine/model/markercollection~MarkerCollection MarkerCollection},\n * all event listeners listening to it should be removed. It is best to do it on\n * {@link module:engine/model/markercollection~MarkerCollection#event:update MarkerCollection update event}.\n *\n * @see module:engine/model/liverange~LiveRange#event:change:content\n * @eventName ~Marker#change:content\n */\nexport type MarkerCollectionChangeContentEvent = LiveRangeChangeContentEvent;\n\n/**\n * Describes `change:range` or `change:content` event.\n */\nexport type MarkerChangeEvent = LiveRangeChangeEvent;\n\n/**\n * Fired whenever marker is added, updated or removed from `MarkerCollection`.\n *\n * @eventName ~MarkerCollection#update\n * @param marker Updated Marker.\n * @param oldRange Marker range before the update. When is not defined it\n * means that marker is just added.\n * @param newRange Marker range after update. When is not defined it\n * means that marker is just removed.\n * @param oldMarkerData Data of the marker before the change.\n */\nexport type MarkerCollectionUpdateEvent = {\n\tname: 'update' | `update:${ string }`;\n\targs: [ marker: Marker, oldRange: Range | null, newRange: Range | null, oldMarkerData: MarkerData ];\n};\n\n/**\n * Cannot use a {@link module:engine/model/markercollection~MarkerCollection#destroy destroyed marker} instance.\n *\n * @error marker-destroyed\n */\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/operation/detachoperation\n */\n\nimport Operation from './operation';\nimport Range from '../range';\nimport { _remove } from './utils';\n\nimport type Position from '../position';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\nimport type { Selectable } from '../selection';\n\n// @if CK_DEBUG_ENGINE // const ModelRange = require( '../range' ).default;\n\n/**\n * Operation to permanently remove node from detached root.\n * Note this operation is only a local operation and won't be send to the other clients.\n */\nexport default class DetachOperation extends Operation {\n\t/**\n\t * Position before the first {@link module:engine/model/item~Item model item} to detach.\n\t */\n\tpublic sourcePosition: Position;\n\n\t/**\n\t * Offset size of moved range.\n\t */\n\tpublic howMany: number;\n\n\t// This class doesn't implement those abstract methods. Let's silence the compiler.\n\tdeclare public clone: never;\n\tdeclare public getReversed: never;\n\n\t/**\n\t * Creates an insert operation.\n\t *\n\t * @param sourcePosition Position before the first {@link module:engine/model/item~Item model item} to move.\n\t * @param howMany Offset size of moved range. Moved range will start from `sourcePosition` and end at\n\t * `sourcePosition` with offset shifted by `howMany`.\n\t */\n\tconstructor( sourcePosition: Position, howMany: number ) {\n\t\tsuper( null );\n\n\t\tthis.sourcePosition = sourcePosition.clone();\n\t\tthis.howMany = howMany;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get type(): 'detach' {\n\t\treturn 'detach';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic get affectedSelectable(): Selectable {\n\t\treturn null;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override toJSON(): unknown {\n\t\tconst json: any = super.toJSON();\n\n\t\tjson.sourcePosition = this.sourcePosition.toJSON();\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic override _validate(): void {\n\t\tif ( this.sourcePosition.root.document ) {\n\t\t\t/**\n\t\t\t * Cannot detach document node.\n\t\t\t *\n\t\t\t * @error detach-operation-on-document-node\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'detach-operation-on-document-node', this );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t * @internal\n\t */\n\tpublic _execute(): void {\n\t\t_remove( Range._createFromPositionAndShift( this.sourcePosition, this.howMany ) );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static override get className(): string {\n\t\treturn 'DetachOperation';\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): string {\n\t// @if CK_DEBUG_ENGINE // \tconst range = ModelRange._createFromPositionAndShift( this.sourcePosition, this.howMany );\n\t// @if CK_DEBUG_ENGINE // \tconst nodes = Array.from( range.getItems() );\n\t// @if CK_DEBUG_ENGINE // \tconst nodeString = nodes.length > 1 ? `[ ${ nodes.length } ]` : nodes[ 0 ];\n\n\t// @if CK_DEBUG_ENGINE // \treturn `DetachOperation( ${ this.baseVersion } ): ${ nodeString } -> ${ range }`;\n\t// @if CK_DEBUG_ENGINE // }\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/documentfragment\n */\n\nimport TypeCheckable from './typecheckable';\nimport Element from './element';\nimport NodeList from './nodelist';\nimport Text from './text';\nimport TextProxy from './textproxy';\n\nimport type Item from './item';\nimport type Node from './node';\nimport type Range from './range';\n\nimport { isIterable } from '@ckeditor/ckeditor5-utils';\n\n// @if CK_DEBUG_ENGINE // const { stringifyMap } = require( '../dev-utils/utils' );\n\n/**\n * DocumentFragment represents a part of model which does not have a common root but its top-level nodes\n * can be seen as siblings. In other words, it is a detached part of model tree, without a root.\n *\n * DocumentFragment has own {@link module:engine/model/markercollection~MarkerCollection}. Markers from this collection\n * will be set to the {@link module:engine/model/model~Model#markers model markers} by a\n * {@link module:engine/model/writer~Writer#insert} function.\n */\nexport default class DocumentFragment extends TypeCheckable implements Iterable<Node> {\n\t/**\n\t * DocumentFragment static markers map. This is a list of names and {@link module:engine/model/range~Range ranges}\n\t * which will be set as Markers to {@link module:engine/model/model~Model#markers model markers collection}\n\t * when DocumentFragment will be inserted to the document.\n\t */\n\tpublic readonly markers: Map<string, Range> = new Map();\n\n\t/**\n\t * Artificial element name. Returns `undefined`. Added for compatibility reasons.\n\t */\n\tdeclare public name?: undefined;\n\n\t/**\n\t * Artificial root name. Returns `undefined`. Added for compatibility reasons.\n\t */\n\tdeclare public rootName?: undefined;\n\n\t/**\n\t * List of nodes contained inside the document fragment.\n\t */\n\tprivate readonly _children: NodeList = new NodeList();\n\n\t/**\n\t * Creates an empty `DocumentFragment`.\n\t *\n\t * **Note:** Constructor of this class shouldn't be used directly in the code.\n\t * Use the {@link module:engine/model/writer~Writer#createDocumentFragment} method instead.\n\t *\n\t * @internal\n\t * @param children Nodes to be contained inside the `DocumentFragment`.\n\t */\n\tconstructor( children?: Node | Iterable<Node> ) {\n\t\tsuper();\n\n\t\tif ( children ) {\n\t\t\tthis._insertChild( 0, children );\n\t\t}\n\t}\n\n\t/**\n\t * Returns an iterator that iterates over all nodes contained inside this document fragment.\n\t */\n\tpublic [ Symbol.iterator ](): IterableIterator<Node> {\n\t\treturn this.getChildren();\n\t}\n\n\t/**\n\t * Number of this document fragment's children.\n\t */\n\tpublic get childCount(): number {\n\t\treturn this._children.length;\n\t}\n\n\t/**\n\t * Sum of {@link module:engine/model/node~Node#offsetSize offset sizes} of all of this document fragment's children.\n\t */\n\tpublic get maxOffset(): number {\n\t\treturn this._children.maxOffset;\n\t}\n\n\t/**\n\t * Is `true` if there are no nodes inside this document fragment, `false` otherwise.\n\t */\n\tpublic get isEmpty(): boolean {\n\t\treturn this.childCount === 0;\n\t}\n\n\t/**\n\t * Artificial next sibling. Returns `null`. Added for compatibility reasons.\n\t */\n\tpublic get nextSibling(): null {\n\t\treturn null;\n\t}\n\n\t/**\n\t * Artificial previous sibling. Returns `null`. Added for compatibility reasons.\n\t */\n\tpublic get previousSibling(): null {\n\t\treturn null;\n\t}\n\n\t/**\n\t * Artificial root of `DocumentFragment`. Returns itself. Added for compatibility reasons.\n\t */\n\tpublic get root(): DocumentFragment {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Artificial parent of `DocumentFragment`. Returns `null`. Added for compatibility reasons.\n\t */\n\tpublic get parent(): null {\n\t\treturn null;\n\t}\n\n\t/**\n\t * Artificial owner of `DocumentFragment`. Returns `null`. Added for compatibility reasons.\n\t */\n\tpublic get document(): null {\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns `false` as `DocumentFragment` by definition is not attached to a document. Added for compatibility reasons.\n\t */\n\tpublic isAttached(): false {\n\t\treturn false;\n\t}\n\n\t/**\n\t * Returns empty array. Added for compatibility reasons.\n\t */\n\tpublic getAncestors(): Array<never> {\n\t\treturn [];\n\t}\n\n\t/**\n\t * Gets the child at the given index. Returns `null` if incorrect index was passed.\n\t *\n\t * @param index Index of child.\n\t * @returns Child node.\n\t */\n\tpublic getChild( index: number ): Node | null {\n\t\treturn this._children.getNode( index );\n\t}\n\n\t/**\n\t * Returns an iterator that iterates over all of this document fragment's children.\n\t */\n\tpublic getChildren(): IterableIterator<Node> {\n\t\treturn this._children[ Symbol.iterator ]();\n\t}\n\n\t/**\n\t * Returns an index of the given child node. Returns `null` if given node is not a child of this document fragment.\n\t *\n\t * @param node Child node to look for.\n\t * @returns Child node's index.\n\t */\n\tpublic getChildIndex( node: Node ): number | null {\n\t\treturn this._children.getNodeIndex( node );\n\t}\n\n\t/**\n\t * Returns the starting offset of given child. Starting offset is equal to the sum of\n\t * {@link module:engine/model/node~Node#offsetSize offset sizes} of all node's siblings that are before it. Returns `null` if\n\t * given node is not a child of this document fragment.\n\t *\n\t * @param node Child node to look for.\n\t * @returns Child node's starting offset.\n\t */\n\tpublic getChildStartOffset( node: Node ): number | null {\n\t\treturn this._children.getNodeStartOffset( node );\n\t}\n\n\t/**\n\t * Returns path to a `DocumentFragment`, which is an empty array. Added for compatibility reasons.\n\t */\n\tpublic getPath(): Array<number> {\n\t\treturn [];\n\t}\n\n\t/**\n\t * Returns a descendant node by its path relative to this element.\n\t *\n\t * ```ts\n\t * // <this>a<b>c</b></this>\n\t * this.getNodeByPath( [ 0 ] ); // -> \"a\"\n\t * this.getNodeByPath( [ 1 ] ); // -> <b>\n\t * this.getNodeByPath( [ 1, 0 ] ); // -> \"c\"\n\t * ```\n\t *\n\t * @param relativePath Path of the node to find, relative to this element.\n\t */\n\tpublic getNodeByPath( relativePath: Array<number> ): Node | DocumentFragment {\n\t\t// eslint-disable-next-line @typescript-eslint/no-this-alias, consistent-this\n\t\tlet node: Node | DocumentFragment = this;\n\n\t\tfor ( const index of relativePath ) {\n\t\t\tnode = ( node as Element | DocumentFragment ).getChild( ( node as Element | DocumentFragment ).offsetToIndex( index ) )!;\n\t\t}\n\n\t\treturn node;\n\t}\n\n\t/**\n\t * Converts offset \"position\" to index \"position\".\n\t *\n\t * Returns index of a node that occupies given offset. If given offset is too low, returns `0`. If given offset is\n\t * too high, returns index after last child.\n\t *\n\t * ```ts\n\t * const textNode = new Text( 'foo' );\n\t * const pElement = new Element( 'p' );\n\t * const docFrag = new DocumentFragment( [ textNode, pElement ] );\n\t * docFrag.offsetToIndex( -1 ); // Returns 0, because offset is too low.\n\t * docFrag.offsetToIndex( 0 ); // Returns 0, because offset 0 is taken by `textNode` which is at index 0.\n\t * docFrag.offsetToIndex( 1 ); // Returns 0, because `textNode` has `offsetSize` equal to 3, so it occupies offset 1 too.\n\t * docFrag.offsetToIndex( 2 ); // Returns 0.\n\t * docFrag.offsetToIndex( 3 ); // Returns 1.\n\t * docFrag.offsetToIndex( 4 ); // Returns 2. There are no nodes at offset 4, so last available index is returned.\n\t * ```\n\t *\n\t * @param offset Offset to look for.\n\t * @returns Index of a node that occupies given offset.\n\t */\n\tpublic offsetToIndex( offset: number ): number {\n\t\treturn this._children.offsetToIndex( offset );\n\t}\n\n\t/**\n\t * Converts `DocumentFragment` instance to plain object and returns it.\n\t * Takes care of converting all of this document fragment's children.\n\t *\n\t * @returns `DocumentFragment` instance converted to plain object.\n\t */\n\tpublic toJSON(): unknown {\n\t\tconst json = [];\n\n\t\tfor ( const node of this._children ) {\n\t\t\tjson.push( node.toJSON() );\n\t\t}\n\n\t\treturn json;\n\t}\n\n\t/**\n\t * Creates a `DocumentFragment` instance from given plain object (i.e. parsed JSON string).\n\t * Converts `DocumentFragment` children to proper nodes.\n\t *\n\t * @param json Plain object to be converted to `DocumentFragment`.\n\t * @returns `DocumentFragment` instance created using given plain object.\n\t */\n\tpublic static fromJSON( json: any ): DocumentFragment {\n\t\tconst children = [];\n\n\t\tfor ( const child of json ) {\n\t\t\tif ( child.name ) {\n\t\t\t\t// If child has name property, it is an Element.\n\t\t\t\tchildren.push( Element.fromJSON( child ) );\n\t\t\t} else {\n\t\t\t\t// Otherwise, it is a Text node.\n\t\t\t\tchildren.push( Text.fromJSON( child ) );\n\t\t\t}\n\t\t}\n\n\t\treturn new DocumentFragment( children );\n\t}\n\n\t/**\n\t * {@link #_insertChild Inserts} one or more nodes at the end of this document fragment.\n\t *\n\t * @internal\n\t * @param items Items to be inserted.\n\t */\n\tpublic _appendChild( items: string | Item | Iterable<string | Item> ): void {\n\t\tthis._insertChild( this.childCount, items );\n\t}\n\n\t/**\n\t * Inserts one or more nodes at the given index and sets {@link module:engine/model/node~Node#parent parent} of these nodes\n\t * to this document fragment.\n\t *\n\t * @internal\n\t * @param index Index at which nodes should be inserted.\n\t * @param items Items to be inserted.\n\t */\n\tpublic _insertChild( index: number, items: string | Item | Iterable<string | Item> ): void {\n\t\tconst nodes = normalize( items );\n\n\t\tfor ( const node of nodes ) {\n\t\t\t// If node that is being added to this element is already inside another element, first remove it from the old parent.\n\t\t\tif ( node.parent !== null ) {\n\t\t\t\tnode._remove();\n\t\t\t}\n\n\t\t\t( node as any ).parent = this;\n\t\t}\n\n\t\tthis._children._insertNodes( index, nodes );\n\t}\n\n\t/**\n\t * Removes one or more nodes starting at the given index\n\t * and sets {@link module:engine/model/node~Node#parent parent} of these nodes to `null`.\n\t *\n\t * @internal\n\t * @param index Index of the first node to remove.\n\t * @param howMany Number of nodes to remove.\n\t * @returns Array containing removed nodes.\n\t */\n\tpublic _removeChildren( index: number, howMany: number = 1 ): Array<Node> {\n\t\tconst nodes = this._children._removeNodes( index, howMany );\n\n\t\tfor ( const node of nodes ) {\n\t\t\t( node as any ).parent = null;\n\t\t}\n\n\t\treturn nodes;\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public override toString(): 'documentFragment' {\n\t// @if CK_DEBUG_ENGINE // \treturn 'documentFragment';\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public log(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( 'ModelDocumentFragment: ' + this );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public printTree(): string {\n\t// @if CK_DEBUG_ENGINE // \tlet string = 'ModelDocumentFragment: [';\n\n\t// @if CK_DEBUG_ENGINE // \tfor ( const child of this.getChildren() as any ) {\n\t// @if CK_DEBUG_ENGINE // \t\tstring += '\\n';\n\n\t// @if CK_DEBUG_ENGINE // \t\tif ( child.is( '$text' ) ) {\n\t// @if CK_DEBUG_ENGINE // \t\t\tconst textAttrs = stringifyMap( child._attrs );\n\n\t// @if CK_DEBUG_ENGINE // \t\t\tstring += '\\t'.repeat( 1 );\n\n\t// @if CK_DEBUG_ENGINE // \t\t\tif ( textAttrs !== '' ) {\n\t// @if CK_DEBUG_ENGINE // \t\t\t\tstring += `<$text${ textAttrs }>` + child.data + '</$text>';\n\t// @if CK_DEBUG_ENGINE // \t\t\t} else {\n\t// @if CK_DEBUG_ENGINE // \t\t\t\tstring += child.data;\n\t// @if CK_DEBUG_ENGINE // \t\t\t}\n\t// @if CK_DEBUG_ENGINE // \t\t} else {\n\t// @if CK_DEBUG_ENGINE // \t\t\tstring += child.printTree( 1 );\n\t// @if CK_DEBUG_ENGINE // \t\t}\n\t// @if CK_DEBUG_ENGINE // \t}\n\n\t// @if CK_DEBUG_ENGINE // \tstring += '\\n]';\n\n\t// @if CK_DEBUG_ENGINE // \treturn string;\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public logTree(): void {\n\t// @if CK_DEBUG_ENGINE // \tconsole.log( this.printTree() );\n\t// @if CK_DEBUG_ENGINE // }\n}\n\n// The magic of type inference using `is` method is centralized in `TypeCheckable` class.\n// Proper overload would interfere with that.\nDocumentFragment.prototype.is = function( type: string ): boolean {\n\treturn type === 'documentFragment' || type === 'model:documentFragment';\n};\n\n/**\n * Converts strings to Text and non-iterables to arrays.\n */\nfunction normalize( nodes: string | Item | Iterable<string | Item> ): Iterable<Node> {\n\t// Separate condition because string is iterable.\n\tif ( typeof nodes == 'string' ) {\n\t\treturn [ new Text( nodes ) ];\n\t}\n\n\tif ( !isIterable( nodes ) ) {\n\t\tnodes = [ nodes ];\n\t}\n\n\t// Array.from to enable .map() on non-arrays.\n\treturn Array.from( nodes )\n\t\t.map( node => {\n\t\t\tif ( typeof node == 'string' ) {\n\t\t\t\treturn new Text( node );\n\t\t\t}\n\n\t\t\tif ( node instanceof TextProxy ) {\n\t\t\t\treturn new Text( node.data, node.getAttributes() );\n\t\t\t}\n\n\t\t\treturn node;\n\t\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/writer\n */\n\nimport AttributeOperation from './operation/attributeoperation';\nimport DetachOperation from './operation/detachoperation';\nimport InsertOperation from './operation/insertoperation';\nimport MarkerOperation from './operation/markeroperation';\nimport MergeOperation from './operation/mergeoperation';\nimport MoveOperation from './operation/moveoperation';\nimport RenameOperation from './operation/renameoperation';\nimport RootAttributeOperation from './operation/rootattributeoperation';\nimport RootOperation from './operation/rootoperation';\nimport SplitOperation from './operation/splitoperation';\n\nimport DocumentFragment from './documentfragment';\nimport DocumentSelection from './documentselection';\nimport Element from './element';\nimport Position, { type PositionOffset, type PositionStickiness } from './position';\nimport Range from './range';\nimport RootElement from './rootelement';\nimport Text from './text';\n\nimport type { Marker } from './markercollection';\nimport type { default as Selection, PlaceOrOffset, Selectable } from './selection';\nimport type Batch from './batch';\nimport type Item from './item';\nimport type Model from './model';\nimport type { default as Node, NodeAttributes } from './node';\n\nimport { CKEditorError, logWarning, toMap } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The model can only be modified by using the writer. It should be used whenever you want to create a node, modify\n * child nodes, attributes or text, set the selection's position and its attributes.\n *\n * The instance of the writer is only available in the {@link module:engine/model/model~Model#change `change()`} or\n * {@link module:engine/model/model~Model#enqueueChange `enqueueChange()`}.\n *\n * ```ts\n * model.change( writer => {\n * \twriter.insertText( 'foo', paragraph, 'end' );\n * } );\n * ```\n *\n * Note that the writer should never be stored and used outside of the `change()` and\n * `enqueueChange()` blocks.\n *\n * Note that writer's methods do not check the {@link module:engine/model/schema~Schema}. It is possible\n * to create incorrect model structures by using the writer. Read more about in\n * {@glink framework/deep-dive/schema#who-checks-the-schema \"Who checks the schema?\"}.\n *\n * @see module:engine/model/model~Model#change\n * @see module:engine/model/model~Model#enqueueChange\n */\nexport default class Writer {\n\t/**\n\t * Instance of the model on which this writer operates.\n\t */\n\tpublic readonly model: Model;\n\n\t/**\n\t * The batch to which this writer will add changes.\n\t */\n\tpublic readonly batch: Batch;\n\n\t/**\n\t * Creates a writer instance.\n\t *\n\t * **Note:** It is not recommended to use it directly. Use {@link module:engine/model/model~Model#change `Model#change()`} or\n\t * {@link module:engine/model/model~Model#enqueueChange `Model#enqueueChange()`} instead.\n\t *\n\t * @internal\n\t */\n\tconstructor( model: Model, batch: Batch ) {\n\t\tthis.model = model;\n\t\tthis.batch = batch;\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/model/text~Text text node}.\n\t *\n\t * ```ts\n\t * writer.createText( 'foo' );\n\t * writer.createText( 'foo', { bold: true } );\n\t * ```\n\t *\n\t * @param data Text data.\n\t * @param attributes Text attributes.\n\t * @returns {module:engine/model/text~Text} Created text node.\n\t */\n\tpublic createText(\n\t\tdata: string,\n\t\tattributes?: NodeAttributes\n\t): Text {\n\t\treturn new Text( data, attributes );\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/model/element~Element element}.\n\t *\n\t * ```ts\n\t * writer.createElement( 'paragraph' );\n\t * writer.createElement( 'paragraph', { alignment: 'center' } );\n\t * ```\n\t *\n\t * @param name Name of the element.\n\t * @param attributes Elements attributes.\n\t * @returns Created element.\n\t */\n\tpublic createElement(\n\t\tname: string,\n\t\tattributes?: NodeAttributes\n\t): Element {\n\t\treturn new Element( name, attributes );\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/model/documentfragment~DocumentFragment document fragment}.\n\t *\n\t * @returns Created document fragment.\n\t */\n\tpublic createDocumentFragment(): DocumentFragment {\n\t\treturn new DocumentFragment();\n\t}\n\n\t/**\n\t * Creates a copy of the element and returns it. Created element has the same name and attributes as the original element.\n\t * If clone is deep, the original element's children are also cloned. If not, then empty element is returned.\n\t *\n\t * @param element The element to clone.\n\t * @param deep If set to `true` clones element and all its children recursively. When set to `false`,\n\t * element will be cloned without any child.\n\t */\n\tpublic cloneElement( element: Element, deep: boolean = true ): Element {\n\t\treturn element._clone( deep );\n\t}\n\n\t/**\n\t * Inserts item on given position.\n\t *\n\t * ```ts\n\t * const paragraph = writer.createElement( 'paragraph' );\n\t * writer.insert( paragraph, position );\n\t * ```\n\t *\n\t * Instead of using position you can use parent and offset:\n\t *\n\t * ```ts\n\t * const text = writer.createText( 'foo' );\n\t * writer.insert( text, paragraph, 5 );\n\t * ```\n\t *\n\t * You can also use `end` instead of the offset to insert at the end:\n\t *\n\t * ```ts\n\t * const text = writer.createText( 'foo' );\n\t * writer.insert( text, paragraph, 'end' );\n\t * ```\n\t *\n\t * Or insert before or after another element:\n\t *\n\t * ```ts\n\t * const paragraph = writer.createElement( 'paragraph' );\n\t * writer.insert( paragraph, anotherParagraph, 'after' );\n\t * ```\n\t *\n\t * These parameters works the same way as {@link #createPositionAt `writer.createPositionAt()`}.\n\t *\n\t * Note that if the item already has parent it will be removed from the previous parent.\n\t *\n\t * Note that you cannot re-insert a node from a document to a different document or a document fragment. In this case,\n\t * `model-writer-insert-forbidden-move` is thrown.\n\t *\n\t * If you want to move {@link module:engine/model/range~Range range} instead of an\n\t * {@link module:engine/model/item~Item item} use {@link module:engine/model/writer~Writer#move `Writer#move()`}.\n\t *\n\t * **Note:** For a paste-like content insertion mechanism see\n\t * {@link module:engine/model/model~Model#insertContent `model.insertContent()`}.\n\t *\n\t * @param item Item or document fragment to insert.\n\t * @param offset Offset or one of the flags. Used only when second parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic insert(\n\t\titem: Item | DocumentFragment,\n\t\titemOrPosition: Item | DocumentFragment | Position,\n\t\toffset: PositionOffset = 0\n\t): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( item instanceof Text && item.data == '' ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst position = Position._createAt( itemOrPosition, offset );\n\n\t\t// If item has a parent already.\n\t\tif ( item.parent ) {\n\t\t\t// We need to check if item is going to be inserted within the same document.\n\t\t\tif ( isSameTree( item.root, position.root ) ) {\n\t\t\t\t// If it's we just need to move it.\n\t\t\t\tthis.move( Range._createOn( item ), position );\n\n\t\t\t\treturn;\n\t\t\t}\n\t\t\t// If it isn't the same root.\n\t\t\telse {\n\t\t\t\tif ( item.root.document ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * Cannot move a node from a document to a different tree.\n\t\t\t\t\t * It is forbidden to move a node that was already in a document outside of it.\n\t\t\t\t\t *\n\t\t\t\t\t * @error model-writer-insert-forbidden-move\n\t\t\t\t\t */\n\t\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t\t'model-writer-insert-forbidden-move',\n\t\t\t\t\t\tthis\n\t\t\t\t\t);\n\t\t\t\t} else {\n\t\t\t\t\t// Move between two different document fragments or from document fragment to a document is possible.\n\t\t\t\t\t// In that case, remove the item from it's original parent.\n\t\t\t\t\tthis.remove( item );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tconst version = position.root.document ? position.root.document.version : null;\n\n\t\tconst insert = new InsertOperation( position, item, version );\n\n\t\tif ( item instanceof Text ) {\n\t\t\tinsert.shouldReceiveAttributes = true;\n\t\t}\n\n\t\tthis.batch.addOperation( insert );\n\t\tthis.model.applyOperation( insert );\n\n\t\t// When element is a DocumentFragment we need to move its markers to Document#markers.\n\t\tif ( item instanceof DocumentFragment ) {\n\t\t\tfor ( const [ markerName, markerRange ] of item.markers ) {\n\t\t\t\t// We need to migrate marker range from DocumentFragment to Document.\n\t\t\t\tconst rangeRootPosition = Position._createAt( markerRange.root, 0 );\n\t\t\t\tconst range = new Range(\n\t\t\t\t\tmarkerRange.start._getCombined( rangeRootPosition, position ),\n\t\t\t\t\tmarkerRange.end._getCombined( rangeRootPosition, position )\n\t\t\t\t);\n\n\t\t\t\tconst options = { range, usingOperation: true, affectsData: true };\n\n\t\t\t\tif ( this.model.markers.has( markerName ) ) {\n\t\t\t\t\tthis.updateMarker( markerName, options );\n\t\t\t\t} else {\n\t\t\t\t\tthis.addMarker( markerName, options );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Creates and inserts text on given position.\n\t *\n\t * ```ts\n\t * writer.insertText( 'foo', position );\n\t * ```\n\t *\n\t * Instead of using position you can use parent and offset or define that text should be inserted at the end\n\t * or before or after other node:\n\t *\n\t * ```ts\n\t * // Inserts 'foo' in paragraph, at offset 5:\n\t * writer.insertText( 'foo', paragraph, 5 );\n\t * // Inserts 'foo' at the end of a paragraph:\n\t * writer.insertText( 'foo', paragraph, 'end' );\n\t * // Inserts 'foo' after an image:\n\t * writer.insertText( 'foo', image, 'after' );\n\t * ```\n\t *\n\t * These parameters work in the same way as {@link #createPositionAt `writer.createPositionAt()`}.\n\t *\n\t * @label WITHOUT_ATTRIBUTES\n\t * @param text Text data.\n\t * @param offset Offset or one of the flags. Used only when second parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic insertText(\n\t\ttext: string,\n\t\titemOrPosition?: Item | Position,\n\t\toffset?: PositionOffset\n\t): void;\n\n\t/**\n\t * Creates and inserts text with specified attributes on given position.\n\t *\n\t * ```ts\n\t * writer.insertText( 'foo', { bold: true }, position );\n\t * ```\n\t *\n\t * Instead of using position you can use parent and offset or define that text should be inserted at the end\n\t * or before or after other node:\n\t *\n\t * ```ts\n\t * // Inserts 'foo' in paragraph, at offset 5:\n\t * writer.insertText( 'foo', { bold: true }, paragraph, 5 );\n\t * // Inserts 'foo' at the end of a paragraph:\n\t * writer.insertText( 'foo', { bold: true }, paragraph, 'end' );\n\t * // Inserts 'foo' after an image:\n\t * writer.insertText( 'foo', { bold: true }, image, 'after' );\n\t * ```\n\t *\n\t * These parameters work in the same way as {@link #createPositionAt `writer.createPositionAt()`}.\n\t *\n\t * @label WITH_ATTRIBUTES\n\t * @param text Text data.\n\t * @param attributes Text attributes.\n\t * @param offset Offset or one of the flags. Used only when third parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic insertText(\n\t\ttext: string,\n\t\tattributes?: NodeAttributes,\n\t\titemOrPosition?: Item | Position,\n\t\toffset?: PositionOffset\n\t): void;\n\n\tpublic insertText(\n\t\ttext: string,\n\t\tattributes?: any, // Too complicated when not using `any`.\n\t\titemOrPosition?: any, // Too complicated when not using `any`.\n\t\toffset?: any // Too complicated when not using `any`.\n\t): void {\n\t\tif ( attributes instanceof DocumentFragment || attributes instanceof Element || attributes instanceof Position ) {\n\t\t\tthis.insert( this.createText( text ), attributes, itemOrPosition );\n\t\t} else {\n\t\t\tthis.insert( this.createText( text, attributes ), itemOrPosition, offset );\n\t\t}\n\t}\n\n\t/**\n\t * Creates and inserts element on given position. You can optionally set attributes:\n\t *\n\t * ```ts\n\t * writer.insertElement( 'paragraph', position );\n\t * ```\n\t *\n\t * Instead of using position you can use parent and offset or define that text should be inserted at the end\n\t * or before or after other node:\n\t *\n\t * ```ts\n\t * // Inserts paragraph in the root at offset 5:\n\t * writer.insertElement( 'paragraph', root, 5 );\n\t * // Inserts paragraph at the end of a blockquote:\n\t * writer.insertElement( 'paragraph', blockquote, 'end' );\n\t * // Inserts after an image:\n\t * writer.insertElement( 'paragraph', image, 'after' );\n\t * ```\n\t *\n\t * These parameters works the same way as {@link #createPositionAt `writer.createPositionAt()`}.\n\t *\n\t * @label WITHOUT_ATTRIBUTES\n\t * @param name Name of the element.\n\t * @param offset Offset or one of the flags. Used only when second parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic insertElement(\n\t\tname: string,\n\t\titemOrPosition: Item | DocumentFragment | Position,\n\t\toffset?: PositionOffset\n\t): void;\n\n\t/**\n\t * Creates and inserts element with specified attributes on given position.\n\t *\n\t * ```ts\n\t * writer.insertElement( 'paragraph', { alignment: 'center' }, position );\n\t * ```\n\t *\n\t * Instead of using position you can use parent and offset or define that text should be inserted at the end\n\t * or before or after other node:\n\t *\n\t * ```ts\n\t * // Inserts paragraph in the root at offset 5:\n\t * writer.insertElement( 'paragraph', { alignment: 'center' }, root, 5 );\n\t * // Inserts paragraph at the end of a blockquote:\n\t * writer.insertElement( 'paragraph', { alignment: 'center' }, blockquote, 'end' );\n\t * // Inserts after an image:\n\t * writer.insertElement( 'paragraph', { alignment: 'center' }, image, 'after' );\n\t * ```\n\t *\n\t * These parameters works the same way as {@link #createPositionAt `writer.createPositionAt()`}.\n\t *\n\t * @label WITH_ATTRIBUTES\n\t * @param name Name of the element.\n\t * @param attributes Elements attributes.\n\t * @param offset Offset or one of the flags. Used only when third parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic insertElement(\n\t\tname: string,\n\t\tattributes: NodeAttributes,\n\t\titemOrPosition: Item | DocumentFragment | Position,\n\t\toffset?: PositionOffset\n\t): void;\n\n\tpublic insertElement(\n\t\tname: string,\n\t\tattributes: any, // Too complicated when not using `any`.\n\t\titemOrPositionOrOffset?: any, // Too complicated when not using `any`.\n\t\toffset?: any // Too complicated when not using `any`.\n\t): void {\n\t\tif ( attributes instanceof DocumentFragment || attributes instanceof Element || attributes instanceof Position ) {\n\t\t\tthis.insert( this.createElement( name ), attributes, itemOrPositionOrOffset );\n\t\t} else {\n\t\t\tthis.insert( this.createElement( name, attributes ), itemOrPositionOrOffset, offset );\n\t\t}\n\t}\n\n\t/**\n\t * Inserts item at the end of the given parent.\n\t *\n\t * ```ts\n\t * const paragraph = writer.createElement( 'paragraph' );\n\t * writer.append( paragraph, root );\n\t * ```\n\t *\n\t * Note that if the item already has parent it will be removed from the previous parent.\n\t *\n\t * If you want to move {@link module:engine/model/range~Range range} instead of an\n\t * {@link module:engine/model/item~Item item} use {@link module:engine/model/writer~Writer#move `Writer#move()`}.\n\t *\n\t * @param item Item or document fragment to insert.\n\t */\n\tpublic append( item: Item | DocumentFragment, parent: Element | DocumentFragment ): void {\n\t\tthis.insert( item, parent, 'end' );\n\t}\n\n\t/**\n\t * Creates text node and inserts it at the end of the parent.\n\t *\n\t * ```ts\n\t * writer.appendText( 'foo', paragraph );\n\t * ```\n\t *\n\t * @label WITHOUT_ATTRIBUTES\n\t * @param text Text data.\n\t */\n\tpublic appendText(\n\t\ttext: string,\n\t\tparent: Element | DocumentFragment\n\t): void;\n\n\t/**\n\t * Creates text node with specified attributes and inserts it at the end of the parent.\n\t *\n\t * ```ts\n\t * writer.appendText( 'foo', { bold: true }, paragraph );\n\t * ```\n\t *\n\t * @label WITH_ATTRIBUTES\n\t * @param text Text data.\n\t * @param attributes Text attributes.\n\t */\n\tpublic appendText(\n\t\ttext: string,\n\t\tattributes: NodeAttributes,\n\t\tparent: Element | DocumentFragment\n\t): void;\n\n\tpublic appendText(\n\t\ttext: string,\n\t\tattributes: NodeAttributes | Element | DocumentFragment,\n\t\tparent?: Element | DocumentFragment\n\t): void {\n\t\tif ( attributes instanceof DocumentFragment || attributes instanceof Element ) {\n\t\t\tthis.insert( this.createText( text ), attributes, 'end' );\n\t\t} else {\n\t\t\tthis.insert( this.createText( text, attributes ), parent!, 'end' );\n\t\t}\n\t}\n\n\t/**\n\t * Creates element and inserts it at the end of the parent.\n\t *\n\t * ```ts\n\t * writer.appendElement( 'paragraph', root );\n\t * ```\n\t *\n\t * @label WITHOUT_ATTRIBUTES\n\t * @param name Name of the element.\n\t */\n\tpublic appendElement(\n\t\tname: string,\n\t\tparent: Element | DocumentFragment\n\t): void;\n\n\t/**\n\t * Creates element with specified attributes and inserts it at the end of the parent.\n\t *\n\t * ```ts\n\t * writer.appendElement( 'paragraph', { alignment: 'center' }, root );\n\t * ```\n\t *\n\t * @label WITH_ATTRIBUTES\n\t * @param name Name of the element.\n\t * @param attributes Elements attributes.\n\t */\n\tpublic appendElement(\n\t\tname: string,\n\t\tattributes: NodeAttributes,\n\t\tparent: Element | DocumentFragment\n\t): void;\n\n\tpublic appendElement(\n\t\tname: string,\n\t\tattributes: NodeAttributes | Element | DocumentFragment,\n\t\tparent?: Element | DocumentFragment\n\t): void {\n\t\tif ( attributes instanceof DocumentFragment || attributes instanceof Element ) {\n\t\t\tthis.insert( this.createElement( name ), attributes, 'end' );\n\t\t} else {\n\t\t\tthis.insert( this.createElement( name, attributes ), parent!, 'end' );\n\t\t}\n\t}\n\n\t/**\n\t * Sets value of the attribute with given key on a {@link module:engine/model/item~Item model item}\n\t * or on a {@link module:engine/model/range~Range range}.\n\t *\n\t * @param key Attribute key.\n\t * @param value Attribute new value.\n\t * @param itemOrRange Model item or range on which the attribute will be set.\n\t */\n\tpublic setAttribute( key: string, value: unknown, itemOrRange: Item | Range ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( itemOrRange instanceof Range ) {\n\t\t\tconst ranges = itemOrRange.getMinimalFlatRanges();\n\n\t\t\tfor ( const range of ranges ) {\n\t\t\t\tsetAttributeOnRange( this, key, value, range );\n\t\t\t}\n\t\t} else {\n\t\t\tsetAttributeOnItem( this, key, value, itemOrRange );\n\t\t}\n\t}\n\n\t/**\n\t * Sets values of attributes on a {@link module:engine/model/item~Item model item}\n\t * or on a {@link module:engine/model/range~Range range}.\n\t *\n\t * ```ts\n\t * writer.setAttributes( {\n\t * \tbold: true,\n\t * \titalic: true\n\t * }, range );\n\t * ```\n\t *\n\t * @param attributes Attributes keys and values.\n\t * @param itemOrRange Model item or range on which the attributes will be set.\n\t */\n\tpublic setAttributes(\n\t\tattributes: NodeAttributes,\n\t\titemOrRange: Item | Range\n\t): void {\n\t\tfor ( const [ key, val ] of toMap( attributes ) ) {\n\t\t\tthis.setAttribute( key, val, itemOrRange );\n\t\t}\n\t}\n\n\t/**\n\t * Removes an attribute with given key from a {@link module:engine/model/item~Item model item}\n\t * or from a {@link module:engine/model/range~Range range}.\n\t *\n\t * @param key Attribute key.\n\t * @param itemOrRange Model item or range from which the attribute will be removed.\n\t */\n\tpublic removeAttribute( key: string, itemOrRange: Item | Range ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( itemOrRange instanceof Range ) {\n\t\t\tconst ranges = itemOrRange.getMinimalFlatRanges();\n\n\t\t\tfor ( const range of ranges ) {\n\t\t\t\tsetAttributeOnRange( this, key, null, range );\n\t\t\t}\n\t\t} else {\n\t\t\tsetAttributeOnItem( this, key, null, itemOrRange );\n\t\t}\n\t}\n\n\t/**\n\t * Removes all attributes from all elements in the range or from the given item.\n\t *\n\t * @param itemOrRange Model item or range from which all attributes will be removed.\n\t */\n\tpublic clearAttributes( itemOrRange: Item | Range ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tconst removeAttributesFromItem = ( item: Item ) => {\n\t\t\tfor ( const attribute of item.getAttributeKeys() ) {\n\t\t\t\tthis.removeAttribute( attribute, item );\n\t\t\t}\n\t\t};\n\n\t\tif ( !( itemOrRange instanceof Range ) ) {\n\t\t\tremoveAttributesFromItem( itemOrRange );\n\t\t} else {\n\t\t\tfor ( const item of itemOrRange.getItems() ) {\n\t\t\t\tremoveAttributesFromItem( item );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Moves all items in the source range to the target position.\n\t *\n\t * ```ts\n\t * writer.move( sourceRange, targetPosition );\n\t * ```\n\t *\n\t * Instead of the target position you can use parent and offset or define that range should be moved to the end\n\t * or before or after chosen item:\n\t *\n\t * ```ts\n\t * // Moves all items in the range to the paragraph at offset 5:\n\t * writer.move( sourceRange, paragraph, 5 );\n\t * // Moves all items in the range to the end of a blockquote:\n\t * writer.move( sourceRange, blockquote, 'end' );\n\t * // Moves all items in the range to a position after an image:\n\t * writer.move( sourceRange, image, 'after' );\n\t * ```\n\t *\n\t * These parameters work the same way as {@link #createPositionAt `writer.createPositionAt()`}.\n\t *\n\t * Note that items can be moved only within the same tree. It means that you can move items within the same root\n\t * (element or document fragment) or between {@link module:engine/model/document~Document#roots documents roots},\n\t * but you can not move items from document fragment to the document or from one detached element to another. Use\n\t * {@link module:engine/model/writer~Writer#insert} in such cases.\n\t *\n\t * @param range Source range.\n\t * @param offset Offset or one of the flags. Used only when second parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic move(\n\t\trange: Range,\n\t\titemOrPosition: Item | Position,\n\t\toffset?: PositionOffset\n\t): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( !( range instanceof Range ) ) {\n\t\t\t/**\n\t\t\t * Invalid range to move.\n\t\t\t *\n\t\t\t * @error writer-move-invalid-range\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-move-invalid-range', this );\n\t\t}\n\n\t\tif ( !range.isFlat ) {\n\t\t\t/**\n\t\t\t * Range to move is not flat.\n\t\t\t *\n\t\t\t * @error writer-move-range-not-flat\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-move-range-not-flat', this );\n\t\t}\n\n\t\tconst position = Position._createAt( itemOrPosition, offset );\n\n\t\t// Do not move anything if the move target is same as moved range start.\n\t\tif ( position.isEqual( range.start ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// If part of the marker is removed, create additional marker operation for undo purposes.\n\t\tthis._addOperationForAffectedMarkers( 'move', range );\n\n\t\tif ( !isSameTree( range.root, position.root ) ) {\n\t\t\t/**\n\t\t\t * Range is going to be moved within not the same document. Please use\n\t\t\t * {@link module:engine/model/writer~Writer#insert insert} instead.\n\t\t\t *\n\t\t\t * @error writer-move-different-document\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-move-different-document', this );\n\t\t}\n\n\t\tconst version = range.root.document ? range.root.document.version : null;\n\t\tconst operation = new MoveOperation( range.start, range.end.offset - range.start.offset, position, version );\n\n\t\tthis.batch.addOperation( operation );\n\t\tthis.model.applyOperation( operation );\n\t}\n\n\t/**\n\t * Removes given model {@link module:engine/model/item~Item item} or {@link module:engine/model/range~Range range}.\n\t *\n\t * @param itemOrRange Model item or range to remove.\n\t */\n\tpublic remove( itemOrRange: Item | Range ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tconst rangeToRemove = itemOrRange instanceof Range ? itemOrRange : Range._createOn( itemOrRange );\n\t\tconst ranges = rangeToRemove.getMinimalFlatRanges().reverse();\n\n\t\tfor ( const flat of ranges ) {\n\t\t\t// If part of the marker is removed, create additional marker operation for undo purposes.\n\t\t\tthis._addOperationForAffectedMarkers( 'move', flat );\n\n\t\t\tapplyRemoveOperation( flat.start, flat.end.offset - flat.start.offset, this.batch, this.model );\n\t\t}\n\t}\n\n\t/**\n\t * Merges two siblings at the given position.\n\t *\n\t * Node before and after the position have to be an element. Otherwise `writer-merge-no-element-before` or\n\t * `writer-merge-no-element-after` error will be thrown.\n\t *\n\t * @param position Position between merged elements.\n\t */\n\tpublic merge( position: Position ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tconst nodeBefore = position.nodeBefore;\n\t\tconst nodeAfter = position.nodeAfter;\n\n\t\t// If part of the marker is removed, create additional marker operation for undo purposes.\n\t\tthis._addOperationForAffectedMarkers( 'merge', position );\n\n\t\tif ( !( nodeBefore instanceof Element ) ) {\n\t\t\t/**\n\t\t\t * Node before merge position must be an element.\n\t\t\t *\n\t\t\t * @error writer-merge-no-element-before\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-merge-no-element-before', this );\n\t\t}\n\n\t\tif ( !( nodeAfter instanceof Element ) ) {\n\t\t\t/**\n\t\t\t * Node after merge position must be an element.\n\t\t\t *\n\t\t\t * @error writer-merge-no-element-after\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-merge-no-element-after', this );\n\t\t}\n\n\t\tif ( !position.root.document ) {\n\t\t\tthis._mergeDetached( position );\n\t\t} else {\n\t\t\tthis._merge( position );\n\t\t}\n\t}\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createPositionFromPath `Model#createPositionFromPath()`}.\n\t *\n\t * @param root Root of the position.\n\t * @param path Position path. See {@link module:engine/model/position~Position#path}.\n\t * @param stickiness Position stickiness. See {@link module:engine/model/position~PositionStickiness}.\n\t */\n\tpublic createPositionFromPath(\n\t\troot: Element | DocumentFragment,\n\t\tpath: ReadonlyArray<number>,\n\t\tstickiness?: PositionStickiness\n\t): Position {\n\t\treturn this.model.createPositionFromPath( root, path, stickiness );\n\t}\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createPositionAt `Model#createPositionAt()`}.\n\t *\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic createPositionAt(\n\t\titemOrPosition: Item | Position | DocumentFragment,\n\t\toffset?: PositionOffset\n\t): Position {\n\t\treturn this.model.createPositionAt( itemOrPosition, offset );\n\t}\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createPositionAfter `Model#createPositionAfter()`}.\n\t *\n\t * @param item Item after which the position should be placed.\n\t */\n\tpublic createPositionAfter( item: Item ): Position {\n\t\treturn this.model.createPositionAfter( item );\n\t}\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createPositionBefore `Model#createPositionBefore()`}.\n\t *\n\t * @param item Item after which the position should be placed.\n\t */\n\tpublic createPositionBefore( item: Item ): Position {\n\t\treturn this.model.createPositionBefore( item );\n\t}\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createRange `Model#createRange()`}.\n\t *\n\t * @param start Start position.\n\t * @param end End position. If not set, range will be collapsed at `start` position.\n\t */\n\tpublic createRange( start: Position, end?: Position ): Range {\n\t\treturn this.model.createRange( start, end );\n\t}\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createRangeIn `Model#createRangeIn()`}.\n\t *\n\t * @param element Element which is a parent for the range.\n\t */\n\tpublic createRangeIn( element: Element | DocumentFragment ): Range {\n\t\treturn this.model.createRangeIn( element );\n\t}\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createRangeOn `Model#createRangeOn()`}.\n\t *\n\t * @param element Element which is a parent for the range.\n\t */\n\tpublic createRangeOn( element: Item ): Range {\n\t\treturn this.model.createRangeOn( element );\n\t}\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createSelection:NODE_OFFSET `Model#createSelection()`}.\n\t *\n\t * @label NODE_OFFSET\n\t */\n\tpublic createSelection( selectable: Node, placeOrOffset: PlaceOrOffset, options?: { backward?: boolean } ): Selection;\n\n\t/**\n\t * Shortcut for {@link module:engine/model/model~Model#createSelection:SELECTABLE `Model#createSelection()`}.\n\t *\n\t * @label SELECTABLE\n\t */\n\tpublic createSelection( selectable?: Exclude<Selectable, Node>, options?: { backward?: boolean } ): Selection;\n\n\tpublic createSelection( ...args: [ any?, any?, any? ] ): Selection {\n\t\treturn this.model.createSelection( ...args );\n\t}\n\n\t/**\n\t * Performs merge action in a detached tree.\n\t *\n\t * @param position Position between merged elements.\n\t */\n\tprivate _mergeDetached( position: Position ): void {\n\t\tconst nodeBefore = position.nodeBefore;\n\t\tconst nodeAfter = position.nodeAfter;\n\n\t\tthis.move( Range._createIn( nodeAfter as any ), Position._createAt( nodeBefore!, 'end' ) );\n\t\tthis.remove( nodeAfter! );\n\t}\n\n\t/**\n\t * Performs merge action in a non-detached tree.\n\t *\n\t * @param position Position between merged elements.\n\t */\n\tprivate _merge( position: Position ): void {\n\t\tconst targetPosition = Position._createAt( position.nodeBefore!, 'end' );\n\t\tconst sourcePosition = Position._createAt( position.nodeAfter!, 0 );\n\n\t\tconst graveyard = position.root.document!.graveyard;\n\t\tconst graveyardPosition = new Position( graveyard, [ 0 ] );\n\n\t\tconst version = position.root.document!.version;\n\n\t\tconst merge = new MergeOperation(\n\t\t\tsourcePosition,\n\t\t\t( position.nodeAfter as any ).maxOffset,\n\t\t\ttargetPosition,\n\t\t\tgraveyardPosition,\n\t\t\tversion\n\t\t);\n\n\t\tthis.batch.addOperation( merge );\n\t\tthis.model.applyOperation( merge );\n\t}\n\n\t/**\n\t * Renames the given element.\n\t *\n\t * @param element The element to rename.\n\t * @param newName New element name.\n\t */\n\tpublic rename( element: Element | DocumentFragment, newName: string ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( !( element instanceof Element ) ) {\n\t\t\t/**\n\t\t\t * Trying to rename an object which is not an instance of Element.\n\t\t\t *\n\t\t\t * @error writer-rename-not-element-instance\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'writer-rename-not-element-instance',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\tconst version = element.root.document ? element.root.document.version : null;\n\t\tconst renameOperation = new RenameOperation( Position._createBefore( element ), element.name, newName, version );\n\n\t\tthis.batch.addOperation( renameOperation );\n\t\tthis.model.applyOperation( renameOperation );\n\t}\n\n\t/**\n\t * Splits elements starting from the given position and going to the top of the model tree as long as given\n\t * `limitElement` is reached. When `limitElement` is not defined then only the parent of the given position will be split.\n\t *\n\t * The element needs to have a parent. It cannot be a root element nor a document fragment.\n\t * The `writer-split-element-no-parent` error will be thrown if you try to split an element with no parent.\n\t *\n\t * @param position Position of split.\n\t * @param limitElement Stop splitting when this element will be reached.\n\t * @returns Split result with properties:\n\t * * `position` - Position between split elements.\n\t * * `range` - Range that stars from the end of the first split element and ends at the beginning of the first copy element.\n\t */\n\tpublic split( position: Position, limitElement?: Node | DocumentFragment ): { position: Position; range: Range } {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tlet splitElement = position.parent;\n\n\t\tif ( !splitElement.parent ) {\n\t\t\t/**\n\t\t\t * Element with no parent can not be split.\n\t\t\t *\n\t\t\t * @error writer-split-element-no-parent\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-split-element-no-parent', this );\n\t\t}\n\n\t\t// When limit element is not defined lets set splitElement parent as limit.\n\t\tif ( !limitElement ) {\n\t\t\tlimitElement = splitElement.parent;\n\t\t}\n\n\t\tif ( !position.parent.getAncestors( { includeSelf: true } ).includes( limitElement! ) ) {\n\t\t\t/**\n\t\t\t * Limit element is not a position ancestor.\n\t\t\t *\n\t\t\t * @error writer-split-invalid-limit-element\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-split-invalid-limit-element', this );\n\t\t}\n\n\t\t// We need to cache elements that will be created as a result of the first split because\n\t\t// we need to create a range from the end of the first split element to the beginning of the\n\t\t// first copy element. This should be handled by LiveRange but it doesn't work on detached nodes.\n\t\tlet firstSplitElement: Element | DocumentFragment | undefined;\n\t\tlet firstCopyElement: Node | null | undefined;\n\n\t\tdo {\n\t\t\tconst version = splitElement.root.document ? splitElement.root.document.version : null;\n\t\t\tconst howMany = splitElement.maxOffset - position.offset;\n\n\t\t\tconst insertionPosition = SplitOperation.getInsertionPosition( position );\n\t\t\tconst split = new SplitOperation( position, howMany, insertionPosition, null, version );\n\n\t\t\tthis.batch.addOperation( split );\n\t\t\tthis.model.applyOperation( split );\n\n\t\t\t// Cache result of the first split.\n\t\t\tif ( !firstSplitElement && !firstCopyElement ) {\n\t\t\t\tfirstSplitElement = splitElement;\n\t\t\t\tfirstCopyElement = position.parent.nextSibling;\n\t\t\t}\n\n\t\t\tposition = this.createPositionAfter( position.parent as any );\n\t\t\tsplitElement = position.parent;\n\t\t} while ( splitElement !== limitElement );\n\n\t\treturn {\n\t\t\tposition,\n\t\t\trange: new Range( Position._createAt( firstSplitElement!, 'end' ), Position._createAt( firstCopyElement!, 0 ) )\n\t\t};\n\t}\n\n\t/**\n\t * Wraps the given range with the given element or with a new element (if a string was passed).\n\t *\n\t * **Note:** range to wrap should be a \"flat range\" (see {@link module:engine/model/range~Range#isFlat `Range#isFlat`}).\n\t * If not, an error will be thrown.\n\t *\n\t * @param range Range to wrap.\n\t * @param elementOrString Element or name of element to wrap the range with.\n\t */\n\tpublic wrap( range: Range, elementOrString: Element | string ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( !range.isFlat ) {\n\t\t\t/**\n\t\t\t * Range to wrap is not flat.\n\t\t\t *\n\t\t\t * @error writer-wrap-range-not-flat\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-wrap-range-not-flat', this );\n\t\t}\n\n\t\tconst element = elementOrString instanceof Element ? elementOrString : new Element( elementOrString );\n\n\t\tif ( element.childCount > 0 ) {\n\t\t\t/**\n\t\t\t * Element to wrap with is not empty.\n\t\t\t *\n\t\t\t * @error writer-wrap-element-not-empty\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-wrap-element-not-empty', this );\n\t\t}\n\n\t\tif ( element.parent !== null ) {\n\t\t\t/**\n\t\t\t * Element to wrap with is already attached to a tree model.\n\t\t\t *\n\t\t\t * @error writer-wrap-element-attached\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-wrap-element-attached', this );\n\t\t}\n\n\t\tthis.insert( element, range.start );\n\n\t\t// Shift the range-to-wrap because we just inserted an element before that range.\n\t\tconst shiftedRange = new Range( range.start.getShiftedBy( 1 ), range.end.getShiftedBy( 1 ) );\n\n\t\tthis.move( shiftedRange, Position._createAt( element, 0 ) );\n\t}\n\n\t/**\n\t * Unwraps children of the given element all its children are moved before it and then the element is removed.\n\t * Throws error if you try to unwrap an element which does not have a parent.\n\t *\n\t * @param element Element to unwrap.\n\t */\n\tpublic unwrap( element: Element ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( element.parent === null ) {\n\t\t\t/**\n\t\t\t * Trying to unwrap an element which has no parent.\n\t\t\t *\n\t\t\t * @error writer-unwrap-element-no-parent\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-unwrap-element-no-parent', this );\n\t\t}\n\n\t\tthis.move( Range._createIn( element ), this.createPositionAfter( element ) );\n\t\tthis.remove( element );\n\t}\n\n\t/**\n\t * Adds a {@link module:engine/model/markercollection~Marker marker}. Marker is a named range, which tracks\n\t * changes in the document and updates its range automatically, when model tree changes.\n\t *\n\t * As the first parameter you can set marker name.\n\t *\n\t * The required `options.usingOperation` parameter lets you decide if the marker should be managed by operations or not. See\n\t * {@link module:engine/model/markercollection~Marker marker class description} to learn about the difference between\n\t * markers managed by operations and not-managed by operations.\n\t *\n\t * The `options.affectsData` parameter, which defaults to `false`, allows you to define if a marker affects the data. It should be\n\t * `true` when the marker change changes the data returned by the\n\t * {@link module:core/editor/utils/dataapimixin~DataApi#getData `editor.getData()`} method.\n\t * When set to `true` it fires the {@link module:engine/model/document~Document#event:change:data `change:data`} event.\n\t * When set to `false` it fires the {@link module:engine/model/document~Document#event:change `change`} event.\n\t *\n\t * Create marker directly base on marker's name:\n\t *\n\t * ```ts\n\t * addMarker( markerName, { range, usingOperation: false } );\n\t * ```\n\t *\n\t * Create marker using operation:\n\t *\n\t * ```ts\n\t * addMarker( markerName, { range, usingOperation: true } );\n\t * ```\n\t *\n\t * Create marker that affects the editor data:\n\t *\n\t * ```ts\n\t * addMarker( markerName, { range, usingOperation: false, affectsData: true } );\n\t * ```\n\t *\n\t * Note: For efficiency reasons, it's best to create and keep as little markers as possible.\n\t *\n\t * @see module:engine/model/markercollection~Marker\n\t * @param name Name of a marker to create - must be unique.\n\t * @param options.usingOperation Flag indicating that the marker should be added by MarkerOperation.\n\t * See {@link module:engine/model/markercollection~Marker#managedUsingOperations}.\n\t * @param options.range Marker range.\n\t * @param options.affectsData Flag indicating that the marker changes the editor data.\n\t * @returns Marker that was set.\n\t */\n\tpublic addMarker(\n\t\tname: string,\n\t\toptions: {\n\t\t\tusingOperation: boolean;\n\t\t\taffectsData?: boolean;\n\t\t\trange: Range;\n\t\t}\n\t): Marker {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( !options || typeof options.usingOperation != 'boolean' ) {\n\t\t\t/**\n\t\t\t * The `options.usingOperation` parameter is required when adding a new marker.\n\t\t\t *\n\t\t\t * @error writer-addmarker-no-usingoperation\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-addmarker-no-usingoperation', this );\n\t\t}\n\n\t\tconst usingOperation = options.usingOperation;\n\t\tconst range = options.range;\n\t\tconst affectsData = options.affectsData === undefined ? false : options.affectsData;\n\n\t\tif ( this.model.markers.has( name ) ) {\n\t\t\t/**\n\t\t\t * Marker with provided name already exists.\n\t\t\t *\n\t\t\t * @error writer-addmarker-marker-exists\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-addmarker-marker-exists', this );\n\t\t}\n\n\t\tif ( !range ) {\n\t\t\t/**\n\t\t\t * Range parameter is required when adding a new marker.\n\t\t\t *\n\t\t\t * @error writer-addmarker-no-range\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-addmarker-no-range', this );\n\t\t}\n\n\t\tif ( !usingOperation ) {\n\t\t\treturn this.model.markers._set( name, range, usingOperation, affectsData );\n\t\t}\n\n\t\tapplyMarkerOperation( this, name, null, range, affectsData );\n\n\t\treturn this.model.markers.get( name )!;\n\t}\n\n\t/**\n\t * Adds, updates or refreshes a {@link module:engine/model/markercollection~Marker marker}. Marker is a named range, which tracks\n\t * changes in the document and updates its range automatically, when model tree changes. Still, it is possible to change the\n\t * marker's range directly using this method.\n\t *\n\t * As the first parameter you can set marker name or instance. If none of them is provided, new marker, with a unique\n\t * name is created and returned.\n\t *\n\t * **Note**: If you want to change the {@link module:engine/view/element~Element view element} of the marker while its data in the model\n\t * remains the same, use the dedicated {@link module:engine/controller/editingcontroller~EditingController#reconvertMarker} method.\n\t *\n\t * The `options.usingOperation` parameter lets you change if the marker should be managed by operations or not. See\n\t * {@link module:engine/model/markercollection~Marker marker class description} to learn about the difference between\n\t * markers managed by operations and not-managed by operations. It is possible to change this option for an existing marker.\n\t *\n\t * The `options.affectsData` parameter, which defaults to `false`, allows you to define if a marker affects the data. It should be\n\t * `true` when the marker change changes the data returned by\n\t * the {@link module:core/editor/utils/dataapimixin~DataApi#getData `editor.getData()`} method.\n\t * When set to `true` it fires the {@link module:engine/model/document~Document#event:change:data `change:data`} event.\n\t * When set to `false` it fires the {@link module:engine/model/document~Document#event:change `change`} event.\n\t *\n\t * Update marker directly base on marker's name:\n\t *\n\t * ```ts\n\t * updateMarker( markerName, { range } );\n\t * ```\n\t *\n\t * Update marker using operation:\n\t *\n\t * ```ts\n\t * updateMarker( marker, { range, usingOperation: true } );\n\t * updateMarker( markerName, { range, usingOperation: true } );\n\t * ```\n\t *\n\t * Change marker's option (start using operations to manage it):\n\t *\n\t * ```ts\n\t * updateMarker( marker, { usingOperation: true } );\n\t * ```\n\t *\n\t * Change marker's option (inform the engine, that the marker does not affect the data anymore):\n\t *\n\t * ```ts\n\t * updateMarker( markerName, { affectsData: false } );\n\t * ```\n\t *\n\t * @see module:engine/model/markercollection~Marker\n\t * @param markerOrName Name of a marker to update, or a marker instance.\n\t * @param options If options object is not defined then marker will be refreshed by triggering\n\t * downcast conversion for this marker with the same data.\n\t * @param options.range Marker range to update.\n\t * @param options.usingOperation Flag indicated whether the marker should be added by MarkerOperation.\n\t * See {@link module:engine/model/markercollection~Marker#managedUsingOperations}.\n\t * @param options.affectsData Flag indicating that the marker changes the editor data.\n\t */\n\tpublic updateMarker(\n\t\tmarkerOrName: string | Marker,\n\t\toptions?: {\n\t\t\trange?: Range;\n\t\t\tusingOperation?: boolean;\n\t\t\taffectsData?: boolean;\n\t\t}\n\t): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tconst markerName = typeof markerOrName == 'string' ? markerOrName : markerOrName.name;\n\t\tconst currentMarker = this.model.markers.get( markerName );\n\n\t\tif ( !currentMarker ) {\n\t\t\t/**\n\t\t\t * Marker with provided name does not exist and will not be updated.\n\t\t\t *\n\t\t\t * @error writer-updatemarker-marker-not-exists\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-updatemarker-marker-not-exists', this );\n\t\t}\n\n\t\tif ( !options ) {\n\t\t\t/**\n\t\t\t * The usage of `writer.updateMarker()` only to reconvert (refresh) a\n\t\t\t * {@link module:engine/model/markercollection~Marker model marker} was deprecated and may not work in the future.\n\t\t\t * Please update your code to use\n\t\t\t * {@link module:engine/controller/editingcontroller~EditingController#reconvertMarker `editor.editing.reconvertMarker()`}\n\t\t\t * instead.\n\t\t\t *\n\t\t\t * @error writer-updatemarker-reconvert-using-editingcontroller\n\t\t\t * @param markerName The name of the updated marker.\n\t\t\t */\n\t\t\tlogWarning( 'writer-updatemarker-reconvert-using-editingcontroller', { markerName } );\n\n\t\t\tthis.model.markers._refresh( currentMarker );\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst hasUsingOperationDefined = typeof options.usingOperation == 'boolean';\n\t\tconst affectsDataDefined = typeof options.affectsData == 'boolean';\n\n\t\t// Use previously defined marker's affectsData if the property is not provided.\n\t\tconst affectsData = affectsDataDefined ? options.affectsData : currentMarker.affectsData;\n\n\t\tif ( !hasUsingOperationDefined && !options.range && !affectsDataDefined ) {\n\t\t\t/**\n\t\t\t * One of the options is required - provide range, usingOperations or affectsData.\n\t\t\t *\n\t\t\t * @error writer-updatemarker-wrong-options\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-updatemarker-wrong-options', this );\n\t\t}\n\n\t\tconst currentRange = currentMarker.getRange();\n\t\tconst updatedRange = options.range ? options.range : currentRange;\n\n\t\tif ( hasUsingOperationDefined && options.usingOperation !== currentMarker.managedUsingOperations ) {\n\t\t\t// The marker type is changed so it's necessary to create proper operations.\n\t\t\tif ( options.usingOperation ) {\n\t\t\t\t// If marker changes to a managed one treat this as synchronizing existing marker.\n\t\t\t\t// Create `MarkerOperation` with `oldRange` set to `null`, so reverse operation will remove the marker.\n\t\t\t\tapplyMarkerOperation( this, markerName, null, updatedRange, affectsData );\n\t\t\t} else {\n\t\t\t\t// If marker changes to a marker that do not use operations then we need to create additional operation\n\t\t\t\t// that removes that marker first.\n\t\t\t\tapplyMarkerOperation( this, markerName, currentRange, null, affectsData );\n\n\t\t\t\t// Although not managed the marker itself should stay in model and its range should be preserver or changed to passed range.\n\t\t\t\tthis.model.markers._set( markerName, updatedRange, undefined, affectsData );\n\t\t\t}\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Marker's type doesn't change so update it accordingly.\n\t\tif ( currentMarker.managedUsingOperations ) {\n\t\t\tapplyMarkerOperation( this, markerName, currentRange, updatedRange, affectsData );\n\t\t} else {\n\t\t\tthis.model.markers._set( markerName, updatedRange, undefined, affectsData );\n\t\t}\n\t}\n\n\t/**\n\t * Removes given {@link module:engine/model/markercollection~Marker marker} or marker with given name.\n\t * The marker is removed accordingly to how it has been created, so if the marker was created using operation,\n\t * it will be destroyed using operation.\n\t *\n\t * @param markerOrName Marker or marker name to remove.\n\t */\n\tpublic removeMarker( markerOrName: string | Marker ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tconst name = typeof markerOrName == 'string' ? markerOrName : markerOrName.name;\n\n\t\tif ( !this.model.markers.has( name ) ) {\n\t\t\t/**\n\t\t\t * Trying to remove marker which does not exist.\n\t\t\t *\n\t\t\t * @error writer-removemarker-no-marker\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-removemarker-no-marker', this );\n\t\t}\n\n\t\tconst marker = this.model.markers.get( name )!;\n\n\t\tif ( !marker.managedUsingOperations ) {\n\t\t\tthis.model.markers._remove( name );\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst oldRange = marker.getRange();\n\n\t\tapplyMarkerOperation( this, name, oldRange, null, marker.affectsData );\n\t}\n\n\t/**\n\t * Adds a new root to the document (or re-attaches a {@link #detachRoot detached root}).\n\t *\n\t * Throws an error, if trying to add a root that is already added and attached.\n\t *\n\t * @param rootName Name of the added root.\n\t * @param elementName The element name. Defaults to `'$root'` which also has some basic schema defined\n\t * (e.g. `$block` elements are allowed inside the `$root`). Make sure to define a proper schema if you use a different name.\n\t * @returns The added root element.\n\t */\n\tpublic addRoot( rootName: string, elementName = '$root' ): RootElement {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tconst root = this.model.document.getRoot( rootName );\n\n\t\tif ( root && root.isAttached() ) {\n\t\t\t/**\n\t\t\t * Root with provided name already exists and is attached.\n\t\t\t *\n\t\t\t * @error writer-addroot-root-exists\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-addroot-root-exists', this );\n\t\t}\n\n\t\tconst document = this.model.document;\n\t\tconst operation = new RootOperation( rootName, elementName, true, document, document.version );\n\n\t\tthis.batch.addOperation( operation );\n\t\tthis.model.applyOperation( operation );\n\n\t\treturn this.model.document.getRoot( rootName )!;\n\t}\n\n\t/**\n\t * Detaches the root from the document.\n\t *\n\t * All content and markers are removed from the root upon detaching. New content and new markers cannot be added to the root, as long\n\t * as it is detached.\n\t *\n\t * A root cannot be fully removed from the document, it can be only detached. A root is permanently removed only after you\n\t * re-initialize the editor and do not specify the root in the initial data.\n\t *\n\t * A detached root can be re-attached using {@link #addRoot}.\n\t *\n\t * Throws an error if the root does not exist or the root is already detached.\n\t *\n\t * @param rootOrName Name of the detached root.\n\t */\n\tpublic detachRoot( rootOrName: string | RootElement ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tconst root = typeof rootOrName == 'string' ? this.model.document.getRoot( rootOrName ) : rootOrName;\n\n\t\tif ( !root || !root.isAttached() ) {\n\t\t\t/**\n\t\t\t * Root with provided name does not exist or is already detached.\n\t\t\t *\n\t\t\t * @error writer-detachroot-no-root\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'writer-detachroot-no-root', this );\n\t\t}\n\n\t\t// First, remove all markers from the root. It is better to do it before removing stuff for undo purposes.\n\t\t// However, looking through all the markers may not be the best performance wise. But there's no better solution for now.\n\t\tfor ( const marker of this.model.markers ) {\n\t\t\tif ( marker.getRange().root === root ) {\n\t\t\t\tthis.removeMarker( marker );\n\t\t\t}\n\t\t}\n\n\t\t// Remove all attributes from the root.\n\t\tfor ( const key of root.getAttributeKeys() ) {\n\t\t\tthis.removeAttribute( key, root );\n\t\t}\n\n\t\t// Remove all contents of the root.\n\t\tthis.remove( this.createRangeIn( root ) );\n\n\t\t// Finally, detach the root.\n\t\tconst document = this.model.document;\n\t\tconst operation = new RootOperation( root.rootName, root.name, false, document, document.version );\n\n\t\tthis.batch.addOperation( operation );\n\t\tthis.model.applyOperation( operation );\n\t}\n\n\t/**\n\t * Sets the document's selection (ranges and direction) to the specified location based on the given\n\t * {@link module:engine/model/selection~Selectable selectable} or creates an empty selection if no arguments were passed.\n\t *\n\t * ```ts\n\t * // Sets collapsed selection at the position of the given node and an offset.\n\t * writer.setSelection( paragraph, offset );\n\t * ```\n\t *\n\t * Creates a range inside an {@link module:engine/model/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * ```ts\n\t * writer.setSelection( paragraph, 'in' );\n\t * ```\n\t *\n\t * Creates a range on an {@link module:engine/model/item~Item item} which starts before the item and ends just after the item.\n\t *\n\t * ```ts\n\t * writer.setSelection( paragraph, 'on' );\n\t * ```\n\t *\n\t * `Writer#setSelection()` allow passing additional options (`backward`) as the last argument.\n\t *\n\t * ```ts\n\t * // Sets selection as backward.\n\t * writer.setSelection( element, 'in', { backward: true } );\n\t * ```\n\t *\n\t * Throws `writer-incorrect-use` error when the writer is used outside the `change()` block.\n\t *\n\t * See also: {@link #setSelection:SELECTABLE `setSelection( selectable, options )`}.\n\t *\n\t * @label NODE_OFFSET\n\t */\n\tpublic setSelection( selectable: Node, placeOrOffset: PlaceOrOffset, options?: { backward?: boolean } ): void;\n\n\t/**\n\t * Sets the document's selection (ranges and direction) to the specified location based on the given\n\t * {@link module:engine/model/selection~Selectable selectable} or creates an empty selection if no arguments were passed.\n\t *\n\t * ```ts\n\t * // Sets selection to the given range.\n\t * const range = writer.createRange( start, end );\n\t * writer.setSelection( range );\n\t *\n\t * // Sets selection to given ranges.\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( star2, end2 ) ];\n\t * writer.setSelection( ranges );\n\t *\n\t * // Sets selection to other selection.\n\t * const otherSelection = writer.createSelection();\n\t * writer.setSelection( otherSelection );\n\t *\n\t * // Sets selection to the given document selection.\n\t * const documentSelection = model.document.selection;\n\t * writer.setSelection( documentSelection );\n\t *\n\t * // Sets collapsed selection at the given position.\n\t * const position = writer.createPosition( root, path );\n\t * writer.setSelection( position );\n\t *\n\t * // Removes all selection's ranges.\n\t * writer.setSelection( null );\n\t * ```\n\t *\n\t * `Writer#setSelection()` allow passing additional options (`backward`) as the last argument.\n\t *\n\t * ```ts\n\t * // Sets selection as backward.\n\t * writer.setSelection( range, { backward: true } );\n\t * ```\n\t *\n\t * Throws `writer-incorrect-use` error when the writer is used outside the `change()` block.\n\t *\n\t * See also: {@link #setSelection:NODE_OFFSET `setSelection( node, placeOrOffset, options )`}.\n\t *\n\t * @label SELECTABLE\n\t */\n\tpublic setSelection( selectable: Exclude<Selectable, Node>, options?: { backward?: boolean } ): void;\n\n\tpublic setSelection( ...args: Parameters<Selection[ 'setTo' ]> ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tthis.model.document.selection._setTo( ...args );\n\t}\n\n\t/**\n\t * Moves {@link module:engine/model/documentselection~DocumentSelection#focus} to the specified location.\n\t *\n\t * The location can be specified in the same form as\n\t * {@link #createPositionAt `writer.createPositionAt()`} parameters.\n\t *\n\t * @param itemOrPosition\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic setSelectionFocus(\n\t\titemOrPosition: Item | Position,\n\t\toffset?: PositionOffset\n\t): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tthis.model.document.selection._setFocus( itemOrPosition, offset );\n\t}\n\n\t/**\n\t * Sets attribute on the selection. If attribute with the same key already is set, it's value is overwritten.\n\t *\n\t * ```ts\n\t * writer.setSelectionAttribute( 'italic', true );\n\t * ```\n\t *\n\t * @label KEY_VALUE\n\t * @param key Key of the attribute to set.\n\t * @param value Attribute value.\n\t */\n\tpublic setSelectionAttribute( key: string, value: unknown ): void;\n\n\t/**\n\t * Sets attributes on the selection. If any attribute with the same key already is set, it's value is overwritten.\n\t *\n\t * Using key-value object:\n\t *\n\t * ```ts\n\t * writer.setSelectionAttribute( { italic: true, bold: false } );\n\t * ```\n\t *\n\t * Using iterable object:\n\t *\n\t * ```ts\n\t * writer.setSelectionAttribute( new Map( [ [ 'italic', true ] ] ) );\n\t * ```\n\t *\n\t * @label OBJECT\n\t * @param objectOrIterable Object / iterable of key => value attribute pairs.\n\t */\n\tpublic setSelectionAttribute( objectOrIterable: NodeAttributes ): void;\n\n\tpublic setSelectionAttribute(\n\t\tkeyOrObjectOrIterable: string | NodeAttributes,\n\t\tvalue?: unknown\n\t): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( typeof keyOrObjectOrIterable === 'string' ) {\n\t\t\tthis._setSelectionAttribute( keyOrObjectOrIterable, value );\n\t\t} else {\n\t\t\tfor ( const [ key, value ] of toMap( keyOrObjectOrIterable ) ) {\n\t\t\t\tthis._setSelectionAttribute( key, value );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Removes attribute(s) with given key(s) from the selection.\n\t *\n\t * Remove one attribute:\n\t *\n\t * ```ts\n\t * writer.removeSelectionAttribute( 'italic' );\n\t * ```\n\t *\n\t * Remove multiple attributes:\n\t *\n\t * ```ts\n\t * writer.removeSelectionAttribute( [ 'italic', 'bold' ] );\n\t * ```\n\t *\n\t * @param keyOrIterableOfKeys Key of the attribute to remove or an iterable of attribute keys to remove.\n\t */\n\tpublic removeSelectionAttribute( keyOrIterableOfKeys: string | Iterable<string> ): void {\n\t\tthis._assertWriterUsedCorrectly();\n\n\t\tif ( typeof keyOrIterableOfKeys === 'string' ) {\n\t\t\tthis._removeSelectionAttribute( keyOrIterableOfKeys );\n\t\t} else {\n\t\t\tfor ( const key of keyOrIterableOfKeys ) {\n\t\t\t\tthis._removeSelectionAttribute( key );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Temporarily changes the {@link module:engine/model/documentselection~DocumentSelection#isGravityOverridden gravity}\n\t * of the selection from left to right.\n\t *\n\t * The gravity defines from which direction the selection inherits its attributes. If it's the default left gravity,\n\t * then the selection (after being moved by the user) inherits attributes from its left-hand side.\n\t * This method allows to temporarily override this behavior by forcing the gravity to the right.\n\t *\n\t * For the following model fragment:\n\t *\n\t * ```xml\n\t * <$text bold=\"true\" linkHref=\"url\">bar[]</$text><$text bold=\"true\">biz</$text>\n\t * ```\n\t *\n\t * * Default gravity: selection will have the `bold` and `linkHref` attributes.\n\t * * Overridden gravity: selection will have `bold` attribute.\n\t *\n\t * **Note**: It returns an unique identifier which is required to restore the gravity. It guarantees the symmetry\n\t * of the process.\n\t *\n\t * @returns The unique id which allows restoring the gravity.\n\t */\n\tpublic overrideSelectionGravity(): string {\n\t\treturn this.model.document.selection._overrideGravity();\n\t}\n\n\t/**\n\t * Restores {@link ~Writer#overrideSelectionGravity} gravity to default.\n\t *\n\t * Restoring the gravity is only possible using the unique identifier returned by\n\t * {@link ~Writer#overrideSelectionGravity}. Note that the gravity remains overridden as long as won't be restored\n\t * the same number of times it was overridden.\n\t *\n\t * @param uid The unique id returned by {@link ~Writer#overrideSelectionGravity}.\n\t */\n\tpublic restoreSelectionGravity( uid: string ): void {\n\t\tthis.model.document.selection._restoreGravity( uid );\n\t}\n\n\t/**\n\t * @param key Key of the attribute to remove.\n\t * @param value Attribute value.\n\t */\n\tprivate _setSelectionAttribute( key: string, value: unknown ): void {\n\t\tconst selection = this.model.document.selection;\n\n\t\t// Store attribute in parent element if the selection is collapsed in an empty node.\n\t\tif ( selection.isCollapsed && selection.anchor!.parent.isEmpty ) {\n\t\t\tconst storeKey = DocumentSelection._getStoreAttributeKey( key );\n\n\t\t\tthis.setAttribute( storeKey, value, selection.anchor!.parent as any );\n\t\t}\n\n\t\tselection._setAttribute( key, value );\n\t}\n\n\t/**\n\t * @param key Key of the attribute to remove.\n\t */\n\tprivate _removeSelectionAttribute( key: string ): void {\n\t\tconst selection = this.model.document.selection;\n\n\t\t// Remove stored attribute from parent element if the selection is collapsed in an empty node.\n\t\tif ( selection.isCollapsed && selection.anchor!.parent.isEmpty ) {\n\t\t\tconst storeKey = DocumentSelection._getStoreAttributeKey( key );\n\n\t\t\tthis.removeAttribute( storeKey, selection.anchor!.parent as any );\n\t\t}\n\n\t\tselection._removeAttribute( key );\n\t}\n\n\t/**\n\t * Throws `writer-detached-writer-tries-to-modify-model` error when the writer is used outside of the `change()` block.\n\t */\n\tprivate _assertWriterUsedCorrectly(): void {\n\t\t/**\n\t\t * Trying to use a writer outside a {@link module:engine/model/model~Model#change `change()`} or\n\t\t * {@link module:engine/model/model~Model#enqueueChange `enqueueChange()`} blocks.\n\t\t *\n\t\t * The writer can only be used inside these blocks which ensures that the model\n\t\t * can only be changed during such \"sessions\".\n\t\t *\n\t\t * @error writer-incorrect-use\n\t\t */\n\t\tif ( ( this.model as any )._currentWriter !== this ) {\n\t\t\tthrow new CKEditorError( 'writer-incorrect-use', this );\n\t\t}\n\t}\n\n\t/**\n\t * For given action `type` and `positionOrRange` where the action happens, this function finds all affected markers\n\t * and applies a marker operation with the new marker range equal to the current range. Thanks to this, the marker range\n\t * can be later correctly processed during undo.\n\t *\n\t * @param type Writer action type.\n\t * @param positionOrRange Position or range where the writer action happens.\n\t */\n\tprivate _addOperationForAffectedMarkers(\n\t\ttype: 'move' | 'merge',\n\t\tpositionOrRange: Position | Range\n\t): void {\n\t\tfor ( const marker of this.model.markers ) {\n\t\t\tif ( !marker.managedUsingOperations ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst markerRange = marker.getRange();\n\t\t\tlet isAffected = false;\n\n\t\t\tif ( type === 'move' ) {\n\t\t\t\tconst range = positionOrRange as Range;\n\t\t\t\tisAffected =\n\t\t\t\t\trange.containsPosition( markerRange.start ) ||\n\t\t\t\t\trange.start.isEqual( markerRange.start ) ||\n\t\t\t\t\trange.containsPosition( markerRange.end ) ||\n\t\t\t\t\trange.end.isEqual( markerRange.end );\n\t\t\t} else {\n\t\t\t\t// if type === 'merge'.\n\t\t\t\tconst position = positionOrRange as Position;\n\t\t\t\tconst elementBefore = position.nodeBefore;\n\t\t\t\tconst elementAfter = position.nodeAfter;\n\n\t\t\t\t// Start: <p>Foo[</p><p>Bar]</p>\n\t\t\t\t// After merge: <p>Foo[Bar]</p>\n\t\t\t\t// After undoing split: <p>Foo</p><p>[Bar]</p> <-- incorrect, needs remembering for undo.\n\t\t\t\t//\n\t\t\t\tconst affectedInLeftElement = markerRange.start.parent == elementBefore && markerRange.start.isAtEnd;\n\n\t\t\t\t// Start: <p>[Foo</p><p>]Bar</p>\n\t\t\t\t// After merge: <p>[Foo]Bar</p>\n\t\t\t\t// After undoing split: <p>[Foo]</p><p>Bar</p> <-- incorrect, needs remembering for undo.\n\t\t\t\t//\n\t\t\t\tconst affectedInRightElement = markerRange.end.parent == elementAfter && markerRange.end.offset == 0;\n\n\t\t\t\t// Start: <p>[Foo</p>]<p>Bar</p>\n\t\t\t\t// After merge: <p>[Foo]Bar</p>\n\t\t\t\t// After undoing split: <p>[Foo]</p><p>Bar</p> <-- incorrect, needs remembering for undo.\n\t\t\t\t//\n\t\t\t\tconst affectedAfterLeftElement = markerRange.end.nodeAfter == elementAfter;\n\n\t\t\t\t// Start: <p>Foo</p>[<p>Bar]</p>\n\t\t\t\t// After merge: <p>Foo[Bar]</p>\n\t\t\t\t// After undoing split: <p>Foo</p><p>[Bar]</p> <-- incorrect, needs remembering for undo.\n\t\t\t\t//\n\t\t\t\tconst affectedBeforeRightElement = markerRange.start.nodeAfter == elementAfter;\n\n\t\t\t\tisAffected = affectedInLeftElement || affectedInRightElement || affectedAfterLeftElement || affectedBeforeRightElement;\n\t\t\t}\n\n\t\t\tif ( isAffected ) {\n\t\t\t\tthis.updateMarker( marker.name, { range: markerRange } );\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Sets given attribute to each node in given range. When attribute value is null then attribute will be removed.\n *\n * Because attribute operation needs to have the same attribute value on the whole range, this function splits\n * the range into smaller parts.\n *\n * Given `range` must be flat.\n */\nfunction setAttributeOnRange( writer: Writer, key: string, value: unknown, range: Range ) {\n\tconst model = writer.model;\n\tconst doc = model.document;\n\n\t// Position of the last split, the beginning of the new range.\n\tlet lastSplitPosition = range.start;\n\n\t// Currently position in the scanning range. Because we need value after the position, it is not a current\n\t// position of the iterator but the previous one (we need to iterate one more time to get the value after).\n\tlet position: Position | undefined;\n\n\t// Value before the currently position.\n\tlet valueBefore: unknown;\n\n\t// Value after the currently position.\n\tlet valueAfter;\n\n\tfor ( const val of range.getWalker( { shallow: true } ) ) {\n\t\tvalueAfter = val.item.getAttribute( key );\n\n\t\t// At the first run of the iterator the position in undefined. We also do not have a valueBefore, but\n\t\t// because valueAfter may be null, valueBefore may be equal valueAfter ( undefined == null ).\n\t\tif ( position && valueBefore != valueAfter ) {\n\t\t\t// if valueBefore == value there is nothing to change, so we add operation only if these values are different.\n\t\t\tif ( valueBefore != value ) {\n\t\t\t\taddOperation();\n\t\t\t}\n\n\t\t\tlastSplitPosition = position;\n\t\t}\n\n\t\tposition = val.nextPosition;\n\t\tvalueBefore = valueAfter;\n\t}\n\n\t// Because position in the loop is not the iterator position (see let position comment), the last position in\n\t// the while loop will be last but one position in the range. We need to check the last position manually.\n\tif ( position instanceof Position && position != lastSplitPosition && valueBefore != value ) {\n\t\taddOperation();\n\t}\n\n\tfunction addOperation() {\n\t\tconst range = new Range( lastSplitPosition, position );\n\t\tconst version = range.root.document ? doc.version : null;\n\t\tconst operation = new AttributeOperation( range, key, valueBefore, value, version );\n\n\t\twriter.batch.addOperation( operation );\n\t\tmodel.applyOperation( operation );\n\t}\n}\n\n/**\n * Sets given attribute to the given node. When attribute value is null then attribute will be removed.\n */\nfunction setAttributeOnItem( writer: Writer, key: string, value: unknown, item: Item ) {\n\tconst model = writer.model;\n\tconst doc = model.document;\n\tconst previousValue = item.getAttribute( key );\n\tlet range, operation;\n\n\tif ( previousValue != value ) {\n\t\tconst isRootChanged = item.root === item;\n\n\t\tif ( isRootChanged ) {\n\t\t\t// If we change attributes of root element, we have to use `RootAttributeOperation`.\n\t\t\tconst version = item.document ? doc.version : null;\n\n\t\t\toperation = new RootAttributeOperation( item as any, key, previousValue, value, version );\n\t\t} else {\n\t\t\trange = new Range( Position._createBefore( item ), writer.createPositionAfter( item ) );\n\n\t\t\tconst version = range.root.document ? doc.version : null;\n\n\t\t\toperation = new AttributeOperation( range, key, previousValue, value, version );\n\t\t}\n\n\t\twriter.batch.addOperation( operation );\n\t\tmodel.applyOperation( operation );\n\t}\n}\n\n/**\n * Creates and applies marker operation to {@link module:engine/model/operation/operation~Operation operation}.\n */\nfunction applyMarkerOperation(\n\twriter: Writer,\n\tname: string,\n\toldRange: Range | null,\n\tnewRange: Range | null,\n\taffectsData: boolean | undefined\n) {\n\tconst model = writer.model;\n\tconst doc = model.document;\n\n\tconst operation = new MarkerOperation( name, oldRange, newRange, model.markers, !!affectsData, doc.version );\n\n\twriter.batch.addOperation( operation );\n\tmodel.applyOperation( operation );\n}\n\n/**\n * Creates `MoveOperation` or `DetachOperation` that removes `howMany` nodes starting from `position`.\n * The operation will be applied on given model instance and added to given operation instance.\n *\n * @param position Position from which nodes are removed.\n * @param howMany Number of nodes to remove.\n * @param batch Batch to which the operation will be added.\n * @param model Model instance on which operation will be applied.\n */\nfunction applyRemoveOperation( position: Position, howMany: number, batch: Batch, model: Model ) {\n\tlet operation;\n\n\tif ( position.root.document ) {\n\t\tconst doc = model.document;\n\t\tconst graveyardPosition = new Position( doc.graveyard, [ 0 ] );\n\n\t\toperation = new MoveOperation( position, howMany, graveyardPosition, doc.version );\n\t} else {\n\t\toperation = new DetachOperation( position, howMany );\n\t}\n\n\tbatch.addOperation( operation );\n\tmodel.applyOperation( operation );\n}\n\n/**\n * Returns `true` if both root elements are the same element or both are documents root elements.\n *\n * Elements in the same tree can be moved (for instance you can move element form one documents root to another, or\n * within the same document fragment), but when element supposed to be moved from document fragment to the document, or\n * to another document it should be removed and inserted to avoid problems with OT. This is because features like undo or\n * collaboration may track changes on the document but ignore changes on detached fragments and should not get\n * unexpected `move` operation.\n */\nfunction isSameTree( rootA: Node | DocumentFragment, rootB: Node | DocumentFragment ): boolean {\n\t// If it is the same root this is the same tree.\n\tif ( rootA === rootB ) {\n\t\treturn true;\n\t}\n\n\t// If both roots are documents root it is operation within the document what we still treat as the same tree.\n\tif ( rootA instanceof RootElement && rootB instanceof RootElement ) {\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/utils/deletecontent\n */\n\nimport DocumentSelection from '../documentselection';\nimport LivePosition from '../liveposition';\nimport Range from '../range';\n\nimport type DocumentFragment from '../documentfragment';\nimport type Element from '../element';\nimport type Model from '../model';\nimport type Position from '../position';\nimport type Schema from '../schema';\nimport type Selection from '../selection';\nimport type Writer from '../writer';\n\n/**\n * Deletes content of the selection and merge siblings. The resulting selection is always collapsed.\n *\n * **Note:** Use {@link module:engine/model/model~Model#deleteContent} instead of this function.\n * This function is only exposed to be reusable in algorithms\n * which change the {@link module:engine/model/model~Model#deleteContent}\n * method's behavior.\n *\n * @param model The model in context of which the insertion should be performed.\n * @param selection Selection of which the content should be deleted.\n * @param options.leaveUnmerged Whether to merge elements after removing the content of the selection.\n *\n * For example `<heading>x[x</heading><paragraph>y]y</paragraph>` will become:\n *\n * * `<heading>x^y</heading>` with the option disabled (`leaveUnmerged == false`)\n * * `<heading>x^</heading><paragraph>y</paragraph>` with enabled (`leaveUnmerged == true`).\n *\n * Note: {@link module:engine/model/schema~Schema#isObject object} and {@link module:engine/model/schema~Schema#isLimit limit}\n * elements will not be merged.\n *\n * @param options.doNotResetEntireContent Whether to skip replacing the entire content with a\n * paragraph when the entire content was selected.\n *\n * For example `<heading>[x</heading><paragraph>y]</paragraph>` will become:\n *\n * * `<paragraph>^</paragraph>` with the option disabled (`doNotResetEntireContent == false`)\n * * `<heading>^</heading>` with enabled (`doNotResetEntireContent == true`).\n *\n * @param options.doNotAutoparagraph Whether to create a paragraph if after content deletion selection is moved\n * to a place where text cannot be inserted.\n *\n * For example `<paragraph>x</paragraph>[<imageBlock src=\"foo.jpg\"></imageBlock>]` will become:\n *\n * * `<paragraph>x</paragraph><paragraph>[]</paragraph>` with the option disabled (`doNotAutoparagraph == false`)\n * * `<paragraph>x</paragraph>[]` with the option enabled (`doNotAutoparagraph == true`).\n *\n * If you use this option you need to make sure to handle invalid selections yourself or leave\n * them to the selection post-fixer (may not always work).\n *\n * **Note:** If there is no valid position for the selection, the paragraph will always be created:\n *\n * `[<imageBlock src=\"foo.jpg\"></imageBlock>]` -> `<paragraph>[]</paragraph>`.\n */\nexport default function deleteContent(\n\tmodel: Model,\n\tselection: Selection | DocumentSelection,\n\toptions: {\n\t\tleaveUnmerged?: boolean;\n\t\tdoNotResetEntireContent?: boolean;\n\t\tdoNotAutoparagraph?: boolean;\n\t} = {}\n): void {\n\tif ( selection.isCollapsed ) {\n\t\treturn;\n\t}\n\n\tconst selRange = selection.getFirstRange()!;\n\n\t// If the selection is already removed, don't do anything.\n\tif ( selRange.root.rootName == '$graveyard' ) {\n\t\treturn;\n\t}\n\n\tconst schema = model.schema;\n\n\tmodel.change( writer => {\n\t\t// 1. Replace the entire content with paragraph.\n\t\t// See: https://github.com/ckeditor/ckeditor5-engine/issues/1012#issuecomment-315017594.\n\t\tif ( !options.doNotResetEntireContent && shouldEntireContentBeReplacedWithParagraph( schema, selection ) ) {\n\t\t\treplaceEntireContentWithParagraph( writer, selection );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Collect attributes to copy in case of autoparagraphing.\n\t\tconst attributesForAutoparagraph = {};\n\n\t\tif ( !options.doNotAutoparagraph ) {\n\t\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\t\tif ( selectedElement ) {\n\t\t\t\tObject.assign( attributesForAutoparagraph, schema.getAttributesWithProperty( selectedElement, 'copyOnReplace', true ) );\n\t\t\t}\n\t\t}\n\n\t\t// Get the live positions for the range adjusted to span only blocks selected from the user perspective.\n\t\tconst [ startPosition, endPosition ] = getLivePositionsForSelectedBlocks( selRange );\n\n\t\t// 2. Remove the content if there is any.\n\t\tif ( !startPosition.isTouching( endPosition ) ) {\n\t\t\twriter.remove( writer.createRange( startPosition, endPosition ) );\n\t\t}\n\n\t\t// 3. Merge elements in the right branch to the elements in the left branch.\n\t\t// The only reasonable (in terms of data and selection correctness) case in which we need to do that is:\n\t\t//\n\t\t// <heading type=1>Fo[</heading><paragraph>]ar</paragraph> => <heading type=1>Fo^ar</heading>\n\t\t//\n\t\t// However, the algorithm supports also merging deeper structures (up to the depth of the shallower branch),\n\t\t// as it's hard to imagine what should actually be the default behavior. Usually, specific features will\n\t\t// want to override that behavior anyway.\n\t\tif ( !options.leaveUnmerged ) {\n\t\t\tmergeBranches( writer, startPosition, endPosition );\n\n\t\t\t// TMP this will be replaced with a postfixer.\n\t\t\t// We need to check and strip disallowed attributes in all nested nodes because after merge\n\t\t\t// some attributes could end up in a path where are disallowed.\n\t\t\t//\n\t\t\t// e.g. bold is disallowed for <H1>\n\t\t\t// <h1>Fo{o</h1><p>b}a<b>r</b><p> -> <h1>Fo{}a<b>r</b><h1> -> <h1>Fo{}ar<h1>.\n\t\t\tschema.removeDisallowedAttributes( startPosition.parent.getChildren(), writer );\n\t\t}\n\n\t\tcollapseSelectionAt( writer, selection, startPosition );\n\n\t\t// 4. Add a paragraph to set selection in it.\n\t\t// Check if a text is allowed in the new container. If not, try to create a new paragraph (if it's allowed here).\n\t\t// If autoparagraphing is off, we assume that you know what you do so we leave the selection wherever it was.\n\t\tif ( !options.doNotAutoparagraph && shouldAutoparagraph( schema, startPosition ) ) {\n\t\t\tinsertParagraph( writer, startPosition, selection, attributesForAutoparagraph );\n\t\t}\n\n\t\tstartPosition.detach();\n\t\tendPosition.detach();\n\t} );\n}\n\n/**\n * Returns the live positions for the range adjusted to span only blocks selected from the user perspective. Example:\n *\n * ```\n * <heading1>[foo</heading1>\n * <paragraph>bar</paragraph>\n * <heading1>]abc</heading1> <-- this block is not considered as selected\n * ```\n *\n * This is the same behavior as in Selection#getSelectedBlocks() \"special case\".\n */\nfunction getLivePositionsForSelectedBlocks( range: Range ): [ startPosition: LivePosition, endPosition: LivePosition ] {\n\tconst model = range.root.document!.model;\n\n\tconst startPosition = range.start;\n\tlet endPosition = range.end;\n\n\t// If the end of selection is at the start position of last block in the selection, then\n\t// shrink it to not include that trailing block. Note that this should happen only for not empty selection.\n\tif ( model.hasContent( range, { ignoreMarkers: true } ) ) {\n\t\tconst endBlock = getParentBlock( endPosition );\n\n\t\tif ( endBlock && endPosition.isTouching( model.createPositionAt( endBlock, 0 ) ) ) {\n\t\t\t// Create forward selection as a probe to find a valid position after excluding last block from the range.\n\t\t\tconst selection = model.createSelection( range );\n\n\t\t\t// Modify the forward selection in backward direction to shrink it and remove first position of following block from it.\n\t\t\t// This is how modifySelection works and here we are making use of it.\n\t\t\tmodel.modifySelection( selection, { direction: 'backward' } );\n\n\t\t\tconst newEndPosition = selection.getLastPosition()!;\n\n\t\t\t// For such a model and selection:\n\t\t\t// <paragraph>A[</paragraph><imageBlock></imageBlock><paragraph>]B</paragraph>\n\t\t\t//\n\t\t\t// After modifySelection(), we would end up with this:\n\t\t\t// <paragraph>A[</paragraph>]<imageBlock></imageBlock><paragraph>B</paragraph>\n\t\t\t//\n\t\t\t// So we need to check if there is no content in the skipped range (because we want to include the <imageBlock>).\n\t\t\tconst skippedRange = model.createRange( newEndPosition, endPosition );\n\n\t\t\tif ( !model.hasContent( skippedRange, { ignoreMarkers: true } ) ) {\n\t\t\t\tendPosition = newEndPosition;\n\t\t\t}\n\t\t}\n\t}\n\n\treturn [\n\t\tLivePosition.fromPosition( startPosition, 'toPrevious' ),\n\t\tLivePosition.fromPosition( endPosition, 'toNext' )\n\t];\n}\n\n/**\n * Finds the lowest element in position's ancestors which is a block.\n * Returns null if a limit element is encountered before reaching a block element.\n */\nfunction getParentBlock( position: Position ): Element | null | undefined {\n\tconst element = position.parent;\n\tconst schema = element.root.document!.model.schema;\n\tconst ancestors = element.getAncestors( { parentFirst: true, includeSelf: true } );\n\n\tfor ( const element of ancestors ) {\n\t\tif ( schema.isLimit( element ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( schema.isBlock( element ) ) {\n\t\t\treturn element as Element;\n\t\t}\n\t}\n}\n\n/**\n * This function is a result of reaching the Ballmer's peak for just the right amount of time.\n * Even I had troubles documenting it after a while and after reading it again I couldn't believe that it really works.\n */\nfunction mergeBranches( writer: Writer, startPosition: Position, endPosition: Position ) {\n\tconst model = writer.model;\n\n\t// Verify if there is a need and possibility to merge.\n\tif ( !checkShouldMerge( writer.model.schema, startPosition, endPosition ) ) {\n\t\treturn;\n\t}\n\n\t// If the start element on the common ancestor level is empty, and the end element on the same level is not empty\n\t// then merge those to the right element so that it's properties are preserved (name, attributes).\n\t// Because of OT merging is used instead of removing elements.\n\t//\n\t// Merge left:\n\t// <heading1>foo[</heading1> -> <heading1>foo[]bar</heading1>\n\t// <paragraph>]bar</paragraph> -> --^\n\t//\n\t// Merge right:\n\t// <heading1>[</heading1> ->\n\t// <paragraph>]bar</paragraph> -> <paragraph>[]bar</paragraph>\n\t//\n\t// Merge left:\n\t// <blockQuote> -> <blockQuote>\n\t// <heading1>foo[</heading1> -> <heading1>foo[]bar</heading1>\n\t// <paragraph>]bar</paragraph> -> --^\n\t// </blockQuote> -> </blockQuote>\n\t//\n\t// Merge right:\n\t// <blockQuote> -> <blockQuote>\n\t// <heading1>[</heading1> ->\n\t// <paragraph>]bar</paragraph> -> <paragraph>[]bar</paragraph>\n\t// </blockQuote> -> </blockQuote>\n\n\t// Merging should not go deeper than common ancestor.\n\tconst [ startAncestor, endAncestor ] = getAncestorsJustBelowCommonAncestor( startPosition, endPosition );\n\n\t// Branches can't be merged if one of the positions is directly inside a common ancestor.\n\t//\n\t// Example:\n\t// <blockQuote>\n\t// <paragraph>[foo</paragraph>]\n\t// <table> ... </table>\n\t// <blockQuote>\n\t//\n\tif ( !startAncestor || !endAncestor ) {\n\t\treturn;\n\t}\n\n\tif ( !model.hasContent( startAncestor, { ignoreMarkers: true } ) && model.hasContent( endAncestor, { ignoreMarkers: true } ) ) {\n\t\tmergeBranchesRight( writer, startPosition, endPosition, startAncestor.parent );\n\t} else {\n\t\tmergeBranchesLeft( writer, startPosition, endPosition, startAncestor.parent );\n\t}\n}\n\n/**\n * Merging blocks to the left (properties of the left block are preserved).\n * Simple example:\n *\n * ```\n * <heading1>foo[</heading1> -> <heading1>foo[bar</heading1>]\n * <paragraph>]bar</paragraph> -> --^\n * ```\n *\n * Nested example:\n *\n * ```\n * <blockQuote> -> <blockQuote>\n * <heading1>foo[</heading1> -> <heading1>foo[bar</heading1>\n * </blockQuote> -> </blockQuote>] ^\n * <blockBlock> -> |\n * <paragraph>]bar</paragraph> -> ---\n * </blockBlock> ->\n * ```\n */\nfunction mergeBranchesLeft(\n\twriter: Writer,\n\tstartPosition: Position,\n\tendPosition: Position,\n\tcommonAncestor: Element | DocumentFragment | null\n) {\n\tconst startElement = startPosition.parent as Element;\n\tconst endElement = endPosition.parent as Element;\n\n\t// Merging reached the common ancestor element, stop here.\n\tif ( startElement == commonAncestor || endElement == commonAncestor ) {\n\t\treturn;\n\t}\n\n\t// Remember next positions to merge in next recursive step (also used as modification points pointers).\n\tstartPosition = writer.createPositionAfter( startElement );\n\tendPosition = writer.createPositionBefore( endElement );\n\n\t// Move endElement just after startElement if they aren't siblings.\n\tif ( !endPosition.isEqual( startPosition ) ) {\n\t\t//\n\t\t// <blockQuote> -> <blockQuote>\n\t\t// <heading1>foo[</heading1> -> <heading1>foo</heading1>[<paragraph>bar</paragraph>\n\t\t// </blockQuote> -> </blockQuote> ^\n\t\t// <blockBlock> -> <blockBlock> |\n\t\t// <paragraph>]bar</paragraph> -> ] ---\n\t\t// </blockBlock> -> </blockBlock>\n\t\t//\n\t\twriter.insert( endElement, startPosition );\n\t}\n\n\t// Merge two siblings (nodes on sides of startPosition):\n\t//\n\t// <blockQuote> -> <blockQuote>\n\t// <heading1>foo</heading1>[<paragraph>bar</paragraph> -> <heading1>foo[bar</heading1>\n\t// </blockQuote> -> </blockQuote>\n\t// <blockBlock> -> <blockBlock>\n\t// ] -> ]\n\t// </blockBlock> -> </blockBlock>\n\t//\n\t// Or in simple case (without moving elements in above if):\n\t// <heading1>foo</heading1>[<paragraph>bar</paragraph>] -> <heading1>foo[bar</heading1>]\n\t//\n\twriter.merge( startPosition );\n\n\t// Remove empty end ancestors:\n\t//\n\t// <blockQuote> -> <blockQuote>\n\t// <heading1>foo[bar</heading1> -> <heading1>foo[bar</heading1>\n\t// </blockQuote> -> </blockQuote>\n\t// <blockBlock> ->\n\t// ] -> ]\n\t// </blockBlock> ->\n\t//\n\twhile ( endPosition.parent.isEmpty ) {\n\t\tconst parentToRemove = endPosition.parent as Element;\n\n\t\tendPosition = writer.createPositionBefore( parentToRemove );\n\n\t\twriter.remove( parentToRemove );\n\t}\n\n\t// Verify if there is a need and possibility to merge next level.\n\tif ( !checkShouldMerge( writer.model.schema, startPosition, endPosition ) ) {\n\t\treturn;\n\t}\n\n\t// Continue merging next level (blockQuote with blockBlock in the examples above if it would not be empty and got removed).\n\tmergeBranchesLeft( writer, startPosition, endPosition, commonAncestor );\n}\n\n/**\n * Merging blocks to the right (properties of the right block are preserved).\n * Simple example:\n *\n * ```\n * <heading1>foo[</heading1> -> --v\n * <paragraph>]bar</paragraph> -> [<paragraph>foo]bar</paragraph>\n * ```\n *\n * Nested example:\n *\n * ```\n * <blockQuote> ->\n * <heading1>foo[</heading1> -> ---\n * </blockQuote> -> |\n * <blockBlock> -> [<blockBlock> v\n * <paragraph>]bar</paragraph> -> <paragraph>foo]bar</paragraph>\n * </blockBlock> -> </blockBlock>\n * ```\n */\nfunction mergeBranchesRight(\n\twriter: Writer,\n\tstartPosition: Position,\n\tendPosition: Position,\n\tcommonAncestor: Element | DocumentFragment | null\n) {\n\tconst startElement = startPosition.parent as Element;\n\tconst endElement = endPosition.parent as Element;\n\n\t// Merging reached the common ancestor element, stop here.\n\tif ( startElement == commonAncestor || endElement == commonAncestor ) {\n\t\treturn;\n\t}\n\n\t// Remember next positions to merge in next recursive step (also used as modification points pointers).\n\tstartPosition = writer.createPositionAfter( startElement );\n\tendPosition = writer.createPositionBefore( endElement );\n\n\t// Move startElement just before endElement if they aren't siblings.\n\tif ( !endPosition.isEqual( startPosition ) ) {\n\t\t//\n\t\t// <blockQuote> -> <blockQuote>\n\t\t// <heading1>foo[</heading1> -> [ ---\n\t\t// </blockQuote> -> </blockQuote> |\n\t\t// <blockBlock> -> <blockBlock> v\n\t\t// <paragraph>]bar</paragraph> -> <heading1>foo</heading1>]<paragraph>bar</paragraph>\n\t\t// </blockBlock> -> </blockBlock>\n\t\t//\n\t\twriter.insert( startElement, endPosition );\n\t}\n\n\t// Remove empty end ancestors:\n\t//\n\t// <blockQuote> ->\n\t// [ -> [\n\t// </blockQuote> ->\n\t// <blockBlock> -> <blockBlock>\n\t// <heading1>foo</heading1>]<paragraph>bar</paragraph> -> <heading1>foo</heading1>]<paragraph>bar</paragraph>\n\t// </blockBlock> -> </blockBlock>\n\t//\n\twhile ( startPosition.parent.isEmpty ) {\n\t\tconst parentToRemove = startPosition.parent as Element;\n\n\t\tstartPosition = writer.createPositionBefore( parentToRemove );\n\n\t\twriter.remove( parentToRemove );\n\t}\n\n\t// Update endPosition after inserting and removing elements.\n\tendPosition = writer.createPositionBefore( endElement );\n\n\t// Merge right two siblings (nodes on sides of endPosition):\n\t// ->\n\t// [ -> [\n\t// ->\n\t// <blockBlock> -> <blockBlock>\n\t// <heading1>foo</heading1>]<paragraph>bar</paragraph> -> <paragraph>foo]bar</paragraph>\n\t// </blockBlock> -> </blockBlock>\n\t//\n\t// Or in simple case (without moving elements in above if):\n\t// [<heading1>foo</heading1>]<paragraph>bar</paragraph> -> [<heading1>foo]bar</heading1>\n\t//\n\tmergeRight( writer, endPosition );\n\n\t// Verify if there is a need and possibility to merge next level.\n\tif ( !checkShouldMerge( writer.model.schema, startPosition, endPosition ) ) {\n\t\treturn;\n\t}\n\n\t// Continue merging next level (blockQuote with blockBlock in the examples above if it would not be empty and got removed).\n\tmergeBranchesRight( writer, startPosition, endPosition, commonAncestor );\n}\n\n/**\n * There is no right merge operation so we need to simulate it.\n */\nfunction mergeRight( writer: Writer, position: Position ) {\n\tconst startElement: any = position.nodeBefore;\n\tconst endElement: any = position.nodeAfter;\n\n\tif ( startElement.name != endElement.name ) {\n\t\twriter.rename( startElement, endElement.name );\n\t}\n\n\twriter.clearAttributes( startElement );\n\twriter.setAttributes( Object.fromEntries( endElement.getAttributes() ), startElement );\n\n\twriter.merge( position );\n}\n\n/**\n * Verifies if merging is needed and possible. It's not needed if both positions are in the same element\n * and it's not possible if some element is a limit or the range crosses a limit element.\n */\nfunction checkShouldMerge( schema: Schema, startPosition: Position, endPosition: Position ): boolean {\n\tconst startElement = startPosition.parent;\n\tconst endElement = endPosition.parent;\n\n\t// If both positions ended up in the same parent, then there's nothing more to merge:\n\t// <$root><p>x[</p><p>]y</p></$root> => <$root><p>xy</p>[]</$root>\n\tif ( startElement == endElement ) {\n\t\treturn false;\n\t}\n\n\t// If one of the positions is a limit element, then there's nothing to merge because we don't want to cross the limit boundaries.\n\tif ( schema.isLimit( startElement ) || schema.isLimit( endElement ) ) {\n\t\treturn false;\n\t}\n\n\t// Check if operations we'll need to do won't need to cross object or limit boundaries.\n\t// E.g., we can't merge endElement into startElement in this case:\n\t// <limit><startElement>x[</startElement></limit><endElement>]</endElement>\n\treturn isCrossingLimitElement( startPosition, endPosition, schema );\n}\n\n/**\n * Returns the elements that are the ancestors of the provided positions that are direct children of the common ancestor.\n */\nfunction getAncestorsJustBelowCommonAncestor( positionA: Position, positionB: Position ) {\n\tconst ancestorsA = positionA.getAncestors();\n\tconst ancestorsB = positionB.getAncestors();\n\n\tlet i = 0;\n\n\twhile ( ancestorsA[ i ] && ancestorsA[ i ] == ancestorsB[ i ] ) {\n\t\ti++;\n\t}\n\n\treturn [ ancestorsA[ i ], ancestorsB[ i ] ];\n}\n\nfunction shouldAutoparagraph( schema: Schema, position: Position ) {\n\tconst isTextAllowed = schema.checkChild( position, '$text' );\n\tconst isParagraphAllowed = schema.checkChild( position, 'paragraph' );\n\n\treturn !isTextAllowed && isParagraphAllowed;\n}\n\n/**\n * Check if parents of two positions can be merged by checking if there are no limit/object\n * boundaries between those two positions.\n *\n * E.g. in <bQ><p>x[]</p></bQ><widget><caption>{}</caption></widget>\n * we'll check <p>, <bQ>, <widget> and <caption>.\n * Usually, widget and caption are marked as objects/limits in the schema, so in this case merging will be blocked.\n */\nfunction isCrossingLimitElement( leftPos: Position, rightPos: Position, schema: Schema ) {\n\tconst rangeToCheck = new Range( leftPos, rightPos );\n\n\tfor ( const value of rangeToCheck.getWalker() ) {\n\t\tif ( schema.isLimit( value.item ) ) {\n\t\t\treturn false;\n\t\t}\n\t}\n\n\treturn true;\n}\n\nfunction insertParagraph(\n\twriter: Writer,\n\tposition: Position,\n\tselection: Selection | DocumentSelection,\n\tattributes = {}\n) {\n\tconst paragraph = writer.createElement( 'paragraph' );\n\n\twriter.model.schema.setAllowedAttributes( paragraph, attributes, writer );\n\n\twriter.insert( paragraph, position );\n\n\tcollapseSelectionAt( writer, selection, writer.createPositionAt( paragraph, 0 ) );\n}\n\nfunction replaceEntireContentWithParagraph( writer: Writer, selection: Selection | DocumentSelection ) {\n\tconst limitElement = writer.model.schema.getLimitElement( selection );\n\n\twriter.remove( writer.createRangeIn( limitElement ) );\n\tinsertParagraph( writer, writer.createPositionAt( limitElement, 0 ), selection );\n}\n\n/**\n * We want to replace the entire content with a paragraph when:\n * * the entire content is selected,\n * * selection contains at least two elements,\n * * whether the paragraph is allowed in schema in the common ancestor.\n */\nfunction shouldEntireContentBeReplacedWithParagraph( schema: Schema, selection: Selection | DocumentSelection ) {\n\tconst limitElement = schema.getLimitElement( selection );\n\n\tif ( !selection.containsEntireContent( limitElement ) ) {\n\t\treturn false;\n\t}\n\n\tconst range = selection.getFirstRange()!;\n\n\tif ( range.start.parent == range.end.parent ) {\n\t\treturn false;\n\t}\n\n\treturn schema.checkChild( limitElement, 'paragraph' );\n}\n\n/**\n * Helper function that sets the selection. Depending whether given `selection` is a document selection or not,\n * uses a different method to set it.\n */\nfunction collapseSelectionAt(\n\twriter: Writer,\n\tselection: Selection | DocumentSelection,\n\tpositionOrRange: Position | Range\n) {\n\tif ( selection instanceof DocumentSelection ) {\n\t\twriter.setSelection( positionOrRange );\n\t} else {\n\t\tselection.setTo( positionOrRange );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport type DocumentFragment from '../documentfragment';\nimport type DocumentSelection from '../documentselection';\nimport type Element from '../element';\nimport type Model from '../model';\nimport type Range from '../range';\nimport type Selection from '../selection';\nimport type Writer from '../writer';\n\n/**\n * @module engine/model/utils/getselectedcontent\n */\n\n/**\n * Gets a clone of the selected content.\n *\n * For example, for the following selection:\n *\n * ```html\n * <p>x</p><quote><p>y</p><h>fir[st</h></quote><p>se]cond</p><p>z</p>\n * ```\n *\n * It will return a document fragment with such a content:\n *\n * ```html\n * <quote><h>st</h></quote><p>se</p>\n * ```\n *\n * @param model The model in context of which the selection modification should be performed.\n * @param selection The selection of which content will be returned.\n */\nexport default function getSelectedContent(\n\tmodel: Model,\n\tselection: Selection | DocumentSelection\n): DocumentFragment {\n\treturn model.change( writer => {\n\t\tconst frag = writer.createDocumentFragment();\n\t\tconst range = selection.getFirstRange();\n\n\t\tif ( !range || range.isCollapsed ) {\n\t\t\treturn frag;\n\t\t}\n\n\t\tconst root = range.start.root;\n\t\tconst commonPath = range.start.getCommonPath( range.end );\n\t\tconst commonParent = root.getNodeByPath( commonPath );\n\n\t\t// ## 1st step\n\t\t//\n\t\t// First, we'll clone a fragment represented by a minimal flat range\n\t\t// containing the original range to be cloned.\n\t\t// E.g. let's consider such a range:\n\t\t//\n\t\t// <p>x</p><quote><p>y</p><h>fir[st</h></quote><p>se]cond</p><p>z</p>\n\t\t//\n\t\t// A minimal flat range containing this one is:\n\t\t//\n\t\t// <p>x</p>[<quote><p>y</p><h>first</h></quote><p>second</p>]<p>z</p>\n\t\t//\n\t\t// We can easily clone this structure, preserving e.g. the <quote> element.\n\t\tlet flatSubtreeRange: Range;\n\n\t\tif ( range.start.parent == range.end.parent ) {\n\t\t\t// The original range is flat, so take it.\n\t\t\tflatSubtreeRange = range;\n\t\t} else {\n\t\t\tflatSubtreeRange = writer.createRange(\n\t\t\t\twriter.createPositionAt( commonParent, range.start.path[ commonPath.length ] ),\n\t\t\t\twriter.createPositionAt( commonParent, range.end.path[ commonPath.length ] + 1 )\n\t\t\t);\n\t\t}\n\n\t\tconst howMany = flatSubtreeRange.end.offset - flatSubtreeRange.start.offset;\n\n\t\t// Clone the whole contents.\n\t\tfor ( const item of flatSubtreeRange.getItems( { shallow: true } ) ) {\n\t\t\tif ( item.is( '$textProxy' ) ) {\n\t\t\t\twriter.appendText( item.data, item.getAttributes(), frag );\n\t\t\t} else {\n\t\t\t\twriter.append( writer.cloneElement( item as Element, true ), frag );\n\t\t\t}\n\t\t}\n\n\t\t// ## 2nd step\n\t\t//\n\t\t// If the original range wasn't flat, then we need to remove the excess nodes from the both ends of the cloned fragment.\n\t\t//\n\t\t// For example, for the range shown in the 1st step comment, we need to remove these pieces:\n\t\t//\n\t\t// <quote>[<p>y</p>]<h>[fir]st</h></quote><p>se[cond]</p>\n\t\t//\n\t\t// So this will be the final copied content:\n\t\t//\n\t\t// <quote><h>st</h></quote><p>se</p>\n\t\t//\n\t\t// In order to do that, we remove content from these two ranges:\n\t\t//\n\t\t// [<quote><p>y</p><h>fir]st</h></quote><p>se[cond</p>]\n\t\tif ( flatSubtreeRange != range ) {\n\t\t\t// Find the position of the original range in the cloned fragment.\n\t\t\tconst newRange = range._getTransformedByMove( flatSubtreeRange.start, writer.createPositionAt( frag, 0 ), howMany )[ 0 ];\n\n\t\t\tconst leftExcessRange = writer.createRange( writer.createPositionAt( frag, 0 ), newRange.start );\n\t\t\tconst rightExcessRange = writer.createRange( newRange.end, writer.createPositionAt( frag, 'end' ) );\n\n\t\t\tremoveRangeContent( rightExcessRange, writer );\n\t\t\tremoveRangeContent( leftExcessRange, writer );\n\t\t}\n\n\t\treturn frag;\n\t} );\n}\n\n// After https://github.com/ckeditor/ckeditor5-engine/issues/690 is fixed,\n// this function will, most likely, be able to rewritten using getMinimalFlatRanges().\nfunction removeRangeContent( range: Range, writer: Writer ) {\n\tconst parentsToCheck: Array<Element | DocumentFragment> = [];\n\n\tArray.from( range.getItems( { direction: 'backward' } ) )\n\t\t// We should better store ranges because text proxies will lose integrity\n\t\t// with the text nodes when we'll start removing content.\n\t\t.map( item => writer.createRangeOn( item ) )\n\t\t// Filter only these items which are fully contained in the passed range.\n\t\t//\n\t\t// E.g. for the following range: [<quote><p>y</p><h>fir]st</h>\n\t\t// the walker will return the entire <h> element, when only the \"fir\" item inside it is fully contained.\n\t\t.filter( itemRange => {\n\t\t\t// We should be able to use Range.containsRange, but https://github.com/ckeditor/ckeditor5-engine/issues/691.\n\t\t\tconst contained =\n\t\t\t\t( itemRange.start.isAfter( range.start ) || itemRange.start.isEqual( range.start ) ) &&\n\t\t\t\t( itemRange.end.isBefore( range.end ) || itemRange.end.isEqual( range.end ) );\n\n\t\t\treturn contained;\n\t\t} )\n\t\t.forEach( itemRange => {\n\t\t\tparentsToCheck.push( itemRange.start.parent );\n\n\t\t\twriter.remove( itemRange );\n\t\t} );\n\n\t// Remove ancestors of the removed items if they turned to be empty now\n\t// (their whole content was contained in the range).\n\tparentsToCheck.forEach( parentToCheck => {\n\t\tlet parent = parentToCheck;\n\n\t\twhile ( parent.parent && parent.isEmpty ) {\n\t\t\tconst removeRange = writer.createRangeOn( parent );\n\n\t\t\tparent = parent.parent;\n\n\t\t\twriter.remove( removeRange );\n\t\t}\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/utils/insertcontent\n */\n\nimport DocumentSelection from '../documentselection';\nimport Element from '../element';\nimport LivePosition from '../liveposition';\nimport LiveRange from '../liverange';\nimport Position from '../position';\nimport Range from '../range';\n\nimport type DocumentFragment from '../documentfragment';\nimport type Item from '../item';\nimport type Model from '../model';\nimport type Schema from '../schema';\nimport type Writer from '../writer';\nimport type Node from '../node';\nimport type Selection from '../selection';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Inserts content into the editor (specified selection) as one would expect the paste functionality to work.\n *\n * It takes care of removing the selected content, splitting elements (if needed), inserting elements and merging elements appropriately.\n *\n * Some examples:\n *\n * ```html\n * <p>x^</p> + <p>y</p> => <p>x</p><p>y</p> => <p>xy[]</p>\n * <p>x^y</p> + <p>z</p> => <p>x</p>^<p>y</p> + <p>z</p> => <p>x</p><p>z</p><p>y</p> => <p>xz[]y</p>\n * <p>x^y</p> + <img /> => <p>x</p>^<p>y</p> + <img /> => <p>x</p><img /><p>y</p>\n * <p>x</p><p>^</p><p>z</p> + <p>y</p> => <p>x</p><p>y[]</p><p>z</p> (no merging)\n * <p>x</p>[<img />]<p>z</p> + <p>y</p> => <p>x</p>^<p>z</p> + <p>y</p> => <p>x</p><p>y[]</p><p>z</p>\n * ```\n *\n * If an instance of {@link module:engine/model/selection~Selection} is passed as `selectable` it will be modified\n * to the insertion selection (equal to a range to be selected after insertion).\n *\n * If `selectable` is not passed, the content will be inserted using the current selection of the model document.\n *\n * **Note:** Use {@link module:engine/model/model~Model#insertContent} instead of this function.\n * This function is only exposed to be reusable in algorithms which change the {@link module:engine/model/model~Model#insertContent}\n * method's behavior.\n *\n * @param model The model in context of which the insertion should be performed.\n * @param content The content to insert.\n * @param selectable Selection into which the content should be inserted.\n * @param placeOrOffset Sets place or offset of the selection.\n * @returns Range which contains all the performed changes. This is a range that, if removed,\n * would return the model to the state before the insertion. If no changes were preformed by `insertContent`, returns a range collapsed\n * at the insertion position.\n */\nexport default function insertContent(\n\tmodel: Model,\n\tcontent: Item | DocumentFragment,\n\tselectable?: Selection | DocumentSelection\n): Range {\n\treturn model.change( writer => {\n\t\tconst selection: Selection | DocumentSelection = selectable ? selectable : model.document.selection;\n\n\t\tif ( !selection.isCollapsed ) {\n\t\t\tmodel.deleteContent( selection, { doNotAutoparagraph: true } );\n\t\t}\n\n\t\tconst insertion = new Insertion( model, writer, selection.anchor! );\n\t\tconst fakeMarkerElements = [];\n\n\t\tlet nodesToInsert: any;\n\n\t\tif ( content.is( 'documentFragment' ) ) {\n\t\t\t// If document fragment has any markers, these markers should be inserted into the model as well.\n\t\t\tif ( content.markers.size ) {\n\t\t\t\tconst markersPosition = [];\n\n\t\t\t\tfor ( const [ name, range ] of content.markers ) {\n\t\t\t\t\tconst { start, end } = range;\n\t\t\t\t\tconst isCollapsed = start.isEqual( end );\n\n\t\t\t\t\tmarkersPosition.push(\n\t\t\t\t\t\t{ position: start, name, isCollapsed },\n\t\t\t\t\t\t{ position: end, name, isCollapsed }\n\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\t// Markers position is sorted backwards to ensure that the insertion of fake markers will not change\n\t\t\t\t// the position of the next markers.\n\t\t\t\tmarkersPosition.sort( ( { position: posA }, { position: posB } ) => posA.isBefore( posB ) ? 1 : -1 );\n\n\t\t\t\tfor ( const { position, name, isCollapsed } of markersPosition ) {\n\t\t\t\t\tlet fakeElement = null;\n\t\t\t\t\tlet collapsed = null;\n\t\t\t\t\tconst isAtBeginning = position.parent === content && position.isAtStart;\n\t\t\t\t\tconst isAtEnd = position.parent === content && position.isAtEnd;\n\n\t\t\t\t\t// We have two ways of handling markers. In general, we want to add temporary <$marker> model elements to\n\t\t\t\t\t// represent marker boundaries. These elements will be inserted into content together with the rest\n\t\t\t\t\t// of the document fragment. After insertion is done, positions for these elements will be read\n\t\t\t\t\t// and proper, actual markers will be created in the model and fake elements will be removed.\n\t\t\t\t\t//\n\t\t\t\t\t// However, if the <$marker> element is at the beginning or at the end of the document fragment,\n\t\t\t\t\t// it may affect how the inserted content is merged with current model, impacting the insertion\n\t\t\t\t\t// result. To avoid that, we don't add <$marker> elements at these positions. Instead, we will use\n\t\t\t\t\t// `Insertion#getAffectedRange()` to figure out new positions for these marker boundaries.\n\t\t\t\t\tif ( !isAtBeginning && !isAtEnd ) {\n\t\t\t\t\t\tfakeElement = writer.createElement( '$marker' );\n\t\t\t\t\t\twriter.insert( fakeElement, position );\n\t\t\t\t\t} else if ( isCollapsed ) {\n\t\t\t\t\t\t// Save whether the collapsed marker was at the beginning or at the end of document fragment\n\t\t\t\t\t\t// to know where to create it after the insertion is done.\n\t\t\t\t\t\tcollapsed = isAtBeginning ? 'start' as const : 'end' as const;\n\t\t\t\t\t}\n\n\t\t\t\t\tfakeMarkerElements.push( {\n\t\t\t\t\t\tname,\n\t\t\t\t\t\telement: fakeElement,\n\t\t\t\t\t\tcollapsed\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tnodesToInsert = content.getChildren();\n\t\t} else {\n\t\t\tnodesToInsert = [ content ];\n\t\t}\n\n\t\tinsertion.handleNodes( nodesToInsert );\n\n\t\tlet newRange = insertion.getSelectionRange();\n\n\t\tif ( content.is( 'documentFragment' ) && fakeMarkerElements.length ) {\n\t\t\t// After insertion was done, the selection was set but the model contains fake <$marker> elements.\n\t\t\t// These <$marker> elements will be now removed. Because of that, we will need to fix the selection.\n\t\t\t// We will create a live range that will automatically be update as <$marker> elements are removed.\n\t\t\tconst selectionLiveRange = newRange ? LiveRange.fromRange( newRange ) : null;\n\n\t\t\t// Marker name -> [ start position, end position ].\n\t\t\tconst markersData: Record<string, Array<Position>> = {};\n\n\t\t\t// Note: `fakeMarkerElements` are sorted backwards. However, now, we want to handle the markers\n\t\t\t// from the beginning, so that existing <$marker> elements do not affect markers positions.\n\t\t\t// This is why we iterate from the end to the start.\n\t\t\tfor ( let i = fakeMarkerElements.length - 1; i >= 0; i-- ) {\n\t\t\t\tconst { name, element, collapsed } = fakeMarkerElements[ i ];\n\t\t\t\tconst isStartBoundary = !markersData[ name ];\n\n\t\t\t\tif ( isStartBoundary ) {\n\t\t\t\t\tmarkersData[ name ] = [];\n\t\t\t\t}\n\n\t\t\t\tif ( element ) {\n\t\t\t\t\t// Read fake marker element position to learn where the marker should be created.\n\t\t\t\t\tconst elementPosition = writer.createPositionAt( element, 'before' );\n\n\t\t\t\t\tmarkersData[ name ].push( elementPosition );\n\n\t\t\t\t\twriter.remove( element );\n\t\t\t\t} else {\n\t\t\t\t\t// If the fake marker element does not exist, it means that the marker boundary was at the beginning or at the end.\n\t\t\t\t\tconst rangeOnInsertion = insertion.getAffectedRange();\n\n\t\t\t\t\tif ( !rangeOnInsertion ) {\n\t\t\t\t\t\t// If affected range is `null` it means that nothing was in the document fragment or all content was filtered out.\n\t\t\t\t\t\t// Some markers that were in the filtered content may be removed (partially or totally).\n\t\t\t\t\t\t// Let's handle only those markers that were at the beginning or at the end of the document fragment.\n\t\t\t\t\t\tif ( collapsed ) {\n\t\t\t\t\t\t\tmarkersData[ name ].push( insertion.position );\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\tif ( collapsed ) {\n\t\t\t\t\t\t// If the marker was collapsed at the beginning or at the end of the document fragment,\n\t\t\t\t\t\t// put both boundaries at the beginning or at the end of inserted range (to keep the marker collapsed).\n\t\t\t\t\t\tmarkersData[ name ].push( rangeOnInsertion[ collapsed ] );\n\t\t\t\t\t} else {\n\t\t\t\t\t\tmarkersData[ name ].push( isStartBoundary ? rangeOnInsertion.start : rangeOnInsertion.end );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfor ( const [ name, [ start, end ] ] of Object.entries( markersData ) ) {\n\t\t\t\t// For now, we ignore markers if they are included in the filtered-out content.\n\t\t\t\t// In the future implementation we will improve that case to create markers that are not filtered out completely.\n\t\t\t\tif ( start && end && start.root === end.root ) {\n\t\t\t\t\twriter.addMarker( name, {\n\t\t\t\t\t\tusingOperation: true,\n\t\t\t\t\t\taffectsData: true,\n\t\t\t\t\t\trange: new Range( start, end )\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( selectionLiveRange ) {\n\t\t\t\tnewRange = selectionLiveRange.toRange();\n\t\t\t\tselectionLiveRange.detach();\n\t\t\t}\n\t\t}\n\n\t\t/* istanbul ignore else -- @preserve */\n\t\tif ( newRange ) {\n\t\t\tif ( selection instanceof DocumentSelection ) {\n\t\t\t\twriter.setSelection( newRange );\n\t\t\t} else {\n\t\t\t\tselection.setTo( newRange );\n\t\t\t}\n\t\t} else {\n\t\t\t// We are not testing else because it's a safe check for unpredictable edge cases:\n\t\t\t// an insertion without proper range to select.\n\t\t\t//\n\t\t\t// @if CK_DEBUG // console.warn( 'Cannot determine a proper selection range after insertion.' );\n\t\t}\n\n\t\tconst affectedRange = insertion.getAffectedRange() || model.createRange( selection.anchor! );\n\n\t\tinsertion.destroy();\n\n\t\treturn affectedRange;\n\t} );\n}\n\n/**\n * Utility class for performing content insertion.\n */\nclass Insertion {\n\t/**\n\t * The model in context of which the insertion should be performed.\n\t */\n\tpublic readonly model: Model;\n\n\t/**\n\t * Batch to which operations will be added.\n\t */\n\tpublic readonly writer: Writer;\n\n\t/**\n\t * The position at which (or near which) the next node will be inserted.\n\t */\n\tpublic position: Position;\n\n\t/**\n\t * Elements with which the inserted elements can be merged.\n\t *\n\t * ```html\n\t * <p>x^</p><p>y</p> + <p>z</p> (can merge to <p>x</p>)\n\t * <p>x</p><p>^y</p> + <p>z</p> (can merge to <p>y</p>)\n\t * <p>x^y</p> + <p>z</p> (can merge to <p>xy</p> which will be split during the action,\n\t * \t\t\t\t\t\tso both its pieces will be added to this set)\n\t * ```\n\t */\n\tpublic readonly canMergeWith: Set<Node | DocumentFragment | null>;\n\n\t/**\n\t * Schema of the model.\n\t */\n\tpublic readonly schema: Schema;\n\n\t/**\n\t * The temporary DocumentFragment used for grouping multiple nodes for single insert operation.\n\t */\n\tprivate readonly _documentFragment: DocumentFragment;\n\n\t/**\n\t * The current position in the temporary DocumentFragment.\n\t */\n\tprivate _documentFragmentPosition: Position;\n\n\t/**\n\t * The reference to the first inserted node.\n\t */\n\tprivate _firstNode: Node | null = null;\n\n\t/**\n\t * The reference to the last inserted node.\n\t */\n\tprivate _lastNode: Node | null = null;\n\n\t/**\n\t * The reference to the last auto paragraph node.\n\t */\n\tprivate _lastAutoParagraph: Element | null = null;\n\n\t/**\n\t * The array of nodes that should be cleaned of not allowed attributes.\n\t */\n\tprivate _filterAttributesOf: Array<Node> = [];\n\n\t/**\n\t * Beginning of the affected range. See {@link module:engine/model/utils/insertcontent~Insertion#getAffectedRange}.\n\t */\n\tprivate _affectedStart: LivePosition | null = null;\n\n\t/**\n\t * End of the affected range. See {@link module:engine/model/utils/insertcontent~Insertion#getAffectedRange}.\n\t */\n\tprivate _affectedEnd: LivePosition | null = null;\n\n\tprivate _nodeToSelect: Node | null = null;\n\n\tconstructor( model: Model, writer: Writer, position: Position ) {\n\t\tthis.model = model;\n\t\tthis.writer = writer;\n\t\tthis.position = position;\n\t\tthis.canMergeWith = new Set( [ this.position.parent ] );\n\t\tthis.schema = model.schema;\n\n\t\tthis._documentFragment = writer.createDocumentFragment();\n\t\tthis._documentFragmentPosition = writer.createPositionAt( this._documentFragment, 0 );\n\t}\n\n\t/**\n\t * Handles insertion of a set of nodes.\n\t *\n\t * @param nodes Nodes to insert.\n\t */\n\tpublic handleNodes( nodes: Iterable<Node> ): void {\n\t\tfor ( const node of Array.from( nodes ) ) {\n\t\t\tthis._handleNode( node );\n\t\t}\n\n\t\t// Insert nodes collected in temporary DocumentFragment.\n\t\tthis._insertPartialFragment();\n\n\t\t// If there was an auto paragraph then we might need to adjust the end of insertion.\n\t\tif ( this._lastAutoParagraph ) {\n\t\t\tthis._updateLastNodeFromAutoParagraph( this._lastAutoParagraph );\n\t\t}\n\n\t\t// After the content was inserted we may try to merge it with its next sibling if the selection was in it initially.\n\t\t// Merging with the previous sibling was performed just after inserting the first node to the document.\n\t\tthis._mergeOnRight();\n\n\t\t// TMP this will become a post-fixer.\n\t\tthis.schema.removeDisallowedAttributes( this._filterAttributesOf, this.writer );\n\t\tthis._filterAttributesOf = [];\n\t}\n\n\t/**\n\t * Updates the last node after the auto paragraphing.\n\t *\n\t * @param node The last auto paragraphing node.\n\t */\n\tprivate _updateLastNodeFromAutoParagraph( node: Node ): void {\n\t\tconst positionAfterLastNode = this.writer.createPositionAfter( this._lastNode! );\n\t\tconst positionAfterNode = this.writer.createPositionAfter( node );\n\n\t\t// If the real end was after the last auto paragraph then update relevant properties.\n\t\tif ( positionAfterNode.isAfter( positionAfterLastNode ) ) {\n\t\t\tthis._lastNode = node;\n\n\t\t\t/* istanbul ignore if -- @preserve */\n\t\t\tif ( this.position.parent != node || !this.position.isAtEnd ) {\n\t\t\t\t// Algorithm's correctness check. We should never end up here but it's good to know that we did.\n\t\t\t\t// At this point the insertion position should be at the end of the last auto paragraph.\n\t\t\t\t// Note: This error is documented in other place in this file.\n\t\t\t\tthrow new CKEditorError( 'insertcontent-invalid-insertion-position', this );\n\t\t\t}\n\n\t\t\tthis.position = positionAfterNode;\n\t\t\tthis._setAffectedBoundaries( this.position );\n\t\t}\n\t}\n\n\t/**\n\t * Returns range to be selected after insertion.\n\t * Returns `null` if there is no valid range to select after insertion.\n\t */\n\tpublic getSelectionRange(): Range | null {\n\t\tif ( this._nodeToSelect ) {\n\t\t\treturn Range._createOn( this._nodeToSelect );\n\t\t}\n\n\t\treturn this.model.schema.getNearestSelectionRange( this.position );\n\t}\n\n\t/**\n\t * Returns a range which contains all the performed changes. This is a range that, if removed, would return the model to the state\n\t * before the insertion. Returns `null` if no changes were done.\n\t */\n\tpublic getAffectedRange(): Range | null {\n\t\tif ( !this._affectedStart ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn new Range( this._affectedStart, this._affectedEnd );\n\t}\n\n\t/**\n\t * Destroys `Insertion` instance.\n\t */\n\tpublic destroy(): void {\n\t\tif ( this._affectedStart ) {\n\t\t\tthis._affectedStart.detach();\n\t\t}\n\n\t\tif ( this._affectedEnd ) {\n\t\t\tthis._affectedEnd.detach();\n\t\t}\n\t}\n\n\t/**\n\t * Handles insertion of a single node.\n\t */\n\tprivate _handleNode( node: Node ): void {\n\t\t// Let's handle object in a special way.\n\t\t// * They should never be merged with other elements.\n\t\t// * If they are not allowed in any of the selection ancestors, they could be either autoparagraphed or totally removed.\n\t\tif ( this.schema.isObject( node ) ) {\n\t\t\tthis._handleObject( node as Element );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Try to find a place for the given node.\n\n\t\t// Check if a node can be inserted in the given position or it would be accepted if a paragraph would be inserted.\n\t\t// Inserts the auto paragraph if it would allow for insertion.\n\t\tlet isAllowed = this._checkAndAutoParagraphToAllowedPosition( node );\n\n\t\tif ( !isAllowed ) {\n\t\t\t// Split the position.parent's branch up to a point where the node can be inserted.\n\t\t\t// If it isn't allowed in the whole branch, then of course don't split anything.\n\t\t\tisAllowed = this._checkAndSplitToAllowedPosition( node );\n\n\t\t\tif ( !isAllowed ) {\n\t\t\t\tthis._handleDisallowedNode( node );\n\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\n\t\t// Add node to the current temporary DocumentFragment.\n\t\tthis._appendToFragment( node );\n\n\t\t// Store the first and last nodes for easy access for merging with sibling nodes.\n\t\tif ( !this._firstNode ) {\n\t\t\tthis._firstNode = node;\n\t\t}\n\n\t\tthis._lastNode = node;\n\t}\n\n\t/**\n\t * Inserts the temporary DocumentFragment into the model.\n\t */\n\tprivate _insertPartialFragment(): void {\n\t\tif ( this._documentFragment.isEmpty ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst livePosition = LivePosition.fromPosition( this.position, 'toNext' );\n\n\t\tthis._setAffectedBoundaries( this.position );\n\n\t\t// If the very first node of the whole insertion process is inserted, insert it separately for OT reasons (undo).\n\t\t// Note: there can be multiple calls to `_insertPartialFragment()` during one insertion process.\n\t\t// Note: only the very first node can be merged so we have to do separate operation only for it.\n\t\tif ( this._documentFragment.getChild( 0 ) == this._firstNode ) {\n\t\t\tthis.writer.insert( this._firstNode!, this.position );\n\n\t\t\t// We must merge the first node just after inserting it to avoid problems with OT.\n\t\t\t// (See: https://github.com/ckeditor/ckeditor5/pull/8773#issuecomment-760945652).\n\t\t\tthis._mergeOnLeft();\n\n\t\t\tthis.position = livePosition.toPosition();\n\t\t}\n\n\t\t// Insert the remaining nodes from document fragment.\n\t\tif ( !this._documentFragment.isEmpty ) {\n\t\t\tthis.writer.insert( this._documentFragment, this.position );\n\t\t}\n\n\t\tthis._documentFragmentPosition = this.writer.createPositionAt( this._documentFragment, 0 );\n\n\t\tthis.position = livePosition.toPosition();\n\t\tlivePosition.detach();\n\t}\n\n\t/**\n\t * @param node The object element.\n\t */\n\tprivate _handleObject( node: Element ): void {\n\t\t// Try finding it a place in the tree.\n\t\tif ( this._checkAndSplitToAllowedPosition( node ) ) {\n\t\t\tthis._appendToFragment( node );\n\t\t}\n\t\t// Try autoparagraphing.\n\t\telse {\n\t\t\tthis._tryAutoparagraphing( node );\n\t\t}\n\t}\n\n\t/**\n\t * @param node The disallowed node which needs to be handled.\n\t */\n\tprivate _handleDisallowedNode( node: Node ): void {\n\t\t// If the node is an element, try inserting its children (strip the parent).\n\t\tif ( node.is( 'element' ) ) {\n\t\t\tthis.handleNodes( node.getChildren() );\n\t\t}\n\t\t// If text is not allowed, try autoparagraphing it.\n\t\telse {\n\t\t\tthis._tryAutoparagraphing( node );\n\t\t}\n\t}\n\n\t/**\n\t * Append a node to the temporary DocumentFragment.\n\t *\n\t * @param node The node to insert.\n\t */\n\tprivate _appendToFragment( node: Node ): void {\n\t\t/* istanbul ignore if -- @preserve */\n\t\tif ( !this.schema.checkChild( this.position, node ) ) {\n\t\t\t// Algorithm's correctness check. We should never end up here but it's good to know that we did.\n\t\t\t// Note that it would often be a silent issue if we insert node in a place where it's not allowed.\n\n\t\t\t/**\n\t\t\t * Given node cannot be inserted on the given position.\n\t\t\t *\n\t\t\t * @error insertcontent-wrong-position\n\t\t\t * @param node Node to insert.\n\t\t\t * @param position Position to insert the node at.\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'insertcontent-wrong-position',\n\t\t\t\tthis,\n\t\t\t\t{ node, position: this.position }\n\t\t\t);\n\t\t}\n\n\t\tthis.writer.insert( node, this._documentFragmentPosition );\n\t\tthis._documentFragmentPosition = this._documentFragmentPosition.getShiftedBy( node.offsetSize );\n\n\t\t// The last inserted object should be selected because we can't put a collapsed selection after it.\n\t\tif ( this.schema.isObject( node ) && !this.schema.checkChild( this.position, '$text' ) ) {\n\t\t\tthis._nodeToSelect = node;\n\t\t} else {\n\t\t\tthis._nodeToSelect = null;\n\t\t}\n\n\t\tthis._filterAttributesOf.push( node );\n\t}\n\n\t/**\n\t * Sets `_affectedStart` and `_affectedEnd` to the given `position`. Should be used before a change is done during insertion process to\n\t * mark the affected range.\n\t *\n\t * This method is used before inserting a node or splitting a parent node. `_affectedStart` and `_affectedEnd` are also changed\n\t * during merging, but the logic there is more complicated so it is left out of this function.\n\t */\n\tprivate _setAffectedBoundaries( position: Position ): void {\n\t\t// Set affected boundaries stickiness so that those position will \"expand\" when something is inserted in between them:\n\t\t// <paragraph>Foo][bar</paragraph> -> <paragraph>Foo]xx[bar</paragraph>\n\t\t// This is why it cannot be a range but two separate positions.\n\t\tif ( !this._affectedStart ) {\n\t\t\tthis._affectedStart = LivePosition.fromPosition( position, 'toPrevious' );\n\t\t}\n\n\t\t// If `_affectedEnd` is before the new boundary position, expand `_affectedEnd`. This can happen if first inserted node was\n\t\t// inserted into the parent but the next node is moved-out of that parent:\n\t\t// (1) <paragraph>Foo][</paragraph> -> <paragraph>Foo]xx[</paragraph>\n\t\t// (2) <paragraph>Foo]xx[</paragraph> -> <paragraph>Foo]xx</paragraph><widget></widget>[\n\t\tif ( !this._affectedEnd || this._affectedEnd.isBefore( position ) ) {\n\t\t\tif ( this._affectedEnd ) {\n\t\t\t\tthis._affectedEnd.detach();\n\t\t\t}\n\n\t\t\tthis._affectedEnd = LivePosition.fromPosition( position, 'toNext' );\n\t\t}\n\t}\n\n\t/**\n\t * Merges the previous sibling of the first node if it should be merged.\n\t *\n\t * After the content was inserted we may try to merge it with its siblings.\n\t * This should happen only if the selection was in those elements initially.\n\t */\n\tprivate _mergeOnLeft(): void {\n\t\tconst node = this._firstNode;\n\n\t\tif ( !( node instanceof Element ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !this._canMergeLeft( node ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst mergePosLeft = LivePosition._createBefore( node );\n\t\tmergePosLeft.stickiness = 'toNext';\n\n\t\tconst livePosition = LivePosition.fromPosition( this.position, 'toNext' );\n\n\t\t// If `_affectedStart` is sames as merge position, it means that the element \"marked\" by `_affectedStart` is going to be\n\t\t// removed and its contents will be moved. This won't transform `LivePosition` so `_affectedStart` needs to be moved\n\t\t// by hand to properly reflect affected range. (Due to `_affectedStart` and `_affectedEnd` stickiness, the \"range\" is\n\t\t// shown as `][`).\n\t\t//\n\t\t// Example - insert `<paragraph>Abc</paragraph><paragraph>Xyz</paragraph>` at the end of `<paragraph>Foo^</paragraph>`:\n\t\t//\n\t\t// <paragraph>Foo</paragraph><paragraph>Bar</paragraph> -->\n\t\t// <paragraph>Foo</paragraph>]<paragraph>Abc</paragraph><paragraph>Xyz</paragraph>[<paragraph>Bar</paragraph> -->\n\t\t// <paragraph>Foo]Abc</paragraph><paragraph>Xyz</paragraph>[<paragraph>Bar</paragraph>\n\t\t//\n\t\t// Note, that if we are here then something must have been inserted, so `_affectedStart` and `_affectedEnd` have to be set.\n\t\tif ( this._affectedStart!.isEqual( mergePosLeft ) ) {\n\t\t\tthis._affectedStart!.detach();\n\t\t\tthis._affectedStart = LivePosition._createAt( mergePosLeft.nodeBefore!, 'end', 'toPrevious' );\n\t\t}\n\n\t\t// We need to update the references to the first and last nodes if they will be merged into the previous sibling node\n\t\t// because the reference would point to the removed node.\n\t\t//\n\t\t// <p>A^A</p> + <p>X</p>\n\t\t//\n\t\t// <p>A</p>^<p>A</p>\n\t\t// <p>A</p><p>X</p><p>A</p>\n\t\t// <p>AX</p><p>A</p>\n\t\t// <p>AXA</p>\n\t\tif ( this._firstNode === this._lastNode ) {\n\t\t\tthis._firstNode = mergePosLeft.nodeBefore;\n\t\t\tthis._lastNode = mergePosLeft.nodeBefore;\n\t\t}\n\n\t\tthis.writer.merge( mergePosLeft );\n\n\t\t// If only one element (the merged one) is in the \"affected range\", also move the affected range end appropriately.\n\t\t//\n\t\t// Example - insert `<paragraph>Abc</paragraph>` at the of `<paragraph>Foo^</paragraph>`:\n\t\t//\n\t\t// <paragraph>Foo</paragraph><paragraph>Bar</paragraph> -->\n\t\t// <paragraph>Foo</paragraph>]<paragraph>Abc</paragraph>[<paragraph>Bar</paragraph> -->\n\t\t// <paragraph>Foo]Abc</paragraph>[<paragraph>Bar</paragraph> -->\n\t\t// <paragraph>Foo]Abc[</paragraph><paragraph>Bar</paragraph>\n\t\tif ( mergePosLeft.isEqual( this._affectedEnd! ) && this._firstNode === this._lastNode ) {\n\t\t\tthis._affectedEnd!.detach();\n\t\t\tthis._affectedEnd = LivePosition._createAt( mergePosLeft.nodeBefore!, 'end', 'toNext' );\n\t\t}\n\n\t\tthis.position = livePosition.toPosition();\n\t\tlivePosition.detach();\n\n\t\t// After merge elements that were marked by _insert() to be filtered might be gone so\n\t\t// we need to mark the new container.\n\t\tthis._filterAttributesOf.push( this.position.parent as any );\n\n\t\tmergePosLeft.detach();\n\t}\n\n\t/**\n\t * Merges the next sibling of the last node if it should be merged.\n\t *\n\t * After the content was inserted we may try to merge it with its siblings.\n\t * This should happen only if the selection was in those elements initially.\n\t */\n\tprivate _mergeOnRight(): void {\n\t\tconst node = this._lastNode;\n\n\t\tif ( !( node instanceof Element ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !this._canMergeRight( node ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst mergePosRight = LivePosition._createAfter( node );\n\t\tmergePosRight.stickiness = 'toNext';\n\n\t\t/* istanbul ignore if -- @preserve */\n\t\tif ( !this.position.isEqual( mergePosRight ) ) {\n\t\t\t// Algorithm's correctness check. We should never end up here but it's good to know that we did.\n\t\t\t// At this point the insertion position should be after the node we'll merge. If it isn't,\n\t\t\t// it should need to be secured as in the left merge case.\n\t\t\t/**\n\t\t\t * An internal error occurred when merging inserted content with its siblings.\n\t\t\t * The insertion position should equal the merge position.\n\t\t\t *\n\t\t\t * If you encountered this error, report it back to the CKEditor 5 team\n\t\t\t * with as many details as possible regarding the content being inserted and the insertion position.\n\t\t\t *\n\t\t\t * @error insertcontent-invalid-insertion-position\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'insertcontent-invalid-insertion-position', this );\n\t\t}\n\n\t\t// Move the position to the previous node, so it isn't moved to the graveyard on merge.\n\t\t// <p>x</p>[]<p>y</p> => <p>x[]</p><p>y</p>\n\t\tthis.position = Position._createAt( mergePosRight.nodeBefore!, 'end' );\n\n\t\t// Explanation of setting position stickiness to `'toPrevious'`:\n\t\t// OK: <p>xx[]</p> + <p>yy</p> => <p>xx[]yy</p> (when sticks to previous)\n\t\t// NOK: <p>xx[]</p> + <p>yy</p> => <p>xxyy[]</p> (when sticks to next)\n\t\tconst livePosition = LivePosition.fromPosition( this.position, 'toPrevious' );\n\n\t\t// See comment in `_mergeOnLeft()` on moving `_affectedStart`.\n\t\tif ( this._affectedEnd!.isEqual( mergePosRight ) ) {\n\t\t\tthis._affectedEnd!.detach();\n\t\t\tthis._affectedEnd = LivePosition._createAt( mergePosRight.nodeBefore!, 'end', 'toNext' );\n\t\t}\n\n\t\t// We need to update the references to the first and last nodes if they will be merged into the previous sibling node\n\t\t// because the reference would point to the removed node.\n\t\t//\n\t\t// <p>A^A</p> + <p>X</p>\n\t\t//\n\t\t// <p>A</p>^<p>A</p>\n\t\t// <p>A</p><p>X</p><p>A</p>\n\t\t// <p>AX</p><p>A</p>\n\t\t// <p>AXA</p>\n\t\tif ( this._firstNode === this._lastNode ) {\n\t\t\tthis._firstNode = mergePosRight.nodeBefore;\n\t\t\tthis._lastNode = mergePosRight.nodeBefore;\n\t\t}\n\n\t\tthis.writer.merge( mergePosRight );\n\n\t\t// See comment in `_mergeOnLeft()` on moving `_affectedStart`.\n\t\tif ( mergePosRight.getShiftedBy( -1 ).isEqual( this._affectedStart! ) && this._firstNode === this._lastNode ) {\n\t\t\tthis._affectedStart!.detach();\n\t\t\tthis._affectedStart = LivePosition._createAt( mergePosRight.nodeBefore!, 0, 'toPrevious' );\n\t\t}\n\n\t\tthis.position = livePosition.toPosition();\n\t\tlivePosition.detach();\n\n\t\t// After merge elements that were marked by _insert() to be filtered might be gone so\n\t\t// we need to mark the new container.\n\t\tthis._filterAttributesOf.push( this.position.parent as any );\n\n\t\tmergePosRight.detach();\n\t}\n\n\t/**\n\t * Checks whether specified node can be merged with previous sibling element.\n\t *\n\t * @param node The node which could potentially be merged.\n\t */\n\tprivate _canMergeLeft( node: Element ): boolean {\n\t\tconst previousSibling = node.previousSibling;\n\n\t\treturn ( previousSibling instanceof Element ) &&\n\t\t\tthis.canMergeWith.has( previousSibling ) &&\n\t\t\tthis.model.schema.checkMerge( previousSibling, node );\n\t}\n\n\t/**\n\t * Checks whether specified node can be merged with next sibling element.\n\t *\n\t * @param node The node which could potentially be merged.\n\t */\n\tprivate _canMergeRight( node: Element ): boolean {\n\t\tconst nextSibling = node.nextSibling;\n\n\t\treturn ( nextSibling instanceof Element ) &&\n\t\t\tthis.canMergeWith.has( nextSibling ) &&\n\t\t\tthis.model.schema.checkMerge( node, nextSibling );\n\t}\n\n\t/**\n\t * Tries wrapping the node in a new paragraph and inserting it this way.\n\t *\n\t * @param node The node which needs to be autoparagraphed.\n\t */\n\tprivate _tryAutoparagraphing( node: Node ): void {\n\t\tconst paragraph = this.writer.createElement( 'paragraph' );\n\n\t\t// Do not autoparagraph if the paragraph won't be allowed there,\n\t\t// cause that would lead to an infinite loop. The paragraph would be rejected in\n\t\t// the next _handleNode() call and we'd be here again.\n\t\tif ( this._getAllowedIn( this.position.parent as any, paragraph ) && this.schema.checkChild( paragraph, node ) ) {\n\t\t\tparagraph._appendChild( node );\n\t\t\tthis._handleNode( paragraph );\n\t\t}\n\t}\n\n\t/**\n\t * Checks if a node can be inserted in the given position or it would be accepted if a paragraph would be inserted.\n\t * It also handles inserting the paragraph.\n\t *\n\t * @returns Whether an allowed position was found.\n\t * `false` is returned if the node isn't allowed at the current position or in auto paragraph, `true` if was.\n\t */\n\tprivate _checkAndAutoParagraphToAllowedPosition( node: Node ): boolean {\n\t\tif ( this.schema.checkChild( this.position.parent as any, node ) ) {\n\t\t\treturn true;\n\t\t}\n\n\t\t// Do not auto paragraph if the paragraph won't be allowed there,\n\t\t// cause that would lead to an infinite loop. The paragraph would be rejected in\n\t\t// the next _handleNode() call and we'd be here again.\n\t\tif ( !this.schema.checkChild( this.position.parent as any, 'paragraph' ) || !this.schema.checkChild( 'paragraph', node ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Insert nodes collected in temporary DocumentFragment if the position parent needs change to process further nodes.\n\t\tthis._insertPartialFragment();\n\n\t\t// Insert a paragraph and move insertion position to it.\n\t\tconst paragraph = this.writer.createElement( 'paragraph' );\n\n\t\tthis.writer.insert( paragraph, this.position );\n\t\tthis._setAffectedBoundaries( this.position );\n\n\t\tthis._lastAutoParagraph = paragraph;\n\t\tthis.position = this.writer.createPositionAt( paragraph, 0 );\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * @returns Whether an allowed position was found.\n\t * `false` is returned if the node isn't allowed at any position up in the tree, `true` if was.\n\t */\n\tprivate _checkAndSplitToAllowedPosition( node: Node ): boolean {\n\t\tconst allowedIn = this._getAllowedIn( this.position.parent as any, node );\n\n\t\tif ( !allowedIn ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Insert nodes collected in temporary DocumentFragment if the position parent needs change to process further nodes.\n\t\tif ( allowedIn != this.position.parent ) {\n\t\t\tthis._insertPartialFragment();\n\t\t}\n\n\t\twhile ( allowedIn != this.position.parent ) {\n\t\t\tif ( this.position.isAtStart ) {\n\t\t\t\t// If insertion position is at the beginning of the parent, move it out instead of splitting.\n\t\t\t\t// <p>^Foo</p> -> ^<p>Foo</p>\n\t\t\t\tconst parent: Element = this.position.parent as Element;\n\n\t\t\t\tthis.position = this.writer.createPositionBefore( parent );\n\n\t\t\t\t// Special case parent is empty (<p>^</p>).\n\t\t\t\t//\n\t\t\t\t// 1. parent.isEmpty\n\t\t\t\t// We can remove the element after moving insertion position out of it.\n\t\t\t\t//\n\t\t\t\t// 2. parent.parent === allowedIn\n\t\t\t\t// However parent should remain in place when allowed element is above limit element in document tree.\n\t\t\t\t// For example there shouldn't be allowed to remove empty paragraph from tableCell, when is pasted\n\t\t\t\t// content allowed in $root.\n\t\t\t\tif ( parent.isEmpty && parent.parent === allowedIn ) {\n\t\t\t\t\tthis.writer.remove( parent );\n\t\t\t\t}\n\t\t\t} else if ( this.position.isAtEnd ) {\n\t\t\t\t// If insertion position is at the end of the parent, move it out instead of splitting.\n\t\t\t\t// <p>Foo^</p> -> <p>Foo</p>^\n\t\t\t\tthis.position = this.writer.createPositionAfter( this.position.parent as Element );\n\t\t\t} else {\n\t\t\t\tconst tempPos = this.writer.createPositionAfter( this.position.parent as Element );\n\n\t\t\t\tthis._setAffectedBoundaries( this.position );\n\t\t\t\tthis.writer.split( this.position );\n\n\t\t\t\tthis.position = tempPos;\n\n\t\t\t\tthis.canMergeWith.add( this.position.nodeAfter );\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Gets the element in which the given node is allowed. It checks the passed element and all its ancestors.\n\t *\n\t * @param contextElement The element in which context the node should be checked.\n\t * @param childNode The node to check.\n\t */\n\tprivate _getAllowedIn( contextElement: Element, childNode: Node ): Element | null {\n\t\tif ( this.schema.checkChild( contextElement, childNode ) ) {\n\t\t\treturn contextElement;\n\t\t}\n\n\t\t// If the child wasn't allowed in the context element and the element is a limit there's no point in\n\t\t// checking any further towards the root. This is it: the limit is unsplittable and there's nothing\n\t\t// we can do about it. Without this check, the algorithm will analyze parent of the limit and may create\n\t\t// an illusion of the child being allowed. There's no way to insert it down there, though. It results in\n\t\t// infinite loops.\n\t\tif ( this.schema.isLimit( contextElement ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn this._getAllowedIn( contextElement.parent as any, childNode );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/utils/findoptimalinsertionrange\n */\n\nimport { first } from '@ckeditor/ckeditor5-utils';\n\nimport type DocumentSelection from '../documentselection';\nimport type Model from '../model';\nimport type Range from '../range';\nimport type Selection from '../selection';\n\n/**\n * Returns a model range which is optimal (in terms of UX) for inserting a widget block.\n *\n * For instance, if a selection is in the middle of a paragraph, the collapsed range before this paragraph\n * will be returned so that it is not split. If the selection is at the end of a paragraph,\n * the collapsed range after this paragraph will be returned.\n *\n * Note: If the selection is placed in an empty block, the range in that block will be returned. If that range\n * is then passed to {@link module:engine/model/model~Model#insertContent}, the block will be fully replaced\n * by the inserted widget block.\n *\n * **Note:** Use {@link module:widget/utils#findOptimalInsertionRange} instead of this function outside engine.\n * This function is only exposed to be used by {@link module:widget/utils#findOptimalInsertionRange findOptimalInsertionRange()}\n * in the `widget` package and inside the `engine` package.\n *\n * @param selection The selection based on which the insertion position should be calculated.\n * @param model Model instance.\n * @param place The place where to look for optimal insertion range.\n * The default `auto` value will determine itself the best position for insertion.\n * The `before` value will try to find a position before selection.\n * The `after` value will try to find a position after selection.\n * @returns The optimal range.\n */\nexport function findOptimalInsertionRange(\n\tselection: Selection | DocumentSelection,\n\tmodel: Model,\n\tplace: 'auto' | 'before' | 'after' = 'auto'\n): Range {\n\tconst selectedElement = selection.getSelectedElement();\n\n\tif ( selectedElement && model.schema.isObject( selectedElement ) && !model.schema.isInline( selectedElement ) ) {\n\t\tif ( place == 'before' || place == 'after' ) {\n\t\t\treturn model.createRange( model.createPositionAt( selectedElement, place ) );\n\t\t}\n\n\t\treturn model.createRangeOn( selectedElement );\n\t}\n\n\tconst firstBlock = first( selection.getSelectedBlocks() );\n\n\t// There are no block elements within ancestors (in the current limit element).\n\tif ( !firstBlock ) {\n\t\treturn model.createRange( selection.focus! );\n\t}\n\n\t// If inserting into an empty block return position in that block. It will get\n\t// replaced with the image by insertContent(). #42.\n\tif ( firstBlock.isEmpty ) {\n\t\treturn model.createRange( model.createPositionAt( firstBlock, 0 ) );\n\t}\n\n\tconst positionAfter = model.createPositionAfter( firstBlock );\n\n\t// If selection is at the end of the block - return position after the block.\n\tif ( selection.focus!.isTouching( positionAfter ) ) {\n\t\treturn model.createRange( positionAfter );\n\t}\n\n\t// Otherwise, return position before the block.\n\treturn model.createRange( model.createPositionBefore( firstBlock ) );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/utils/insertobject\n */\n\nimport { findOptimalInsertionRange } from './findoptimalinsertionrange';\nimport type DocumentSelection from '../documentselection';\nimport type Selection from '../selection';\n\nimport type Element from '../element';\nimport type Model from '../model';\nimport type Range from '../range';\nimport type Writer from '../writer';\n\nimport { CKEditorError, first } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Inserts an {@glink framework/deep-dive/schema#object-elements object element} at a specific position in the editor content.\n *\n * **Note:** Use {@link module:engine/model/model~Model#insertObject} instead of this function.\n * This function is only exposed to be reusable in algorithms which change the {@link module:engine/model/model~Model#insertObject}\n * method's behavior.\n *\n * **Note**: For more documentation and examples, see {@link module:engine/model/model~Model#insertObject}.\n *\n * @param model The model in context of which the insertion should be performed.\n * @param object An object to be inserted into the model document.\n * @param selectable A selectable where the content should be inserted. If not specified, the current\n * {@link module:engine/model/document~Document#selection document selection} will be used instead.\n * @param placeOrOffset Specifies the exact place or offset for the insertion to take place, relative to `selectable`.\n * @param options Additional options.\n * @param options.findOptimalPosition An option that, when set, adjusts the insertion position (relative to\n * `selectable` and `placeOrOffset`) so that the content of `selectable` is not split upon insertion (a.k.a. non-destructive insertion).\n * * When `'auto'`, the algorithm will decide whether to insert the object before or after `selectable` to avoid content splitting.\n * * When `'before'`, the closest position before `selectable` will be used that will not result in content splitting.\n * * When `'after'`, the closest position after `selectable` will be used that will not result in content splitting.\n *\n * Note that this option works only for block objects. Inline objects are inserted into text and do not split blocks.\n * @param options.setSelection An option that, when set, moves the\n * {@link module:engine/model/document~Document#selection document selection} after inserting the object.\n * * When `'on'`, the document selection will be set on the inserted object.\n * * When `'after'`, the document selection will move to the closest text node after the inserted object. If there is no\n * such text node, a paragraph will be created and the document selection will be moved inside it.\n * @returns A range which contains all the performed changes. This is a range that, if removed,\n * would return the model to the state before the insertion. If no changes were preformed by `insertObject()`, returns a range collapsed\n * at the insertion position.\n */\nexport default function insertObject(\n\tmodel: Model,\n\tobject: Element,\n\tselectable?: Selection | DocumentSelection | null,\n\toptions: {\n\t\tfindOptimalPosition?: 'auto' | 'before' | 'after';\n\t\tsetSelection?: 'on' | 'after';\n\t} = {}\n): Range {\n\tif ( !model.schema.isObject( object ) ) {\n\t\t/**\n\t\t * Tried to insert an element with {@link module:engine/model/utils/insertobject insertObject()} function\n\t\t * that is not defined as an object in schema.\n\t\t * See {@link module:engine/model/schema~SchemaItemDefinition#isObject `SchemaItemDefinition`}.\n\t\t * If you want to insert content that is not an object you might want to use\n\t\t * {@link module:engine/model/utils/insertcontent insertContent()} function.\n\t\t * @error insertobject-element-not-an-object\n\t\t */\n\t\tthrow new CKEditorError( 'insertobject-element-not-an-object', model, { object } );\n\t}\n\n\t// Normalize selectable to a selection instance.\n\tconst originalSelection: Selection | DocumentSelection = selectable ? selectable : model.document.selection;\n\n\t// Adjust the insertion selection.\n\tlet insertionSelection = originalSelection;\n\n\tif ( options.findOptimalPosition && model.schema.isBlock( object ) ) {\n\t\tinsertionSelection = model.createSelection( findOptimalInsertionRange( originalSelection, model, options.findOptimalPosition ) );\n\t}\n\n\t// Collect attributes to be copied on the inserted object.\n\tconst firstSelectedBlock = first( originalSelection.getSelectedBlocks() );\n\tconst attributesToCopy = {};\n\n\tif ( firstSelectedBlock ) {\n\t\tObject.assign( attributesToCopy, model.schema.getAttributesWithProperty( firstSelectedBlock, 'copyOnReplace', true ) );\n\t}\n\n\treturn model.change( writer => {\n\t\t// Remove the selected content to find out what the parent of the inserted object would be.\n\t\t// It would be removed inside model.insertContent() anyway.\n\t\tif ( !insertionSelection.isCollapsed ) {\n\t\t\tmodel.deleteContent( insertionSelection, { doNotAutoparagraph: true } );\n\t\t}\n\n\t\tlet elementToInsert = object;\n\t\tconst insertionPositionParent = insertionSelection.anchor!.parent;\n\n\t\t// Autoparagraphing of an inline objects.\n\t\tif (\n\t\t\t!model.schema.checkChild( insertionPositionParent as any, object ) &&\n\t\t\tmodel.schema.checkChild( insertionPositionParent as any, 'paragraph' ) &&\n\t\t\tmodel.schema.checkChild( 'paragraph', object )\n\t\t) {\n\t\t\telementToInsert = writer.createElement( 'paragraph' );\n\n\t\t\twriter.insert( object, elementToInsert );\n\t\t}\n\n\t\t// Apply attributes that are allowed on the inserted object (or paragraph if autoparagraphed).\n\t\tmodel.schema.setAllowedAttributes( elementToInsert, attributesToCopy, writer );\n\n\t\t// Insert the prepared content at the optionally adjusted selection.\n\t\tconst affectedRange = model.insertContent( elementToInsert, insertionSelection );\n\n\t\t// Nothing got inserted.\n\t\tif ( affectedRange.isCollapsed ) {\n\t\t\treturn affectedRange;\n\t\t}\n\n\t\tif ( options.setSelection ) {\n\t\t\tupdateSelection( writer, object, options.setSelection, attributesToCopy );\n\t\t}\n\n\t\treturn affectedRange;\n\t} );\n}\n\n/**\n * Updates document selection based on given `place` parameter in relation to `contextElement` element.\n *\n * @param writer An instance of the model writer.\n * @param contextElement An element to set the attributes on.\n * @param place The place where selection should be set in relation to the `contextElement` element.\n * Value `on` will set selection on the passed `contextElement`. Value `after` will set selection after `contextElement`.\n * @param attributes Attributes keys and values to set on a paragraph that this function can create when\n * `place` parameter is equal to `after` but there is no element with `$text` node to set selection in.\n */\nfunction updateSelection(\n\twriter: Writer,\n\tcontextElement: Element,\n\tplace: 'after' | 'on',\n\tparagraphAttributes: Record<string, unknown>\n) {\n\tconst model = writer.model;\n\n\tif ( place == 'on' ) {\n\t\twriter.setSelection( contextElement, 'on' );\n\n\t\treturn;\n\t}\n\n\tif ( place != 'after' ) {\n\t\t/**\n\t\t * The unsupported `options.setSelection` parameter was passed\n\t\t * to the {@link module:engine/model/utils/insertobject insertObject()} function.\n\t\t * Check the {@link module:engine/model/utils/insertobject insertObject()} API documentation for allowed\n\t\t * `options.setSelection` parameter values.\n\t\t *\n\t\t * @error insertobject-invalid-place-parameter-value\n\t\t */\n\t\tthrow new CKEditorError( 'insertobject-invalid-place-parameter-value', model );\n\t}\n\n\tlet nextElement = contextElement.nextSibling;\n\n\tif ( model.schema.isInline( contextElement ) ) {\n\t\twriter.setSelection( contextElement, 'after' );\n\n\t\treturn;\n\t}\n\n\t// Check whether an element next to the inserted element is defined and can contain a text.\n\tconst canSetSelection = nextElement && model.schema.checkChild( nextElement, '$text' );\n\n\t// If the element is missing, but a paragraph could be inserted next to the element, let's add it.\n\tif ( !canSetSelection && model.schema.checkChild( contextElement.parent as any, 'paragraph' ) ) {\n\t\tnextElement = writer.createElement( 'paragraph' );\n\n\t\tmodel.schema.setAllowedAttributes( nextElement, paragraphAttributes, writer );\n\t\tmodel.insertContent( nextElement, writer.createPositionAfter( contextElement ) );\n\t}\n\n\t// Put the selection inside the element, at the beginning.\n\tif ( nextElement ) {\n\t\twriter.setSelection( nextElement, 0 );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/utils/modifyselection\n */\n\nimport DocumentSelection from '../documentselection';\nimport Position from '../position';\nimport Range from '../range';\nimport TreeWalker, { type TreeWalkerValue } from '../treewalker';\n\nimport type Model from '../model';\nimport type Schema from '../schema';\nimport type Selection from '../selection';\nimport type Text from '../text';\nimport type Node from '../node';\n\nimport { isInsideSurrogatePair, isInsideCombinedSymbol, isInsideEmojiSequence } from '@ckeditor/ckeditor5-utils';\n\nconst wordBoundaryCharacters = ' ,.?!:;\"-()';\n\n/**\n * Modifies the selection. Currently, the supported modifications are:\n *\n * * Extending. The selection focus is moved in the specified `options.direction` with a step specified in `options.unit`.\n * Possible values for `unit` are:\n * * `'character'` (default) - moves selection by one user-perceived character. In most cases this means moving by one\n * character in `String` sense. However, unicode also defines \"combing marks\". These are special symbols, that combines\n * with a symbol before it (\"base character\") to create one user-perceived character. For example, `q̣̇` is a normal\n * letter `q` with two \"combining marks\": upper dot (`Ux0307`) and lower dot (`Ux0323`). For most actions, i.e. extending\n * selection by one position, it is correct to include both \"base character\" and all of it's \"combining marks\". That is\n * why `'character'` value is most natural and common method of modifying selection.\n * * `'codePoint'` - moves selection by one unicode code point. In contrary to, `'character'` unit, this will insert\n * selection between \"base character\" and \"combining mark\", because \"combining marks\" have their own unicode code points.\n * However, for technical reasons, unicode code points with values above `UxFFFF` are represented in native `String` by\n * two characters, called \"surrogate pairs\". Halves of \"surrogate pairs\" have a meaning only when placed next to each other.\n * For example `𨭎` is represented in `String` by `\\uD862\\uDF4E`. Both `\\uD862` and `\\uDF4E` do not have any meaning\n * outside the pair (are rendered as ? when alone). Position between them would be incorrect. In this case, selection\n * extension will include whole \"surrogate pair\".\n * * `'word'` - moves selection by a whole word.\n *\n * **Note:** if you extend a forward selection in a backward direction you will in fact shrink it.\n *\n * **Note:** Use {@link module:engine/model/model~Model#modifySelection} instead of this function.\n * This function is only exposed to be reusable in algorithms\n * which change the {@link module:engine/model/model~Model#modifySelection}\n * method's behavior.\n *\n * @param model The model in context of which the selection modification should be performed.\n * @param selection The selection to modify.\n * @param options.direction The direction in which the selection should be modified. Default 'forward'.\n * @param options.unit The unit by which selection should be modified. Default 'character'.\n * @param options.treatEmojiAsSingleUnit Whether multi-characer emoji sequences should be handled as single unit.\n */\nexport default function modifySelection(\n\tmodel: Model,\n\tselection: Selection | DocumentSelection,\n\toptions: {\n\t\tdirection?: 'forward' | 'backward';\n\t\tunit?: 'character' | 'codePoint' | 'word';\n\t\ttreatEmojiAsSingleUnit?: boolean;\n\t} = {}\n): void {\n\tconst schema = model.schema;\n\tconst isForward = options.direction != 'backward';\n\tconst unit = options.unit ? options.unit : 'character';\n\tconst treatEmojiAsSingleUnit = !!options.treatEmojiAsSingleUnit;\n\n\tconst focus = selection.focus!;\n\n\tconst walker = new TreeWalker( {\n\t\tboundaries: getSearchRange( focus, isForward ),\n\t\tsingleCharacters: true,\n\t\tdirection: isForward ? 'forward' : 'backward'\n\t} );\n\n\tconst data = { walker, schema, isForward, unit, treatEmojiAsSingleUnit };\n\n\tlet next;\n\n\twhile ( ( next = walker.next() ) ) {\n\t\tif ( next.done ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst position = tryExtendingTo( data, next.value );\n\n\t\tif ( position ) {\n\t\t\tif ( selection instanceof DocumentSelection ) {\n\t\t\t\tmodel.change( writer => {\n\t\t\t\t\twriter.setSelectionFocus( position );\n\t\t\t\t} );\n\t\t\t} else {\n\t\t\t\tselection.setFocus( position );\n\t\t\t}\n\n\t\t\treturn;\n\t\t}\n\t}\n}\n\n/**\n * Checks whether the selection can be extended to the the walker's next value (next position).\n */\nfunction tryExtendingTo(\n\tdata: {\n\t\twalker: TreeWalker;\n\t\tschema: Schema;\n\t\tisForward: boolean;\n\t\tunit: 'character' | 'codePoint' | 'word';\n\t\ttreatEmojiAsSingleUnit: boolean;\n\t},\n\tvalue: TreeWalkerValue\n): Position | undefined {\n\tconst { isForward, walker, unit, schema, treatEmojiAsSingleUnit } = data;\n\tconst { type, item, nextPosition } = value;\n\n\t// If found text, we can certainly put the focus in it. Let's just find a correct position\n\t// based on the unit.\n\tif ( type == 'text' ) {\n\t\tif ( data.unit === 'word' ) {\n\t\t\treturn getCorrectWordBreakPosition( walker, isForward );\n\t\t}\n\n\t\treturn getCorrectPosition( walker, unit, treatEmojiAsSingleUnit );\n\t}\n\n\t// Entering an element.\n\tif ( type == ( isForward ? 'elementStart' : 'elementEnd' ) ) {\n\t\t// If it's a selectable, we can select it now.\n\t\tif ( schema.isSelectable( item ) ) {\n\t\t\treturn Position._createAt( item, isForward ? 'after' : 'before' );\n\t\t}\n\n\t\t// If text allowed on this position, extend to this place.\n\t\tif ( schema.checkChild( nextPosition, '$text' ) ) {\n\t\t\treturn nextPosition;\n\t\t}\n\t}\n\t// Leaving an element.\n\telse {\n\t\t// If leaving a limit element, stop.\n\t\tif ( schema.isLimit( item ) ) {\n\t\t\t// NOTE: Fast-forward the walker until the end.\n\t\t\twalker.skip( () => true );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// If text allowed on this position, extend to this place.\n\t\tif ( schema.checkChild( nextPosition, '$text' ) ) {\n\t\t\treturn nextPosition;\n\t\t}\n\t}\n}\n\n/**\n * Finds a correct position by walking in a text node and checking whether selection can be extended to given position\n * or should be extended further.\n */\nfunction getCorrectPosition(\n\twalker: TreeWalker,\n\tunit: 'character' | 'codePoint' | 'word',\n\ttreatEmojiAsSingleUnit: boolean\n): Position {\n\tconst textNode = walker.position.textNode;\n\n\tif ( textNode ) {\n\t\tconst data = textNode.data;\n\t\tlet offset = walker.position.offset - textNode.startOffset!;\n\n\t\twhile (\n\t\t\tisInsideSurrogatePair( data, offset ) ||\n\t\t\t( unit == 'character' && isInsideCombinedSymbol( data, offset ) ) ||\n\t\t\t( treatEmojiAsSingleUnit && isInsideEmojiSequence( data, offset ) )\n\t\t) {\n\t\t\twalker.next();\n\n\t\t\toffset = walker.position.offset - textNode.startOffset!;\n\t\t}\n\t}\n\n\treturn walker.position;\n}\n\n/**\n * Finds a correct position of a word break by walking in a text node and checking whether selection can be extended to given position\n * or should be extended further.\n */\nfunction getCorrectWordBreakPosition( walker: TreeWalker, isForward: boolean ): Position {\n\tlet textNode: Node | null = walker.position.textNode;\n\n\tif ( !textNode ) {\n\t\ttextNode = isForward ? walker.position.nodeAfter : walker.position.nodeBefore;\n\t}\n\n\twhile ( textNode && textNode.is( '$text' ) ) {\n\t\tconst offset = walker.position.offset - textNode.startOffset!;\n\n\t\t// Check of adjacent text nodes with different attributes (like BOLD).\n\t\t// Example : 'foofoo []bar<$text bold=\"true\">bar</$text> bazbaz'\n\t\t// should expand to : 'foofoo [bar<$text bold=\"true\">bar</$text>] bazbaz'.\n\t\tif ( isAtNodeBoundary( textNode, offset, isForward ) ) {\n\t\t\ttextNode = isForward ? walker.position.nodeAfter : walker.position.nodeBefore;\n\t\t}\n\t\t// Check if this is a word boundary.\n\t\telse if ( isAtWordBoundary( textNode.data, offset, isForward ) ) {\n\t\t\tbreak;\n\t\t}\n\t\t// Maybe one more character.\n\t\telse {\n\t\t\twalker.next();\n\t\t}\n\t}\n\n\treturn walker.position;\n}\n\nfunction getSearchRange( start: Position, isForward: boolean ) {\n\tconst root = start.root;\n\tconst searchEnd = Position._createAt( root, isForward ? 'end' : 0 );\n\n\tif ( isForward ) {\n\t\treturn new Range( start, searchEnd );\n\t} else {\n\t\treturn new Range( searchEnd, start );\n\t}\n}\n\n/**\n * Checks if selection is on word boundary.\n */\nfunction isAtWordBoundary( data: string, offset: number, isForward: boolean ) {\n\t// The offset to check depends on direction.\n\tconst offsetToCheck = offset + ( isForward ? 0 : -1 );\n\n\treturn wordBoundaryCharacters.includes( data.charAt( offsetToCheck ) );\n}\n\n/**\n * Checks if selection is on node boundary.\n */\nfunction isAtNodeBoundary( textNode: Text, offset: number, isForward: boolean ) {\n\treturn offset === ( isForward ? textNode.offsetSize : 0 );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/model/model\n */\n\nimport Batch, { type BatchType } from './batch';\nimport Document from './document';\nimport MarkerCollection from './markercollection';\nimport ModelPosition, { type PositionOffset, type PositionStickiness } from './position';\nimport ModelRange from './range';\nimport ModelSelection, { type PlaceOrOffset, type Selectable } from './selection';\nimport OperationFactory from './operation/operationfactory';\nimport DocumentSelection from './documentselection';\nimport Schema from './schema';\nimport Writer from './writer';\nimport Node from './node';\n\nimport { autoParagraphEmptyRoots } from './utils/autoparagraphing';\nimport { injectSelectionPostFixer } from './utils/selection-post-fixer';\nimport deleteContent from './utils/deletecontent';\nimport getSelectedContent from './utils/getselectedcontent';\nimport insertContent from './utils/insertcontent';\nimport insertObject from './utils/insertobject';\nimport modifySelection from './utils/modifyselection';\n\nimport type ModelDocumentFragment from './documentfragment';\nimport type Item from './item';\nimport type ModelElement from './element';\nimport type Operation from './operation/operation';\n\nimport {\n\tCKEditorError,\n\tObservableMixin,\n\ttype DecoratedMethodEvent\n} from '@ckeditor/ckeditor5-utils';\n\n// @if CK_DEBUG_ENGINE // const { dumpTrees, initDocumentDumping } = require( '../dev-utils/utils' );\n// @if CK_DEBUG_ENGINE // const { OperationReplayer } = require( '../dev-utils/operationreplayer' ).default;\n\n/**\n * Editor's data model. Read about the model in the\n * {@glink framework/architecture/editing-engine engine architecture} guide.\n */\nexport default class Model extends ObservableMixin() {\n\t/**\n\t * Model's marker collection.\n\t */\n\tpublic readonly markers: MarkerCollection;\n\n\t/**\n\t * Model's document.\n\t */\n\tpublic readonly document: Document;\n\n\t/**\n\t * Model's schema.\n\t */\n\tpublic readonly schema: Schema;\n\n\t/**\n\t * All callbacks added by {@link module:engine/model/model~Model#change} or\n\t * {@link module:engine/model/model~Model#enqueueChange} methods waiting to be executed.\n\t */\n\tprivate readonly _pendingChanges: Array<{ batch: Batch; callback: ( writer: Writer ) => any }>;\n\n\t/**\n\t * The last created and currently used writer instance.\n\t */\n\tprivate _currentWriter: Writer | null;\n\n\t// @if CK_DEBUG_ENGINE // private _operationLogs: Array<string>;\n\t// @if CK_DEBUG_ENGINE // private _appliedOperations: Array<Operation>;\n\n\tconstructor() {\n\t\tsuper();\n\n\t\tthis.markers = new MarkerCollection();\n\t\tthis.document = new Document( this );\n\t\tthis.schema = new Schema();\n\n\t\tthis._pendingChanges = [];\n\t\tthis._currentWriter = null;\n\n\t\t( [ 'deleteContent', 'modifySelection', 'getSelectedContent', 'applyOperation' ] as const )\n\t\t\t.forEach( methodName => this.decorate( methodName ) );\n\n\t\t// Adding operation validation with `highest` priority, so it is called before any other feature would like\n\t\t// to do anything with the operation. If the operation has incorrect parameters it should throw on the earliest occasion.\n\t\tthis.on<ModelApplyOperationEvent>( 'applyOperation', ( evt, args ) => {\n\t\t\tconst operation = args[ 0 ];\n\n\t\t\toperation._validate();\n\t\t}, { priority: 'highest' } );\n\n\t\t// Register some default abstract entities.\n\t\tthis.schema.register( '$root', {\n\t\t\tisLimit: true\n\t\t} );\n\n\t\tthis.schema.register( '$container', {\n\t\t\tallowIn: [ '$root', '$container' ]\n\t\t} );\n\n\t\tthis.schema.register( '$block', {\n\t\t\tallowIn: [ '$root', '$container' ],\n\t\t\tisBlock: true\n\t\t} );\n\n\t\tthis.schema.register( '$blockObject', {\n\t\t\tallowWhere: '$block',\n\t\t\tisBlock: true,\n\t\t\tisObject: true\n\t\t} );\n\n\t\tthis.schema.register( '$inlineObject', {\n\t\t\tallowWhere: '$text',\n\t\t\tallowAttributesOf: '$text',\n\t\t\tisInline: true,\n\t\t\tisObject: true\n\t\t} );\n\n\t\tthis.schema.register( '$text', {\n\t\t\tallowIn: '$block',\n\t\t\tisInline: true,\n\t\t\tisContent: true\n\t\t} );\n\n\t\tthis.schema.register( '$clipboardHolder', {\n\t\t\tallowContentOf: '$root',\n\t\t\tallowChildren: '$text',\n\t\t\tisLimit: true\n\t\t} );\n\n\t\tthis.schema.register( '$documentFragment', {\n\t\t\tallowContentOf: '$root',\n\t\t\tallowChildren: '$text',\n\t\t\tisLimit: true\n\t\t} );\n\n\t\t// An element needed by the `upcastElementToMarker` converter.\n\t\t// This element temporarily represents a marker boundary during the conversion process and is removed\n\t\t// at the end of the conversion. `UpcastDispatcher` or at least `Conversion` class looks like a\n\t\t// better place for this registration but both know nothing about `Schema`.\n\t\tthis.schema.register( '$marker' );\n\t\tthis.schema.addChildCheck( ( context, childDefinition ) => {\n\t\t\tif ( childDefinition.name === '$marker' ) {\n\t\t\t\treturn true;\n\t\t\t}\n\t\t} );\n\n\t\tinjectSelectionPostFixer( this );\n\n\t\t// Post-fixer which takes care of adding empty paragraph elements to the empty roots.\n\t\tthis.document.registerPostFixer( autoParagraphEmptyRoots );\n\n\t\t// The base implementation for \"decorated\" method with remapped arguments.\n\t\tthis.on<ModelInsertContentEvent>( 'insertContent', ( evt, [ content, selectable ] ) => {\n\t\t\tevt.return = insertContent( this, content, selectable );\n\t\t} );\n\n\t\t// The base implementation for \"decorated\" method with remapped arguments.\n\t\tthis.on<ModelInsertObjectEvent>( 'insertObject', ( evt, [ element, selection, options ] ) => {\n\t\t\tevt.return = insertObject( this, element, selection, options );\n\t\t} );\n\n\t\t// The base implementation for \"decorated\" method with remapped arguments.\n\t\tthis.on<ModelCanEditAtEvent>( 'canEditAt', evt => {\n\t\t\tconst canEditAt = !this.document.isReadOnly;\n\n\t\t\tevt.return = canEditAt;\n\n\t\t\tif ( !canEditAt ) {\n\t\t\t\t// Prevent further processing if the selection is at non-editable place.\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t} );\n\n\t\t// @if CK_DEBUG_ENGINE // initDocumentDumping( this.document );\n\t\t// @if CK_DEBUG_ENGINE // this.on( 'applyOperation', () => {\n\t\t// @if CK_DEBUG_ENGINE // \tdumpTrees( this.document, this.document.version );\n\t\t// @if CK_DEBUG_ENGINE // }, { priority: 'lowest' } );\n\n\t\t// @if CK_DEBUG_ENGINE // this._operationLogs = [];\n\t\t// @if CK_DEBUG_ENGINE // this._appliedOperations = [];\n\t}\n\n\t/**\n\t * The `change()` method is the primary way of changing the model. You should use it to modify all document nodes\n\t * (including detached nodes i.e. nodes not added to the {@link module:engine/model/model~Model#document model document}),\n\t * the {@link module:engine/model/document~Document#selection document's selection}, and\n\t * {@link module:engine/model/model~Model#markers model markers}.\n\t *\n\t * ```ts\n\t * model.change( writer => {\n\t * \twriter.insertText( 'foo', paragraph, 'end' );\n\t * } );\n\t * ```\n\t *\n\t * All changes inside the change block use the same {@link module:engine/model/batch~Batch} so they are combined\n\t * into a single undo step.\n\t *\n\t * ```ts\n\t * model.change( writer => {\n\t * \twriter.insertText( 'foo', paragraph, 'end' ); // foo.\n\t *\n\t * \tmodel.change( writer => {\n\t * \t\twriter.insertText( 'bar', paragraph, 'end' ); // foobar.\n\t * \t} );\n\t *\n\t * \twriter.insertText( 'bom', paragraph, 'end' ); // foobarbom.\n\t * } );\n\t * ```\n\t *\n\t * The callback of the `change()` block is executed synchronously.\n\t *\n\t * You can also return a value from the change block.\n\t *\n\t * ```ts\n\t * const img = model.change( writer => {\n\t * \treturn writer.createElement( 'img' );\n\t * } );\n\t * ```\n\t *\n\t * @see #enqueueChange\n\t * @typeParam TReturn The return type of the provided callback.\n\t * @param callback Callback function which may modify the model.\n\t */\n\tpublic change<TReturn>( callback: ( writer: Writer ) => TReturn ): TReturn {\n\t\ttry {\n\t\t\tif ( this._pendingChanges.length === 0 ) {\n\t\t\t\t// If this is the outermost block, create a new batch and start `_runPendingChanges` execution flow.\n\t\t\t\tthis._pendingChanges.push( { batch: new Batch(), callback } );\n\n\t\t\t\treturn this._runPendingChanges()[ 0 ];\n\t\t\t} else {\n\t\t\t\t// If this is not the outermost block, just execute the callback.\n\t\t\t\treturn callback( this._currentWriter! );\n\t\t\t}\n\t\t} catch ( err: any ) {\n\t\t\t// @if CK_DEBUG // throw err;\n\t\t\t/* istanbul ignore next -- @preserve */\n\t\t\tCKEditorError.rethrowUnexpectedError( err, this );\n\t\t}\n\t}\n\n\t/**\n\t * The `enqueueChange()` method performs similar task as the {@link #change `change()` method}, with two major differences.\n\t *\n\t * First, the callback of `enqueueChange()` is executed when all other enqueued changes are done. It might be executed\n\t * immediately if it is not nested in any other change block, but if it is nested in another (enqueue)change block,\n\t * it will be delayed and executed after the outermost block.\n\t *\n\t * ```ts\n\t * model.change( writer => {\n\t * \tconsole.log( 1 );\n\t *\n\t * \tmodel.enqueueChange( writer => {\n\t * \t\tconsole.log( 2 );\n\t * \t} );\n\t *\n\t * \tconsole.log( 3 );\n\t * } ); // Will log: 1, 3, 2.\n\t * ```\n\t *\n\t * In addition to that, the changes enqueued with `enqueueChange()` will be converted separately from the changes\n\t * done in the outer `change()` block.\n\t *\n\t * By default, a new batch with the default {@link module:engine/model/batch~Batch#constructor batch type} is created.\n\t * To define the {@link module:engine/model/batch~Batch} into which you want to add your changes,\n\t * use {@link #enqueueChange:CUSTOM_BATCH `enqueueChange( batchOrType, callback )`}.\n\t *\n\t * @label DEFAULT_BATCH\n\t * @param callback Callback function which may modify the model.\n\t */\n\tpublic enqueueChange(\n\t\tcallback: ( writer: Writer ) => unknown\n\t): void;\n\n\t/**\n\t * The `enqueueChange()` method performs similar task as the {@link #change `change()` method}, with two major differences.\n\t *\n\t * First, the callback of `enqueueChange()` is executed when all other enqueued changes are done. It might be executed\n\t * immediately if it is not nested in any other change block, but if it is nested in another (enqueue)change block,\n\t * it will be delayed and executed after the outermost block.\n\t *\n\t * ```ts\n\t * model.change( new Batch(), writer => {\n\t * \tconsole.log( 1 );\n\t *\n\t * \tmodel.enqueueChange( new Batch(), writer => {\n\t * \t\tconsole.log( 2 );\n\t * \t} );\n\t *\n\t * \tconsole.log( 3 );\n\t * } ); // Will log: 1, 3, 2.\n\t * ```\n\t *\n\t * In addition to that, the changes enqueued with `enqueueChange()` will be converted separately from the changes\n\t * done in the outer `change()` block.\n\t *\n\t * Second, it lets you define the {@link module:engine/model/batch~Batch} into which you want to add your changes.\n\t * If you want to use default {@link module:engine/model/batch~Batch#constructor batch type}, use\n\t * {@link #enqueueChange:DEFAULT_BATCH `enqueueChange( callback )`}.\n\t *\n\t * ```ts\n\t * model.enqueueChange( { isUndoable: false }, writer => {\n\t * \twriter.insertText( 'foo', paragraph, 'end' );\n\t * } );\n\t * ```\n\t *\n\t * When using the `enqueueChange()` block you can also add some changes to the batch you used before.\n\t *\n\t * ```ts\n\t * model.enqueueChange( batch, writer => {\n\t * \twriter.insertText( 'foo', paragraph, 'end' );\n\t * } );\n\t * ```\n\t *\n\t * In order to make a nested `enqueueChange()` create a single undo step together with the changes done in the outer `change()`\n\t * block, you can obtain the batch instance from the {@link module:engine/model/writer~Writer#batch writer} of the outer block.\n\t *\n\t * @label CUSTOM_BATCH\n\t * @param batchOrType A batch or a {@link module:engine/model/batch~Batch#constructor batch type} that should be used in the callback.\n\t * If not defined, a new batch with the default type will be created.\n\t * @param callback Callback function which may modify the model.\n\t */\n\tpublic enqueueChange(\n\t\tbatchOrType: Batch | BatchType | undefined,\n\t\tcallback: ( writer: Writer ) => unknown\n\t): void;\n\n\tpublic enqueueChange(\n\t\tbatchOrType: Batch | BatchType | ( ( writer: Writer ) => unknown ) | undefined,\n\t\tcallback?: ( writer: Writer ) => unknown\n\t): void {\n\t\ttry {\n\t\t\tif ( !batchOrType ) {\n\t\t\t\tbatchOrType = new Batch();\n\t\t\t} else if ( typeof batchOrType === 'function' ) {\n\t\t\t\tcallback = batchOrType;\n\t\t\t\tbatchOrType = new Batch();\n\t\t\t} else if ( !( batchOrType instanceof Batch ) ) {\n\t\t\t\tbatchOrType = new Batch( batchOrType );\n\t\t\t}\n\n\t\t\tthis._pendingChanges.push( { batch: batchOrType, callback } as any );\n\n\t\t\tif ( this._pendingChanges.length == 1 ) {\n\t\t\t\tthis._runPendingChanges();\n\t\t\t}\n\t\t} catch ( err: any ) {\n\t\t\t// @if CK_DEBUG // throw err;\n\t\t\t/* istanbul ignore next -- @preserve */\n\t\t\tCKEditorError.rethrowUnexpectedError( err, this );\n\t\t}\n\t}\n\n\t/**\n\t * {@link module:utils/observablemixin~Observable#decorate Decorated} function for applying\n\t * {@link module:engine/model/operation/operation~Operation operations} to the model.\n\t *\n\t * This is a low-level way of changing the model. It is exposed for very specific use cases (like the undo feature).\n\t * Normally, to modify the model, you will want to use {@link module:engine/model/writer~Writer `Writer`}.\n\t * See also {@glink framework/architecture/editing-engine#changing-the-model Changing the model} section\n\t * of the {@glink framework/architecture/editing-engine Editing architecture} guide.\n\t *\n\t * @param operation The operation to apply.\n\t */\n\tpublic applyOperation( operation: Operation ): void {\n\t\t// @if CK_DEBUG_ENGINE // console.log( 'Applying ' + operation );\n\n\t\t// @if CK_DEBUG_ENGINE // this._operationLogs.push( JSON.stringify( operation ) );\n\t\t// @if CK_DEBUG_ENGINE // this._appliedOperations.push( operation );\n\n\t\toperation._execute();\n\t}\n\n\t// @if CK_DEBUG_ENGINE // public getAppliedOperation(): string {\n\t// @if CK_DEBUG_ENGINE // \tif ( !this._appliedOperations ) {\n\t// @if CK_DEBUG_ENGINE // \t\treturn '';\n\t// @if CK_DEBUG_ENGINE // \t}\n\n\t// @if CK_DEBUG_ENGINE // \treturn this._appliedOperations.map( operation => JSON.stringify( operation ) ).join( '-------' );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t// @if CK_DEBUG_ENGINE // public createReplayer( stringifiedOperations: string ): typeof OperationReplayer {\n\t// @if CK_DEBUG_ENGINE // \treturn new OperationReplayer( this, '-------', stringifiedOperations );\n\t// @if CK_DEBUG_ENGINE // }\n\n\t/**\n\t * Inserts content at the position in the editor specified by the selection, as one would expect the paste\n\t * functionality to work.\n\t *\n\t * **Note**: If you want to insert an {@glink framework/deep-dive/schema#object-elements object element}\n\t * (e.g. a {@link module:widget/utils~toWidget widget}), see {@link #insertObject} instead.\n\t *\n\t * This is a high-level method. It takes the {@link #schema schema} into consideration when inserting\n\t * the content, clears the given selection's content before inserting nodes and moves the selection\n\t * to its target position at the end of the process.\n\t * It can split elements, merge them, wrap bare text nodes with paragraphs, etc. &mdash; just like the\n\t * pasting feature should do.\n\t *\n\t * For lower-level methods see {@link module:engine/model/writer~Writer `Writer`}.\n\t *\n\t * This method, unlike {@link module:engine/model/writer~Writer `Writer`}'s methods, does not have to be used\n\t * inside a {@link #change `change()` block}.\n\t *\n\t * # Conversion and schema\n\t *\n\t * Inserting elements and text nodes into the model is not enough to make CKEditor 5 render that content\n\t * to the user. CKEditor 5 implements a model-view-controller architecture and what `model.insertContent()` does\n\t * is only adding nodes to the model. Additionally, you need to define\n\t * {@glink framework/architecture/editing-engine#conversion converters} between the model and view\n\t * and define those nodes in the {@glink framework/architecture/editing-engine#schema schema}.\n\t *\n\t * So, while this method may seem similar to CKEditor 4 `editor.insertHtml()` (in fact, both methods\n\t * are used for paste-like content insertion), the CKEditor 5 method cannot be use to insert arbitrary HTML\n\t * unless converters are defined for all elements and attributes in that HTML.\n\t *\n\t * # Examples\n\t *\n\t * Using `insertContent()` with a manually created model structure:\n\t *\n\t * ```ts\n\t * // Let's create a document fragment containing such content as:\n\t * //\n\t * // <paragraph>foo</paragraph>\n\t * // <blockQuote>\n\t * // <paragraph>bar</paragraph>\n\t * // </blockQuote>\n\t * const docFrag = editor.model.change( writer => {\n\t * \tconst p1 = writer.createElement( 'paragraph' );\n\t * \tconst p2 = writer.createElement( 'paragraph' );\n\t * \tconst blockQuote = writer.createElement( 'blockQuote' );\n\t * \tconst docFrag = writer.createDocumentFragment();\n\t *\n\t * \twriter.append( p1, docFrag );\n\t * \twriter.append( blockQuote, docFrag );\n\t * \twriter.append( p2, blockQuote );\n\t * \twriter.insertText( 'foo', p1 );\n\t * \twriter.insertText( 'bar', p2 );\n\t *\n\t * \treturn docFrag;\n\t * } );\n\t *\n\t * // insertContent() does not have to be used in a change() block. It can, though,\n\t * // so this code could be moved to the callback defined above.\n\t * editor.model.insertContent( docFrag );\n\t * ```\n\t *\n\t * Using `insertContent()` with an HTML string converted to a model document fragment (similar to the pasting mechanism):\n\t *\n\t * ```ts\n\t * // You can create your own HtmlDataProcessor instance or use editor.data.processor\n\t * // if you have not overridden the default one (which is the HtmlDataProcessor instance).\n\t * const htmlDP = new HtmlDataProcessor( viewDocument );\n\t *\n\t * // Convert an HTML string to a view document fragment:\n\t * const viewFragment = htmlDP.toView( htmlString );\n\t *\n\t * // Convert the view document fragment to a model document fragment\n\t * // in the context of $root. This conversion takes the schema into\n\t * // account so if, for example, the view document fragment contained a bare text node,\n\t * // this text node cannot be a child of $root, so it will be automatically\n\t * // wrapped with a <paragraph>. You can define the context yourself (in the second parameter),\n\t * // and e.g. convert the content like it would happen in a <paragraph>.\n\t * // Note: The clipboard feature uses a custom context called $clipboardHolder\n\t * // which has a loosened schema.\n\t * const modelFragment = editor.data.toModel( viewFragment );\n\t *\n\t * editor.model.insertContent( modelFragment );\n\t * ```\n\t *\n\t * By default this method will use the document selection but it can also be used with a position, range or selection instance.\n\t *\n\t * ```ts\n\t * // Insert text at the current document selection position.\n\t * editor.model.change( writer => {\n\t * \teditor.model.insertContent( writer.createText( 'x' ) );\n\t * } );\n\t *\n\t * // Insert text at a given position - the document selection will not be modified.\n\t * editor.model.change( writer => {\n\t * \teditor.model.insertContent( writer.createText( 'x' ), doc.getRoot(), 2 );\n\t *\n\t * \t// Which is a shorthand for:\n\t * \teditor.model.insertContent( writer.createText( 'x' ), writer.createPositionAt( doc.getRoot(), 2 ) );\n\t * } );\n\t * ```\n\t *\n\t * If you want the document selection to be moved to the inserted content, use the\n\t * {@link module:engine/model/writer~Writer#setSelection `setSelection()`} method of the writer after inserting\n\t * the content:\n\t *\n\t * ```ts\n\t * editor.model.change( writer => {\n\t * \tconst paragraph = writer.createElement( 'paragraph' );\n\t *\n\t * \t// Insert an empty paragraph at the beginning of the root.\n\t * \teditor.model.insertContent( paragraph, writer.createPositionAt( editor.model.document.getRoot(), 0 ) );\n\t *\n\t * \t// Move the document selection to the inserted paragraph.\n\t * \twriter.setSelection( paragraph, 'in' );\n\t * } );\n\t * ```\n\t *\n\t * If an instance of the {@link module:engine/model/selection~Selection model selection} is passed as `selectable`,\n\t * the new content will be inserted at the passed selection (instead of document selection):\n\t *\n\t * ```ts\n\t * editor.model.change( writer => {\n\t * \t// Create a selection in a paragraph that will be used as a place of insertion.\n\t * \tconst selection = writer.createSelection( paragraph, 'in' );\n\t *\n\t * \t// Insert the new text at the created selection.\n\t * \teditor.model.insertContent( writer.createText( 'x' ), selection );\n\t *\n\t * \t// insertContent() modifies the passed selection instance so it can be used to set the document selection.\n\t * \t// Note: This is not necessary when you passed the document selection to insertContent().\n\t * \twriter.setSelection( selection );\n\t * } );\n\t * ```\n\t *\n\t * @fires insertContent\n\t * @param content The content to insert.\n\t * @param selectable The selection into which the content should be inserted.\n\t * If not provided the current model document selection will be used.\n\t * @param placeOrOffset To be used when a model item was passed as `selectable`.\n\t * This param defines a position in relation to that item.\n\t * at the insertion position.\n\t */\n\tpublic insertContent(\n\t\tcontent: Item | ModelDocumentFragment,\n\t\tselectable?: Selectable,\n\t\tplaceOrOffset?: PlaceOrOffset,\n\t\t...rest: Array<unknown>\n\t): ModelRange {\n\t\tconst selection = normalizeSelectable( selectable, placeOrOffset );\n\n\t\t// Passing all call arguments so it acts like decorated method.\n\t\treturn this.fire<ModelInsertContentEvent>( 'insertContent', [ content, selection, placeOrOffset, ...rest ] )!;\n\t}\n\n\t/**\n\t * Inserts an {@glink framework/deep-dive/schema#object-elements object element} at a specific position in the editor content.\n\t *\n\t * This is a high-level API:\n\t * * It takes the {@link #schema schema} into consideration,\n\t * * It clears the content of passed `selectable` before inserting,\n\t * * It can move the selection at the end of the process,\n\t * * It will copy the selected block's attributes to preserve them upon insertion,\n\t * * It can split elements or wrap inline objects with paragraphs if they are not allowed in target position,\n\t * * etc.\n\t *\n\t * # Notes\n\t *\n\t * * If you want to insert a non-object content, see {@link #insertContent} instead.\n\t * * For lower-level API, see {@link module:engine/model/writer~Writer `Writer`}.\n\t * * Unlike {@link module:engine/model/writer~Writer `Writer`}, this method does not have to be used inside\n\t * a {@link #change `change()` block}.\n\t * * Inserting object into the model is not enough to make CKEditor 5 render that content to the user.\n\t * CKEditor 5 implements a model-view-controller architecture and what `model.insertObject()` does\n\t * is only adding nodes to the model. Additionally, you need to define\n\t * {@glink framework/architecture/editing-engine#conversion converters} between the model and view\n\t * and define those nodes in the {@glink framework/architecture/editing-engine#schema schema}.\n\t *\n\t * # Examples\n\t *\n\t * Use the following code to insert an object at the current selection and keep the selection on the inserted element:\n\t *\n\t * ```ts\n\t * const rawHtmlEmbedElement = writer.createElement( 'rawHtml' );\n\t *\n\t * model.insertObject( rawHtmlEmbedElement, null, null, {\n\t * \tsetSelection: 'on'\n\t * } );\n\t * ```\n\t *\n\t * Use the following code to insert an object at the current selection and nudge the selection after the inserted object:\n\t *\n\t * ```ts\n\t * const pageBreakElement = writer.createElement( 'pageBreak' );\n \t *\n\t * model.insertObject( pageBreakElement, null, null, {\n\t * \tsetSelection: 'after'\n\t * } );\n\t * ```\n\t *\n\t * Use the following code to insert an object at the current selection and avoid splitting the content (non-destructive insertion):\n\t *\n\t * ```ts\n\t * const tableElement = writer.createElement( 'table' );\n \t *\n\t * model.insertObject( tableElement, null, null, {\n\t * \tfindOptimalPosition: 'auto'\n\t * } );\n\t * ```\n\t *\n\t * Use the following code to insert an object at the specific range (also: replace the content of the range):\n\t *\n\t * ```ts\n\t * const tableElement = writer.createElement( 'table' );\n\t * const range = model.createRangeOn( model.document.getRoot().getChild( 1 ) );\n \t *\n\t * model.insertObject( tableElement, range );\n\t * ```\n\t *\n\t * @param element An object to be inserted into the model document.\n\t * @param selectable A selectable where the content should be inserted. If not specified, the current\n\t * {@link module:engine/model/document~Document#selection document selection} will be used instead.\n\t * @param placeOrOffset Specifies the exact place or offset for the insertion to take place, relative to `selectable`.\n\t * @param options Additional options.\n\t * @param options.findOptimalPosition An option that, when set, adjusts the insertion position (relative to\n\t * `selectable` and `placeOrOffset`) so that the content of `selectable` is not split upon insertion (a.k.a. non-destructive insertion).\n\t * * When `'auto'`, the algorithm will decide whether to insert the object before or after `selectable` to avoid content splitting.\n\t * * When `'before'`, the closest position before `selectable` will be used that will not result in content splitting.\n\t * * When `'after'`, the closest position after `selectable` will be used that will not result in content splitting.\n\t *\n\t * Note that this option only works for block objects. Inline objects are inserted into text and do not split blocks.\n\t * @param options.setSelection An option that, when set, moves the\n\t * {@link module:engine/model/document~Document#selection document selection} after inserting the object.\n\t * * When `'on'`, the document selection will be set on the inserted object.\n\t * * When `'after'`, the document selection will move to the closest text node after the inserted object. If there is no\n\t * such text node, a paragraph will be created and the document selection will be moved inside it.\n\t * at the insertion position.\n\t */\n\tpublic insertObject(\n\t\telement: ModelElement,\n\t\tselectable?: Selectable,\n\t\tplaceOrOffset?: PlaceOrOffset | null,\n\t\toptions?: {\n\t\t\tfindOptimalPosition?: 'auto' | 'before' | 'after';\n\t\t\tsetSelection?: 'on' | 'after';\n\t\t},\n\t\t...rest: Array<unknown>\n\t): ModelRange {\n\t\tconst selection = normalizeSelectable( selectable, placeOrOffset );\n\n\t\t// Note that options are fired as 2 arguments for backward compatibility with the decorated method.\n\t\t// Passing all call arguments so it acts like decorated method.\n\t\treturn this.fire<ModelInsertObjectEvent>( 'insertObject', [ element, selection, options, options, ...rest ] )!;\n\t}\n\n\t/**\n\t * Deletes content of the selection and merge siblings. The resulting selection is always collapsed.\n\t *\n\t * **Note:** For the sake of predictability, the resulting selection should always be collapsed.\n\t * In cases where a feature wants to modify deleting behavior so selection isn't collapsed\n\t * (e.g. a table feature may want to keep row selection after pressing <kbd>Backspace</kbd>),\n\t * then that behavior should be implemented in the view's listener. At the same time, the table feature\n\t * will need to modify this method's behavior too, e.g. to \"delete contents and then collapse\n\t * the selection inside the last selected cell\" or \"delete the row and collapse selection somewhere near\".\n\t * That needs to be done in order to ensure that other features which use `deleteContent()` will work well with tables.\n\t *\n\t * @fires deleteContent\n\t * @param selection Selection of which the content should be deleted.\n\t * @param options.leaveUnmerged Whether to merge elements after removing the content of the selection.\n\t *\n\t * For example `<heading1>x[x</heading1><paragraph>y]y</paragraph>` will become:\n\t *\n\t * * `<heading1>x^y</heading1>` with the option disabled (`leaveUnmerged == false`)\n\t * * `<heading1>x^</heading1><paragraph>y</paragraph>` with enabled (`leaveUnmerged == true`).\n\t *\n\t * Note: {@link module:engine/model/schema~Schema#isObject object} and {@link module:engine/model/schema~Schema#isLimit limit}\n\t * elements will not be merged.\n\t *\n\t * @param options.doNotResetEntireContent Whether to skip replacing the entire content with a\n\t * paragraph when the entire content was selected.\n\t *\n\t * For example `<heading1>[x</heading1><paragraph>y]</paragraph>` will become:\n\t *\n\t * * `<paragraph>^</paragraph>` with the option disabled (`doNotResetEntireContent == false`)\n\t * * `<heading1>^</heading1>` with enabled (`doNotResetEntireContent == true`)\n\t *\n\t * @param options.doNotAutoparagraph Whether to create a paragraph if after content deletion selection is moved\n\t * to a place where text cannot be inserted.\n\t *\n\t * For example `<paragraph>x</paragraph>[<imageBlock src=\"foo.jpg\"></imageBlock>]` will become:\n\t *\n\t * * `<paragraph>x</paragraph><paragraph>[]</paragraph>` with the option disabled (`doNotAutoparagraph == false`)\n\t * * `<paragraph>x[]</paragraph>` with the option enabled (`doNotAutoparagraph == true`).\n\t *\n\t * **Note:** if there is no valid position for the selection, the paragraph will always be created:\n\t *\n\t * `[<imageBlock src=\"foo.jpg\"></imageBlock>]` -> `<paragraph>[]</paragraph>`.\n\t *\n\t * @param options.direction The direction in which the content is being consumed.\n\t * Deleting backward corresponds to using the <kbd>Backspace</kbd> key, while deleting content forward corresponds to\n\t * the <kbd>Shift</kbd>+<kbd>Backspace</kbd> keystroke.\n\t */\n\tpublic deleteContent(\n\t\tselection: ModelSelection | DocumentSelection,\n\t\toptions?: {\n\t\t\tleaveUnmerged?: boolean;\n\t\t\tdoNotResetEntireContent?: boolean;\n\t\t\tdoNotAutoparagraph?: boolean;\n\t\t\tdirection?: 'forward' | 'backward';\n\t\t\t[ i: string ]: unknown;\n\t\t}\n\t): void {\n\t\tdeleteContent( this, selection, options );\n\t}\n\n\t/**\n\t * Modifies the selection. Currently, the supported modifications are:\n\t *\n\t * * Extending. The selection focus is moved in the specified `options.direction` with a step specified in `options.unit`.\n\t * Possible values for `unit` are:\n\t * * `'character'` (default) - moves selection by one user-perceived character. In most cases this means moving by one\n\t * character in `String` sense. However, unicode also defines \"combing marks\". These are special symbols, that combines\n\t * with a symbol before it (\"base character\") to create one user-perceived character. For example, `q̣̇` is a normal\n\t * letter `q` with two \"combining marks\": upper dot (`Ux0307`) and lower dot (`Ux0323`). For most actions, i.e. extending\n\t * selection by one position, it is correct to include both \"base character\" and all of it's \"combining marks\". That is\n\t * why `'character'` value is most natural and common method of modifying selection.\n\t * * `'codePoint'` - moves selection by one unicode code point. In contrary to, `'character'` unit, this will insert\n\t * selection between \"base character\" and \"combining mark\", because \"combining marks\" have their own unicode code points.\n\t * However, for technical reasons, unicode code points with values above `UxFFFF` are represented in native `String` by\n\t * two characters, called \"surrogate pairs\". Halves of \"surrogate pairs\" have a meaning only when placed next to each other.\n\t * For example `𨭎` is represented in `String` by `\\uD862\\uDF4E`. Both `\\uD862` and `\\uDF4E` do not have any meaning\n\t * outside the pair (are rendered as ? when alone). Position between them would be incorrect. In this case, selection\n\t * extension will include whole \"surrogate pair\".\n\t * * `'word'` - moves selection by a whole word.\n\t *\n\t * **Note:** if you extend a forward selection in a backward direction you will in fact shrink it.\n\t *\n\t * @fires modifySelection\n\t * @param selection The selection to modify.\n\t * @param options.direction The direction in which the selection should be modified.\n\t * @param options.unit The unit by which selection should be modified.\n\t * @param options.treatEmojiAsSingleUnit Whether multi-characer emoji sequences should be handled as single unit.\n\t */\n\tpublic modifySelection(\n\t\tselection: ModelSelection | DocumentSelection,\n\t\toptions?: {\n\t\t\tdirection?: 'forward' | 'backward';\n\t\t\tunit?: 'character' | 'codePoint' | 'word';\n\t\t\ttreatEmojiAsSingleUnit?: boolean;\n\t\t}\n\t): void {\n\t\tmodifySelection( this, selection, options );\n\t}\n\n\t/**\n\t * Gets a clone of the selected content.\n\t *\n\t * For example, for the following selection:\n\t *\n\t * ```html\n\t * <paragraph>x</paragraph>\n\t * <blockQuote>\n\t * \t<paragraph>y</paragraph>\n\t * \t<heading1>fir[st</heading1>\n\t * </blockQuote>\n\t * <paragraph>se]cond</paragraph>\n\t * <paragraph>z</paragraph>\n\t * ```\n\t *\n\t * It will return a document fragment with such a content:\n\t *\n\t * ```html\n\t * <blockQuote>\n\t * \t<heading1>st</heading1>\n\t * </blockQuote>\n\t * <paragraph>se</paragraph>\n\t * ```\n\t *\n\t * @fires getSelectedContent\n\t * @param selection The selection of which content will be returned.\n\t */\n\tpublic getSelectedContent( selection: ModelSelection | DocumentSelection ): ModelDocumentFragment {\n\t\treturn getSelectedContent( this, selection );\n\t}\n\n\t/**\n\t * Checks whether the given {@link module:engine/model/range~Range range} or\n\t * {@link module:engine/model/element~Element element} has any meaningful content.\n\t *\n\t * Meaningful content is:\n\t *\n\t * * any text node (`options.ignoreWhitespaces` allows controlling whether this text node must also contain\n\t * any non-whitespace characters),\n\t * * or any {@link module:engine/model/schema~Schema#isContent content element},\n\t * * or any {@link module:engine/model/markercollection~Marker marker} which\n\t * {@link module:engine/model/markercollection~Marker#_affectsData affects data}.\n\t *\n\t * This means that a range containing an empty `<paragraph></paragraph>` is not considered to have a meaningful content.\n\t * However, a range containing an `<imageBlock></imageBlock>` (which would normally be marked in the schema as an object element)\n\t * is considered non-empty.\n\t *\n\t * @param rangeOrElement Range or element to check.\n\t * @param options.ignoreWhitespaces Whether text node with whitespaces only should be considered empty.\n\t * @param options.ignoreMarkers Whether markers should be ignored.\n\t */\n\tpublic hasContent(\n\t\trangeOrElement: ModelRange | ModelElement | ModelDocumentFragment,\n\t\toptions: {\n\t\t\tignoreWhitespaces?: boolean;\n\t\t\tignoreMarkers?: boolean;\n\t\t} = {}\n\t): boolean {\n\t\tconst range = rangeOrElement instanceof ModelRange ? rangeOrElement : ModelRange._createIn( rangeOrElement );\n\n\t\tif ( range.isCollapsed ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst { ignoreWhitespaces = false, ignoreMarkers = false } = options;\n\n\t\t// Check if there are any markers which affects data in this given range.\n\t\tif ( !ignoreMarkers ) {\n\t\t\tfor ( const intersectingMarker of this.markers.getMarkersIntersectingRange( range ) ) {\n\t\t\t\tif ( intersectingMarker.affectsData ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tfor ( const item of range.getItems() ) {\n\t\t\tif ( this.schema.isContent( item ) ) {\n\t\t\t\tif ( item.is( '$textProxy' ) ) {\n\t\t\t\t\tif ( !ignoreWhitespaces ) {\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t} else if ( item.data.search( /\\S/ ) !== -1 ) {\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Check whether given selectable is at a place in the model where it can be edited (returns `true`) or not (returns `false`).\n\t *\n\t * Should be used instead of {@link module:core/editor/editor~Editor#isReadOnly} to check whether a user action can happen at\n\t * given selectable. It may be decorated and used differently in different environment (e.g. multi-root editor can disable\n\t * a particular root).\n\t *\n\t * This method is decorated. Although this method accepts any parameter of `Selectable` type, the\n\t * {@link ~Model#event:canEditAt `canEditAt` event} is fired with `selectable` normalized to an instance of\n\t * {@link module:engine/model/selection~Selection} or {@link module:engine/model/documentselection~DocumentSelection}\n\t *\n\t * @fires canEditAt\n\t */\n\tpublic canEditAt( selectable: Selectable ): boolean {\n\t\tconst selection = normalizeSelectable( selectable );\n\n\t\treturn this.fire<ModelCanEditAtEvent>( 'canEditAt', [ selection ] )!;\n\t}\n\n\t/**\n\t * Creates a position from the given root and path in that root.\n\t *\n\t * Note: This method is also available as\n\t * {@link module:engine/model/writer~Writer#createPositionFromPath `Writer#createPositionFromPath()`}.\n\t *\n\t * @param root Root of the position.\n\t * @param path Position path. See {@link module:engine/model/position~Position#path}.\n\t * @param stickiness Position stickiness. See {@link module:engine/model/position~PositionStickiness}.\n\t */\n\tpublic createPositionFromPath(\n\t\troot: ModelElement | ModelDocumentFragment,\n\t\tpath: ReadonlyArray<number>,\n\t\tstickiness?: PositionStickiness\n\t): ModelPosition {\n\t\treturn new ModelPosition( root, path, stickiness );\n\t}\n\n\t/**\n\t * Creates position at the given location. The location can be specified as:\n\t *\n\t * * a {@link module:engine/model/position~Position position},\n\t * * a parent element and offset in that element,\n\t * * a parent element and `'end'` (the position will be set at the end of that element),\n\t * * a {@link module:engine/model/item~Item model item} and `'before'` or `'after'`\n\t * (the position will be set before or after the given model item).\n\t *\n\t * This method is a shortcut to other factory methods such as:\n\t *\n\t * * {@link module:engine/model/model~Model#createPositionBefore `createPositionBefore()`},\n\t * * {@link module:engine/model/model~Model#createPositionAfter `createPositionAfter()`}.\n\t *\n\t * Note: This method is also available as\n\t * {@link module:engine/model/writer~Writer#createPositionAt `Writer#createPositionAt()`},\n\t *\n\t * @param itemOrPosition\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/model/item~Item model item}.\n\t */\n\tpublic createPositionAt(\n\t\titemOrPosition: Item | ModelPosition | ModelDocumentFragment,\n\t\toffset?: PositionOffset\n\t): ModelPosition {\n\t\treturn ModelPosition._createAt( itemOrPosition, offset );\n\t}\n\n\t/**\n\t * Creates a new position after the given {@link module:engine/model/item~Item model item}.\n\t *\n\t * Note: This method is also available as\n\t * {@link module:engine/model/writer~Writer#createPositionAfter `Writer#createPositionAfter()`}.\n\t *\n\t * @param item Item after which the position should be placed.\n\t */\n\tpublic createPositionAfter( item: Item ): ModelPosition {\n\t\treturn ModelPosition._createAfter( item );\n\t}\n\n\t/**\n\t * Creates a new position before the given {@link module:engine/model/item~Item model item}.\n\t *\n\t * Note: This method is also available as\n\t * {@link module:engine/model/writer~Writer#createPositionBefore `Writer#createPositionBefore()`}.\n\t *\n\t * @param item Item before which the position should be placed.\n\t */\n\tpublic createPositionBefore( item: Item ): ModelPosition {\n\t\treturn ModelPosition._createBefore( item );\n\t}\n\n\t/**\n\t * Creates a range spanning from the `start` position to the `end` position.\n\t *\n\t * Note: This method is also available as\n\t * {@link module:engine/model/writer~Writer#createRange `Writer#createRange()`}:\n\t *\n\t * ```ts\n\t * model.change( writer => {\n\t * \tconst range = writer.createRange( start, end );\n\t * } );\n\t * ```\n\t *\n\t * @param start Start position.\n\t * @param end End position. If not set, the range will be collapsed to the `start` position.\n\t */\n\tpublic createRange( start: ModelPosition, end?: ModelPosition ): ModelRange {\n\t\treturn new ModelRange( start, end );\n\t}\n\n\t/**\n\t * Creates a range inside the given element which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * Note: This method is also available as\n\t * {@link module:engine/model/writer~Writer#createRangeIn `Writer#createRangeIn()`}:\n\t *\n\t * ```ts\n\t * model.change( writer => {\n\t * \tconst range = writer.createRangeIn( paragraph );\n\t * } );\n\t * ```\n\t *\n\t * @param element Element which is a parent for the range.\n\t */\n\tpublic createRangeIn( element: ModelElement | ModelDocumentFragment ): ModelRange {\n\t\treturn ModelRange._createIn( element );\n\t}\n\n\t/**\n\t * Creates a range that starts before the given {@link module:engine/model/item~Item model item} and ends after it.\n\t *\n\t * Note: This method is also available on `writer` instance as\n\t * {@link module:engine/model/writer~Writer#createRangeOn `Writer.createRangeOn()`}:\n\t *\n\t * ```ts\n\t * model.change( writer => {\n\t * \tconst range = writer.createRangeOn( paragraph );\n\t * } );\n\t * ```\n\t *\n\t * @param item\n\t */\n\tpublic createRangeOn( item: Item ): ModelRange {\n\t\treturn ModelRange._createOn( item );\n\t}\n\n\t/**\n\t * Creates a new selection instance based on the given {@link module:engine/model/selection~Selectable selectable}\n\t * or creates an empty selection if no arguments were passed.\n\t *\n\t * Note: This method is also available as\n\t * {@link module:engine/model/writer~Writer#createSelection `Writer#createSelection()`}.\n\t *\n\t * ```ts\n\t * // Creates selection at the given offset in the given element.\n\t * const paragraph = writer.createElement( 'paragraph' );\n\t * const selection = writer.createSelection( paragraph, offset );\n\t *\n\t * // Creates a range inside an {@link module:engine/model/element~Element element} which starts before the\n\t * // first child of that element and ends after the last child of that element.\n\t * const selection = writer.createSelection( paragraph, 'in' );\n\t *\n\t * // Creates a range on an {@link module:engine/model/item~Item item} which starts before the item and ends\n\t * // just after the item.\n\t * const selection = writer.createSelection( paragraph, 'on' );\n\t *\n\t * // Additional options (`'backward'`) can be specified as the last argument.\n\t *\n\t * // Creates backward selection.\n\t * const selection = writer.createSelection( element, 'in', { backward: true } );\n\t * ```\n\t *\n\t * See also: {@link #createSelection:SELECTABLE `createSelection( selectable, options )`}.\n\t *\n\t * @label NODE_OFFSET\n\t */\n\tpublic createSelection( selectable: Node, placeOrOffset: PlaceOrOffset, options?: { backward?: boolean } ): ModelSelection;\n\n\t/**\n\t * Creates a new selection instance based on the given {@link module:engine/model/selection~Selectable selectable}\n\t * or creates an empty selection if no arguments were passed.\n\t *\n\t * Note: This method is also available as\n\t * {@link module:engine/model/writer~Writer#createSelection `Writer#createSelection()`}.\n\t *\n\t * ```ts\n\t * // Creates empty selection without ranges.\n\t * const selection = writer.createSelection();\n\t *\n\t * // Creates selection at the given range.\n\t * const range = writer.createRange( start, end );\n\t * const selection = writer.createSelection( range );\n\t *\n\t * // Creates selection at the given ranges\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( star2, end2 ) ];\n\t * const selection = writer.createSelection( ranges );\n\t *\n\t * // Creates selection from the other selection.\n\t * // Note: It doesn't copies selection attributes.\n\t * const otherSelection = writer.createSelection();\n\t * const selection = writer.createSelection( otherSelection );\n\t *\n\t * // Creates selection from the given document selection.\n\t * // Note: It doesn't copies selection attributes.\n\t * const documentSelection = model.document.selection;\n\t * const selection = writer.createSelection( documentSelection );\n\t *\n\t * // Creates selection at the given position.\n\t * const position = writer.createPositionFromPath( root, path );\n\t * const selection = writer.createSelection( position );\n\t *\n\t * // Additional options (`'backward'`) can be specified as the last argument.\n\t *\n\t * // Creates backward selection.\n\t * const selection = writer.createSelection( range, { backward: true } );\n\t * ```\n\t *\n\t * See also: {@link #createSelection:NODE_OFFSET `createSelection( node, placeOrOffset, options )`}.\n\t *\n\t * @label SELECTABLE\n\t */\n\tpublic createSelection( selectable?: Exclude<Selectable, Node>, options?: { backward?: boolean } ): ModelSelection;\n\n\tpublic createSelection( ...args: ConstructorParameters<typeof ModelSelection> ): ModelSelection {\n\t\treturn new ModelSelection( ...args );\n\t}\n\n\t/**\n\t * Creates a {@link module:engine/model/batch~Batch} instance.\n\t *\n\t * **Note:** In most cases creating a batch instance is not necessary as they are created when using:\n\t *\n\t * * {@link #change `change()`},\n\t * * {@link #enqueueChange `enqueueChange()`}.\n\t *\n\t * @param type {@link module:engine/model/batch~Batch#constructor The type} of the batch.\n\t */\n\tpublic createBatch( type?: BatchType ): Batch {\n\t\treturn new Batch( type );\n\t}\n\n\t/**\n\t * Creates an operation instance from a JSON object (parsed JSON string).\n\t *\n\t * This is an alias for {@link module:engine/model/operation/operationfactory~OperationFactory.fromJSON `OperationFactory.fromJSON()`}.\n\t *\n\t * @param json Deserialized JSON object.\n\t */\n\tpublic createOperationFromJSON( json: unknown ): Operation {\n\t\treturn OperationFactory.fromJSON( json, this.document );\n\t}\n\n\t/**\n\t * Removes all events listeners set by model instance and destroys {@link module:engine/model/document~Document}.\n\t */\n\tpublic destroy(): void {\n\t\tthis.document.destroy();\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Common part of {@link module:engine/model/model~Model#change} and {@link module:engine/model/model~Model#enqueueChange}\n\t * which calls callbacks and returns array of values returned by these callbacks.\n\t */\n\tprivate _runPendingChanges() {\n\t\tconst ret = [];\n\n\t\tthis.fire( '_beforeChanges' );\n\n\t\ttry {\n\t\t\twhile ( this._pendingChanges.length ) {\n\t\t\t\t// Create a new writer using batch instance created for this chain of changes.\n\t\t\t\tconst currentBatch = this._pendingChanges[ 0 ].batch;\n\t\t\t\tthis._currentWriter = new Writer( this, currentBatch );\n\n\t\t\t\t// Execute changes callback and gather the returned value.\n\t\t\t\tconst callbackReturnValue = this._pendingChanges[ 0 ].callback( this._currentWriter );\n\t\t\t\tret.push( callbackReturnValue );\n\n\t\t\t\tthis.document._handleChangeBlock( this._currentWriter );\n\n\t\t\t\tthis._pendingChanges.shift();\n\t\t\t\tthis._currentWriter = null;\n\t\t\t}\n\t\t} finally {\n\t\t\tthis._pendingChanges.length = 0;\n\t\t\tthis._currentWriter = null;\n\n\t\t\tthis.fire( '_afterChanges' );\n\t\t}\n\n\t\treturn ret;\n\t}\n}\n\n/**\n * Normalizes a selectable to a Selection or DocumentSelection.\n */\nfunction normalizeSelectable(\n\tselectable?: Selectable,\n\tplaceOrOffset?: PlaceOrOffset | null\n): ModelSelection | DocumentSelection | undefined {\n\tif ( !selectable ) {\n\t\treturn;\n\t}\n\n\tif ( selectable instanceof ModelSelection || selectable instanceof DocumentSelection ) {\n\t\treturn selectable;\n\t}\n\n\tif ( selectable instanceof Node ) {\n\t\tif ( placeOrOffset || placeOrOffset === 0 ) {\n\t\t\treturn new ModelSelection( selectable, placeOrOffset );\n\t\t} else if ( selectable.is( 'rootElement' ) ) {\n\t\t\treturn new ModelSelection( selectable, 'in' );\n\t\t} else {\n\t\t\treturn new ModelSelection( selectable, 'on' );\n\t\t}\n\t}\n\n\treturn new ModelSelection( selectable );\n}\n\n/**\n * Fired when entering the outermost {@link module:engine/model/model~Model#enqueueChange} or\n * {@link module:engine/model/model~Model#change} block.\n *\n * @internal\n * @eventName ~Model#_beforeChanges\n */\nexport type BeforeChangesEvent = {\n\tname: '_beforeChanges';\n\targs: [];\n};\n\n/**\n * Fired when leaving the outermost {@link module:engine/model/model~Model#enqueueChange} or\n * {@link module:engine/model/model~Model#change} block.\n *\n * @internal\n * @eventName ~Model#_afterChanges\n */\nexport type AfterChangesEvent = {\n\tname: '_afterChanges';\n\targs: [];\n};\n\n/**\n * Fired every time any {@link module:engine/model/operation/operation~Operation operation} is applied on the model\n * using {@link ~Model#applyOperation}.\n *\n * Note that this event is suitable only for very specific use-cases. Use it if you need to listen to every single operation\n * applied on the document. However, in most cases {@link module:engine/model/document~Document#event:change} should\n * be used.\n *\n * A few callbacks are already added to this event by engine internal classes:\n *\n * * with `highest` priority operation is validated,\n * * with `normal` priority operation is executed,\n * * with `low` priority the {@link module:engine/model/document~Document} updates its version,\n * * with `low` priority {@link module:engine/model/liveposition~LivePosition} and {@link module:engine/model/liverange~LiveRange}\n * update themselves.\n *\n * @eventName ~Model#applyOperation\n * @param args Arguments of the `applyOperation` which is an array with a single element - applied\n * {@link module:engine/model/operation/operation~Operation operation}.\n */\nexport type ModelApplyOperationEvent = DecoratedMethodEvent<Model, 'applyOperation'>;\n\n/**\n * Event fired when {@link ~Model#insertContent} method is called.\n *\n * The {@link ~Model#insertContent default action of that method} is implemented as a\n * listener to this event so it can be fully customized by the features.\n *\n * **Note** The `selectable` parameter for the {@link ~Model#insertContent} is optional. When `undefined` value is passed the method uses\n * {@link module:engine/model/document~Document#selection document selection}.\n *\n * @eventName ~Model#insertContent\n * @param args The arguments passed to the original method.\n */\nexport type ModelInsertContentEvent = {\n\tname: 'insertContent';\n\targs: [ [\n\t\tcontent: Item | ModelDocumentFragment,\n\t\tselectable?: ModelSelection | DocumentSelection,\n\t\t...rest: Array<unknown>\n\t] ];\n\treturn: ModelRange;\n};\n\n/**\n * Event fired when the {@link ~Model#insertObject} method is called.\n *\n * The {@link ~Model#insertObject default action of that method} is implemented as a\n * listener to this event so it can be fully customized by the features.\n *\n * **Note** The `selectable` parameter for the {@link ~Model#insertObject} is optional. When `undefined` value is passed the method uses\n * {@link module:engine/model/document~Document#selection document selection}.\n *\n * @eventName ~Model#insertObject\n * @param args The arguments passed to the original method.\n */\nexport type ModelInsertObjectEvent = {\n\tname: 'insertObject';\n\targs: [ [\n\t\telement: ModelElement,\n\t\tselectable?: ModelSelection | DocumentSelection | null,\n\t\toptions?: {\n\t\t\tfindOptimalPosition?: 'auto' | 'before' | 'after';\n\t\t\tsetSelection?: 'on' | 'after';\n\t\t},\n\t\t...rest: Array<unknown>\n\t] ];\n\treturn: ModelRange;\n};\n\n/**\n * Event fired when {@link ~Model#deleteContent} method is called.\n *\n * The {@link ~Model#deleteContent default action of that method} is implemented as a\n * listener to this event, so it can be fully customized by the features.\n *\n * @eventName ~Model#deleteContent\n * @param args The arguments passed to the original method.\n */\nexport type ModelDeleteContentEvent = DecoratedMethodEvent<Model, 'deleteContent'>;\n\n/**\n * Event fired when {@link ~Model#modifySelection} method is called.\n *\n * The {@link ~Model#modifySelection default action of that method} is implemented as a\n * listener to this event, so it can be fully customized by the features.\n *\n * @eventName ~Model#modifySelection\n * @param args The arguments passed to the original method.\n */\nexport type ModelModifySelectionEvent = DecoratedMethodEvent<Model, 'modifySelection'>;\n\n/**\n * Event fired when {@link ~Model#getSelectedContent} method is called.\n *\n * The {@link ~Model#getSelectedContent default action of that method} is implemented as a\n * listener to this event, so it can be fully customized by the features.\n *\n * @eventName ~Model#getSelectedContent\n * @param args The arguments passed to the original method.\n */\nexport type ModelGetSelectedContentEvent = DecoratedMethodEvent<Model, 'getSelectedContent'>;\n\n/**\n * Event fired when {@link ~Model#canEditAt} method is called.\n *\n * The {@link ~Model#canEditAt default action of that method} is implemented as a\n * listener to this event, so it can be fully customized by the features.\n *\n * Although the original method accepts any parameter of `Selectable` type, this event is fired with `selectable` normalized\n * to an instance of {@link module:engine/model/selection~Selection} or {@link module:engine/model/documentselection~DocumentSelection}.\n *\n * @eventName ~Model#canEditAt\n * @param args The arguments passed to the original method.\n */\nexport type ModelCanEditAtEvent = {\n\tname: 'canEditAt';\n\targs: [ [\n\t\tselectable?: ModelSelection | DocumentSelection\n\t] ];\n\treturn: boolean;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/clickobserver\n */\n\nimport DomEventObserver from './domeventobserver';\nimport type DomEventData from './domeventdata';\nimport type { BubblingEvent } from './bubblingemittermixin';\n\n/**\n * {@link module:engine/view/document~Document#event:click Click} event observer.\n *\n * Note that this observer is not available by default. To make it available it needs to be added to\n * {@link module:engine/view/view~View view controller} by a {@link module:engine/view/view~View#addObserver} method.\n */\nexport default class ClickObserver extends DomEventObserver<'click'> {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic readonly domEventType = 'click' as const;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic onDomEvent( domEvent: MouseEvent ): void {\n\t\tthis.fire( domEvent.type, domEvent );\n\t}\n}\n\n/**\n * Fired when one of the editables has been clicked.\n *\n * Introduced by {@link module:engine/view/observer/clickobserver~ClickObserver}.\n *\n * Note that this event is not available by default. To make it available\n * {@link module:engine/view/observer/clickobserver~ClickObserver} needs to be added\n * to {@link module:engine/view/view~View} by a {@link module:engine/view/view~View#addObserver} method.\n *\n * @see module:engine/view/observer/clickobserver~ClickObserver\n * @eventName module:engine/view/document~Document#click\n * @param data Event data.\n */\nexport type ViewDocumentClickEvent = BubblingEvent<{\n\tname: 'click';\n\targs: [ data: DomEventData<MouseEvent> ];\n}>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/observer/mouseobserver\n */\n\nimport DomEventObserver from './domeventobserver';\nimport type DomEventData from './domeventdata';\n\n/**\n * Mouse events observer.\n *\n * Note that this observer is not available by default. To make it available it needs to be added to\n * {@link module:engine/view/view~View} by {@link module:engine/view/view~View#addObserver} method.\n */\nexport default class MouseObserver extends DomEventObserver<'mousedown' | 'mouseup' | 'mouseover' | 'mouseout'> {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic readonly domEventType = [ 'mousedown', 'mouseup', 'mouseover', 'mouseout' ] as const;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic onDomEvent( domEvent: MouseEvent ): void {\n\t\tthis.fire( domEvent.type, domEvent );\n\t}\n}\n\n/**\n * Fired when the mouse button is pressed down on one of the editing roots of the editor.\n *\n * Introduced by {@link module:engine/view/observer/mouseobserver~MouseObserver}.\n *\n * Note that this event is not available by default. To make it available, {@link module:engine/view/observer/mouseobserver~MouseObserver}\n * needs to be added to {@link module:engine/view/view~View} by the {@link module:engine/view/view~View#addObserver} method.\n *\n * @see module:engine/view/observer/mouseobserver~MouseObserver\n * @eventName module:engine/view/document~Document#mousedown\n * @param data The event data.\n */\nexport type ViewDocumentMouseDownEvent = {\n\tname: 'mousedown';\n\targs: [ data: DomEventData<MouseEvent> ];\n};\n\n/**\n * Fired when the mouse button is released over one of the editing roots of the editor.\n *\n * Introduced by {@link module:engine/view/observer/mouseobserver~MouseObserver}.\n *\n * Note that this event is not available by default. To make it available, {@link module:engine/view/observer/mouseobserver~MouseObserver}\n * needs to be added to {@link module:engine/view/view~View} by the {@link module:engine/view/view~View#addObserver} method.\n *\n * @see module:engine/view/observer/mouseobserver~MouseObserver\n * @eventName module:engine/view/document~Document#mouseup\n * @param data The event data.\n */\nexport type ViewDocumentMouseUpEvent = {\n\tname: 'mouseup';\n\targs: [ data: DomEventData<MouseEvent> ];\n};\n\n/**\n * Fired when the mouse is over one of the editing roots of the editor.\n *\n * Introduced by {@link module:engine/view/observer/mouseobserver~MouseObserver}.\n *\n * Note that this event is not available by default. To make it available, {@link module:engine/view/observer/mouseobserver~MouseObserver}\n * needs to be added to {@link module:engine/view/view~View} by the {@link module:engine/view/view~View#addObserver} method.\n *\n * @see module:engine/view/observer/mouseobserver~MouseObserver\n * @eventName module:engine/view/document~Document#mouseover\n * @param data The event data.\n */\nexport type ViewDocumentMouseOverEvent = {\n\tname: 'mouseover';\n\targs: [ data: DomEventData<MouseEvent> ];\n};\n\n/**\n * Fired when the mouse is moved out of the one of the editing roots of the editor.\n *\n * Introduced by {@link module:engine/view/observer/mouseobserver~MouseObserver}.\n *\n * Note that this event is not available by default. To make it available, {@link module:engine/view/observer/mouseobserver~MouseObserver}\n * needs to be added to {@link module:engine/view/view~View} by the {@link module:engine/view/view~View#addObserver} method.\n *\n * @see module:engine/view/observer/mouseobserver~MouseObserver\n * @eventName module:engine/view/document~Document#mouseout\n * @param data The event data.\n */\nexport type ViewDocumentMouseOutEvent = {\n\tname: 'mouseout';\n\targs: [ data: DomEventData<MouseEvent> ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module engine/view/upcastwriter\n */\n\nimport DocumentFragment from './documentfragment';\nimport Element, { type ElementAttributes } from './element';\nimport Text from './text';\nimport { isPlainObject } from 'lodash-es';\nimport Position, { type PositionOffset } from './position';\nimport Range from './range';\nimport Selection, {\n\ttype PlaceOrOffset,\n\ttype Selectable,\n\ttype SelectionOptions\n} from './selection';\n\nimport type Document from './document';\nimport type Item from './item';\nimport type Node from './node';\n\n/**\n * View upcast writer. It provides a set of methods used to manipulate non-semantic view trees.\n *\n * It should be used only while working on a non-semantic view\n * (e.g. a view created from HTML string on paste).\n * To manipulate a view which was or is being downcasted from the the model use the\n * {@link module:engine/view/downcastwriter~DowncastWriter downcast writer}.\n *\n * Read more about changing the view in the {@glink framework/architecture/editing-engine#changing-the-view Changing the view}\n * section of the {@glink framework/architecture/editing-engine Editing engine architecture} guide.\n *\n * Unlike `DowncastWriter`, which is available in the {@link module:engine/view/view~View#change `View#change()`} block,\n * `UpcastWriter` can be created wherever you need it:\n *\n * ```ts\n * const writer = new UpcastWriter( viewDocument );\n * const text = writer.createText( 'foo!' );\n *\n * writer.appendChild( text, someViewElement );\n * ```\n */\nexport default class UpcastWriter {\n\t/**\n\t * The view document instance in which this upcast writer operates.\n\t */\n\tpublic readonly document: Document;\n\n\t/**\n\t * @param document The view document instance in which this upcast writer operates.\n\t */\n\tconstructor( document: Document ) {\n\t\tthis.document = document;\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/documentfragment~DocumentFragment} instance.\n\t *\n\t * @param children A list of nodes to be inserted into the created document fragment.\n\t * @returns The created document fragment.\n\t */\n\tpublic createDocumentFragment( children?: Node | Iterable<Node> ): DocumentFragment {\n\t\treturn new DocumentFragment( this.document, children );\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/element~Element} instance.\n\t *\n\t * Attributes can be passed in various formats:\n\t *\n\t * ```ts\n\t * upcastWriter.createElement( 'div', { class: 'editor', contentEditable: 'true' } ); // object\n\t * upcastWriter.createElement( 'div', [ [ 'class', 'editor' ], [ 'contentEditable', 'true' ] ] ); // map-like iterator\n\t * upcastWriter.createElement( 'div', mapOfAttributes ); // map\n\t * ```\n\t *\n\t * @param name Node name.\n\t * @param attrs Collection of attributes.\n\t * @param children A list of nodes to be inserted into created element.\n\t * @returns Created element.\n\t */\n\tpublic createElement(\n\t\tname: string,\n\t\tattrs?: ElementAttributes,\n\t\tchildren?: Node | Iterable<Node>\n\t): Element {\n\t\treturn new Element( this.document, name, attrs, children );\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/text~Text} instance.\n\t *\n\t * @param data The text's data.\n\t * @returns The created text node.\n\t */\n\tpublic createText( data: string ): Text {\n\t\treturn new Text( this.document, data );\n\t}\n\n\t/**\n\t * Clones the provided element.\n\t *\n\t * @see module:engine/view/element~Element#_clone\n\t * @param element Element to be cloned.\n\t * @param deep If set to `true` clones element and all its children recursively. When set to `false`,\n\t * element will be cloned without any children.\n\t * @returns Clone of this element.\n\t */\n\tpublic clone( element: Element, deep: boolean = false ): Element {\n\t\treturn element._clone( deep );\n\t}\n\n\t/**\n\t * Appends a child node or a list of child nodes at the end of this node\n\t * and sets the parent of these nodes to this element.\n\t *\n\t * @see module:engine/view/element~Element#_appendChild\n\t * @param items Items to be inserted.\n\t * @param element Element to which items will be appended.\n\t * @returns Number of appended nodes.\n\t */\n\tpublic appendChild( items: Item | string | Iterable<Item | string>, element: Element | DocumentFragment ): number {\n\t\treturn element._appendChild( items );\n\t}\n\n\t/**\n\t * Inserts a child node or a list of child nodes on the given index and sets the parent of these nodes to\n\t * this element.\n\t *\n\t * @see module:engine/view/element~Element#_insertChild\n\t * @param index Offset at which nodes should be inserted.\n\t * @param items Items to be inserted.\n\t * @param element Element to which items will be inserted.\n\t * @returns Number of inserted nodes.\n\t */\n\tpublic insertChild( index: number, items: Item | Iterable<Item>, element: Element | DocumentFragment ): number {\n\t\treturn element._insertChild( index, items );\n\t}\n\n\t/**\n\t * Removes the given number of child nodes starting at the given index and set the parent of these nodes to `null`.\n\t *\n\t * @see module:engine/view/element~Element#_removeChildren\n\t * @param index Offset from which nodes will be removed.\n\t * @param howMany Number of nodes to remove.\n\t * @param element Element which children will be removed.\n\t * @returns The array containing removed nodes.\n\t */\n\tpublic removeChildren( index: number, howMany: number, element: Element | DocumentFragment ): Array<Node> {\n\t\treturn element._removeChildren( index, howMany );\n\t}\n\n\t/**\n\t * Removes given element from the view structure. Will not have effect on detached elements.\n\t *\n\t * @param element Element which will be removed.\n\t * @returns The array containing removed nodes.\n\t */\n\tpublic remove( element: Node ): Array<Node> {\n\t\tconst parent = element.parent;\n\n\t\tif ( parent ) {\n\t\t\treturn this.removeChildren( parent.getChildIndex( element ), 1, parent );\n\t\t}\n\n\t\treturn [];\n\t}\n\n\t/**\n\t * Replaces given element with the new one in the view structure. Will not have effect on detached elements.\n\t *\n\t * @param oldElement Element which will be replaced.\n\t * @param newElement Element which will be inserted in the place of the old element.\n\t * @returns Whether old element was successfully replaced.\n\t */\n\tpublic replace( oldElement: Element, newElement: Element ): boolean {\n\t\tconst parent = oldElement.parent;\n\n\t\tif ( parent ) {\n\t\t\tconst index = parent.getChildIndex( oldElement )!;\n\n\t\t\tthis.removeChildren( index, 1, parent );\n\t\t\tthis.insertChild( index, newElement, parent );\n\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Removes given element from view structure and places its children in its position.\n\t * It does nothing if element has no parent.\n\t *\n\t * @param element Element to unwrap.\n\t */\n\tpublic unwrapElement( element: Element ): void {\n\t\tconst parent = element.parent;\n\n\t\tif ( parent ) {\n\t\t\tconst index = parent.getChildIndex( element )!;\n\n\t\t\tthis.remove( element );\n\t\t\tthis.insertChild( index, element.getChildren(), parent );\n\t\t}\n\t}\n\n\t/**\n\t * Renames element by creating a copy of a given element but with its name changed and then moving contents of the\n\t * old element to the new one.\n\t *\n\t * Since this function creates a new element and removes the given one, the new element is returned to keep reference.\n\t *\n\t * @param newName New element name.\n\t * @param element Element to be renamed.\n\t * @returns New element or null if the old element was not replaced (happens for detached elements).\n\t */\n\tpublic rename( newName: string, element: Element ): Element | null {\n\t\tconst newElement = new Element( this.document, newName, element.getAttributes(), element.getChildren() );\n\n\t\treturn this.replace( element, newElement ) ? newElement : null;\n\t}\n\n\t/**\n\t * Adds or overwrites element's attribute with a specified key and value.\n\t *\n\t * ```ts\n\t * writer.setAttribute( 'href', 'http://ckeditor.com', linkElement );\n\t * ```\n\t *\n\t * @see module:engine/view/element~Element#_setAttribute\n\t * @param key Attribute key.\n\t * @param value Attribute value.\n\t * @param element Element for which attribute will be set.\n\t */\n\tpublic setAttribute( key: string, value: unknown, element: Element ): void {\n\t\telement._setAttribute( key, value );\n\t}\n\n\t/**\n\t * Removes attribute from the element.\n\t *\n\t * ```ts\n\t * writer.removeAttribute( 'href', linkElement );\n\t * ```\n\t *\n\t * @see module:engine/view/element~Element#_removeAttribute\n\t * @param key Attribute key.\n\t * @param element Element from which attribute will be removed.\n\t */\n\tpublic removeAttribute( key: string, element: Element ): void {\n\t\telement._removeAttribute( key );\n\t}\n\n\t/**\n\t * Adds specified class to the element.\n\t *\n\t * ```ts\n\t * writer.addClass( 'foo', linkElement );\n\t * writer.addClass( [ 'foo', 'bar' ], linkElement );\n\t * ```\n\t *\n\t * @see module:engine/view/element~Element#_addClass\n\t * @param className Single class name or array of class names which will be added.\n\t * @param element Element for which class will be added.\n\t */\n\tpublic addClass( className: string | Array<string>, element: Element ): void {\n\t\telement._addClass( className );\n\t}\n\n\t/**\n\t * Removes specified class from the element.\n\t *\n\t * ```ts\n\t * writer.removeClass( 'foo', linkElement );\n\t * writer.removeClass( [ 'foo', 'bar' ], linkElement );\n\t * ```\n\t *\n\t * @see module:engine/view/element~Element#_removeClass\n\t * @param className Single class name or array of class names which will be removed.\n\t * @param element Element from which class will be removed.\n\t */\n\tpublic removeClass( className: string | Array<string>, element: Element ): void {\n\t\telement._removeClass( className );\n\t}\n\n\t/**\n\t * Adds style to the element.\n\t *\n\t * ```ts\n\t * writer.setStyle( 'color', 'red', element );\n\t * ```\n\t *\n\t * **Note**: This method can work with normalized style names if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#set `StylesMap#set()`} for details.\n\t *\n\t * @see module:engine/view/element~Element#_setStyle\n\t * @label KEY_VALUE\n\t * @param property Property name.\n\t * @param value Value to set.\n\t * @param element Element for which style will be added.\n\t */\n\tpublic setStyle( property: string, value: string, element: Element ): void;\n\n\t/**\n\t * Adds style to the element.\n\t *\n\t * ```ts\n\t * writer.setStyle( {\n\t * \tcolor: 'red',\n\t * \tposition: 'fixed'\n\t * }, element );\n\t * ```\n\t *\n\t * **Note**: This method can work with normalized style names if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#set `StylesMap#set()`} for details.\n\t *\n\t * @see module:engine/view/element~Element#_setStyle\n\t * @label OBJECT\n\t * @param properties Object with key - value pairs.\n\t * @param element Element for which style will be added.\n\t */\n\tpublic setStyle( properties: Record<string, string>, element: Element ): void;\n\n\tpublic setStyle( property: string | Record<string, string>, valueOrElement: string | Element, element?: Element ): void {\n\t\tif ( isPlainObject( property ) && element === undefined ) {\n\t\t\t( valueOrElement as Element )._setStyle( property as Record<string, string> );\n\t\t} else {\n\t\t\telement!._setStyle( property as string, valueOrElement as string );\n\t\t}\n\t}\n\n\t/**\n\t * Removes specified style from the element.\n\t *\n\t * ```ts\n\t * writer.removeStyle( 'color', element ); // Removes 'color' style.\n\t * writer.removeStyle( [ 'color', 'border-top' ], element ); // Removes both 'color' and 'border-top' styles.\n\t * ```\n\t *\n\t * **Note**: This method can work with normalized style names if\n\t * {@link module:engine/controller/datacontroller~DataController#addStyleProcessorRules a particular style processor rule is enabled}.\n\t * See {@link module:engine/view/stylesmap~StylesMap#remove `StylesMap#remove()`} for details.\n\t *\n\t * @see module:engine/view/element~Element#_removeStyle\n\t * @param property Style property name or names to be removed.\n\t * @param element Element from which style will be removed.\n\t */\n\tpublic removeStyle( property: string | Array<string>, element: Element ): void {\n\t\telement._removeStyle( property );\n\t}\n\n\t/**\n\t * Sets a custom property on element. Unlike attributes, custom properties are not rendered to the DOM,\n\t * so they can be used to add special data to elements.\n\t *\n\t * @see module:engine/view/element~Element#_setCustomProperty\n\t * @param key Custom property name/key.\n\t * @param value Custom property value to be stored.\n\t * @param element Element for which custom property will be set.\n\t */\n\tpublic setCustomProperty( key: string | symbol, value: unknown, element: Element | DocumentFragment ): void {\n\t\telement._setCustomProperty( key, value );\n\t}\n\n\t/**\n\t * Removes a custom property stored under the given key.\n\t *\n\t * @see module:engine/view/element~Element#_removeCustomProperty\n\t * @param key Name/key of the custom property to be removed.\n\t * @param element Element from which the custom property will be removed.\n\t * @returns Returns true if property was removed.\n\t */\n\tpublic removeCustomProperty( key: string | symbol, element: Element | DocumentFragment ): boolean {\n\t\treturn element._removeCustomProperty( key );\n\t}\n\n\t/**\n\t * Creates position at the given location. The location can be specified as:\n\t *\n\t * * a {@link module:engine/view/position~Position position},\n\t * * parent element and offset (offset defaults to `0`),\n\t * * parent element and `'end'` (sets position at the end of that element),\n\t * * {@link module:engine/view/item~Item view item} and `'before'` or `'after'` (sets position before or after given view item).\n\t *\n\t * This method is a shortcut to other constructors such as:\n\t *\n\t * * {@link #createPositionBefore},\n\t * * {@link #createPositionAfter},\n\t *\n\t * @param offset Offset or one of the flags. Used only when first parameter is a {@link module:engine/view/item~Item view item}.\n\t */\n\tpublic createPositionAt( itemOrPosition: Item | Position, offset?: PositionOffset ): Position {\n\t\treturn Position._createAt( itemOrPosition, offset );\n\t}\n\n\t/**\n\t * Creates a new position after given view item.\n\t *\n\t * @param item View item after which the position should be located.\n\t */\n\tpublic createPositionAfter( item: Item ): Position {\n\t\treturn Position._createAfter( item );\n\t}\n\n\t/**\n\t * Creates a new position before given view item.\n\t *\n\t * @param item View item before which the position should be located.\n\t */\n\tpublic createPositionBefore( item: Item ): Position {\n\t\treturn Position._createBefore( item );\n\t}\n\n\t/**\n\t * Creates a range spanning from `start` position to `end` position.\n\t *\n\t * **Note:** This factory method creates it's own {@link module:engine/view/position~Position} instances basing on passed values.\n\t *\n\t * @param start Start position.\n\t * @param end End position. If not set, range will be collapsed at `start` position.\n\t */\n\tpublic createRange( start: Position, end: Position ): Range {\n\t\treturn new Range( start, end );\n\t}\n\n\t/**\n\t * Creates a range that starts before given {@link module:engine/view/item~Item view item} and ends after it.\n\t */\n\tpublic createRangeOn( item: Item ): Range {\n\t\treturn Range._createOn( item );\n\t}\n\n\t/**\n\t * Creates a range inside an {@link module:engine/view/element~Element element} which starts before the first child of\n\t * that element and ends after the last child of that element.\n\t *\n\t * @param element Element which is a parent for the range.\n\t */\n\tpublic createRangeIn( element: Element | DocumentFragment ): Range {\n\t\treturn Range._createIn( element );\n\t}\n\n\t/**\n\t * Creates a new {@link module:engine/view/selection~Selection} instance.\n\t *\n\t * ```ts\n\t * // Creates collapsed selection at the position of given item and offset.\n\t * const paragraph = writer.createContainerElement( 'paragraph' );\n\t * const selection = writer.createSelection( paragraph, offset );\n\t *\n\t * // Creates a range inside an {@link module:engine/view/element~Element element} which starts before the\n\t * // first child of that element and ends after the last child of that element.\n\t * const selection = writer.createSelection( paragraph, 'in' );\n\t *\n\t * // Creates a range on an {@link module:engine/view/item~Item item} which starts before the item and ends\n\t * // just after the item.\n\t * const selection = writer.createSelection( paragraph, 'on' );\n\t * ```\n\t *\n\t * `Selection`'s constructor allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = writer.createSelection( element, 'in', { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = writer.createSelection( element, 'in', { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #createSelection:SELECTABLE `createSelection( selectable, options )`}.\n\t *\n\t * @label NODE_OFFSET\n\t */\n\tpublic createSelection( selectable: Node, placeOrOffset: PlaceOrOffset, options?: SelectionOptions ): Selection;\n\n\t/**\n\t * Creates a new {@link module:engine/view/selection~Selection} instance.\n\t *\n\t * ```ts\n\t * // Creates empty selection without ranges.\n\t * const selection = writer.createSelection();\n\t *\n\t * // Creates selection at the given range.\n\t * const range = writer.createRange( start, end );\n\t * const selection = writer.createSelection( range );\n\t *\n\t * // Creates selection at the given ranges\n\t * const ranges = [ writer.createRange( start1, end2 ), writer.createRange( star2, end2 ) ];\n\t * const selection = writer.createSelection( ranges );\n\t *\n\t * // Creates selection from the other selection.\n\t * const otherSelection = writer.createSelection();\n\t * const selection = writer.createSelection( otherSelection );\n\t *\n\t * // Creates selection from the document selection.\n\t * const selection = writer.createSelection( editor.editing.view.document.selection );\n\t *\n\t * // Creates selection at the given position.\n\t * const position = writer.createPositionFromPath( root, path );\n\t * const selection = writer.createSelection( position );\n\t * ```\n\t *\n\t * `Selection`'s constructor allow passing additional options (`backward`, `fake` and `label`) as the last argument.\n\t *\n\t * ```ts\n\t * // Creates backward selection.\n\t * const selection = writer.createSelection( range, { backward: true } );\n\t * ```\n\t *\n\t * Fake selection does not render as browser native selection over selected elements and is hidden to the user.\n\t * This way, no native selection UI artifacts are displayed to the user and selection over elements can be\n\t * represented in other way, for example by applying proper CSS class.\n\t *\n\t * Additionally fake's selection label can be provided. It will be used to describe fake selection in DOM\n\t * (and be properly handled by screen readers).\n\t *\n\t * ```ts\n\t * // Creates fake selection with label.\n\t * const selection = writer.createSelection( range, { fake: true, label: 'foo' } );\n\t * ```\n\t *\n\t * See also: {@link #createSelection:NODE_OFFSET `createSelection( node, placeOrOffset, options )`}.\n\t *\n\t * @label SELECTABLE\n\t */\n\tpublic createSelection( selectable?: Exclude<Selectable, Node>, options?: SelectionOptions ): Selection;\n\n\tpublic createSelection( ...args: ConstructorParameters<typeof Selection> ): Selection {\n\t\treturn new Selection( ...args );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/commandcollection\n */\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\nimport type Command from './command';\n\n/**\n * Collection of commands. Its instance is available in {@link module:core/editor/editor~Editor#commands `editor.commands`}.\n */\nexport default class CommandCollection implements Iterable<[ string, Command ]> {\n\t/**\n\t * Command map.\n\t */\n\tprivate _commands: Map<string, Command>;\n\n\t/**\n\t * Creates collection instance.\n\t */\n\tconstructor() {\n\t\tthis._commands = new Map();\n\t}\n\n\t/**\n\t * Registers a new command.\n\t *\n\t * @param commandName The name of the command.\n\t */\n\tpublic add<TName extends string>(\n\t\tcommandName: TName,\n\t\tcommand: CommandsMap[ TName ]\n\t): void {\n\t\tthis._commands.set( commandName, command );\n\t}\n\n\t/**\n\t * Retrieves a command from the collection.\n\t *\n\t * @param commandName The name of the command.\n\t */\n\tpublic get<TName extends string>( commandName: TName ): CommandsMap[ TName ] | undefined {\n\t\treturn this._commands.get( commandName );\n\t}\n\n\t/**\n\t * Executes a command.\n\t *\n\t * @param commandName The name of the command.\n\t * @param commandParams Command parameters.\n\t * @returns The value returned by the {@link module:core/command~Command#execute `command.execute()`}.\n\t */\n\tpublic execute<TName extends string>(\n\t\tcommandName: TName,\n\t\t...commandParams: Parameters<CommandsMap[ TName ][ 'execute' ]>\n\t): ReturnType<CommandsMap[ TName ][ 'execute' ]> {\n\t\tconst command = this.get( commandName );\n\n\t\tif ( !command ) {\n\t\t\t/**\n\t\t\t * Command does not exist.\n\t\t\t *\n\t\t\t * @error commandcollection-command-not-found\n\t\t\t * @param commandName Name of the command.\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'commandcollection-command-not-found', this, { commandName } );\n\t\t}\n\n\t\treturn command.execute( ...commandParams ) as any;\n\t}\n\n\t/**\n\t * Returns iterator of command names.\n\t */\n\tpublic* names(): IterableIterator<string> {\n\t\tyield* this._commands.keys();\n\t}\n\n\t/**\n\t * Returns iterator of command instances.\n\t */\n\tpublic* commands(): IterableIterator<Command> {\n\t\tyield* this._commands.values();\n\t}\n\n\t/**\n\t * Iterable interface.\n\t *\n\t * Returns `[ commandName, commandInstance ]` pairs.\n\t */\n\tpublic [ Symbol.iterator ](): Iterator<[ string, Command ]> {\n\t\treturn this._commands[ Symbol.iterator ]();\n\t}\n\n\t/**\n\t * Destroys all collection commands.\n\t */\n\tpublic destroy(): void {\n\t\tfor ( const command of this.commands() ) {\n\t\t\tcommand.destroy();\n\t\t}\n\t}\n}\n\n/**\n * Helper type that maps command names to their types.\n * It is meant to be extended with module augmentation.\n *\n * ```ts\n * class MyCommand extends Command {\n * \tpublic execute( parameter: A ): B {\n * \t\t// ...\n * \t}\n * }\n *\n * declare module '@ckeditor/ckeditor5-core' {\n * \tinterface CommandsMap {\n * \t\tmyCommand: MyCommand;\n * \t}\n * }\n *\n * // Returns `MyCommand | undefined`.\n * const myCommand = editor.commands.get( 'myCommand' );\n *\n * // Expects `A` type as parameter and returns `B`.\n * const value = editor.commands.execute( 'myCommand', new A() );\n * ```\n */\nexport interface CommandsMap {\n\t[ name: string ]: Command;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/editor/editor\n */\n\nimport {\n\tConfig,\n\tCKEditorError,\n\tObservableMixin,\n\ttype Locale,\n\ttype LocaleTranslate,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport {\n\tConversion,\n\tDataController,\n\tEditingController,\n\tModel,\n\tStylesProcessor\n} from '@ckeditor/ckeditor5-engine';\n\nimport type { EditorUI } from '@ckeditor/ckeditor5-ui';\n\nimport Context from '../context';\nimport PluginCollection from '../plugincollection';\nimport CommandCollection, { type CommandsMap } from '../commandcollection';\nimport EditingKeystrokeHandler from '../editingkeystrokehandler';\n\nimport type { LoadedPlugins, PluginConstructor } from '../plugin';\nimport type { EditorConfig } from './editorconfig';\n\n/**\n * The class representing a basic, generic editor.\n *\n * Check out the list of its subclasses to learn about specific editor implementations.\n *\n * All editor implementations (like {@link module:editor-classic/classiceditor~ClassicEditor} or\n * {@link module:editor-inline/inlineeditor~InlineEditor}) should extend this class. They can add their\n * own methods and properties.\n *\n * When you are implementing a plugin, this editor represents the API\n * which your plugin can expect to get when using its {@link module:core/plugin~Plugin#editor} property.\n *\n * This API should be sufficient in order to implement the \"editing\" part of your feature\n * (schema definition, conversion, commands, keystrokes, etc.).\n * It does not define the editor UI, which is available only if\n * the specific editor implements also the {@link ~Editor#ui} property\n * (as most editor implementations do).\n */\nexport default abstract class Editor extends ObservableMixin() {\n\t/**\n\t * Commands registered to the editor.\n\t *\n\t * Use the shorthand {@link #execute `editor.execute()`} method to execute commands:\n\t *\n\t * ```ts\n\t * // Execute the bold command:\n\t * editor.execute( 'bold' );\n\t *\n\t * // Check the state of the bold command:\n\t * editor.commands.get( 'bold' ).value;\n\t * ```\n\t */\n\tpublic readonly commands: CommandCollection;\n\n\t/**\n\t * Stores all configurations specific to this editor instance.\n\t *\n\t * ```ts\n\t * editor.config.get( 'image.toolbar' );\n\t * // -> [ 'imageStyle:block', 'imageStyle:side', '|', 'toggleImageCaption', 'imageTextAlternative' ]\n\t * ```\n\t */\n\tpublic readonly config: Config<EditorConfig>;\n\n\t/**\n\t * Conversion manager through which you can register model-to-view and view-to-model converters.\n\t *\n\t * See the {@link module:engine/conversion/conversion~Conversion} documentation to learn how to add converters.\n\t */\n\tpublic readonly conversion: Conversion;\n\n\t/**\n\t * The {@link module:engine/controller/datacontroller~DataController data controller}.\n\t * Used e.g. for setting and retrieving the editor data.\n\t */\n\tpublic readonly data: DataController;\n\n\t/**\n\t * The {@link module:engine/controller/editingcontroller~EditingController editing controller}.\n\t * Controls user input and rendering the content for editing.\n\t */\n\tpublic readonly editing: EditingController;\n\n\t/**\n\t * The locale instance.\n\t */\n\tpublic readonly locale: Locale;\n\n\t/**\n\t * The editor's model.\n\t *\n\t * The central point of the editor's abstract data model.\n\t */\n\tpublic readonly model: Model;\n\n\t/**\n\t * The plugins loaded and in use by this editor instance.\n\t *\n\t * ```ts\n\t * editor.plugins.get( 'ClipboardPipeline' ); // -> An instance of the clipboard pipeline plugin.\n\t * ```\n\t */\n\tpublic readonly plugins: PluginCollection<Editor>;\n\n\t/**\n\t * An instance of the {@link module:core/editingkeystrokehandler~EditingKeystrokeHandler}.\n\t *\n\t * It allows setting simple keystrokes:\n\t *\n\t * ```ts\n\t * // Execute the bold command on Ctrl+E:\n\t * editor.keystrokes.set( 'Ctrl+E', 'bold' );\n\t *\n\t * // Execute your own callback:\n\t * editor.keystrokes.set( 'Ctrl+E', ( data, cancel ) => {\n\t * \tconsole.log( data.keyCode );\n\t *\n\t * \t// Prevent the default (native) action and stop the underlying keydown event\n\t * \t// so no other editor feature will interfere.\n\t * \tcancel();\n\t * } );\n\t * ```\n\t *\n\t * Note: Certain typing-oriented keystrokes (like <kbd>Backspace</kbd> or <kbd>Enter</kbd>) are handled\n\t * by a low-level mechanism and trying to listen to them via the keystroke handler will not work reliably.\n\t * To handle these specific keystrokes, see the events fired by the\n\t * {@link module:engine/view/document~Document editing view document} (`editor.editing.view.document`).\n\t */\n\tpublic readonly keystrokes: EditingKeystrokeHandler;\n\n\t/**\n\t * Shorthand for {@link module:utils/locale~Locale#t}.\n\t *\n\t * @see module:utils/locale~Locale#t\n\t */\n\tpublic readonly t: LocaleTranslate;\n\n\tpublic declare readonly id: string;\n\n\t/**\n\t * Indicates the editor life-cycle state.\n\t *\n\t * The editor is in one of the following states:\n\t *\n\t * * `initializing` &ndash; During the editor initialization (before\n\t * {@link module:core/editor/editor~Editor.create `Editor.create()`}) finished its job.\n\t * * `ready` &ndash; After the promise returned by the {@link module:core/editor/editor~Editor.create `Editor.create()`}\n\t * method is resolved.\n\t * * `destroyed` &ndash; Once the {@link #destroy `editor.destroy()`} method was called.\n\t *\n\t * @observable\n\t */\n\tpublic declare state: 'initializing' | 'ready' | 'destroyed';\n\n\t/**\n\t * The default configuration which is built into the editor class.\n\t *\n\t * It is used in CKEditor 5 builds to provide the default configuration options which are later used during the editor initialization.\n\t *\n\t * ```ts\n\t * ClassicEditor.defaultConfig = {\n\t * \tfoo: 1,\n\t * \tbar: 2\n\t * };\n\t *\n\t * ClassicEditor\n\t * \t.create( sourceElement )\n\t * \t.then( editor => {\n\t * \t\teditor.config.get( 'foo' ); // -> 1\n\t * \t\teditor.config.get( 'bar' ); // -> 2\n\t * \t} );\n\t *\n\t * // The default options can be overridden by the configuration passed to create().\n\t * ClassicEditor\n\t * \t.create( sourceElement, { bar: 3 } )\n\t * \t.then( editor => {\n\t * \t\teditor.config.get( 'foo' ); // -> 1\n\t * \t\teditor.config.get( 'bar' ); // -> 3\n\t * \t} );\n\t * ```\n\t *\n\t * See also {@link module:core/editor/editor~Editor.builtinPlugins}.\n\t */\n\tpublic static defaultConfig?: EditorConfig;\n\n\t/**\n\t * An array of plugins built into this editor class.\n\t *\n\t * It is used in CKEditor 5 builds to provide a list of plugins which are later automatically initialized\n\t * during the editor initialization.\n\t *\n\t * They will be automatically initialized by the editor, unless listed in `config.removePlugins` and\n\t * unless `config.plugins` is passed.\n\t *\n\t * ```ts\n\t * // Build some plugins into the editor class first.\n\t * ClassicEditor.builtinPlugins = [ FooPlugin, BarPlugin ];\n\t *\n\t * // Normally, you need to define config.plugins, but since ClassicEditor.builtinPlugins was\n\t * // defined, now you can call create() without any configuration.\n\t * ClassicEditor\n\t * \t.create( sourceElement )\n\t * \t.then( editor => {\n\t * \t\teditor.plugins.get( FooPlugin ); // -> An instance of the Foo plugin.\n\t * \t\teditor.plugins.get( BarPlugin ); // -> An instance of the Bar plugin.\n\t * \t} );\n\t *\n\t * ClassicEditor\n\t * \t.create( sourceElement, {\n\t * \t\t// Do not initialize these plugins (note: it is defined by a string):\n\t * \t\tremovePlugins: [ 'Foo' ]\n\t * \t} )\n\t * \t.then( editor => {\n\t * \t\teditor.plugins.get( FooPlugin ); // -> Undefined.\n\t * \t\teditor.config.get( BarPlugin ); // -> An instance of the Bar plugin.\n\t * \t} );\n\t *\n\t * ClassicEditor\n\t * \t.create( sourceElement, {\n\t * \t\t// Load only this plugin. It can also be defined by a string if\n\t * \t\t// this plugin was built into the editor class.\n\t * \t\tplugins: [ FooPlugin ]\n\t * \t} )\n\t * \t.then( editor => {\n\t * \t\teditor.plugins.get( FooPlugin ); // -> An instance of the Foo plugin.\n\t * \t\teditor.config.get( BarPlugin ); // -> Undefined.\n\t * \t} );\n\t * ```\n\t *\n\t * See also {@link module:core/editor/editor~Editor.defaultConfig}.\n\t */\n\tpublic static builtinPlugins?: Array<PluginConstructor<Editor>>;\n\n\t/**\n\t * The editor UI instance.\n\t */\n\tpublic abstract get ui(): EditorUI;\n\n\t/**\n\t * The editor context.\n\t * When it is not provided through the configuration, the editor creates it.\n\t */\n\tprotected readonly _context: Context;\n\n\t/**\n\t * A set of lock IDs for the {@link #isReadOnly} getter.\n\t */\n\tprotected readonly _readOnlyLocks: Set<symbol | string>;\n\n\t/**\n\t * Creates a new instance of the editor class.\n\t *\n\t * Usually, not to be used directly. See the static {@link module:core/editor/editor~Editor.create `create()`} method.\n\t *\n\t * @param config The editor configuration.\n\t */\n\tconstructor( config: EditorConfig = {} ) {\n\t\tsuper();\n\n\t\tconst constructor = this.constructor as typeof Editor;\n\n\t\t// Prefer the language passed as the argument to the constructor instead of the constructor's `defaultConfig`, if both are set.\n\t\tconst language = config.language || ( constructor.defaultConfig && constructor.defaultConfig.language );\n\n\t\tthis._context = config.context || new Context( { language } );\n\t\tthis._context._addEditor( this, !config.context );\n\n\t\t// Clone the plugins to make sure that the plugin array will not be shared\n\t\t// between editors and make the watchdog feature work correctly.\n\t\tconst availablePlugins = Array.from( constructor.builtinPlugins || [] );\n\n\t\tthis.config = new Config<EditorConfig>( config, constructor.defaultConfig );\n\t\tthis.config.define( 'plugins', availablePlugins );\n\t\tthis.config.define( this._context._getEditorConfig() );\n\n\t\tthis.plugins = new PluginCollection<Editor>( this, availablePlugins, this._context.plugins );\n\n\t\tthis.locale = this._context.locale;\n\t\tthis.t = this.locale.t;\n\n\t\tthis._readOnlyLocks = new Set();\n\n\t\tthis.commands = new CommandCollection();\n\n\t\tthis.set( 'state', 'initializing' );\n\t\tthis.once<EditorReadyEvent>( 'ready', () => ( this.state = 'ready' ), { priority: 'high' } );\n\t\tthis.once<EditorDestroyEvent>( 'destroy', () => ( this.state = 'destroyed' ), { priority: 'high' } );\n\n\t\tthis.model = new Model();\n\n\t\tthis.on( 'change:isReadOnly', () => {\n\t\t\tthis.model.document.isReadOnly = this.isReadOnly;\n\t\t} );\n\n\t\tconst stylesProcessor = new StylesProcessor();\n\n\t\tthis.data = new DataController( this.model, stylesProcessor );\n\n\t\tthis.editing = new EditingController( this.model, stylesProcessor );\n\t\tthis.editing.view.document.bind( 'isReadOnly' ).to( this );\n\n\t\tthis.conversion = new Conversion( [ this.editing.downcastDispatcher, this.data.downcastDispatcher ], this.data.upcastDispatcher );\n\t\tthis.conversion.addAlias( 'dataDowncast', this.data.downcastDispatcher );\n\t\tthis.conversion.addAlias( 'editingDowncast', this.editing.downcastDispatcher );\n\n\t\tthis.keystrokes = new EditingKeystrokeHandler( this );\n\t\tthis.keystrokes.listenTo( this.editing.view.document );\n\t}\n\n\t/**\n\t * Defines whether the editor is in the read-only mode.\n\t *\n\t * In read-only mode the editor {@link #commands commands} are disabled so it is not possible\n\t * to modify the document by using them. Also, the editable element(s) become non-editable.\n\t *\n\t * In order to make the editor read-only, you need to call the {@link #enableReadOnlyMode} method:\n\t *\n\t * ```ts\n\t * editor.enableReadOnlyMode( 'feature-id' );\n\t * ```\n\t *\n * Later, to turn off the read-only mode, call {@link #disableReadOnlyMode}:\n\t *\n\t * ```ts\n\t * editor.disableReadOnlyMode( 'feature-id' );\n\t * ```\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tpublic get isReadOnly(): boolean {\n\t\treturn this._readOnlyLocks.size > 0;\n\t}\n\n\tpublic set isReadOnly( value: boolean ) {\n\t\t/**\n\t\t * The {@link module:core/editor/editor~Editor#isReadOnly Editor#isReadOnly} property is read-only since version `34.0.0`\n\t\t * and can be set only using {@link module:core/editor/editor~Editor#enableReadOnlyMode `Editor#enableReadOnlyMode( lockId )`} and\n\t\t * {@link module:core/editor/editor~Editor#disableReadOnlyMode `Editor#disableReadOnlyMode( lockId )`}.\n\t\t *\n\t\t * Usage before version `34.0.0`:\n\t\t *\n\t\t * ```ts\n\t\t * editor.isReadOnly = true;\n\t\t * editor.isReadOnly = false;\n\t\t * ```\n\t\t *\n\t\t * Usage since version `34.0.0`:\n\t\t *\n\t\t * ```ts\n\t\t * editor.enableReadOnlyMode( 'my-feature-id' );\n\t\t * editor.disableReadOnlyMode( 'my-feature-id' );\n\t\t * ```\n\t\t *\n\t\t * @error editor-isreadonly-has-no-setter\n\t\t */\n\t\tthrow new CKEditorError( 'editor-isreadonly-has-no-setter' );\n\t}\n\n\t/**\n\t * Turns on the read-only mode in the editor.\n\t *\n\t * Editor can be switched to or out of the read-only mode by many features, under various circumstances. The editor supports locking\n\t * mechanism for the read-only mode. It enables easy control over the read-only mode when many features wants to turn it on or off at\n\t * the same time, without conflicting with each other. It guarantees that you will not make the editor editable accidentally (which\n\t * could lead to errors).\n\t *\n\t * Each read-only mode request is identified by a unique id (also called \"lock\"). If multiple plugins requested to turn on the\n\t * read-only mode, then, the editor will become editable only after all these plugins turn the read-only mode off (using the same ids).\n\t *\n\t * Note, that you cannot force the editor to disable the read-only mode if other plugins set it.\n\t *\n\t * After the first `enableReadOnlyMode()` call, the {@link #isReadOnly `isReadOnly` property} will be set to `true`:\n\t *\n\t * ```ts\n\t * editor.isReadOnly; // `false`.\n\t * editor.enableReadOnlyMode( 'my-feature-id' );\n\t * editor.isReadOnly; // `true`.\n\t * ```\n\t *\n\t * You can turn off the read-only mode (\"clear the lock\") using the {@link #disableReadOnlyMode `disableReadOnlyMode()`} method:\n\t *\n\t * ```ts\n\t * editor.enableReadOnlyMode( 'my-feature-id' );\n\t * // ...\n\t * editor.disableReadOnlyMode( 'my-feature-id' );\n\t * editor.isReadOnly; // `false`.\n\t * ```\n\t *\n\t * All \"locks\" need to be removed to enable editing:\n\t *\n\t * ```ts\n\t * editor.enableReadOnlyMode( 'my-feature-id' );\n\t * editor.enableReadOnlyMode( 'my-other-feature-id' );\n\t * // ...\n\t * editor.disableReadOnlyMode( 'my-feature-id' );\n\t * editor.isReadOnly; // `true`.\n\t * editor.disableReadOnlyMode( 'my-other-feature-id' );\n\t * editor.isReadOnly; // `false`.\n\t * ```\n\t *\n\t * @param lockId A unique ID for setting the editor to the read-only state.\n\t */\n\tpublic enableReadOnlyMode( lockId: string | symbol ): void {\n\t\tif ( typeof lockId !== 'string' && typeof lockId !== 'symbol' ) {\n\t\t\t/**\n\t\t\t * The lock ID is missing or it is not a string or symbol.\n\t\t\t *\n\t\t\t * @error editor-read-only-lock-id-invalid\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'editor-read-only-lock-id-invalid', null, { lockId } );\n\t\t}\n\n\t\tif ( this._readOnlyLocks.has( lockId ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._readOnlyLocks.add( lockId );\n\n\t\tif ( this._readOnlyLocks.size === 1 ) {\n\t\t\t// Manually fire the `change:isReadOnly` event as only getter is provided.\n\t\t\tthis.fire<ObservableChangeEvent<boolean>>( 'change:isReadOnly', 'isReadOnly', true, false );\n\t\t}\n\t}\n\n\t/**\n\t * Removes the read-only lock from the editor with given lock ID.\n\t *\n\t * When no lock is present on the editor anymore, then the {@link #isReadOnly `isReadOnly` property} will be set to `false`.\n\t *\n\t * @param lockId The lock ID for setting the editor to the read-only state.\n\t */\n\tpublic disableReadOnlyMode( lockId: string | symbol ): void {\n\t\tif ( typeof lockId !== 'string' && typeof lockId !== 'symbol' ) {\n\t\t\tthrow new CKEditorError( 'editor-read-only-lock-id-invalid', null, { lockId } );\n\t\t}\n\n\t\tif ( !this._readOnlyLocks.has( lockId ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._readOnlyLocks.delete( lockId );\n\n\t\tif ( this._readOnlyLocks.size === 0 ) {\n\t\t\t// Manually fire the `change:isReadOnly` event as only getter is provided.\n\t\t\tthis.fire<ObservableChangeEvent<boolean>>( 'change:isReadOnly', 'isReadOnly', false, true );\n\t\t}\n\t}\n\n\t/**\n\t * Loads and initializes plugins specified in the configuration.\n\t *\n\t * @returns A promise which resolves once the initialization is completed, providing an array of loaded plugins.\n\t */\n\tpublic initPlugins(): Promise<LoadedPlugins> {\n\t\tconst config = this.config;\n\t\tconst plugins = config.get( 'plugins' )!;\n\t\tconst removePlugins = config.get( 'removePlugins' ) || [];\n\t\tconst extraPlugins = config.get( 'extraPlugins' ) || [];\n\t\tconst substitutePlugins = config.get( 'substitutePlugins' ) || [];\n\n\t\treturn this.plugins.init( plugins.concat( extraPlugins ), removePlugins, substitutePlugins );\n\t}\n\n\t/**\n\t * Destroys the editor instance, releasing all resources used by it.\n\t *\n\t * **Note** The editor cannot be destroyed during the initialization phase so if it is called\n\t * while the editor {@link #state is being initialized}, it will wait for the editor initialization before destroying it.\n\t *\n\t * @fires destroy\n\t * @returns A promise that resolves once the editor instance is fully destroyed.\n\t */\n\tpublic destroy(): Promise<unknown> {\n\t\tlet readyPromise: Promise<unknown> = Promise.resolve();\n\n\t\tif ( this.state == 'initializing' ) {\n\t\t\treadyPromise = new Promise( resolve => this.once<EditorReadyEvent>( 'ready', resolve ) );\n\t\t}\n\n\t\treturn readyPromise\n\t\t\t.then( () => {\n\t\t\t\tthis.fire<EditorDestroyEvent>( 'destroy' );\n\t\t\t\tthis.stopListening();\n\t\t\t\tthis.commands.destroy();\n\t\t\t} )\n\t\t\t.then( () => this.plugins.destroy() )\n\t\t\t.then( () => {\n\t\t\t\tthis.model.destroy();\n\t\t\t\tthis.data.destroy();\n\t\t\t\tthis.editing.destroy();\n\t\t\t\tthis.keystrokes.destroy();\n\t\t\t} )\n\t\t\t// Remove the editor from the context.\n\t\t\t// When the context was created by this editor, the context will be destroyed.\n\t\t\t.then( () => this._context._removeEditor( this ) );\n\t}\n\n\t/**\n\t * Executes the specified command with given parameters.\n\t *\n\t * Shorthand for:\n\t *\n\t * ```ts\n\t * editor.commands.get( commandName ).execute( ... );\n\t * ```\n\t *\n\t * @param commandName The name of the command to execute.\n\t * @param commandParams Command parameters.\n\t * @returns The value returned by the {@link module:core/commandcollection~CommandCollection#execute `commands.execute()`}.\n\t */\n\tpublic execute<TName extends string>(\n\t\tcommandName: TName,\n\t\t...commandParams: Parameters<CommandsMap[ TName ][ 'execute' ]>\n\t): ReturnType<CommandsMap[ TName ][ 'execute' ]> {\n\t\ttry {\n\t\t\treturn this.commands.execute( commandName, ...commandParams );\n\t\t} catch ( err: any ) {\n\t\t\t// @if CK_DEBUG // throw err;\n\t\t\t/* istanbul ignore next -- @preserve */\n\t\t\tCKEditorError.rethrowUnexpectedError( err, this );\n\t\t}\n\t}\n\n\t/**\n\t * Focuses the editor.\n\t *\n\t * **Note** To explicitly focus the editing area of the editor, use the\n\t * {@link module:engine/view/view~View#focus `editor.editing.view.focus()`} method of the editing view.\n\t *\n\t * Check out the {@glink framework/deep-dive/ui/focus-tracking#focus-in-the-editor-ui Focus in the editor UI} section\n\t * of the {@glink framework/deep-dive/ui/focus-tracking Deep dive into focus tracking} guide to learn more.\n\t */\n\tpublic focus(): void {\n\t\tthis.editing.view.focus();\n\t}\n\n\t/* istanbul ignore next -- @preserve */\n\t/**\n\t * Creates and initializes a new editor instance.\n\t *\n\t * This is an abstract method. Every editor type needs to implement its own initialization logic.\n\t *\n\t * See the `create()` methods of the existing editor types to learn how to use them:\n\t *\n\t * * {@link module:editor-classic/classiceditor~ClassicEditor.create `ClassicEditor.create()`}\n\t * * {@link module:editor-balloon/ballooneditor~BalloonEditor.create `BalloonEditor.create()`}\n\t * * {@link module:editor-decoupled/decouplededitor~DecoupledEditor.create `DecoupledEditor.create()`}\n\t * * {@link module:editor-inline/inlineeditor~InlineEditor.create `InlineEditor.create()`}\n\t */\n\tpublic static create( ...args: Array<unknown> ): void {\n\t\tthrow new Error( 'This is an abstract method.' );\n\t}\n}\n\n/**\n * Fired when the {@link module:engine/controller/datacontroller~DataController#event:ready data} and all additional\n * editor components are ready.\n *\n * Note: This event is most useful for plugin developers. When integrating the editor with your website or\n * application, you do not have to listen to `editor#ready` because when the promise returned by the static\n * {@link module:core/editor/editor~Editor.create `Editor.create()`} event is resolved, the editor is already ready.\n * In fact, since the first moment when the editor instance is available to you is inside `then()`'s callback,\n * you cannot even add a listener to the `editor#ready` event.\n *\n * See also the {@link module:core/editor/editor~Editor#state `editor.state`} property.\n *\n * @eventName ~Editor#ready\n */\nexport type EditorReadyEvent = {\n\tname: 'ready';\n\targs: [];\n};\n\n/**\n * Fired when this editor instance is destroyed. The editor at this point is not usable and this event should be used to\n * perform the clean-up in any plugin.\n *\n * See also the {@link module:core/editor/editor~Editor#state `editor.state`} property.\n *\n * @eventName ~Editor#destroy\n */\nexport type EditorDestroyEvent = {\n\tname: 'destroy';\n\targs: [];\n};\n\n/**\n * This error is thrown when trying to pass a `<textarea>` element to a `create()` function of an editor class.\n *\n * The only editor type which can be initialized on `<textarea>` elements is\n * the {@glink installation/getting-started/predefined-builds#classic-editor classic editor}.\n * This editor hides the passed element and inserts its own UI next to it. Other types of editors reuse the passed element as their root\n * editable element and therefore `<textarea>` is not appropriate for them. Use a `<div>` or another text container instead:\n *\n * ```html\n * <div id=\"editor\">\n * \t<p>Initial content.</p>\n * </div>\n * ```\n *\n * @error editor-wrong-element\n */\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/editor/utils/dataapimixin\n */\n\nimport type Editor from '../editor';\n\nimport type { Constructor, Mixed } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Implementation of the {@link module:core/editor/utils/dataapimixin~DataApi}.\n */\nexport default function DataApiMixin<Base extends Constructor<Editor>>( base: Base ): Mixed<Base, DataApi> {\n\tabstract class Mixin extends base implements DataApi {\n\t\tpublic setData( data: string | Record<string, string> ): void {\n\t\t\tthis.data.set( data );\n\t\t}\n\n\t\tpublic getData( options?: {\n\t\t\trootName?: string;\n\t\t\ttrim?: 'empty' | 'none';\n\t\t\t[ key: string ]: unknown;\n\t\t} ): string {\n\t\t\treturn this.data.get( options );\n\t\t}\n\t}\n\n\treturn Mixin as any;\n}\n\n// Backward compatibility with `mix`.\n{\n\tconst mixin = ( DataApiMixin as any )( Object );\n\n\t( DataApiMixin as any ).setData = mixin.prototype.setData;\n\t( DataApiMixin as any ).getData = mixin.prototype.getData;\n}\n\n/**\n * Interface defining editor methods for setting and getting data to and from the editor's main root element\n * using the {@link module:core/editor/editor~Editor#data data pipeline}.\n *\n * This interface is not a part of the {@link module:core/editor/editor~Editor} class because one may want to implement\n * an editor with multiple root elements, in which case the methods for setting and getting data will need to be implemented\n * differently.\n */\nexport interface DataApi {\n\n\t/**\n\t * Sets the data in the editor.\n\t *\n\t * ```ts\n\t * editor.setData( '<p>This is editor!</p>' );\n\t * ```\n\t *\n\t * If your editor implementation uses multiple roots, you should pass an object with keys corresponding\n\t * to the editor root names and values equal to the data that should be set in each root:\n\t *\n\t * ```ts\n\t * editor.setData( {\n\t * header: '<p>Content for header part.</p>',\n\t * content: '<p>Content for main part.</p>',\n\t * footer: '<p>Content for footer part.</p>'\n\t * } );\n\t * ```\n\t *\n\t * By default the editor accepts HTML. This can be controlled by injecting a different data processor.\n\t * See the {@glink features/markdown Markdown output} guide for more details.\n\t *\n\t * @param data Input data.\n\t */\n\tsetData( data: string | Record<string, string> ): void;\n\n\t/**\n\t * Gets the data from the editor.\n\t *\n\t * ```ts\n\t * editor.getData(); // -> '<p>This is editor!</p>'\n\t * ```\n\t *\n\t * If your editor implementation uses multiple roots, you should pass root name as one of the options:\n\t *\n\t * ```ts\n\t * editor.getData( { rootName: 'header' } ); // -> '<p>Content for header part.</p>'\n\t * ```\n\t *\n\t * By default, the editor outputs HTML. This can be controlled by injecting a different data processor.\n\t * See the {@glink features/markdown Markdown output} guide for more details.\n\t *\n\t * A warning is logged when you try to retrieve data for a detached root, as most probably this is a mistake. A detached root should\n\t * be treated like it is removed, and you should not save its data. Note, that the detached root data is always an empty string.\n\t *\n\t * @param options Additional configuration for the retrieved data.\n\t * Editor features may introduce more configuration options that can be set through this parameter.\n\t * @param options.rootName Root name. Default to `'main'`.\n\t * @param options.trim Whether returned data should be trimmed. This option is set to `'empty'` by default,\n\t * which means that whenever editor content is considered empty, an empty string is returned. To turn off trimming\n\t * use `'none'`. In such cases exact content will be returned (for example `'<p>&nbsp;</p>'` for an empty editor).\n\t * @returns Output data.\n\t */\n\tgetData( options?: Record<string, unknown> ): string;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/editor/utils/elementapimixin\n */\n\nimport {\n\tCKEditorError,\n\tsetDataInElement,\n\ttype Constructor,\n\ttype Mixed\n} from '@ckeditor/ckeditor5-utils';\n\nimport type Editor from '../editor';\n\n/**\n * Implementation of the {@link module:core/editor/utils/elementapimixin~ElementApi}.\n */\nexport default function ElementApiMixin<Base extends Constructor<Editor>>( base: Base ): Mixed<Base, ElementApi> {\n\tabstract class Mixin extends base implements ElementApi {\n\t\tpublic sourceElement: HTMLElement | undefined;\n\n\t\tpublic updateSourceElement( data: string = this.data.get() ): void {\n\t\t\tif ( !this.sourceElement ) {\n\t\t\t\t/**\n\t\t\t\t * Cannot update the source element of a detached editor.\n\t\t\t\t *\n\t\t\t\t * The {@link module:core/editor/utils/elementapimixin~ElementApi#updateSourceElement `updateSourceElement()`}\n\t\t\t\t * method cannot be called if you did not pass an element to `Editor.create()`.\n\t\t\t\t *\n\t\t\t\t * @error editor-missing-sourceelement\n\t\t\t\t */\n\t\t\t\tthrow new CKEditorError(\n\t\t\t\t\t'editor-missing-sourceelement',\n\t\t\t\t\tthis\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tconst shouldUpdateSourceElement = this.config.get( 'updateSourceElementOnDestroy' );\n\t\t\tconst isSourceElementTextArea = this.sourceElement instanceof HTMLTextAreaElement;\n\n\t\t\t// The data returned by the editor might be unsafe, so we want to prevent rendering\n\t\t\t// unsafe content inside the source element different than <textarea>, which is considered\n\t\t\t// secure. This behaviour could be changed by setting the `updateSourceElementOnDestroy`\n\t\t\t// configuration option to `true`.\n\t\t\tif ( !shouldUpdateSourceElement && !isSourceElementTextArea ) {\n\t\t\t\tsetDataInElement( this.sourceElement, '' );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tsetDataInElement( this.sourceElement, data );\n\t\t}\n\t}\n\n\treturn Mixin as any;\n}\n\n// Backward compatibility with `mix`.\n( ElementApiMixin as any ).updateSourceElement = ( ElementApiMixin as any )( Object ).prototype.updateSourceElement;\n\n/**\n * Interface describing an editor that replaced a DOM element (was \"initialized on an element\").\n *\n * Such an editor should provide a method to\n * {@link module:core/editor/utils/elementapimixin~ElementApi#updateSourceElement update the replaced element with the current data}.\n */\nexport interface ElementApi {\n\n\t/**\n\t * The element on which the editor has been initialized.\n\t *\n\t * @readonly\n\t */\n\tsourceElement: HTMLElement | undefined;\n\n\t/**\n\t * Updates the {@link #sourceElement editor source element}'s content with the data.\n\t */\n\tupdateSourceElement( data?: string ): void;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/pendingactions\n */\n\nimport ContextPlugin from './contextplugin';\n\nimport {\n\tCKEditorError,\n\tCollection,\n\tObservableMixin,\n\ttype CollectionAddEvent,\n\ttype CollectionRemoveEvent,\n\ttype Observable\n} from '@ckeditor/ckeditor5-utils';\n\n/**\n * The list of pending editor actions.\n *\n * This plugin should be used to synchronise plugins that execute long-lasting actions\n * (e.g. file upload) with the editor integration. It gives the developer who integrates the editor\n * an easy way to check if there are any actions pending whenever such information is needed.\n * All plugins that register a pending action also provide a message about the action that is ongoing\n * which can be displayed to the user. This lets them decide if they want to interrupt the action or wait.\n *\n * Adding and updating a pending action:\n *\n * ```ts\n * const pendingActions = editor.plugins.get( 'PendingActions' );\n * const action = pendingActions.add( 'Upload in progress: 0%.' );\n *\n * // You can update the message:\n * action.message = 'Upload in progress: 10%.';\n * ```\n *\n * Removing a pending action:\n *\n * ```ts\n * const pendingActions = editor.plugins.get( 'PendingActions' );\n * const action = pendingActions.add( 'Unsaved changes.' );\n *\n * pendingActions.remove( action );\n * ```\n *\n * Getting pending actions:\n *\n * ```ts\n * const pendingActions = editor.plugins.get( 'PendingActions' );\n *\n * const action1 = pendingActions.add( 'Action 1' );\n * const action2 = pendingActions.add( 'Action 2' );\n *\n * pendingActions.first; // Returns action1\n * Array.from( pendingActions ); // Returns [ action1, action2 ]\n * ```\n *\n * This plugin is used by features like {@link module:upload/filerepository~FileRepository} to register their ongoing actions\n * and by features like {@link module:autosave/autosave~Autosave} to detect whether there are any ongoing actions.\n * Read more about saving the data in the {@glink installation/getting-started/getting-and-setting-data Saving and getting data} guide.\n */\nexport default class PendingActions extends ContextPlugin implements Iterable<PendingAction> {\n\t/**\n\t * Defines whether there is any registered pending action.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public hasAny: boolean;\n\n\t/**\n\t * A list of pending actions.\n\t */\n\tprivate _actions!: Collection<PendingAction>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'PendingActions' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tthis.set( 'hasAny', false );\n\n\t\tthis._actions = new Collection( { idProperty: '_id' } );\n\t\tthis._actions.delegate( 'add', 'remove' ).to( this );\n\t}\n\n\t/**\n\t * Adds an action to the list of pending actions.\n\t *\n\t * This method returns an action object with an observable message property.\n\t * The action object can be later used in the {@link #remove} method. It also allows you to change the message.\n\t *\n\t * @param message The action message.\n\t * @returns An observable object that represents a pending action.\n\t */\n\tpublic add( message: string ): PendingAction {\n\t\tif ( typeof message !== 'string' ) {\n\t\t\t/**\n\t\t\t * The message must be a string.\n\t\t\t *\n\t\t\t * @error pendingactions-add-invalid-message\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'pendingactions-add-invalid-message', this );\n\t\t}\n\n\t\tconst action = new ( ObservableMixin() )() as PendingAction;\n\n\t\taction.set( 'message', message );\n\t\tthis._actions.add( action );\n\t\tthis.hasAny = true;\n\n\t\treturn action;\n\t}\n\n\t/**\n\t * Removes an action from the list of pending actions.\n\t *\n\t * @param action An action object.\n\t */\n\tpublic remove( action: PendingAction ): void {\n\t\tthis._actions.remove( action );\n\t\tthis.hasAny = !!this._actions.length;\n\t}\n\n\t/**\n\t * Returns the first action from the list or null if the list is empty\n\t *\n\t * @returns The pending action object.\n\t */\n\tpublic get first(): PendingAction | null {\n\t\treturn this._actions.get( 0 );\n\t}\n\n\t/**\n\t * Iterable interface.\n\t */\n\tpublic [ Symbol.iterator ](): Iterator<PendingAction> {\n\t\treturn this._actions[ Symbol.iterator ]();\n\t}\n}\n\nexport interface PendingAction extends Observable {\n\tmessage: string;\n}\n\n/**\n * Fired when an action is added to the list.\n *\n * @eventName ~PendingActions#add\n * @param action The added action.\n */\nexport type PendingActionsAddEvent = CollectionAddEvent<PendingAction>;\n\n/**\n * Fired when an action is removed from the list.\n *\n * @eventName ~PendingActions#remove\n * @param action The removed action.\n */\nexport type PendingActionsRemoveEvent = CollectionRemoveEvent<PendingAction>;\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m11.591 10.177 4.243 4.242a1 1 0 0 1-1.415 1.415l-4.242-4.243-4.243 4.243a1 1 0 0 1-1.414-1.415l4.243-4.242L4.52 5.934A1 1 0 0 1 5.934 4.52l4.243 4.243 4.242-4.243a1 1 0 1 1 1.415 1.414l-4.243 4.243z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core\n */\n\nexport { default as Plugin, type PluginDependencies, type PluginConstructor } from './plugin';\nexport { default as Command, type CommandExecuteEvent } from './command';\nexport { default as MultiCommand } from './multicommand';\nexport type { CommandsMap } from './commandcollection';\nexport type { PluginsMap, default as PluginCollection } from './plugincollection';\n\nexport { default as Context, type ContextConfig } from './context';\nexport { default as ContextPlugin, type ContextPluginDependencies } from './contextplugin';\nexport { type EditingKeystrokeCallback } from './editingkeystrokehandler';\n\nexport { default as Editor, type EditorReadyEvent, type EditorDestroyEvent } from './editor/editor';\nexport type {\n\tEditorConfig,\n\tLanguageConfig,\n\tToolbarConfig,\n\tToolbarConfigItem,\n\tUiConfig\n} from './editor/editorconfig';\n\nexport { default as attachToForm } from './editor/utils/attachtoform';\nexport { default as DataApiMixin, type DataApi } from './editor/utils/dataapimixin';\nexport { default as ElementApiMixin, type ElementApi } from './editor/utils/elementapimixin';\nexport { default as secureSourceElement } from './editor/utils/securesourceelement';\n\nexport { default as PendingActions, type PendingAction } from './pendingactions';\n\nimport cancel from './../theme/icons/cancel.svg';\nimport caption from './../theme/icons/caption.svg';\nimport check from './../theme/icons/check.svg';\nimport cog from './../theme/icons/cog.svg';\nimport eraser from './../theme/icons/eraser.svg';\nimport lowVision from './../theme/icons/low-vision.svg';\nimport image from './../theme/icons/image.svg';\n\nimport alignBottom from './../theme/icons/align-bottom.svg';\nimport alignMiddle from './../theme/icons/align-middle.svg';\nimport alignTop from './../theme/icons/align-top.svg';\nimport alignLeft from './../theme/icons/align-left.svg';\nimport alignCenter from './../theme/icons/align-center.svg';\nimport alignRight from './../theme/icons/align-right.svg';\nimport alignJustify from './../theme/icons/align-justify.svg';\n\nimport objectBlockLeft from './../theme/icons/object-left.svg';\nimport objectCenter from './../theme/icons/object-center.svg';\nimport objectBlockRight from './../theme/icons/object-right.svg';\nimport objectFullWidth from './../theme/icons/object-full-width.svg';\nimport objectInline from './../theme/icons/object-inline.svg';\nimport objectLeft from './../theme/icons/object-inline-left.svg';\nimport objectRight from './../theme/icons/object-inline-right.svg';\n\nimport objectSizeFull from './../theme/icons/object-size-full.svg';\nimport objectSizeLarge from './../theme/icons/object-size-large.svg';\nimport objectSizeSmall from './../theme/icons/object-size-small.svg';\nimport objectSizeMedium from './../theme/icons/object-size-medium.svg';\n\nimport pencil from './../theme/icons/pencil.svg';\nimport pilcrow from './../theme/icons/pilcrow.svg';\nimport quote from './../theme/icons/quote.svg';\nimport threeVerticalDots from './../theme/icons/three-vertical-dots.svg';\n\nimport bold from './../theme/icons/bold.svg';\nimport paragraph from './../theme/icons/paragraph.svg';\nimport plus from './../theme/icons/plus.svg';\nimport text from './../theme/icons/text.svg';\nimport importExport from './../theme/icons/importexport.svg';\n\nexport const icons = {\n\tbold,\n\tcancel,\n\tcaption,\n\tcheck,\n\tcog,\n\teraser,\n\timage,\n\tlowVision,\n\timportExport,\n\tparagraph,\n\tplus,\n\ttext,\n\n\talignBottom,\n\talignMiddle,\n\talignTop,\n\talignLeft,\n\talignCenter,\n\talignRight,\n\talignJustify,\n\n\tobjectLeft,\n\tobjectCenter,\n\tobjectRight,\n\tobjectFullWidth,\n\tobjectInline,\n\tobjectBlockLeft,\n\tobjectBlockRight,\n\n\tobjectSizeFull,\n\tobjectSizeLarge,\n\tobjectSizeSmall,\n\tobjectSizeMedium,\n\n\tpencil,\n\tpilcrow,\n\tquote,\n\tthreeVerticalDots\n};\n\nimport './augmentation';\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M10.187 17H5.773c-.637 0-1.092-.138-1.364-.415-.273-.277-.409-.718-.409-1.323V4.738c0-.617.14-1.062.419-1.332.279-.27.73-.406 1.354-.406h4.68c.69 0 1.288.041 1.793.124.506.083.96.242 1.36.478.341.197.644.447.906.75a3.262 3.262 0 0 1 .808 2.162c0 1.401-.722 2.426-2.167 3.075C15.05 10.175 16 11.315 16 13.01a3.756 3.756 0 0 1-2.296 3.504 6.1 6.1 0 0 1-1.517.377c-.571.073-1.238.11-2 .11zm-.217-6.217H7v4.087h3.069c1.977 0 2.965-.69 2.965-2.072 0-.707-.256-1.22-.768-1.537-.512-.319-1.277-.478-2.296-.478zM7 5.13v3.619h2.606c.729 0 1.292-.067 1.69-.2a1.6 1.6 0 0 0 .91-.765c.165-.267.247-.566.247-.897 0-.707-.26-1.176-.778-1.409-.519-.232-1.31-.348-2.375-.348H7z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2 16h9a1 1 0 0 1 0 2H2a1 1 0 0 1 0-2z\\\"/><path d=\\\"M17 1a2 2 0 0 1 2 2v9a2 2 0 0 1-2 2H3a2 2 0 0 1-2-2V3a2 2 0 0 1 2-2h14zm0 1.5H3a.5.5 0 0 0-.492.41L2.5 3v9a.5.5 0 0 0 .41.492L3 12.5h14a.5.5 0 0 0 .492-.41L17.5 12V3a.5.5 0 0 0-.41-.492L17 2.5z\\\" fill-opacity=\\\".6\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M6.972 16.615a.997.997 0 0 1-.744-.292l-4.596-4.596a1 1 0 1 1 1.414-1.414l3.926 3.926 9.937-9.937a1 1 0 0 1 1.414 1.415L7.717 16.323a.997.997 0 0 1-.745.292z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m11.333 2 .19 2.263a5.899 5.899 0 0 1 1.458.604L14.714 3.4 16.6 5.286l-1.467 1.733c.263.452.468.942.605 1.46L18 8.666v2.666l-2.263.19a5.899 5.899 0 0 1-.604 1.458l1.467 1.733-1.886 1.886-1.733-1.467a5.899 5.899 0 0 1-1.46.605L11.334 18H8.667l-.19-2.263a5.899 5.899 0 0 1-1.458-.604L5.286 16.6 3.4 14.714l1.467-1.733a5.899 5.899 0 0 1-.604-1.458L2 11.333V8.667l2.262-.189a5.899 5.899 0 0 1 .605-1.459L3.4 5.286 5.286 3.4l1.733 1.467a5.899 5.899 0 0 1 1.46-.605L8.666 2h2.666zM10 6.267a3.733 3.733 0 1 0 0 7.466 3.733 3.733 0 0 0 0-7.466z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m8.636 9.531-2.758 3.94a.5.5 0 0 0 .122.696l3.224 2.284h1.314l2.636-3.736L8.636 9.53zm.288 8.451L5.14 15.396a2 2 0 0 1-.491-2.786l6.673-9.53a2 2 0 0 1 2.785-.49l3.742 2.62a2 2 0 0 1 .491 2.785l-7.269 10.053-2.147-.066z\\\"/><path d=\\\"M4 18h5.523v-1H4zm-2 0h1v-1H2z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M6.91 10.54c.26-.23.64-.21.88.03l3.36 3.14 2.23-2.06a.64.64 0 0 1 .87 0l2.52 2.97V4.5H3.2v10.12l3.71-4.08zm10.27-7.51c.6 0 1.09.47 1.09 1.05v11.84c0 .59-.49 1.06-1.09 1.06H2.79c-.6 0-1.09-.47-1.09-1.06V4.08c0-.58.49-1.05 1.1-1.05h14.38zm-5.22 5.56a1.96 1.96 0 1 1 3.4-1.96 1.96 1.96 0 0 1-3.4 1.96z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M5.085 6.22 2.943 4.078a.75.75 0 1 1 1.06-1.06l2.592 2.59A11.094 11.094 0 0 1 10 5.068c4.738 0 8.578 3.101 8.578 5.083 0 1.197-1.401 2.803-3.555 3.887l1.714 1.713a.75.75 0 0 1-.09 1.138.488.488 0 0 1-.15.084.75.75 0 0 1-.821-.16L6.17 7.304c-.258.11-.51.233-.757.365l6.239 6.24-.006.005.78.78c-.388.094-.78.166-1.174.215l-1.11-1.11h.011L4.55 8.197a7.2 7.2 0 0 0-.665.514l-.112.098 4.897 4.897-.005.006 1.276 1.276a10.164 10.164 0 0 1-1.477-.117l-.479-.479-.009.009-4.863-4.863-.022.031a2.563 2.563 0 0 0-.124.2c-.043.077-.08.158-.108.241a.534.534 0 0 0-.028.133.29.29 0 0 0 .008.072.927.927 0 0 0 .082.226c.067.133.145.26.234.379l3.242 3.365.025.01.59.623c-3.265-.918-5.59-3.155-5.59-4.668 0-1.194 1.448-2.838 3.663-3.93zm7.07.531a4.632 4.632 0 0 1 1.108 5.992l.345.344.046-.018a9.313 9.313 0 0 0 2-1.112c.256-.187.5-.392.727-.613.137-.134.27-.277.392-.431.072-.091.141-.185.203-.286.057-.093.107-.19.148-.292a.72.72 0 0 0 .036-.12.29.29 0 0 0 .008-.072.492.492 0 0 0-.028-.133.999.999 0 0 0-.036-.096 2.165 2.165 0 0 0-.071-.145 2.917 2.917 0 0 0-.125-.2 3.592 3.592 0 0 0-.263-.335 5.444 5.444 0 0 0-.53-.523 7.955 7.955 0 0 0-1.054-.768 9.766 9.766 0 0 0-1.879-.891c-.337-.118-.68-.219-1.027-.301zm-2.85.21-.069.002a.508.508 0 0 0-.254.097.496.496 0 0 0-.104.679.498.498 0 0 0 .326.199l.045.005c.091.003.181.003.272.012a2.45 2.45 0 0 1 2.017 1.513c.024.061.043.125.069.185a.494.494 0 0 0 .45.287h.008a.496.496 0 0 0 .35-.158.482.482 0 0 0 .13-.335.638.638 0 0 0-.048-.219 3.379 3.379 0 0 0-.36-.723 3.438 3.438 0 0 0-2.791-1.543l-.028-.001h-.013z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M19 4.5 14 0H3v12.673l.868-1.041c.185-.222.4-.402.632-.54V1.5h8v5h5v7.626a2.24 2.24 0 0 1 1.5.822V4.5ZM14 5V2l3.3 3H14Zm-3.692 12.5c.062.105.133.206.213.303L11.52 19H8v-.876a2.243 2.243 0 0 0 1.82-.624h.488Zm7.518-.657a.75.75 0 0 0-1.152-.96L15.5 17.29V12H14v5.29l-1.174-1.408a.75.75 0 0 0-1.152.96l2.346 2.816a.95.95 0 0 0 1.46 0l2.346-2.815Zm-15.056-.38a.75.75 0 0 1-.096-1.056l2.346-2.815a.95.95 0 0 1 1.46 0l2.346 2.815a.75.75 0 1 1-1.152.96L6.5 14.96V20H5v-5.04l-1.174 1.408a.75.75 0 0 1-1.056.096Z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M10.5 5.5H7v5h3.5a2.5 2.5 0 1 0 0-5zM5 3h6.5v.025a5 5 0 0 1 0 9.95V13H7v4a1 1 0 0 1-1 1H5a1 1 0 0 1-1-1V4a1 1 0 0 1 1-1z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M10 2a1 1 0 0 0-1 1v6H3a1 1 0 1 0 0 2h6v6a1 1 0 1 0 2 0v-6h6a1 1 0 1 0 0-2h-6V3a1 1 0 0 0-1-1Z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M9.816 11.5 7.038 4.785 4.261 11.5h5.555Zm.62 1.5H3.641l-1.666 4.028H.312l5.789-14h1.875l5.789 14h-1.663L10.436 13Z\\\"/><path d=\\\"m12.09 17-.534-1.292.848-1.971.545 1.319L12.113 17h-.023Zm1.142-5.187.545 1.319L15.5 9.13l1.858 4.316h-3.45l.398.965h3.467L18.887 17H20l-3.873-9h-1.254l-1.641 3.813Z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m9.239 13.938-2.88-1.663a.75.75 0 0 1 .75-1.3L9 12.067V4.75a.75.75 0 1 1 1.5 0v7.318l1.89-1.093a.75.75 0 0 1 .75 1.3l-2.879 1.663a.752.752 0 0 1-.511.187.752.752 0 0 1-.511-.187zM4.25 17a.75.75 0 1 1 0-1.5h10.5a.75.75 0 0 1 0 1.5H4.25z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M9.75 11.875a.752.752 0 0 1 .508.184l2.883 1.666a.75.75 0 0 1-.659 1.344l-.091-.044-1.892-1.093.001 4.318a.75.75 0 1 1-1.5 0v-4.317l-1.89 1.092a.75.75 0 0 1-.75-1.3l2.879-1.663a.752.752 0 0 1 .51-.187zM15.25 9a.75.75 0 1 1 0 1.5H4.75a.75.75 0 1 1 0-1.5h10.5zM9.75.375a.75.75 0 0 1 .75.75v4.318l1.89-1.093.092-.045a.75.75 0 0 1 .659 1.344l-2.883 1.667a.752.752 0 0 1-.508.184.752.752 0 0 1-.511-.187L6.359 5.65a.75.75 0 0 1 .75-1.299L9 5.442V1.125a.75.75 0 0 1 .75-.75z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m10.261 7.062 2.88 1.663a.75.75 0 0 1-.75 1.3L10.5 8.933v7.317a.75.75 0 1 1-1.5 0V8.932l-1.89 1.093a.75.75 0 0 1-.75-1.3l2.879-1.663a.752.752 0 0 1 .511-.187.752.752 0 0 1 .511.187zM15.25 4a.75.75 0 1 1 0 1.5H4.75a.75.75 0 0 1 0-1.5h10.5z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2 3.75c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm0 8c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm0 4c0 .414.336.75.75.75h9.929a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm0-8c0 .414.336.75.75.75h9.929a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2 3.75c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm0 8c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm2.286 4c0 .414.336.75.75.75h9.928a.75.75 0 1 0 0-1.5H5.036a.75.75 0 0 0-.75.75zm0-8c0 .414.336.75.75.75h9.928a.75.75 0 1 0 0-1.5H5.036a.75.75 0 0 0-.75.75z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M18 3.75a.75.75 0 0 1-.75.75H2.75a.75.75 0 1 1 0-1.5h14.5a.75.75 0 0 1 .75.75zm0 8a.75.75 0 0 1-.75.75H2.75a.75.75 0 1 1 0-1.5h14.5a.75.75 0 0 1 .75.75zm0 4a.75.75 0 0 1-.75.75H7.321a.75.75 0 1 1 0-1.5h9.929a.75.75 0 0 1 .75.75zm0-8a.75.75 0 0 1-.75.75H7.321a.75.75 0 1 1 0-1.5h9.929a.75.75 0 0 1 .75.75z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2 3.75c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm0 8c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm0 4c0 .414.336.75.75.75h9.929a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm0-8c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path opacity=\\\".5\\\" d=\\\"M2 3h16v1.5H2zm11.5 9H18v1.5h-4.5zm0-3H18v1.5h-4.5zm0-3H18v1.5h-4.5zM2 15h16v1.5H2z\\\"/><path d=\\\"M12.003 7v5.5a1 1 0 0 1-1 1H2.996a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1h8.007a1 1 0 0 1 1 1zm-1.506.5H3.5V12h6.997V7.5z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path opacity=\\\".5\\\" d=\\\"M2 3h16v1.5H2zm0 12h16v1.5H2z\\\"/><path d=\\\"M15.003 7v5.5a1 1 0 0 1-1 1H5.996a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1h8.007a1 1 0 0 1 1 1zm-1.506.5H6.5V12h6.997V7.5z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path opacity=\\\".5\\\" d=\\\"M2 3h16v1.5H2zm0 12h16v1.5H2zm0-9h5v1.5H2zm0 3h5v1.5H2zm0 3h5v1.5H2z\\\"/><path d=\\\"M18.003 7v5.5a1 1 0 0 1-1 1H8.996a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1h8.007a1 1 0 0 1 1 1zm-1.506.5H9.5V12h6.997V7.5z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path opacity=\\\".5\\\" d=\\\"M2 3h16v1.5H2zm0 12h16v1.5H2z\\\"/><path d=\\\"M18 7v5.5a1 1 0 0 1-1 1H3a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1h14a1 1 0 0 1 1 1zm-1.505.5H3.504V12h12.991V7.5z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path opacity=\\\".5\\\" d=\\\"M2 3h16v1.5H2zm11.5 9H18v1.5h-4.5zM2 15h16v1.5H2z\\\"/><path d=\\\"M12.003 7v5.5a1 1 0 0 1-1 1H2.996a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1h8.007a1 1 0 0 1 1 1zm-1.506.5H3.5V12h6.997V7.5z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path opacity=\\\".5\\\" d=\\\"M2 3h16v1.5H2zm0 12h16v1.5H2z\\\"/><path d=\\\"M12.003 7v5.5a1 1 0 0 1-1 1H2.996a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1h8.007a1 1 0 0 1 1 1zm-1.506.5H3.5V12h6.997V7.5z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path opacity=\\\".5\\\" d=\\\"M2 3h16v1.5H2zm0 12h16v1.5H2z\\\"/><path d=\\\"M18.003 7v5.5a1 1 0 0 1-1 1H8.996a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1h8.007a1 1 0 0 1 1 1zm-1.506.5H9.5V12h6.997V7.5z\\\"/></svg>\";","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" viewBox=\\\"0 0 20 20\\\"><path d=\\\"M2.5 17v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zM1 15.5v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm0-2v1h-1v-1h1zm-19 0v1H0v-1h1zM14.5 2v1h-1V2h1zm2 0v1h-1V2h1zm2 0v1h-1V2h1zm-8 0v1h-1V2h1zm-2 0v1h-1V2h1zm-2 0v1h-1V2h1zm-2 0v1h-1V2h1zm8 0v1h-1V2h1zm-10 0v1h-1V2h1z\\\"/><path d=\\\"M18.095 2H1.905C.853 2 0 2.895 0 4v12c0 1.105.853 2 1.905 2h16.19C19.147 18 20 17.105 20 16V4c0-1.105-.853-2-1.905-2zm0 1.5c.263 0 .476.224.476.5v12c0 .276-.213.5-.476.5H1.905a.489.489 0 0 1-.476-.5V4c0-.276.213-.5.476-.5h16.19z\\\"/></svg>\";","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" viewBox=\\\"0 0 20 20\\\"><path d=\\\"M2.5 17v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zM1 15.5v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm0-2v1h-1v-1h1zm-19 0v1H0v-1h1zM14.5 2v1h-1V2h1zm2 0v1h-1V2h1zm2 0v1h-1V2h1zm-8 0v1h-1V2h1zm-2 0v1h-1V2h1zm-2 0v1h-1V2h1zm-2 0v1h-1V2h1zm8 0v1h-1V2h1zm-10 0v1h-1V2h1z\\\"/><path d=\\\"M13 6H2a2 2 0 0 0-2 2v8a2 2 0 0 0 2 2h11a2 2 0 0 0 2-2V8a2 2 0 0 0-2-2zm0 1.5a.5.5 0 0 1 .5.5v8a.5.5 0 0 1-.5.5H2a.5.5 0 0 1-.5-.5V8a.5.5 0 0 1 .5-.5h11z\\\"/></svg>\";","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" viewBox=\\\"0 0 20 20\\\"><path d=\\\"M2.5 17v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zM1 15.5v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm0-2v1h-1v-1h1zm-19 0v1H0v-1h1zM14.5 2v1h-1V2h1zm2 0v1h-1V2h1zm2 0v1h-1V2h1zm-8 0v1h-1V2h1zm-2 0v1h-1V2h1zm-2 0v1h-1V2h1zm-2 0v1h-1V2h1zm8 0v1h-1V2h1zm-10 0v1h-1V2h1z\\\"/><path d=\\\"M7 10H2a2 2 0 0 0-2 2v4a2 2 0 0 0 2 2h5a2 2 0 0 0 2-2v-4a2 2 0 0 0-2-2zm0 1.5a.5.5 0 0 1 .5.5v4a.5.5 0 0 1-.5.5H2a.5.5 0 0 1-.5-.5v-4a.5.5 0 0 1 .5-.5h5z\\\"/></svg>\";","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" viewBox=\\\"0 0 20 20\\\"><path d=\\\"M2.5 17v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zm2 0v1h-1v-1h1zM1 15.5v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm-19-2v1H0v-1h1zm19 0v1h-1v-1h1zm0-2v1h-1v-1h1zm-19 0v1H0v-1h1zM14.5 2v1h-1V2h1zm2 0v1h-1V2h1zm2 0v1h-1V2h1zm-8 0v1h-1V2h1zm-2 0v1h-1V2h1zm-2 0v1h-1V2h1zm-2 0v1h-1V2h1zm8 0v1h-1V2h1zm-10 0v1h-1V2h1z\\\"/><path d=\\\"M10 8H2a2 2 0 0 0-2 2v6a2 2 0 0 0 2 2h8a2 2 0 0 0 2-2v-6a2 2 0 0 0-2-2zm0 1.5a.5.5 0 0 1 .5.5v6a.5.5 0 0 1-.5.5H2a.5.5 0 0 1-.5-.5v-6a.5.5 0 0 1 .5-.5h8z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m7.3 17.37-.061.088a1.518 1.518 0 0 1-.934.535l-4.178.663-.806-4.153a1.495 1.495 0 0 1 .187-1.058l.056-.086L8.77 2.639c.958-1.351 2.803-1.076 4.296-.03 1.497 1.047 2.387 2.693 1.433 4.055L7.3 17.37zM9.14 4.728l-5.545 8.346 3.277 2.294 5.544-8.346L9.14 4.728zM6.07 16.512l-3.276-2.295.53 2.73 2.746-.435zM9.994 3.506 13.271 5.8c.316-.452-.16-1.333-1.065-1.966-.905-.634-1.895-.78-2.212-.328zM8 18.5 9.375 17H19v1.5H8z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M6.999 2H15a1 1 0 0 1 0 2h-1.004v13a1 1 0 1 1-2 0V4H8.999v13a1 1 0 1 1-2 0v-7A4 4 0 0 1 3 6a4 4 0 0 1 3.999-4z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M3 10.423a6.5 6.5 0 0 1 6.056-6.408l.038.67C6.448 5.423 5.354 7.663 5.22 10H9c.552 0 .5.432.5.986v4.511c0 .554-.448.503-1 .503h-5c-.552 0-.5-.449-.5-1.003v-4.574zm8 0a6.5 6.5 0 0 1 6.056-6.408l.038.67c-2.646.739-3.74 2.979-3.873 5.315H17c.552 0 .5.432.5.986v4.511c0 .554-.448.503-1 .503h-5c-.552 0-.5-.449-.5-1.003v-4.574z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><circle cx=\\\"9.5\\\" cy=\\\"4.5\\\" r=\\\"1.5\\\"/><circle cx=\\\"9.5\\\" cy=\\\"10.5\\\" r=\\\"1.5\\\"/><circle cx=\\\"9.5\\\" cy=\\\"16.5\\\" r=\\\"1.5\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/bindings/clickoutsidehandler\n */\n\nimport type { DomEmitter } from '@ckeditor/ckeditor5-utils';\n\n/* global document */\n\n/**\n * Handles clicking **outside** of a specified set of elements, then fires an action.\n *\n * **Note**: Actually, the action is executed upon `mousedown`, not `click`. It prevents\n * certain issues when the user keeps holding the mouse button and the UI cannot react\n * properly.\n *\n * @param options Configuration options.\n * @param options.emitter The emitter to which this behavior should be added.\n * @param options.activator Function returning a `Boolean`, to determine whether the handler is active.\n * @param options.contextElements Array of HTML elements or a callback returning an array of HTML elements\n * that determine the scope of the handler. Clicking any of them or their descendants will **not** fire the callback.\n * @param options.callback An action executed by the handler.\n */\nexport default function clickOutsideHandler(\n\t{ emitter, activator, callback, contextElements }: {\n\t\temitter: DomEmitter;\n\t\tactivator: () => boolean;\n\t\tcontextElements: Array<HTMLElement> | ( () => Array<HTMLElement> );\n\t\tcallback: () => void;\n\t}\n): void {\n\temitter.listenTo( document, 'mousedown', ( evt, domEvt ) => {\n\t\tif ( !activator() ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Check if `composedPath` is `undefined` in case the browser does not support native shadow DOM.\n\t\t// Can be removed when all supported browsers support native shadow DOM.\n\t\tconst path = typeof domEvt.composedPath == 'function' ? domEvt.composedPath() : [];\n\n\t\tconst contextElementsList = typeof contextElements == 'function' ? contextElements() : contextElements;\n\n\t\tfor ( const contextElement of contextElementsList ) {\n\t\t\tif ( contextElement.contains( domEvt.target as Node ) || path.includes( contextElement ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\n\t\tcallback();\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/bindings/csstransitiondisablermixin\n */\n\nimport type { Constructor, Mixed } from '@ckeditor/ckeditor5-utils';\n\nimport type View from '../view';\n\n/**\n * A mixin that brings the possibility to temporarily disable CSS transitions using\n * {@link module:ui/view~View} methods. It is helpful when, for instance, the transitions should not happen\n * when the view is first displayed but they should work normal in other cases.\n *\n * The methods to control the CSS transitions are:\n * * `disableCssTransitions()` Adds the `.ck-transitions-disabled` class to the\n * {@link module:ui/view~View#element view element}.\n * * `enableCssTransitions()` Removes the `.ck-transitions-disabled` class from the\n * {@link module:ui/view~View#element view element}.\n *\n * The usage comes down to:\n *\n * ```ts\n * const MyViewWithCssTransitionDisabler = CssTransitionDisablerMixin( MyView );\n * const view = new MyViewWithCssTransitionDisabler();\n *\n * // ...\n *\n * view.disableCssTransitions();\n * view.show();\n * view.enableCssTransitions();\n * ```\n *\n * @param view View instance that should get this functionality.\n */\nexport default function CssTransitionDisablerMixin<Base extends Constructor<View>>( view: Base ): Mixed<Base, ViewWithCssTransitionDisabler>\n{\n\tabstract class Mixin extends view {\n\t\tdeclare public _isCssTransitionsDisabled: boolean;\n\n\t\tpublic disableCssTransitions() {\n\t\t\tthis._isCssTransitionsDisabled = true;\n\t\t}\n\n\t\tpublic enableCssTransitions() {\n\t\t\tthis._isCssTransitionsDisabled = false;\n\t\t}\n\n\t\tconstructor( ...args: Array<any> ) {\n\t\t\tsuper( ...args );\n\n\t\t\tthis.set( '_isCssTransitionsDisabled', false );\n\t\t\tthis.initializeCssTransitionDisablerMixin();\n\t\t}\n\n\t\tprotected initializeCssTransitionDisablerMixin(): void {\n\t\t\tthis.extendTemplate( {\n\t\t\t\tattributes: {\n\t\t\t\t\tclass: [\n\t\t\t\t\t\tthis.bindTemplate.if( '_isCssTransitionsDisabled', 'ck-transitions-disabled' )\n\t\t\t\t\t]\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t}\n\n\treturn Mixin as any;\n}\n\nexport type ViewWithCssTransitionDisabler = View & {\n\tdisableCssTransitions(): void;\n\tenableCssTransitions(): void;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/bindings/submithandler\n */\n\nimport type View from '../view';\n\n/**\n * A handler useful for {@link module:ui/view~View views} working as HTML forms. It intercepts a native DOM\n * `submit` event, prevents the default web browser behavior (navigation and page reload) and\n * fires the `submit` event on a view instead. Such a custom event can be then used by any\n * {@link module:utils/dom/emittermixin~DomEmitter emitter}, e.g. to serialize the form data.\n *\n * ```ts\n * import submitHandler from '@ckeditor/ckeditor5-ui/src/bindings/submithandler';\n *\n * // ...\n *\n * class AnyFormView extends View {\n * \tconstructor() {\n * \t\tsuper();\n *\n * \t\t// ...\n *\n * \t\tsubmitHandler( {\n * \t\t\tview: this\n * \t\t} );\n * \t}\n * }\n *\n * // ...\n *\n * const view = new AnyFormView();\n *\n * // A sample listener attached by an emitter working with the view.\n * this.listenTo( view, 'submit', () => {\n * \tsaveTheFormData();\n * \thideTheForm();\n * } );\n * ```\n *\n * @param options Configuration options.\n * @param options.view The view which DOM `submit` events should be handled.\n */\nexport default function submitHandler( { view }: { view: View } ): void {\n\tview.listenTo( view.element!, 'submit', ( evt, domEvt ) => {\n\t\tdomEvt.preventDefault();\n\t\tview.fire<SubmitHandlerEvent>( 'submit' );\n\t}, { useCapture: true } );\n}\n\n/**\n * Fired by {@link module:ui/bindings/submithandler~submitHandler} helper.\n *\n * @eventName module:ui/view~View#submit\n */\nexport type SubmitHandlerEvent = {\n\tname: 'submit';\n\targs: [];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/viewcollection\n */\n\nimport {\n\tCKEditorError,\n\tCollection,\n\ttype CollectionAddEvent,\n\ttype CollectionRemoveEvent,\n\ttype EmitterMixinDelegateChain\n} from '@ckeditor/ckeditor5-utils';\n\nimport type View from './view';\n\n/**\n * Collects {@link module:ui/view~View} instances.\n *\n * ```ts\n * const parentView = new ParentView( locale );\n * const collection = new ViewCollection( locale );\n *\n * collection.setParent( parentView.element );\n *\n * const viewA = new ChildView( locale );\n * const viewB = new ChildView( locale );\n * ```\n *\n * View collection renders and manages view {@link module:ui/view~View#element elements}:\n *\n * ```ts\n * collection.add( viewA );\n * collection.add( viewB );\n *\n * console.log( parentView.element.firsChild ); // -> viewA.element\n * console.log( parentView.element.lastChild ); // -> viewB.element\n * ```\n *\n * It {@link module:ui/viewcollection~ViewCollection#delegate propagates} DOM events too:\n *\n * ```ts\n * // Delegate #click and #keydown events from viewA and viewB to the parentView.\n * collection.delegate( 'click' ).to( parentView );\n *\n * parentView.on( 'click', ( evt ) => {\n * \tconsole.log( `${ evt.source } has been clicked.` );\n * } );\n *\n * // This event will be delegated to the parentView.\n * viewB.fire( 'click' );\n * ```\n *\n * **Note**: A view collection can be used directly in the {@link module:ui/template~TemplateDefinition definition}\n * of a {@link module:ui/template~Template template}.\n */\nexport default class ViewCollection<TView extends View = View> extends Collection<TView> {\n\tpublic id?: string;\n\n\t/**\n\t * A parent element within which child views are rendered and managed in DOM.\n\t */\n\tprivate _parentElement: DocumentFragment | HTMLElement | null;\n\n\t/**\n\t * Creates a new instance of the {@link module:ui/viewcollection~ViewCollection}.\n\t *\n\t * @param initialItems The initial items of the collection.\n\t */\n\tconstructor( initialItems: Iterable<TView> = [] ) {\n\t\tsuper( initialItems, {\n\t\t\t// An #id Number attribute should be legal and not break the `ViewCollection` instance.\n\t\t\t// https://github.com/ckeditor/ckeditor5-ui/issues/93\n\t\t\tidProperty: 'viewUid'\n\t\t} );\n\n\t\t// Handle {@link module:ui/view~View#element} in DOM when a new view is added to the collection.\n\t\tthis.on<CollectionAddEvent<TView>>( 'add', ( evt, view, index ) => {\n\t\t\tthis._renderViewIntoCollectionParent( view, index );\n\t\t} );\n\n\t\t// Handle {@link module:ui/view~View#element} in DOM when a view is removed from the collection.\n\t\tthis.on<CollectionRemoveEvent<TView>>( 'remove', ( evt, view ) => {\n\t\t\tif ( view.element && this._parentElement ) {\n\t\t\t\tview.element.remove();\n\t\t\t}\n\t\t} );\n\n\t\tthis._parentElement = null;\n\t}\n\n\t/**\n\t * Destroys the view collection along with child views.\n\t * See the view {@link module:ui/view~View#destroy} method.\n\t */\n\tpublic destroy(): void {\n\t\tthis.map( view => view.destroy() );\n\t}\n\n\t/**\n\t * Sets the parent HTML element of this collection. When parent is set, {@link #add adding} and\n\t * {@link #remove removing} views in the collection synchronizes their\n\t * {@link module:ui/view~View#element elements} in the parent element.\n\t *\n\t * @param element A new parent element.\n\t */\n\tpublic setParent( elementOrDocFragment: DocumentFragment | HTMLElement ): void {\n\t\tthis._parentElement = elementOrDocFragment;\n\n\t\t// Take care of the initial collection items passed to the constructor.\n\t\tfor ( const view of this ) {\n\t\t\tthis._renderViewIntoCollectionParent( view );\n\t\t}\n\t}\n\n\t/**\n\t * Delegates selected events coming from within views in the collection to any\n\t * {@link module:utils/emittermixin~Emitter}.\n\t *\n\t * For the following views and collection:\n\t *\n\t * ```ts\n\t * const viewA = new View();\n\t * const viewB = new View();\n\t * const viewC = new View();\n\t *\n\t * const views = parentView.createCollection();\n\t *\n\t * views.delegate( 'eventX' ).to( viewB );\n\t * views.delegate( 'eventX', 'eventY' ).to( viewC );\n\t *\n\t * views.add( viewA );\n\t * ```\n\t *\n\t * the `eventX` is delegated (fired by) `viewB` and `viewC` along with `customData`:\n\t *\n\t * ```ts\n\t * viewA.fire( 'eventX', customData );\n\t * ```\n\t *\n\t * and `eventY` is delegated (fired by) `viewC` along with `customData`:\n\t *\n\t * ```ts\n\t * viewA.fire( 'eventY', customData );\n\t * ```\n\t *\n\t * See {@link module:utils/emittermixin~Emitter#delegate}.\n\t *\n\t * @param events {@link module:ui/view~View} event names to be delegated to another\n\t * {@link module:utils/emittermixin~Emitter}.\n\t * @returns Object with `to` property, a function which accepts the destination\n\t * of {@link module:utils/emittermixin~Emitter#delegate delegated} events.\n\t */\n\tpublic override delegate( ...events: Array<string> ): EmitterMixinDelegateChain {\n\t\tif ( !events.length || !isStringArray( events ) ) {\n\t\t\t/**\n\t\t\t * All event names must be strings.\n\t\t\t *\n\t\t\t * @error ui-viewcollection-delegate-wrong-events\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'ui-viewcollection-delegate-wrong-events',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\treturn {\n\t\t\tto: dest => {\n\t\t\t\t// Activate delegating on existing views in this collection.\n\t\t\t\tfor ( const view of this ) {\n\t\t\t\t\tfor ( const evtName of events ) {\n\t\t\t\t\t\tview.delegate( evtName ).to( dest );\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Activate delegating on future views in this collection.\n\t\t\t\tthis.on<CollectionAddEvent<View>>( 'add', ( evt, view ) => {\n\t\t\t\t\tfor ( const evtName of events ) {\n\t\t\t\t\t\tview.delegate( evtName ).to( dest );\n\t\t\t\t\t}\n\t\t\t\t} );\n\n\t\t\t\t// Deactivate delegating when view is removed from this collection.\n\t\t\t\tthis.on<CollectionRemoveEvent<View>>( 'remove', ( evt, view ) => {\n\t\t\t\t\tfor ( const evtName of events ) {\n\t\t\t\t\t\tview.stopDelegating( evtName, dest );\n\t\t\t\t\t}\n\t\t\t\t} );\n\t\t\t}\n\t\t};\n\t}\n\n\t/**\n\t * This method {@link module:ui/view~View#render renders} a new view added to the collection.\n\t *\n\t * If the {@link #_parentElement parent element} of the collection is set, this method also adds\n\t * the view's {@link module:ui/view~View#element} as a child of the parent in DOM at a specified index.\n\t *\n\t * **Note**: If index is not specified, the view's element is pushed as the last child\n\t * of the parent element.\n\t *\n\t * @param view A new view added to the collection.\n\t * @param index An index the view holds in the collection. When not specified,\n\t * the view is added at the end.\n\t */\n\tprivate _renderViewIntoCollectionParent( view: View, index?: number ): void {\n\t\tif ( !view.isRendered ) {\n\t\t\tview.render();\n\t\t}\n\n\t\tif ( view.element && this._parentElement ) {\n\t\t\tthis._parentElement.insertBefore( view.element, this._parentElement.children[ index! ] );\n\t\t}\n\t}\n\n\t/**\n\t * Removes a child view from the collection. If the {@link #setParent parent element} of the\n\t * collection has been set, the {@link module:ui/view~View#element element} of the view is also removed\n\t * in DOM, reflecting the order of the collection.\n\t *\n\t * See the {@link #add} method.\n\t *\n\t * @param subject The view to remove, its id or index in the collection.\n\t * @returns The removed view.\n\t */\n\tpublic override remove( subject: TView | number | string ): TView {\n\t\treturn super.remove( subject );\n\t}\n}\n\n/**\n * Check if all entries of the array are of `String` type.\n *\n * @param arr An array to be checked.\n */\nfunction isStringArray( arr: Array<unknown> ): arr is Array<string> {\n\treturn arr.every( a => typeof a == 'string' );\n}\n","import api from \"!../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../node_modules/css-loader/dist/cjs.js!../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./globals.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* eslint-disable @typescript-eslint/no-invalid-void-type */\n\n/**\n * @module ui/view\n */\n\nimport ViewCollection from './viewcollection';\nimport Template, { type BindChain, type TemplateDefinition } from './template';\n\nimport {\n\tCKEditorError,\n\tCollection,\n\tDomEmitterMixin,\n\tObservableMixin,\n\tisIterable,\n\ttype CollectionAddEvent,\n\ttype DecoratedMethodEvent,\n\ttype Locale,\n\ttype LocaleTranslate\n} from '@ckeditor/ckeditor5-utils';\n\nimport '../theme/globals/globals.css';\n\n/**\n * The basic view class, which represents an HTML element created out of a\n * {@link module:ui/view~View#template}. Views are building blocks of the user interface and handle\n * interaction\n *\n * Views {@link module:ui/view~View#registerChild aggregate} children in\n * {@link module:ui/view~View#createCollection collections} and manage the life cycle of DOM\n * listeners e.g. by handling rendering and destruction.\n *\n * See the {@link module:ui/template~TemplateDefinition} syntax to learn more about shaping view\n * elements, attributes and listeners.\n *\n * ```ts\n * class SampleView extends View {\n * \tconstructor( locale ) {\n * \t\tsuper( locale );\n *\n * \t\tconst bind = this.bindTemplate;\n *\n * \t\t// Views define their interface (state) using observable attributes.\n * \t\tthis.set( 'elementClass', 'bar' );\n *\n * \t\tthis.setTemplate( {\n * \t\t\ttag: 'p',\n *\n * \t\t\t// The element of the view can be defined with its children.\n * \t\t\tchildren: [\n * \t\t\t\t'Hello',\n * \t\t\t\t{\n * \t\t\t\t\ttag: 'b',\n * \t\t\t\t\tchildren: [ 'world!' ]\n * \t\t\t\t}\n * \t\t\t],\n * \t\t\tattributes: {\n * \t\t\t\tclass: [\n * \t\t\t\t\t'foo',\n *\n * \t\t\t\t\t// Observable attributes control the state of the view in DOM.\n * \t\t\t\t\tbind.to( 'elementClass' )\n * \t\t\t\t]\n * \t\t\t},\n * \t\t\ton: {\n * \t\t\t\t// Views listen to DOM events and propagate them.\n * \t\t\t\tclick: bind.to( 'clicked' )\n * \t\t\t}\n * \t\t} );\n * \t}\n * }\n *\n * const view = new SampleView( locale );\n *\n * view.render();\n *\n * // Append <p class=\"foo bar\">Hello<b>world</b></p> to the <body>\n * document.body.appendChild( view.element );\n *\n * // Change the class attribute to <p class=\"foo baz\">Hello<b>world</b></p>\n * view.elementClass = 'baz';\n *\n * // Respond to the \"click\" event in DOM by executing a custom action.\n * view.on( 'clicked', () => {\n * \tconsole.log( 'The view has been clicked!' );\n * } );\n * ```\n */\nexport default class View<TElement extends HTMLElement = HTMLElement> extends DomEmitterMixin( ObservableMixin() ) {\n\t/**\n\t * An HTML element of the view. `null` until {@link #render rendered}\n\t * from the {@link #template}.\n\t *\n\t * ```ts\n\t * class SampleView extends View {\n\t * \tconstructor() {\n\t * \t\tsuper();\n\t *\n\t * \t\t// A template instance the #element will be created from.\n\t * \t\tthis.setTemplate( {\n\t * \t\t\ttag: 'p'\n\t *\n\t * \t\t\t// ...\n\t * \t\t} );\n\t * \t}\n\t * }\n\t *\n\t * const view = new SampleView();\n\t *\n\t * // Renders the #template.\n\t * view.render();\n\t *\n\t * // Append the HTML element of the view to <body>.\n\t * document.body.appendChild( view.element );\n\t * ```\n\t *\n\t * **Note**: The element of the view can also be assigned directly:\n\t *\n\t * ```ts\n\t * view.element = document.querySelector( '#my-container' );\n\t * ```\n\t */\n\tpublic element: TElement | null;\n\n\t/**\n\t * Set `true` when the view has already been {@link module:ui/view~View#render rendered}.\n\t *\n\t * @readonly\n\t */\n\tpublic isRendered: boolean;\n\n\t/**\n\t * A set of tools to localize the user interface.\n\t *\n\t * Also see {@link module:core/editor/editor~Editor#locale}.\n\t *\n\t * @readonly\n\t */\n\tpublic locale: Locale | undefined;\n\n\t/**\n\t * Shorthand for {@link module:utils/locale~Locale#t}.\n\t *\n\t * Note: If {@link #locale} instance hasn't been passed to the view this method may not\n\t * be available.\n\t *\n\t * @see module:utils/locale~Locale#t\n\t */\n\tpublic t: LocaleTranslate | undefined;\n\n\t/**\n\t * Template of this view. It provides the {@link #element} representing\n\t * the view in DOM, which is {@link #render rendered}.\n\t */\n\tpublic template?: Template;\n\n\tpublic viewUid?: string;\n\n\t/**\n\t * Collections registered with {@link #createCollection}.\n\t */\n\tprotected _viewCollections: Collection<ViewCollection>;\n\n\t/**\n\t * A collection of view instances, which have been added directly\n\t * into the {@link module:ui/template~Template#children}.\n\t */\n\tprotected _unboundChildren: ViewCollection;\n\n\t/**\n\t * Cached {@link module:ui/template~BindChain bind chain} object created by the\n\t * {@link #template}. See {@link #bindTemplate}.\n\t */\n\tprivate _bindTemplate?: BindChain<this>;\n\n\t/**\n\t * Creates an instance of the {@link module:ui/view~View} class.\n\t *\n\t * Also see {@link #render}.\n\t *\n\t * @param locale The localization services instance.\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper();\n\n\t\tthis.element = null;\n\t\tthis.isRendered = false;\n\n\t\tthis.locale = locale;\n\t\tthis.t = locale && locale.t;\n\n\t\tthis._viewCollections = new Collection();\n\t\tthis._unboundChildren = this.createCollection();\n\n\t\t// Pass parent locale to its children.\n\t\tthis._viewCollections.on<CollectionAddEvent<View>>( 'add', ( evt, collection ) => {\n\t\t\tcollection.locale = locale;\n\t\t\tcollection.t = locale && locale.t;\n\t\t} );\n\n\t\tthis.decorate( 'render' );\n\t}\n\n\t/**\n\t * Shorthand for {@link module:ui/template~Template.bind}, a binding\n\t * {@link module:ui/template~BindChain interface} preconfigured for the view instance.\n\t *\n\t * It provides {@link module:ui/template~BindChain#to `to()`} and\n\t * {@link module:ui/template~BindChain#if `if()`} methods that initialize bindings with\n\t * observable attributes and attach DOM listeners.\n\t *\n\t * ```ts\n\t * class SampleView extends View {\n\t * \tconstructor( locale ) {\n\t * \t\tsuper( locale );\n\t *\n\t * \t\tconst bind = this.bindTemplate;\n\t *\n\t * \t\t// These {@link module:utils/observablemixin~Observable observable} attributes will control\n\t * \t\t// the state of the view in DOM.\n\t * \t\tthis.set( {\n\t * \t\t\telementClass: 'foo',\n\t * \t\t \tisEnabled: true\n\t * \t\t } );\n\t *\n\t * \t\tthis.setTemplate( {\n\t * \t\t\ttag: 'p',\n\t *\n\t * \t\t\tattributes: {\n\t * \t\t\t\t// The class HTML attribute will follow elementClass\n\t * \t\t\t\t// and isEnabled view attributes.\n\t * \t\t\t\tclass: [\n\t * \t\t\t\t\tbind.to( 'elementClass' )\n\t * \t\t\t\t\tbind.if( 'isEnabled', 'present-when-enabled' )\n\t * \t\t\t\t]\n\t * \t\t\t},\n\t *\n\t * \t\t\ton: {\n\t * \t\t\t\t// The view will fire the \"clicked\" event upon clicking <p> in DOM.\n\t * \t\t\t\tclick: bind.to( 'clicked' )\n\t * \t\t\t}\n\t * \t\t} );\n\t * \t}\n\t * }\n\t * ```\n\t */\n\tpublic get bindTemplate(): BindChain<this> {\n\t\tif ( this._bindTemplate ) {\n\t\t\treturn this._bindTemplate;\n\t\t}\n\n\t\treturn ( this._bindTemplate = Template.bind( this, this ) );\n\t}\n\n\t/**\n\t * Creates a new collection of views, which can be used as\n\t * {@link module:ui/template~Template#children} of this view.\n\t *\n\t * ```ts\n\t * class SampleView extends View {\n\t * \tconstructor( locale ) {\n\t * \t\tsuper( locale );\n\t *\n\t * \t\tconst child = new ChildView( locale );\n\t * \t\tthis.items = this.createCollection( [ child ] );\n \t *\n\t * \t\tthis.setTemplate( {\n\t * \t\t\ttag: 'p',\n\t *\n\t * \t\t\t// `items` collection will render here.\n\t * \t\t\tchildren: this.items\n\t * \t\t} );\n\t * \t}\n\t * }\n\t *\n\t * const view = new SampleView( locale );\n\t * view.render();\n\t *\n\t * // It will append <p><child#element></p> to the <body>.\n\t * document.body.appendChild( view.element );\n\t * ```\n\t *\n\t * @param views Initial views of the collection.\n\t * @returns A new collection of view instances.\n\t */\n\tpublic createCollection<T extends View = View>( views?: Iterable<T> ): ViewCollection<T> {\n\t\tconst collection = new ViewCollection<T>( views );\n\n\t\tthis._viewCollections.add( collection );\n\n\t\treturn collection;\n\t}\n\n\t/**\n\t * Registers a new child view under the view instance. Once registered, a child\n\t * view is managed by its parent, including {@link #render rendering}\n\t * and {@link #destroy destruction}.\n\t *\n\t * To revert this, use {@link #deregisterChild}.\n\t *\n\t * ```ts\n\t * class SampleView extends View {\n\t * \tconstructor( locale ) {\n\t * \t\tsuper( locale );\n\t *\n\t * \t\tthis.childA = new SomeChildView( locale );\n\t * \t\tthis.childB = new SomeChildView( locale );\n\t *\n\t * \t\tthis.setTemplate( { tag: 'p' } );\n\t *\n\t * \t\t// Register the children.\n\t * \t\tthis.registerChild( [ this.childA, this.childB ] );\n\t * \t}\n\t *\n\t * \trender() {\n\t * \t\tsuper.render();\n\t *\n\t * \t\tthis.element.appendChild( this.childA.element );\n\t * \t\tthis.element.appendChild( this.childB.element );\n\t * \t}\n\t * }\n\t *\n\t * const view = new SampleView( locale );\n\t *\n\t * view.render();\n\t *\n\t * // Will append <p><childA#element><b></b><childB#element></p>.\n\t * document.body.appendChild( view.element );\n\t * ```\n\t *\n\t * **Note**: There's no need to add child views if they're already referenced in the\n\t * {@link #template}:\n\t *\n\t * ```ts\n\t * class SampleView extends View {\n\t * \tconstructor( locale ) {\n\t * \t\tsuper( locale );\n\t *\n\t * \t\tthis.childA = new SomeChildView( locale );\n\t * \t\tthis.childB = new SomeChildView( locale );\n\t *\n\t * \t\tthis.setTemplate( {\n\t * \t\t\ttag: 'p',\n\t *\n \t * \t\t\t// These children will be added automatically. There's no\n \t * \t\t\t// need to call {@link #registerChild} for any of them.\n\t * \t\t\tchildren: [ this.childA, this.childB ]\n\t * \t\t} );\n\t * \t}\n\t *\n\t * \t// ...\n\t * }\n\t * ```\n\t *\n\t * @param children Children views to be registered.\n\t */\n\tpublic registerChild( children: View | Iterable<View> ): void {\n\t\tif ( !isIterable( children ) ) {\n\t\t\tchildren = [ children ];\n\t\t}\n\n\t\tfor ( const child of children ) {\n\t\t\tthis._unboundChildren.add( child );\n\t\t}\n\t}\n\n\t/**\n\t * The opposite of {@link #registerChild}. Removes a child view from this view instance.\n\t * Once removed, the child is no longer managed by its parent, e.g. it can safely\n\t * become a child of another parent view.\n\t *\n\t * @see #registerChild\n\t * @param children Child views to be removed.\n\t */\n\tpublic deregisterChild( children: View | Iterable<View> ): void {\n\t\tif ( !isIterable( children ) ) {\n\t\t\tchildren = [ children ];\n\t\t}\n\n\t\tfor ( const child of children ) {\n\t\t\tthis._unboundChildren.remove( child );\n\t\t}\n\t}\n\n\t/**\n\t * Sets the {@link #template} of the view with with given definition.\n\t *\n\t * A shorthand for:\n\t *\n\t * ```ts\n\t * view.setTemplate( definition );\n\t * ```\n\t *\n\t * @param definition Definition of view's template.\n\t */\n\tpublic setTemplate( definition: TemplateDefinition ): void {\n\t\tthis.template = new Template( definition );\n\t}\n\n\t/**\n\t * {@link module:ui/template~Template.extend Extends} the {@link #template} of the view with\n\t * with given definition.\n\t *\n\t * A shorthand for:\n\t *\n\t * ```ts\n\t * Template.extend( view.template, definition );\n\t * ```\n\t *\n\t * **Note**: Is requires the {@link #template} to be already set. See {@link #setTemplate}.\n\t *\n\t * @param definition Definition which extends the {@link #template}.\n\t */\n\tpublic extendTemplate( definition: Partial<TemplateDefinition> ): void {\n\t\tTemplate.extend( this.template!, definition );\n\t}\n\n\t/**\n\t * Recursively renders the view.\n\t *\n\t * Once the view is rendered:\n\t * * the {@link #element} becomes an HTML element out of {@link #template},\n\t * * the {@link #isRendered} flag is set `true`.\n\t *\n\t * **Note**: The children of the view:\n\t * * defined directly in the {@link #template}\n\t * * residing in collections created by the {@link #createCollection} method,\n\t * * and added by {@link #registerChild}\n\t * are also rendered in the process.\n\t *\n\t * In general, `render()` method is the right place to keep the code which refers to the\n\t * {@link #element} and should be executed at the very beginning of the view's life cycle.\n\t *\n\t * It is possible to {@link module:ui/template~Template.extend} the {@link #template} before\n\t * the view is rendered. To allow an early customization of the view (e.g. by its parent),\n\t * such references should be done in `render()`.\n\t *\n\t * ```ts\n\t * class SampleView extends View {\n\t * \tconstructor() {\n\t * \t\tthis.setTemplate( {\n\t * \t\t\t// ...\n\t * \t\t} );\n\t * \t},\n\t *\n\t * \trender() {\n\t * \t\t// View#element becomes available.\n\t * \t\tsuper.render();\n\t *\n\t * \t\t// The \"scroll\" listener depends on #element.\n\t * \t\tthis.listenTo( window, 'scroll', () => {\n\t * \t\t\t// A reference to #element would render the #template and make it non-extendable.\n\t * \t\t\tif ( window.scrollY > 0 ) {\n\t * \t\t\t\tthis.element.scrollLeft = 100;\n\t * \t\t\t} else {\n\t * \t\t\t\tthis.element.scrollLeft = 0;\n\t * \t\t\t}\n\t * \t\t} );\n\t * \t}\n\t * }\n\t *\n\t * const view = new SampleView();\n\t *\n\t * // Let's customize the view before it gets rendered.\n\t * view.extendTemplate( {\n\t * \tattributes: {\n\t * \t\tclass: [\n\t * \t\t\t'additional-class'\n\t * \t\t]\n\t * \t}\n\t * } );\n\t *\n\t * // Late rendering allows customization of the view.\n\t * view.render();\n\t * ```\n\t */\n\tpublic render(): void {\n\t\tif ( this.isRendered ) {\n\t\t\t/**\n\t\t\t * This View has already been rendered.\n\t\t\t *\n\t\t\t * @error ui-view-render-already-rendered\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'ui-view-render-already-rendered', this );\n\t\t}\n\n\t\t// Render #element of the view.\n\t\tif ( this.template ) {\n\t\t\tthis.element = this.template.render() as TElement;\n\n\t\t\t// Autoregister view children from #template.\n\t\t\tthis.registerChild( this.template.getViews() );\n\t\t}\n\n\t\tthis.isRendered = true;\n\t}\n\n\t/**\n\t * Recursively destroys the view instance and child views added by {@link #registerChild} and\n\t * residing in collections created by the {@link #createCollection}.\n\t *\n\t * Destruction disables all event listeners:\n\t * * created on the view, e.g. `view.on( 'event', () => {} )`,\n\t * * defined in the {@link #template} for DOM events.\n\t */\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\n\t\tthis._viewCollections.map( c => c.destroy() );\n\n\t\t// Template isn't obligatory for views.\n\t\tif ( this.template && ( this.template as any )._revertData ) {\n\t\t\tthis.template.revert( this.element! );\n\t\t}\n\t}\n}\n\n/**\n * Event fired by the {@link module:ui/view~View#render} method. Actual rendering is executed as a listener to\n * this event with the default priority.\n *\n * See {@link module:utils/observablemixin~Observable#decorate} for more information and samples.\n *\n * @eventName ~View#render\n */\nexport type UIViewRenderEvent = DecoratedMethodEvent<View, 'render'>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/template\n */\n\n/* global document */\n\nimport View from './view';\nimport ViewCollection from './viewcollection';\n\nimport {\n\tCKEditorError,\n\tEmitterMixin,\n\tisNode,\n\ttoArray,\n\ttype ArrayOrItem,\n\ttype Emitter,\n\ttype Observable,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport { isObject, cloneDeepWith } from 'lodash-es';\n\nconst xhtmlNs = 'http://www.w3.org/1999/xhtml';\n\n/**\n * A basic Template class. It renders a DOM HTML element or text from a\n * {@link module:ui/template~TemplateDefinition definition} and supports element attributes, children,\n * bindings to {@link module:utils/observablemixin~Observable observables} and DOM event propagation.\n *\n * A simple template can look like this:\n *\n * ```ts\n * const bind = Template.bind( observable, emitter );\n *\n * new Template( {\n * \ttag: 'p',\n * \tattributes: {\n * \t\tclass: 'foo',\n * \t\tstyle: {\n * \t\t\tbackgroundColor: 'yellow'\n * \t\t}\n * \t},\n * \ton: {\n * \t\tclick: bind.to( 'clicked' )\n * \t},\n * \tchildren: [\n * \t\t'A paragraph.'\n * \t]\n * } ).render();\n * ```\n *\n * and it will render the following HTML element:\n *\n * ```html\n * <p class=\"foo\" style=\"background-color: yellow;\">A paragraph.</p>\n * ```\n *\n * Additionally, the `observable` will always fire `clicked` upon clicking `<p>` in the DOM.\n *\n * See {@link module:ui/template~TemplateDefinition} to know more about templates and complex\n * template definitions.\n */\nexport default class Template extends EmitterMixin() {\n\tpublic ns?: string;\n\n\t/**\n\t * The tag (`tagName`) of this template, e.g. `div`. It also indicates that the template\n\t * renders to an HTML element.\n\t */\n\tpublic tag?: string;\n\n\t/**\n\t * The text of the template. It also indicates that the template renders to a DOM text node.\n\t */\n\tpublic text?: Array<TemplateSimpleValue | TemplateBinding>;\n\n\t/**\n\t * The attributes of the template, e.g. `{ id: [ 'ck-id' ] }`, corresponding with\n\t * the attributes of an HTML element.\n\t *\n\t * **Note**: This property only makes sense when {@link #tag} is defined.\n\t */\n\tpublic attributes?: Record<string, AttributeValues>;\n\n\t/**\n\t * The children of the template. They can be either:\n\t * * independent instances of {@link ~Template} (subtemplates),\n\t * * native DOM Nodes.\n\t *\n\t * **Note**: This property only makes sense when {@link #tag} is defined.\n\t */\n\tpublic children?: Array<ViewCollection | View | Node | Template>;\n\n\t/**\n\t * The DOM event listeners of the template.\n\t */\n\tpublic eventListeners?: Record<string, Array<TemplateToBinding>>;\n\n\t/**\n\t * Indicates whether this particular Template instance has been\n\t * {@link #render rendered}.\n\t */\n\tprivate _isRendered: boolean;\n\n\t/**\n\t * The data used by the {@link #revert} method to restore a node to its original state.\n\t *\n\t * See: {@link #apply}.\n\t */\n\tprivate _revertData: RevertData | null;\n\n\t/**\n\t * Creates an instance of the {@link ~Template} class.\n\t *\n\t * @param def The definition of the template.\n\t */\n\tconstructor( def: TemplateDefinition ) {\n\t\tsuper();\n\n\t\tObject.assign( this, normalize( clone( def ) ) );\n\n\t\tthis._isRendered = false;\n\t\tthis._revertData = null;\n\t}\n\n\t/**\n\t * Renders a DOM Node (an HTML element or text) out of the template.\n\t *\n\t * ```ts\n\t * const domNode = new Template( { ... } ).render();\n\t * ```\n\t *\n\t * See: {@link #apply}.\n\t */\n\tpublic render(): HTMLElement | Text {\n\t\tconst node = this._renderNode( {\n\t\t\tintoFragment: true\n\t\t} as any );\n\n\t\tthis._isRendered = true;\n\n\t\treturn node;\n\t}\n\n\t/**\n\t * Applies the template to an existing DOM Node, either HTML element or text.\n\t *\n\t * **Note:** No new DOM nodes will be created. Applying extends:\n\t *\n\t * {@link module:ui/template~TemplateDefinition attributes},\n\t * {@link module:ui/template~TemplateDefinition event listeners}, and\n\t * `textContent` of {@link module:ui/template~TemplateDefinition children} only.\n\t *\n\t * **Note:** Existing `class` and `style` attributes are extended when a template\n\t * is applied to an HTML element, while other attributes and `textContent` are overridden.\n\t *\n\t * **Note:** The process of applying a template can be easily reverted using the\n\t * {@link module:ui/template~Template#revert} method.\n\t *\n\t * ```ts\n\t * const element = document.createElement( 'div' );\n\t * const observable = new Model( { divClass: 'my-div' } );\n\t * const emitter = Object.create( EmitterMixin );\n\t * const bind = Template.bind( observable, emitter );\n\t *\n\t * new Template( {\n\t * \tattributes: {\n\t * \t\tid: 'first-div',\n\t * \t\tclass: bind.to( 'divClass' )\n\t * \t},\n\t * \ton: {\n\t * \t\tclick: bind( 'elementClicked' ) // Will be fired by the observable.\n\t * \t},\n\t * \tchildren: [\n\t * \t\t'Div text.'\n\t * \t]\n\t * } ).apply( element );\n\t *\n\t * console.log( element.outerHTML ); // -> '<div id=\"first-div\" class=\"my-div\"></div>'\n\t * ```\n\t *\n\t * @see module:ui/template~Template#render\n\t * @see module:ui/template~Template#revert\n\t * @param node Root node for the template to apply.\n\t */\n\tpublic apply( node: HTMLElement | Text ): HTMLElement | Text {\n\t\tthis._revertData = getEmptyRevertData();\n\n\t\tthis._renderNode( {\n\t\t\tnode,\n\t\t\tintoFragment: false,\n\t\t\tisApplying: true,\n\t\t\trevertData: this._revertData\n\t\t} );\n\n\t\treturn node;\n\t}\n\n\t/**\n\t * Reverts a template {@link module:ui/template~Template#apply applied} to a DOM node.\n\t *\n\t * @param node The root node for the template to revert. In most of the cases, it is the\n\t * same node used by {@link module:ui/template~Template#apply}.\n\t */\n\tpublic revert( node: HTMLElement | Text ): void {\n\t\tif ( !this._revertData ) {\n\t\t\t/**\n\t\t\t * Attempting to revert a template which has not been applied yet.\n\t\t\t *\n\t\t\t * @error ui-template-revert-not-applied\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'ui-template-revert-not-applied',\n\t\t\t\t[ this, node ]\n\t\t\t);\n\t\t}\n\n\t\tthis._revertTemplateFromNode( node, this._revertData );\n\t}\n\n\t/**\n\t * Returns an iterator which traverses the template in search of {@link module:ui/view~View}\n\t * instances and returns them one by one.\n\t *\n\t * ```ts\n\t * const viewFoo = new View();\n\t * const viewBar = new View();\n\t * const viewBaz = new View();\n\t * const template = new Template( {\n\t * \ttag: 'div',\n\t * \tchildren: [\n\t * \t\tviewFoo,\n\t * \t\t{\n\t * \t\t\ttag: 'div',\n\t * \t\t\tchildren: [\n\t * \t\t\t\tviewBar\n\t * \t\t\t]\n\t * \t\t},\n\t * \t\tviewBaz\n\t * \t]\n\t * } );\n\t *\n\t * // Logs: viewFoo, viewBar, viewBaz\n\t * for ( const view of template.getViews() ) {\n\t * \tconsole.log( view );\n\t * }\n\t * ```\n\t */\n\tpublic* getViews(): IterableIterator<View> {\n\t\tfunction* search( def: Template ): Iterable<View> {\n\t\t\tif ( def.children ) {\n\t\t\t\tfor ( const child of def.children ) {\n\t\t\t\t\tif ( isView( child ) ) {\n\t\t\t\t\t\tyield child;\n\t\t\t\t\t} else if ( isTemplate( child ) ) {\n\t\t\t\t\t\tyield* search( child );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tyield* search( this );\n\t}\n\n\t/**\n\t * An entry point to the interface which binds DOM nodes to\n\t * {@link module:utils/observablemixin~Observable observables}.\n\t * There are two types of bindings:\n\t *\n\t * * HTML element attributes or text `textContent` synchronized with attributes of an\n\t * {@link module:utils/observablemixin~Observable}. Learn more about {@link module:ui/template~BindChain#to}\n\t * and {@link module:ui/template~BindChain#if}.\n\t *\n\t * ```ts\n\t * const bind = Template.bind( observable, emitter );\n\t *\n\t * new Template( {\n\t * \tattributes: {\n\t * \t\t// Binds the element \"class\" attribute to observable#classAttribute.\n\t * \t\tclass: bind.to( 'classAttribute' )\n\t * \t}\n\t * } ).render();\n\t * ```\n\t *\n\t * * DOM events fired on HTML element propagated through\n\t * {@link module:utils/observablemixin~Observable}. Learn more about {@link module:ui/template~BindChain#to}.\n\t *\n\t * ```ts\n\t * const bind = Template.bind( observable, emitter );\n\t *\n\t * new Template( {\n\t * \ton: {\n\t * \t\t// Will be fired by the observable.\n\t * \t\tclick: bind( 'elementClicked' )\n\t * \t}\n\t * } ).render();\n\t * ```\n\t *\n\t * Also see {@link module:ui/view~View#bindTemplate}.\n\t *\n\t * @param observable An observable which provides boundable attributes.\n\t * @param emitter An emitter that listens to observable attribute\n\t * changes or DOM Events (depending on the kind of the binding). Usually, a {@link module:ui/view~View} instance.\n\t */\n\tpublic static override bind<TObservable extends Observable>(\n\t\tobservable: TObservable,\n\t\temitter: Emitter\n\t): BindChain<TObservable> {\n\t\treturn {\n\t\t\tto( eventNameOrFunctionOrAttribute: any, callback?: any ) {\n\t\t\t\treturn new TemplateToBinding( {\n\t\t\t\t\teventNameOrFunction: eventNameOrFunctionOrAttribute,\n\t\t\t\t\tattribute: eventNameOrFunctionOrAttribute,\n\t\t\t\t\tobservable, emitter, callback\n\t\t\t\t} );\n\t\t\t},\n\n\t\t\tif( attribute: string, valueIfTrue?: string, callback?: any ) {\n\t\t\t\treturn new TemplateIfBinding( {\n\t\t\t\t\tobservable, emitter, attribute, valueIfTrue, callback\n\t\t\t\t} );\n\t\t\t}\n\t\t} as any;\n\t}\n\n\t/**\n\t * Extends an existing {@link module:ui/template~Template} instance with some additional content\n\t * from another {@link module:ui/template~TemplateDefinition}.\n\t *\n\t * ```ts\n\t * const bind = Template.bind( observable, emitter );\n\t *\n\t * const template = new Template( {\n\t * \ttag: 'p',\n\t * \tattributes: {\n\t * \t\tclass: 'a',\n\t * \t\tdata-x: bind.to( 'foo' )\n\t * \t},\n\t * \tchildren: [\n\t * \t\t{\n\t * \t\t\ttag: 'span',\n\t * \t\t\tattributes: {\n\t * \t\t\t\tclass: 'b'\n\t * \t\t\t},\n\t * \t\t\tchildren: [\n\t * \t\t\t\t'Span'\n\t * \t\t\t]\n\t * \t\t}\n\t * \t]\n\t * } );\n\t *\n\t * // Instance-level extension.\n\t * Template.extend( template, {\n\t * \tattributes: {\n\t * \t\tclass: 'b',\n\t * \t\tdata-x: bind.to( 'bar' )\n\t * \t},\n\t * \tchildren: [\n\t * \t\t{\n\t * \t\t\tattributes: {\n\t * \t\t\t\tclass: 'c'\n\t * \t\t\t}\n\t * \t\t}\n\t * \t]\n\t * } );\n\t *\n\t * // Child extension.\n\t * Template.extend( template.children[ 0 ], {\n\t * \tattributes: {\n\t * \t\tclass: 'd'\n\t * \t}\n\t * } );\n\t * ```\n\t *\n\t * the `outerHTML` of `template.render()` is:\n\t *\n\t * ```html\n\t * <p class=\"a b\" data-x=\"{ observable.foo } { observable.bar }\">\n\t * \t<span class=\"b c d\">Span</span>\n\t * </p>\n\t * ```\n\t *\n\t * @param template An existing template instance to be extended.\n\t * @param def Additional definition to be applied to a template.\n\t */\n\tpublic static extend( template: Template, def: Partial<TemplateDefinition> ): void {\n\t\tif ( template._isRendered ) {\n\t\t\t/**\n\t\t\t * Extending a template after rendering may not work as expected. To make sure\n\t\t\t * the {@link module:ui/template~Template.extend extending} works for an element,\n\t\t\t * make sure it happens before {@link module:ui/template~Template#render} is called.\n\t\t\t *\n\t\t\t * @error template-extend-render\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'template-extend-render',\n\t\t\t\t[ this, template ]\n\t\t\t);\n\t\t}\n\n\t\textendTemplate( template, normalize( clone( def ) ) );\n\t}\n\n\t/**\n\t * Renders a DOM Node (either an HTML element or text) out of the template.\n\t *\n\t * @param data Rendering data.\n\t */\n\tprivate _renderNode( data: RenderData ) {\n\t\tlet isInvalid;\n\n\t\tif ( data.node ) {\n\t\t\t// When applying, a definition cannot have \"tag\" and \"text\" at the same time.\n\t\t\tisInvalid = this.tag && this.text;\n\t\t} else {\n\t\t\t// When rendering, a definition must have either \"tag\" or \"text\": XOR( this.tag, this.text ).\n\t\t\tisInvalid = this.tag ? this.text : !this.text;\n\t\t}\n\n\t\tif ( isInvalid ) {\n\t\t\t/**\n\t\t\t * Node definition cannot have the \"tag\" and \"text\" properties at the same time.\n\t\t\t * Node definition must have either \"tag\" or \"text\" when rendering a new Node.\n\t\t\t *\n\t\t\t * @error ui-template-wrong-syntax\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'ui-template-wrong-syntax',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\tif ( this.text ) {\n\t\t\treturn this._renderText( data );\n\t\t} else {\n\t\t\treturn this._renderElement( data );\n\t\t}\n\t}\n\n\t/**\n\t * Renders an HTML element out of the template.\n\t *\n\t * @param data Rendering data.\n\t */\n\tprivate _renderElement( data: RenderData ) {\n\t\tlet node = data.node;\n\n\t\tif ( !node ) {\n\t\t\tnode = data.node = document.createElementNS( this.ns || xhtmlNs, this.tag! ) as any;\n\t\t}\n\n\t\tthis._renderAttributes( data );\n\t\tthis._renderElementChildren( data );\n\t\tthis._setUpListeners( data );\n\n\t\treturn node;\n\t}\n\n\t/**\n\t * Renders a text node out of {@link module:ui/template~Template#text}.\n\t *\n\t * @param data Rendering data.\n\t */\n\tprivate _renderText( data: RenderData ) {\n\t\tlet node = data.node;\n\n\t\t// Save the original textContent to revert it in #revert().\n\t\tif ( node ) {\n\t\t\tdata.revertData!.text = node.textContent;\n\t\t} else {\n\t\t\tnode = data.node = document.createTextNode( '' );\n\t\t}\n\n\t\t// Check if this Text Node is bound to Observable. Cases:\n\t\t//\n\t\t//\t\ttext: [ Template.bind( ... ).to( ... ) ]\n\t\t//\n\t\t//\t\ttext: [\n\t\t//\t\t\t'foo',\n\t\t//\t\t\tTemplate.bind( ... ).to( ... ),\n\t\t//\t\t\t...\n\t\t//\t\t]\n\t\t//\n\t\tif ( hasTemplateBinding( this.text ) ) {\n\t\t\tthis._bindToObservable( {\n\t\t\t\tschema: this.text!,\n\t\t\t\tupdater: getTextUpdater( node ),\n\t\t\t\tdata\n\t\t\t} );\n\t\t}\n\t\t// Simply set text. Cases:\n\t\t//\n\t\t//\t\ttext: [ 'all', 'are', 'static' ]\n\t\t//\n\t\t//\t\ttext: [ 'foo' ]\n\t\t//\n\t\telse {\n\t\t\tnode.textContent = this.text!.join( '' );\n\t\t}\n\n\t\treturn node;\n\t}\n\n\t/**\n\t * Renders HTML element attributes out of {@link module:ui/template~Template#attributes}.\n\t *\n\t * @param data Rendering data.\n\t */\n\tprivate _renderAttributes( data: RenderData ) {\n\t\tif ( !this.attributes ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst node = data.node as HTMLElement;\n\t\tconst revertData = data.revertData;\n\n\t\tfor ( const attrName in this.attributes ) {\n\t\t\t// Current attribute value in DOM.\n\t\t\tconst domAttrValue = node.getAttribute( attrName );\n\n\t\t\t// The value to be set.\n\t\t\tconst attrValue = this.attributes[ attrName ];\n\n\t\t\t// Save revert data.\n\t\t\tif ( revertData ) {\n\t\t\t\trevertData.attributes[ attrName ] = domAttrValue;\n\t\t\t}\n\n\t\t\t// Detect custom namespace:\n\t\t\t//\n\t\t\t//\t\tclass: {\n\t\t\t//\t\t\tns: 'abc',\n\t\t\t//\t\t\tvalue: Template.bind( ... ).to( ... )\n\t\t\t//\t\t}\n\t\t\t//\n\t\t\tconst attrNs = isNamespaced( attrValue ) ? attrValue[ 0 ].ns : null;\n\n\t\t\t// Activate binding if one is found. Cases:\n\t\t\t//\n\t\t\t//\t\tclass: [\n\t\t\t//\t\t\tTemplate.bind( ... ).to( ... )\n\t\t\t//\t\t]\n\t\t\t//\n\t\t\t//\t\tclass: [\n\t\t\t//\t\t\t'bar',\n\t\t\t//\t\t\tTemplate.bind( ... ).to( ... ),\n\t\t\t//\t\t\t'baz'\n\t\t\t//\t\t]\n\t\t\t//\n\t\t\t//\t\tclass: {\n\t\t\t//\t\t\tns: 'abc',\n\t\t\t//\t\t\tvalue: Template.bind( ... ).to( ... )\n\t\t\t//\t\t}\n\t\t\t//\n\t\t\tif ( hasTemplateBinding( attrValue ) ) {\n\t\t\t\t// Normalize attributes with additional data like namespace:\n\t\t\t\t//\n\t\t\t\t//\t\tclass: {\n\t\t\t\t//\t\t\tns: 'abc',\n\t\t\t\t//\t\t\tvalue: [ ... ]\n\t\t\t\t//\t\t}\n\t\t\t\t//\n\t\t\t\tconst valueToBind = isNamespaced( attrValue ) ? attrValue[ 0 ].value : attrValue;\n\n\t\t\t\t// Extend the original value of attributes like \"style\" and \"class\",\n\t\t\t\t// don't override them.\n\t\t\t\tif ( revertData && shouldExtend( attrName ) ) {\n\t\t\t\t\tvalueToBind.unshift( domAttrValue );\n\t\t\t\t}\n\n\t\t\t\tthis._bindToObservable( {\n\t\t\t\t\tschema: valueToBind,\n\t\t\t\t\tupdater: getAttributeUpdater( node, attrName, attrNs ),\n\t\t\t\t\tdata\n\t\t\t\t} );\n\t\t\t}\n\n\t\t\t// Style attribute could be an Object so it needs to be parsed in a specific way.\n\t\t\t//\n\t\t\t//\t\tstyle: {\n\t\t\t//\t\t\twidth: '100px',\n\t\t\t//\t\t\theight: Template.bind( ... ).to( ... )\n\t\t\t//\t\t}\n\t\t\t//\n\t\t\telse if ( attrName == 'style' && typeof attrValue[ 0 ] !== 'string' ) {\n\t\t\t\tthis._renderStyleAttribute( attrValue[ 0 ] as any, data );\n\t\t\t}\n\n\t\t\t// Otherwise simply set the static attribute:\n\t\t\t//\n\t\t\t//\t\tclass: [ 'foo' ]\n\t\t\t//\n\t\t\t//\t\tclass: [ 'all', 'are', 'static' ]\n\t\t\t//\n\t\t\t//\t\tclass: [\n\t\t\t//\t\t\t{\n\t\t\t//\t\t\t\tns: 'abc',\n\t\t\t//\t\t\t\tvalue: [ 'foo' ]\n\t\t\t//\t\t\t}\n\t\t\t//\t\t]\n\t\t\t//\n\t\t\telse {\n\t\t\t\t// Extend the original value of attributes like \"style\" and \"class\",\n\t\t\t\t// don't override them.\n\t\t\t\tif ( revertData && domAttrValue && shouldExtend( attrName ) ) {\n\t\t\t\t\tattrValue.unshift( domAttrValue as any );\n\t\t\t\t}\n\n\t\t\t\tconst value: any = attrValue\n\t\t\t\t\t// Retrieve \"values\" from:\n\t\t\t\t\t//\n\t\t\t\t\t//\t\tclass: [\n\t\t\t\t\t//\t\t\t{\n\t\t\t\t\t//\t\t\t\tns: 'abc',\n\t\t\t\t\t//\t\t\t\tvalue: [ ... ]\n\t\t\t\t\t//\t\t\t}\n\t\t\t\t\t//\t\t]\n\t\t\t\t\t//\n\t\t\t\t\t.map( ( val: any ) => val ? ( val.value || val ) : val )\n\t\t\t\t\t// Flatten the array.\n\t\t\t\t\t.reduce( ( prev, next ) => prev.concat( next ), [] )\n\t\t\t\t\t// Convert into string.\n\t\t\t\t\t.reduce( arrayValueReducer, '' );\n\n\t\t\t\tif ( !isFalsy( value ) ) {\n\t\t\t\t\tnode.setAttributeNS( attrNs, attrName, value );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Renders the `style` attribute of an HTML element based on\n\t * {@link module:ui/template~Template#attributes}.\n\t *\n\t * A style attribute is an object with static values:\n\t *\n\t * ```ts\n\t * attributes: {\n\t * \tstyle: {\n\t * \t\tcolor: 'red'\n\t * \t}\n\t * }\n\t * ```\n\t *\n\t * or values bound to {@link module:ui/model~Model} properties:\n\t *\n\t * ```ts\n\t * attributes: {\n\t * \tstyle: {\n\t * \t\tcolor: bind.to( ... )\n\t * \t}\n\t * }\n\t * ```\n\t *\n\t * Note: The `style` attribute is rendered without setting the namespace. It does not seem to be\n\t * needed.\n\t *\n\t * @param styles Styles located in `attributes.style` of {@link module:ui/template~TemplateDefinition}.\n\t * @param data Rendering data.\n\t */\n\tprivate _renderStyleAttribute( styles: Record<string, TemplateSimpleValue | TemplateBinding>, data: RenderData ) {\n\t\tconst node = data.node;\n\n\t\tfor ( const styleName in styles ) {\n\t\t\tconst styleValue = styles[ styleName ];\n\n\t\t\t// Cases:\n\t\t\t//\n\t\t\t//\t\tstyle: {\n\t\t\t//\t\t\tcolor: bind.to( 'attribute' )\n\t\t\t//\t\t}\n\t\t\t//\n\t\t\tif ( hasTemplateBinding( styleValue ) ) {\n\t\t\t\tthis._bindToObservable( {\n\t\t\t\t\tschema: [ styleValue ],\n\t\t\t\t\tupdater: getStyleUpdater( node, styleName ),\n\t\t\t\t\tdata\n\t\t\t\t} );\n\t\t\t}\n\n\t\t\t// Cases:\n\t\t\t//\n\t\t\t//\t\tstyle: {\n\t\t\t//\t\t\tcolor: 'red'\n\t\t\t//\t\t}\n\t\t\t//\n\t\t\telse {\n\t\t\t\t( node as any ).style[ styleName ] = styleValue;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Recursively renders HTML element's children from {@link module:ui/template~Template#children}.\n\t *\n\t * @param data Rendering data.\n\t */\n\tprivate _renderElementChildren( data: RenderData ) {\n\t\tconst node = data.node as HTMLElement;\n\t\tconst container = data.intoFragment ? document.createDocumentFragment() : node;\n\t\tconst isApplying = data.isApplying;\n\t\tlet childIndex = 0;\n\n\t\tfor ( const child of this.children! ) {\n\t\t\tif ( isViewCollection( child ) ) {\n\t\t\t\tif ( !isApplying ) {\n\t\t\t\t\tchild.setParent( node );\n\n\t\t\t\t\t// Note: ViewCollection renders its children.\n\t\t\t\t\tfor ( const view of child ) {\n\t\t\t\t\t\tcontainer.appendChild( view.element! );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} else if ( isView( child ) ) {\n\t\t\t\tif ( !isApplying ) {\n\t\t\t\t\tif ( !child.isRendered ) {\n\t\t\t\t\t\tchild.render();\n\t\t\t\t\t}\n\n\t\t\t\t\tcontainer.appendChild( child.element! );\n\t\t\t\t}\n\t\t\t} else if ( isNode( child ) ) {\n\t\t\t\tcontainer.appendChild( child );\n\t\t\t} else {\n\t\t\t\tif ( isApplying ) {\n\t\t\t\t\tconst revertData = data.revertData!;\n\t\t\t\t\tconst childRevertData = getEmptyRevertData();\n\n\t\t\t\t\trevertData.children.push( childRevertData );\n\n\t\t\t\t\tchild._renderNode( {\n\t\t\t\t\t\tintoFragment: false,\n\t\t\t\t\t\tnode: container.childNodes[ childIndex++ ] as any,\n\t\t\t\t\t\tisApplying: true,\n\t\t\t\t\t\trevertData: childRevertData\n\t\t\t\t\t} );\n\t\t\t\t} else {\n\t\t\t\t\tcontainer.appendChild( child.render() );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif ( data.intoFragment ) {\n\t\t\tnode.appendChild( container );\n\t\t}\n\t}\n\n\t/**\n\t * Activates `on` event listeners from the {@link module:ui/template~TemplateDefinition}\n\t * on an HTML element.\n\t *\n\t * @param data Rendering data.\n\t */\n\tprivate _setUpListeners( data: RenderData ) {\n\t\tif ( !this.eventListeners ) {\n\t\t\treturn;\n\t\t}\n\n\t\tfor ( const key in this.eventListeners ) {\n\t\t\tconst revertBindings = this.eventListeners[ key ].map( schemaItem => {\n\t\t\t\tconst [ domEvtName, domSelector ] = key.split( '@' );\n\n\t\t\t\treturn schemaItem.activateDomEventListener( domEvtName, domSelector, data );\n\t\t\t} );\n\n\t\t\tif ( data.revertData ) {\n\t\t\t\tdata.revertData.bindings.push( revertBindings );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * For a given {@link module:ui/template~TemplateValueSchema} containing {@link module:ui/template~TemplateBinding}\n\t * activates the binding and sets its initial value.\n\t *\n\t * Note: {@link module:ui/template~TemplateValueSchema} can be for HTML element attributes or\n\t * text node `textContent`.\n\t *\n\t * @param options Binding options.\n\t * @param options.updater A function which updates the DOM (like attribute or text).\n\t * @param options.data Rendering data.\n\t */\n\tprivate _bindToObservable( { schema, updater, data }: {\n\t\tschema: Array<TemplateSimpleValue | TemplateBinding>;\n\t\tupdater: Updater;\n\t\tdata: RenderData;\n\t} ) {\n\t\tconst revertData = data.revertData;\n\n\t\t// Set initial values.\n\t\tsyncValueSchemaValue( schema, updater, data );\n\n\t\tconst revertBindings = schema\n\t\t\t// Filter \"falsy\" (false, undefined, null, '') value schema components out.\n\t\t\t.filter( item => !isFalsy( item ) )\n\t\t\t// Filter inactive bindings from schema, like static strings ('foo'), numbers (42), etc.\n\t\t\t.filter( ( item: any ): item is TemplateBinding => item.observable )\n\t\t\t// Once only the actual binding are left, let the emitter listen to observable change:attribute event.\n\t\t\t// TODO: Reduce the number of listeners attached as many bindings may listen\n\t\t\t// to the same observable attribute.\n\t\t\t.map( templateBinding => templateBinding.activateAttributeListener( schema, updater, data ) );\n\n\t\tif ( revertData ) {\n\t\t\trevertData.bindings.push( revertBindings );\n\t\t}\n\t}\n\n\t/**\n\t * Reverts {@link module:ui/template~RenderData#revertData template data} from a node to\n\t * return it to the original state.\n\t *\n\t * @param node A node to be reverted.\n\t * @param revertData An object that stores information about what changes have been made by\n\t * {@link #apply} to the node. See {@link module:ui/template~RenderData#revertData} for more information.\n\t */\n\tprivate _revertTemplateFromNode( node: HTMLElement | Text, revertData: RevertData ) {\n\t\tfor ( const binding of revertData.bindings ) {\n\t\t\t// Each binding may consist of several observable+observable#attribute.\n\t\t\t// like the following has 2:\n\t\t\t//\n\t\t\t//\t\tclass: [\n\t\t\t//\t\t\t'x',\n\t\t\t//\t\t\tbind.to( 'foo' ),\n\t\t\t//\t\t\t'y',\n\t\t\t//\t\t\tbind.to( 'bar' )\n\t\t\t//\t\t]\n\t\t\t//\n\t\t\tfor ( const revertBinding of binding ) {\n\t\t\t\trevertBinding();\n\t\t\t}\n\t\t}\n\n\t\tif ( revertData.text ) {\n\t\t\tnode.textContent = revertData.text;\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst element = node as HTMLElement;\n\n\t\tfor ( const attrName in revertData.attributes ) {\n\t\t\tconst attrValue = revertData.attributes[ attrName ];\n\n\t\t\t// When the attribute has **not** been set before #apply().\n\t\t\tif ( attrValue === null ) {\n\t\t\t\telement.removeAttribute( attrName );\n\t\t\t} else {\n\t\t\t\telement.setAttribute( attrName, attrValue );\n\t\t\t}\n\t\t}\n\n\t\tfor ( let i = 0; i < revertData.children.length; ++i ) {\n\t\t\tthis._revertTemplateFromNode( element.childNodes[ i ] as any, revertData.children[ i ] );\n\t\t}\n\t}\n}\n\ntype AttributeValues = Array<TemplateSimpleValue | TemplateBinding> |\n\t[ NamespacedValue<Array<TemplateSimpleValue | TemplateBinding>> ];\n\n/**\n * Describes a binding created by the {@link module:ui/template~Template.bind} interface.\n *\n * @internal\n */\nexport abstract class TemplateBinding {\n\t/**\n\t * The name of the {@link module:ui/template~TemplateBinding#observable observed attribute}.\n\t */\n\tpublic readonly attribute: string;\n\n\t/**\n\t * An observable instance of the binding. It either:\n\t *\n\t * * provides the attribute with the value,\n\t * * or passes the event when a corresponding DOM event is fired.\n\t */\n\tpublic readonly observable: Observable;\n\n\t/**\n\t * An {@link module:utils/emittermixin~Emitter} used by the binding to:\n\t *\n\t * * listen to the attribute change in the {@link module:ui/template~TemplateBinding#observable},\n\t * * or listen to the event in the DOM.\n\t */\n\tpublic readonly emitter: Emitter;\n\n\t/**\n\t * A custom function to process the value of the {@link module:ui/template~TemplateBinding#attribute}.\n\t */\n\tpublic readonly callback?: ( value: any, node: Node ) => TemplateSimpleValue;\n\n\t/**\n\t * Creates an instance of the {@link module:ui/template~TemplateBinding} class.\n\t *\n\t * @param def The definition of the binding.\n\t */\n\tconstructor( def: {\n\t\tattribute: string;\n\t\tobservable: Observable;\n\t\temitter: Emitter;\n\t\tcallback?: ( value: any, node: Node ) => TemplateSimpleValue;\n\t} ) {\n\t\tthis.attribute = def.attribute;\n\t\tthis.observable = def.observable;\n\t\tthis.emitter = def.emitter;\n\t\tthis.callback = def.callback;\n\t}\n\n\t/**\n\t * Returns the value of the binding. It is the value of the {@link module:ui/template~TemplateBinding#attribute} in\n\t * {@link module:ui/template~TemplateBinding#observable}. The value may be processed by the\n\t * {@link module:ui/template~TemplateBinding#callback}, if such has been passed to the binding.\n\t *\n\t * @param node A native DOM node, passed to the custom {@link module:ui/template~TemplateBinding#callback}.\n\t * @returns The value of {@link module:ui/template~TemplateBinding#attribute} in\n\t * {@link module:ui/template~TemplateBinding#observable}.\n\t */\n\tpublic getValue( node: Node ): TemplateSimpleValue {\n\t\tconst value = ( this.observable as any )[ this.attribute ];\n\n\t\treturn this.callback ? this.callback( value, node ) : value;\n\t}\n\n\t/**\n\t * Activates the listener which waits for changes of the {@link module:ui/template~TemplateBinding#attribute} in\n\t * {@link module:ui/template~TemplateBinding#observable}, then updates the DOM with the aggregated\n\t * value of {@link module:ui/template~TemplateValueSchema}.\n\t *\n\t * @param schema A full schema to generate an attribute or text in the DOM.\n\t * @param updater A DOM updater function used to update the native DOM attribute or text.\n\t * @param data Rendering data.\n\t * @returns A function to sever the listener binding.\n\t */\n\tpublic activateAttributeListener(\n\t\tschema: Array<TemplateSimpleValue | TemplateBinding>,\n\t\tupdater: Updater,\n\t\tdata: RenderData\n\t): () => void {\n\t\tconst callback = () => syncValueSchemaValue( schema, updater, data );\n\n\t\tthis.emitter.listenTo<ObservableChangeEvent>( this.observable, `change:${ this.attribute }`, callback );\n\n\t\t// Allows revert of the listener.\n\t\treturn () => {\n\t\t\tthis.emitter.stopListening( this.observable, `change:${ this.attribute }`, callback );\n\t\t};\n\t}\n}\n\n/**\n * Describes either:\n *\n * * a binding to an {@link module:utils/observablemixin~Observable},\n * * or a native DOM event binding.\n *\n * It is created by the {@link module:ui/template~BindChain#to} method.\n *\n * @internal\n */\nexport class TemplateToBinding extends TemplateBinding {\n\tpublic readonly eventNameOrFunction: string | ( ( domEvent: Event ) => void );\n\n\tconstructor( def: ConstructorParameters<typeof TemplateBinding>[ 0 ] & {\n\t\teventNameOrFunction: string | ( ( domEvent: Event ) => void );\n\t} ) {\n\t\tsuper( def );\n\n\t\tthis.eventNameOrFunction = def.eventNameOrFunction;\n\t}\n\n\t/**\n\t * Activates the listener for the native DOM event, which when fired, is propagated by\n\t * the {@link module:ui/template~TemplateBinding#emitter}.\n\t *\n\t * @param domEvtName The name of the native DOM event.\n\t * @param domSelector The selector in the DOM to filter delegated events.\n\t * @param data Rendering data.\n\t * @returns A function to sever the listener binding.\n\t */\n\tpublic activateDomEventListener(\n\t\tdomEvtName: string,\n\t\tdomSelector: string,\n\t\tdata: { node: any }\n\t): () => void {\n\t\tconst callback = ( evt: unknown, domEvt: Event ) => {\n\t\t\tif ( !domSelector || ( domEvt.target as Element ).matches( domSelector ) ) {\n\t\t\t\tif ( typeof this.eventNameOrFunction == 'function' ) {\n\t\t\t\t\tthis.eventNameOrFunction( domEvt );\n\t\t\t\t} else {\n\t\t\t\t\tthis.observable.fire( this.eventNameOrFunction, domEvt );\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\n\t\tthis.emitter.listenTo( data.node, domEvtName, callback );\n\n\t\t// Allows revert of the listener.\n\t\treturn () => {\n\t\t\tthis.emitter.stopListening( data.node, domEvtName, callback );\n\t\t};\n\t}\n}\n\n/**\n * Describes a binding to {@link module:utils/observablemixin~Observable} created by the {@link module:ui/template~BindChain#if}\n * method.\n *\n * @internal\n */\nexport class TemplateIfBinding extends TemplateBinding {\n\t/**\n\t * The value of the DOM attribute or text to be set if the {@link module:ui/template~TemplateBinding#attribute} in\n\t * {@link module:ui/template~TemplateBinding#observable} is `true`.\n\t */\n\tpublic readonly valueIfTrue?: string;\n\n\tconstructor( def: ConstructorParameters<typeof TemplateBinding>[ 0 ] & {\n\t\tvalueIfTrue?: string;\n\t} ) {\n\t\tsuper( def );\n\n\t\tthis.valueIfTrue = def.valueIfTrue;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override getValue( node: Node ): TemplateSimpleValue {\n\t\tconst value = super.getValue( node );\n\n\t\treturn isFalsy( value ) ? false : ( this.valueIfTrue || true ) as any;\n\t}\n}\n\n/**\n * Checks whether given {@link module:ui/template~TemplateValueSchema} contains a\n * {@link module:ui/template~TemplateBinding}.\n */\nfunction hasTemplateBinding( schema: any ) {\n\tif ( !schema ) {\n\t\treturn false;\n\t}\n\n\t// Normalize attributes with additional data like namespace:\n\t//\n\t//\t\tclass: {\n\t//\t\t\tns: 'abc',\n\t//\t\t\tvalue: [ ... ]\n\t//\t\t}\n\t//\n\tif ( schema.value ) {\n\t\tschema = schema.value;\n\t}\n\n\tif ( Array.isArray( schema ) ) {\n\t\treturn schema.some( hasTemplateBinding );\n\t} else if ( schema instanceof TemplateBinding ) {\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n\n/**\n * Assembles the value using {@link module:ui/template~TemplateValueSchema} and stores it in a form of\n * an Array. Each entry of the Array corresponds to one of {@link module:ui/template~TemplateValueSchema}\n * items.\n *\n * @param node DOM Node updated when {@link module:utils/observablemixin~Observable} changes.\n */\nfunction getValueSchemaValue( schema: Array<TemplateSimpleValue | TemplateBinding>, node: Node ) {\n\treturn schema.map( schemaItem => {\n\t\t// Process {@link module:ui/template~TemplateBinding} bindings.\n\t\tif ( schemaItem instanceof TemplateBinding ) {\n\t\t\treturn schemaItem.getValue( node );\n\t\t}\n\n\t\t// All static values like strings, numbers, and \"falsy\" values (false, null, undefined, '', etc.) just pass.\n\t\treturn schemaItem;\n\t} );\n}\n\n/**\n * A function executed each time the bound Observable attribute changes, which updates the DOM with a value\n * constructed from {@link module:ui/template~TemplateValueSchema}.\n *\n * @param updater A function which updates the DOM (like attribute or text).\n * @param node DOM Node updated when {@link module:utils/observablemixin~Observable} changes.\n */\nfunction syncValueSchemaValue(\n\tschema: Array<TemplateSimpleValue | TemplateBinding>,\n\tupdater: Updater,\n\t{ node }: { node: Node }\n) {\n\tconst values = getValueSchemaValue( schema, node );\n\tlet value: TemplateSimpleValue;\n\n\t// Check if schema is a single Template.bind.if, like:\n\t//\n\t//\t\tclass: Template.bind.if( 'foo' )\n\t//\n\tif ( schema.length == 1 && schema[ 0 ] instanceof TemplateIfBinding ) {\n\t\tvalue = values[ 0 ];\n\t} else {\n\t\tvalue = values.reduce( arrayValueReducer, '' );\n\t}\n\n\tif ( isFalsy( value ) ) {\n\t\tupdater.remove();\n\t} else {\n\t\tupdater.set( value );\n\t}\n}\n\ninterface Updater {\n\tset( value: any ): void;\n\tremove(): void;\n}\n\n/**\n * Returns an object consisting of `set` and `remove` functions, which\n * can be used in the context of DOM Node to set or reset `textContent`.\n * @see module:ui/view~View#_bindToObservable\n *\n * @param node DOM Node to be modified.\n */\nfunction getTextUpdater( node: Node ): Updater {\n\treturn {\n\t\tset( value ) {\n\t\t\tnode.textContent = value;\n\t\t},\n\n\t\tremove() {\n\t\t\tnode.textContent = '';\n\t\t}\n\t};\n}\n\n/**\n * Returns an object consisting of `set` and `remove` functions, which\n * can be used in the context of DOM Node to set or reset an attribute.\n * @see module:ui/view~View#_bindToObservable\n *\n * @param el DOM Node to be modified.\n * @param attrName Name of the attribute to be modified.\n * @param ns Namespace to use.\n */\nfunction getAttributeUpdater( el: Element, attrName: string, ns: string | null ): Updater {\n\treturn {\n\t\tset( value ) {\n\t\t\tel.setAttributeNS( ns, attrName, value );\n\t\t},\n\n\t\tremove() {\n\t\t\tel.removeAttributeNS( ns, attrName );\n\t\t}\n\t};\n}\n\n/**\n * Returns an object consisting of `set` and `remove` functions, which\n * can be used in the context of CSSStyleDeclaration to set or remove a style.\n * @see module:ui/view~View#_bindToObservable\n *\n * @param el DOM Node to be modified.\n * @param styleName Name of the style to be modified.\n */\nfunction getStyleUpdater( el: any, styleName: string ): Updater {\n\treturn {\n\t\tset( value ) {\n\t\t\tel.style[ styleName ] = value;\n\t\t},\n\n\t\tremove() {\n\t\t\tel.style[ styleName ] = null;\n\t\t}\n\t};\n}\n\n/**\n * Clones definition of the template.\n */\nfunction clone( def: unknown ) {\n\tconst clone = cloneDeepWith( def, value => {\n\t\t// Don't clone the `Template.bind`* bindings because of the references to Observable\n\t\t// and DomEmitterMixin instances inside, which would also be traversed and cloned by greedy\n\t\t// cloneDeepWith algorithm. There's no point in cloning Observable/DomEmitterMixins\n\t\t// along with the definition.\n\t\t//\n\t\t// Don't clone Template instances if provided as a child. They're simply #render()ed\n\t\t// and nothing should interfere.\n\t\t//\n\t\t// Also don't clone View instances if provided as a child of the Template. The template\n\t\t// instance will be extracted from the View during the normalization and there's no need\n\t\t// to clone it.\n\t\tif ( value && ( value instanceof TemplateBinding || isTemplate( value ) || isView( value ) || isViewCollection( value ) ) ) {\n\t\t\treturn value;\n\t\t}\n\t} );\n\n\treturn clone;\n}\n\n/**\n * Normalizes given {@link module:ui/template~TemplateDefinition}.\n *\n * See:\n * * {@link normalizeAttributes}\n * * {@link normalizeListeners}\n * * {@link normalizePlainTextDefinition}\n * * {@link normalizeTextDefinition}\n *\n * @param def A template definition.\n * @returns Normalized definition.\n */\nfunction normalize( def: any ) {\n\tif ( typeof def == 'string' ) {\n\t\tdef = normalizePlainTextDefinition( def );\n\t} else if ( def.text ) {\n\t\tnormalizeTextDefinition( def );\n\t}\n\n\tif ( def.on ) {\n\t\tdef.eventListeners = normalizeListeners( def.on );\n\n\t\t// Template mixes EmitterMixin, so delete #on to avoid collision.\n\t\tdelete def.on;\n\t}\n\n\tif ( !def.text ) {\n\t\tif ( def.attributes ) {\n\t\t\tnormalizeAttributes( def.attributes );\n\t\t}\n\n\t\tconst children = [];\n\n\t\tif ( def.children ) {\n\t\t\tif ( isViewCollection( def.children ) ) {\n\t\t\t\tchildren.push( def.children );\n\t\t\t} else {\n\t\t\t\tfor ( const child of def.children ) {\n\t\t\t\t\tif ( isTemplate( child ) || isView( child ) || isNode( child ) ) {\n\t\t\t\t\t\tchildren.push( child );\n\t\t\t\t\t} else {\n\t\t\t\t\t\tchildren.push( new Template( child ) );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tdef.children = children;\n\t}\n\n\treturn def;\n}\n\n/**\n * Normalizes \"attributes\" section of {@link module:ui/template~TemplateDefinition}.\n *\n * ```\n * attributes: {\n * \ta: 'bar',\n * \tb: {@link module:ui/template~TemplateBinding},\n * \tc: {\n * \t\tvalue: 'bar'\n * \t}\n * }\n * ```\n *\n * becomes\n *\n * ```\n * attributes: {\n * \ta: [ 'bar' ],\n * \tb: [ {@link module:ui/template~TemplateBinding} ],\n * \tc: {\n * \t\tvalue: [ 'bar' ]\n * \t}\n * }\n * ```\n */\nfunction normalizeAttributes( attributes: any ) {\n\tfor ( const a in attributes ) {\n\t\tif ( attributes[ a ].value ) {\n\t\t\tattributes[ a ].value = toArray( attributes[ a ].value );\n\t\t}\n\n\t\tarrayify( attributes, a );\n\t}\n}\n\n/**\n * Normalizes \"on\" section of {@link module:ui/template~TemplateDefinition}.\n *\n * ```\n * on: {\n * \ta: 'bar',\n * \tb: {@link module:ui/template~TemplateBinding},\n * \tc: [ {@link module:ui/template~TemplateBinding}, () => { ... } ]\n * }\n * ```\n *\n * becomes\n *\n * ```\n * on: {\n * \ta: [ 'bar' ],\n * \tb: [ {@link module:ui/template~TemplateBinding} ],\n * \tc: [ {@link module:ui/template~TemplateBinding}, () => { ... } ]\n * }\n * ```\n *\n * @returns Object containing normalized listeners.\n */\nfunction normalizeListeners( listeners: any ) {\n\tfor ( const l in listeners ) {\n\t\tarrayify( listeners, l );\n\t}\n\n\treturn listeners;\n}\n\n/**\n * Normalizes \"string\" {@link module:ui/template~TemplateDefinition}.\n *\n * ```\n * \"foo\"\n * ```\n *\n * becomes\n *\n * ```\n * { text: [ 'foo' ] },\n * ```\n *\n * @returns Normalized template definition.\n */\nfunction normalizePlainTextDefinition( def: string ) {\n\treturn {\n\t\ttext: [ def ]\n\t};\n}\n\n/**\n * Normalizes text {@link module:ui/template~TemplateDefinition}.\n *\n * ```\n * children: [\n * \t{ text: 'def' },\n * \t{ text: {@link module:ui/template~TemplateBinding} }\n * ]\n * ```\n *\n * becomes\n *\n * ```\n * children: [\n * \t{ text: [ 'def' ] },\n * \t{ text: [ {@link module:ui/template~TemplateBinding} ] }\n * ]\n * ```\n */\nfunction normalizeTextDefinition( def: any ) {\n\tdef.text = toArray( def.text );\n}\n\n/**\n * Wraps an entry in Object in an Array, if not already one.\n *\n * ```\n * {\n * \tx: 'y',\n * \ta: [ 'b' ]\n * }\n * ```\n *\n * becomes\n *\n * ```\n * {\n * \tx: [ 'y' ],\n * \ta: [ 'b' ]\n * }\n * ```\n */\nfunction arrayify( obj: any, key: string ) {\n\tobj[ key ] = toArray( obj[ key ] );\n}\n\n/**\n * A helper which concatenates the value avoiding unwanted\n * leading white spaces.\n */\nfunction arrayValueReducer( prev: TemplateSimpleValue, cur: TemplateSimpleValue ) {\n\tif ( isFalsy( cur ) ) {\n\t\treturn prev;\n\t} else if ( isFalsy( prev ) ) {\n\t\treturn cur;\n\t} else {\n\t\treturn `${ prev } ${ cur }`;\n\t}\n}\n\n/**\n * Extends one object defined in the following format:\n *\n * ```\n * {\n * \tkey1: [Array1],\n * \tkey2: [Array2],\n * \t...\n * \tkeyN: [ArrayN]\n * }\n * ```\n *\n * with another object of the same data format.\n *\n * @param obj Base object.\n * @param ext Object extending base.\n */\nfunction extendObjectValueArray( obj: any, ext: any ) {\n\tfor ( const a in ext ) {\n\t\tif ( obj[ a ] ) {\n\t\t\tobj[ a ].push( ...ext[ a ] );\n\t\t} else {\n\t\t\tobj[ a ] = ext[ a ];\n\t\t}\n\t}\n}\n\n/**\n * A helper for {@link module:ui/template~Template#extend}. Recursively extends {@link module:ui/template~Template} instance\n * with content from {@link module:ui/template~TemplateDefinition}. See {@link module:ui/template~Template#extend} to learn more.\n *\n * @param def A template instance to be extended.\n * @param def A definition which is to extend the template instance.\n * @param Error context.\n */\nfunction extendTemplate( template: Template, def: any ) {\n\tif ( def.attributes ) {\n\t\tif ( !template.attributes ) {\n\t\t\ttemplate.attributes = {};\n\t\t}\n\n\t\textendObjectValueArray( template.attributes, def.attributes );\n\t}\n\n\tif ( def.eventListeners ) {\n\t\tif ( !template.eventListeners ) {\n\t\t\ttemplate.eventListeners = {};\n\t\t}\n\n\t\textendObjectValueArray( template.eventListeners, def.eventListeners );\n\t}\n\n\tif ( def.text ) {\n\t\ttemplate.text!.push( ...def.text );\n\t}\n\n\tif ( def.children && def.children.length ) {\n\t\tif ( template.children!.length != def.children.length ) {\n\t\t\t/**\n\t\t\t * The number of children in extended definition does not match.\n\t\t\t *\n\t\t\t * @error ui-template-extend-children-mismatch\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'ui-template-extend-children-mismatch',\n\t\t\t\ttemplate\n\t\t\t);\n\t\t}\n\n\t\tlet childIndex = 0;\n\n\t\tfor ( const childDef of def.children ) {\n\t\t\textendTemplate( template.children![ childIndex++ ] as any, childDef );\n\t\t}\n\t}\n}\n\n/**\n * Checks if value is \"falsy\".\n * Note: 0 (Number) is not \"falsy\" in this context.\n *\n * @param value Value to be checked.\n */\nfunction isFalsy( value: unknown ): value is FalsyValue {\n\treturn !value && value !== 0;\n}\n\n/**\n * Checks if the item is an instance of {@link module:ui/view~View}\n *\n * @param value Value to be checked.\n */\nfunction isView( item: unknown ): item is View {\n\treturn item instanceof View;\n}\n\n/**\n * Checks if the item is an instance of {@link module:ui/template~Template}\n *\n * @param value Value to be checked.\n */\nfunction isTemplate( item: unknown ): item is Template {\n\treturn item instanceof Template;\n}\n\n/**\n * Checks if the item is an instance of {@link module:ui/viewcollection~ViewCollection}\n *\n * @param value Value to be checked.\n */\nfunction isViewCollection( item: unknown ): item is ViewCollection {\n\treturn item instanceof ViewCollection;\n}\n\n/**\n * Checks if value array contains the one with namespace.\n */\nfunction isNamespaced(\n\tattrValue: AttributeValues\n): attrValue is [ NamespacedValue<Array<TemplateSimpleValue | TemplateBinding>> ] {\n\treturn isObject( attrValue[ 0 ] ) && ( attrValue[ 0 ] as any ).ns;\n}\n\n/**\n * Creates an empty skeleton for {@link module:ui/template~Template#revert}\n * data.\n */\nfunction getEmptyRevertData(): RevertData {\n\treturn {\n\t\tchildren: [],\n\t\tbindings: [],\n\t\tattributes: {}\n\t};\n}\n\n/**\n * Checks whether an attribute should be extended when\n * {@link module:ui/template~Template#apply} is called.\n *\n * @param attrName Attribute name to check.\n */\nfunction shouldExtend( attrName: string ) {\n\treturn attrName == 'class' || attrName == 'style';\n}\n\n/**\n * A definition of the {@link module:ui/template~Template}. It describes what kind of\n * node a template will render (HTML element or text), attributes of an element, DOM event\n * listeners and children.\n *\n * Also see:\n * * {@link module:ui/template~TemplateValueSchema} to learn about HTML element attributes,\n * * {@link module:ui/template~TemplateListenerSchema} to learn about DOM event listeners.\n *\n * A sample definition on an HTML element can look like this:\n *\n * ```ts\n * new Template( {\n * \ttag: 'p',\n * \tchildren: [\n * \t\t{\n * \t\t\ttag: 'span',\n * \t\t\tattributes: { ... },\n * \t\t\tchildren: [ ... ],\n * \t\t},\n * \t\t{\n * \t\t\ttext: 'statictext'\n * \t\t},\n * \t\t'also-statictext',\n * \t],\n * \tattributes: {\n * \t\tclass: {@link module:ui/template~TemplateValueSchema},\n * \t\tid: {@link module:ui/template~TemplateValueSchema},\n * \t\tstyle: {@link module:ui/template~TemplateValueSchema}\n *\n * \t\t// ...\n * \t},\n * \ton: {\n * \t\t'click': {@link module:ui/template~TemplateListenerSchema}\n *\n * \t\t// Document.querySelector format is also accepted.\n * \t\t'keyup@a.some-class': {@link module:ui/template~TemplateListenerSchema}\n *\n * \t\t// ...\n * \t}\n * } );\n * ```\n *\n * A {@link module:ui/view~View}, another {@link module:ui/template~Template} or a native DOM node\n * can also become a child of a template. When a view is passed, its {@link module:ui/view~View#element} is used:\n *\n * ```ts\n * const view = new SomeView();\n * const childTemplate = new Template( { ... } );\n * const childNode = document.createElement( 'b' );\n *\n * new Template( {\n * \ttag: 'p',\n *\n * \tchildren: [\n * \t\t// view#element will be added as a child of this <p>.\n * \t\tview,\n *\n * \t\t// The output of childTemplate.render() will be added here.\n * \t\tchildTemplate,\n *\n * \t\t// Native DOM nodes are included directly in the rendered output.\n * \t\tchildNode\n * \t]\n * } );\n * ```\n *\n * An entire {@link module:ui/viewcollection~ViewCollection} can be used as a child in the definition:\n *\n * ```ts\n * const collection = new ViewCollection();\n * collection.add( someView );\n *\n * new Template( {\n * \ttag: 'p',\n *\n * \tchildren: collection\n * } );\n * ```\n */\nexport type TemplateDefinition = string |\n\tTemplate |\n\tTemplateElementDefinition |\n\tTemplateTextDefinition;\n\nexport type TemplateElementDefinition = {\n\n\t/**\n\t * See the template {@link module:ui/template~Template#tag} property.\n\t */\n\ttag: string;\n\n\t/**\n\t * See the template {@link module:ui/template~Template#attributes} property.\n\t */\n\tattributes?: Record<string, TemplateValueSchema>;\n\n\t/**\n\t * See the template {@link module:ui/template~Template#children} property.\n\t */\n\tchildren?: Iterable<View | Node | Template | TemplateDefinition>;\n\n\t/**\n\t * See the template {@link module:ui/template~Template#eventListeners} property.\n\t */\n\ton?: Record<string, TemplateListenerSchema>;\n};\n\nexport type TemplateTextDefinition = {\n\n\t/**\n\t * See the template {@link module:ui/template~Template#text} property.\n\t */\n\ttext: ArrayOrItem<TemplateSimpleValueSchema>;\n};\n\nexport type FalsyValue = false | null | undefined | '';\n\nexport type NamespacedValue<T> = { ns: string; value: T };\n\nexport type TemplateSimpleValue = string | boolean | number | null | undefined;\n\nexport type TemplateSimpleValueSchema = TemplateSimpleValue | AttributeBinding;\n\n/**\n * Describes a value of an HTML element attribute or `textContent`. It allows combining multiple\n * data sources like static values and {@link module:utils/observablemixin~Observable} attributes.\n *\n * Also see:\n * * {@link module:ui/template~TemplateDefinition} to learn where to use it,\n * * {@link module:ui/template~Template.bind} to learn how to configure\n * {@link module:utils/observablemixin~Observable} attribute bindings,\n * * {@link module:ui/template~Template#render} to learn how to render a template,\n * * {@link module:ui/template~BindChain#to `to()`} and {@link module:ui/template~BindChain#if `if()`}\n * methods to learn more about bindings.\n *\n * Attribute values can be described in many different ways:\n *\n * ```ts\n * // Bind helper will create bindings to attributes of the observable.\n * const bind = Template.bind( observable, emitter );\n *\n * new Template( {\n * \ttag: 'p',\n * \tattributes: {\n * \t\t// A plain string schema.\n * \t\t'class': 'static-text',\n *\n * \t\t// An object schema, binds to the \"foo\" attribute of the\n * \t\t// observable and follows its value.\n * \t\t'class': bind.to( 'foo' ),\n *\n * \t\t// An array schema, combines the above.\n * \t\t'class': [\n * \t\t\t'static-text',\n * \t\t\tbind.to( 'bar', () => { ... } ),\n *\n * \t\t\t// Bindings can also be conditional.\n * \t\t\tbind.if( 'baz', 'class-when-baz-is-true' )\n * \t\t],\n *\n * \t\t// An array schema, with a custom namespace, e.g. useful for creating SVGs.\n * \t\t'class': {\n * \t\t\tns: 'http://ns.url',\n * \t\t\tvalue: [\n * \t\t\t\tbind.if( 'baz', 'value-when-true' ),\n * \t\t\t\t'static-text'\n * \t\t\t]\n * \t\t},\n *\n * \t\t// An object schema, specific for styles.\n * \t\tstyle: {\n * \t\t\tcolor: 'red',\n * \t\t\tbackgroundColor: bind.to( 'qux', () => { ... } )\n * \t\t}\n * \t}\n * } );\n * ```\n *\n * Text nodes can also have complex values:\n *\n * ```ts\n * const bind = Template.bind( observable, emitter );\n *\n * // Will render a \"foo\" text node.\n * new Template( {\n * \ttext: 'foo'\n * } );\n *\n * // Will render a \"static text: {observable.foo}\" text node.\n * // The text of the node will be updated as the \"foo\" attribute changes.\n * new Template( {\n * \ttext: [\n * \t\t'static text: ',\n * \t\tbind.to( 'foo', () => { ... } )\n * \t]\n * } );\n * ```\n */\nexport type TemplateValueSchema = ArrayOrItem<\n\tTemplateSimpleValueSchema |\n\tNamespacedValue<TemplateSimpleValueSchema>\n> |\n\tRecord<string, TemplateSimpleValueSchema>;\n\n/**\n * Describes an event listener attached to an HTML element. Such listener can propagate DOM events\n * through an {@link module:utils/observablemixin~Observable} instance, execute custom callbacks\n * or both, if necessary.\n *\n * Also see:\n * * {@link module:ui/template~TemplateDefinition} to learn more about template definitions,\n * * {@link module:ui/template~BindChain#to `to()`} method to learn more about bindings.\n *\n * Check out different ways of attaching event listeners below:\n *\n * ```ts\n * // Bind helper will propagate events through the observable.\n * const bind = Template.bind( observable, emitter );\n *\n * new Template( {\n * \ttag: 'p',\n * \ton: {\n * \t\t// An object schema. The observable will fire the \"clicked\" event upon DOM \"click\".\n * \t\tclick: bind.to( 'clicked' )\n *\n * \t\t// An object schema. It will work for \"click\" event on \"a.foo\" children only.\n * \t\t'click@a.foo': bind.to( 'clicked' )\n *\n * \t\t// An array schema, makes the observable propagate multiple events.\n * \t\tclick: [\n * \t\t\tbind.to( 'clicked' ),\n * \t\t\tbind.to( 'executed' )\n * \t\t],\n *\n * \t\t// An array schema with a custom callback.\n * \t\t'click@a.foo': {\n * \t\t\tbind.to( 'clicked' ),\n * \t\t\tbind.to( evt => {\n * \t\t\t\tconsole.log( `${ evt.target } has been clicked!` );\n * \t\t\t} }\n * \t\t}\n * \t}\n * } );\n * ```\n */\nexport type TemplateListenerSchema = ArrayOrItem<ListenerBinding>;\n\n// `Template.bind( observable, emitter ).to( name )` is used in two different contexts:\n// - in `attribute` (or `text`) section of the `TemplateDefinition` - `name` is an observed property of the provided `observable`, or\n// - in `on` section - `name` is the event to be fired.\n//\n// In both cases, the returned type is `TemplateToBinding` which can be misleading.\n// Moreover, some forms of `to()` can be used in specific context only:\n// - `to( name, callback )` can only be used in `attribute` or `text`\n// - `to( callback )` can only be used in `on`.\n//\n// But note, that the fact it's `Template{To,If}Binding` is not relevant to the user.\n// The instances are both produced and consumed by this file only.\n// So let's invent some opaque types for outside use and only keep using `TemplateBinding` internally. They can be different for\n// `attribute` and `on` contexts.\n// To make them opaque, a non-exported symbol is used, so it's not possible to accidentally create an instance outside.\ndeclare const AttributeBindingSymbol: unique symbol;\ndeclare const ListenerBindingSymbol: unique symbol;\n\nexport interface AttributeBinding { _opaqueAttributeBinding: typeof AttributeBindingSymbol }\nexport interface ListenerBinding { _opaqueListenerBinding: typeof ListenerBindingSymbol }\n\n/**\n * The return value of {@link ~Template.bind `Template.bind()`}. It provides `to()` and `if()`\n * methods to create the {@link module:utils/observablemixin~Observable observable} attribute and event bindings.\n */\nexport interface BindChain<TObservable> {\n\n\t/**\n\t * Binds an {@link module:utils/observablemixin~Observable observable} to either:\n\t *\n\t * * an HTML element attribute or a text node `textContent`, so it remains in sync with the observable\n\t * attribute as it changes,\n\t * * or an HTML element DOM event, so the DOM events are propagated through an observable.\n\t *\n\t * Some common use cases of `to()` bindings are presented below:\n\t *\n\t * ```ts\n\t * const bind = Template.bind( observable, emitter );\n\t *\n\t * new Template( {\n\t * \ttag: 'p',\n\t * \tattributes: {\n\t * \t\t// class=\"...\" attribute gets bound to `observable#a`\n\t * \t\tclass: bind.to( 'a' )\n\t * \t},\n\t * \tchildren: [\n\t * \t\t// <p>...</p> gets bound to observable#b; always `toUpperCase()`.\n\t * \t\t{\n\t * \t\t\ttext: bind.to( 'b', ( value, node ) => value.toUpperCase() )\n\t * \t\t}\n\t * \t],\n\t * \ton: {\n\t * \t\tclick: [\n\t * \t\t\t// An observable will fire \"clicked\" upon \"click\" in the DOM.\n\t * \t\t\tbind.to( 'clicked' ),\n\t *\n\t * \t\t\t// A custom callback will be executed upon \"click\" in the DOM.\n\t * \t\t\tbind.to( () => {\n\t * \t\t\t\t...\n\t * \t\t\t} )\n\t * \t\t]\n\t * \t}\n\t * } ).render();\n\t * ```\n\t *\n\t * Learn more about using `to()` in the {@link module:ui/template~TemplateValueSchema} and\n\t * {@link module:ui/template~TemplateListenerSchema}.\n\t *\n\t * @label ATTRIBUTE\n\t * @param attribute An attribute name of {@link module:utils/observablemixin~Observable}.\n\t */\n\tto<TAttribute extends keyof TObservable & string>(\n\t\tattribute: TAttribute\n\t): AttributeBinding & ListenerBinding;\n\n\t/**\n\t * Binds an {@link module:utils/observablemixin~Observable observable} to either:\n\t *\n\t * * an HTML element attribute or a text node `textContent`, so it remains in sync with the observable\n\t * attribute as it changes,\n\t * * or an HTML element DOM event, so the DOM events are propagated through an observable.\n\t *\n\t * Some common use cases of `to()` bindings are presented below:\n\t *\n\t * ```ts\n\t * const bind = Template.bind( observable, emitter );\n\t *\n\t * new Template( {\n\t * \ttag: 'p',\n\t * \tattributes: {\n\t * \t\t// class=\"...\" attribute gets bound to `observable#a`\n\t * \t\tclass: bind.to( 'a' )\n\t * \t},\n\t * \tchildren: [\n\t * \t\t// <p>...</p> gets bound to observable#b; always `toUpperCase()`.\n\t * \t\t{\n\t * \t\t\ttext: bind.to( 'b', ( value, node ) => value.toUpperCase() )\n\t * \t\t}\n\t * \t],\n\t * \ton: {\n\t * \t\tclick: [\n\t * \t\t\t// An observable will fire \"clicked\" upon \"click\" in the DOM.\n\t * \t\t\tbind.to( 'clicked' ),\n\t *\n\t * \t\t\t// A custom callback will be executed upon \"click\" in the DOM.\n\t * \t\t\tbind.to( () => {\n\t * \t\t\t\t...\n\t * \t\t\t} )\n\t * \t\t]\n\t * \t}\n\t * } ).render();\n\t * ```\n\t *\n\t * Learn more about using `to()` in the {@link module:ui/template~TemplateValueSchema} and\n\t * {@link module:ui/template~TemplateListenerSchema}.\n\t *\n\t * @label ATTRIBUTE_CALLBACK\n\t * @param attribute An attribute name of {@link module:utils/observablemixin~Observable}.\n\t * @param callback Allows for processing of the value. Accepts `Node` and `value` as arguments.\n\t */\n\tto<TAttribute extends keyof TObservable & string>(\n\t\tattribute: TAttribute,\n\t\tcallback: ( value: TObservable[ TAttribute ], node: Node ) => ( TemplateSimpleValue )\n\t): AttributeBinding;\n\n\t/**\n\t * Binds an {@link module:utils/observablemixin~Observable observable} to either:\n\t *\n\t * * an HTML element attribute or a text node `textContent`, so it remains in sync with the observable\n\t * attribute as it changes,\n\t * * or an HTML element DOM event, so the DOM events are propagated through an observable.\n\t *\n\t * Some common use cases of `to()` bindings are presented below:\n\t *\n\t * ```ts\n\t * const bind = Template.bind( observable, emitter );\n\t *\n\t * new Template( {\n\t * \ttag: 'p',\n\t * \tattributes: {\n\t * \t\t// class=\"...\" attribute gets bound to `observable#a`\n\t * \t\tclass: bind.to( 'a' )\n\t * \t},\n\t * \tchildren: [\n\t * \t\t// <p>...</p> gets bound to observable#b; always `toUpperCase()`.\n\t * \t\t{\n\t * \t\t\ttext: bind.to( 'b', ( value, node ) => value.toUpperCase() )\n\t * \t\t}\n\t * \t],\n\t * \ton: {\n\t * \t\tclick: [\n\t * \t\t\t// An observable will fire \"clicked\" upon \"click\" in the DOM.\n\t * \t\t\tbind.to( 'clicked' ),\n\t *\n\t * \t\t\t// A custom callback will be executed upon \"click\" in the DOM.\n\t * \t\t\tbind.to( () => {\n\t * \t\t\t\t...\n\t * \t\t\t} )\n\t * \t\t]\n\t * \t}\n\t * } ).render();\n\t * ```\n\t *\n\t * Learn more about using `to()` in the {@link module:ui/template~TemplateValueSchema} and\n\t * {@link module:ui/template~TemplateListenerSchema}.\n\t *\n\t * @label EVENT\n\t * @param eventNameOrCallback A DOM event name or an event callback.\n\t */\n\tto( eventNameOrCallback: string | ( ( domEvent: Event ) => void ) ): ListenerBinding;\n\n\t/**\n\t * Binds an {@link module:utils/observablemixin~Observable observable} to an HTML element attribute or a text\n\t * node `textContent` so it remains in sync with the observable attribute as it changes.\n\t *\n\t * Unlike {@link module:ui/template~BindChain#to}, it controls the presence of the attribute or `textContent`\n\t * depending on the \"falseness\" of an {@link module:utils/observablemixin~Observable} attribute.\n\t *\n\t * ```ts\n\t * const bind = Template.bind( observable, emitter );\n\t *\n\t * new Template( {\n\t * \ttag: 'input',\n\t * \tattributes: {\n\t * \t\t// <input checked> when `observable#a` is not undefined/null/false/''\n\t * \t\t// <input> when `observable#a` is undefined/null/false\n\t * \t\tchecked: bind.if( 'a' )\n\t * \t},\n\t * \tchildren: [\n\t * \t\t{\n\t * \t\t\t// <input>\"b-is-not-set\"</input> when `observable#b` is undefined/null/false/''\n\t * \t\t\t// <input></input> when `observable#b` is not \"falsy\"\n\t * \t\t\ttext: bind.if( 'b', 'b-is-not-set', ( value, node ) => !value )\n\t * \t\t}\n\t * \t]\n\t * } ).render();\n\t * ```\n\t *\n\t * Learn more about using `if()` in the {@link module:ui/template~TemplateValueSchema}.\n\t *\n\t * @param attribute An attribute name of {@link module:utils/observablemixin~Observable} used in the binding.\n\t * @param valueIfTrue Value set when the {@link module:utils/observablemixin~Observable} attribute is not\n\t * undefined/null/false/'' (empty string).\n\t * @param callback Allows for processing of the value. Accepts `Node` and `value` as arguments.\n\t */\n\tif<TAttribute extends keyof TObservable & string>(\n\t\tattribute: TAttribute,\n\t\tvalueIfTrue?: unknown,\n\t\tcallback?: ( value: TObservable[ TAttribute ], node: Node ) => ( boolean | FalsyValue )\n\t): AttributeBinding;\n}\n\n/**\n * The {@link module:ui/template~Template#_renderNode} configuration.\n */\nexport interface RenderData {\n\n\t/**\n\t * A node which is being rendered.\n\t */\n\tnode: HTMLElement | Text;\n\n\t/**\n\t * Tells {@link module:ui/template~Template#_renderNode} to render\n\t * children into `DocumentFragment` first and then append the fragment\n\t * to the parent element. It is a speed optimization.\n\t */\n\tintoFragment: boolean;\n\n\t/**\n\t * Indicates whether the {@link #node} has been provided by {@link module:ui/template~Template#apply}.\n\t */\n\tisApplying: boolean;\n\n\t/**\n\t * An object storing the data that helps {@link module:ui/template~Template#revert}\n\t * bringing back an element to its initial state, i.e. before\n\t * {@link module:ui/template~Template#apply} was called.\n\t */\n\trevertData?: RevertData;\n}\n\ninterface RevertData {\n\ttext?: string | null;\n\tchildren: Array<RevertData>;\n\tbindings: Array<Array<() => void>>;\n\tattributes: Record<string, string | null>;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/editorui/bodycollection\n */\n\n/* globals document */\n\nimport Template from '../template';\nimport ViewCollection from '../viewcollection';\nimport type View from '../view';\n\nimport { createElement, type Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * This is a special {@link module:ui/viewcollection~ViewCollection} dedicated to elements that are detached\n * from the DOM structure of the editor, like panels, icons, etc.\n *\n * The body collection is available in the {@link module:ui/editorui/editoruiview~EditorUIView#body `editor.ui.view.body`} property.\n * Any plugin can add a {@link module:ui/view~View view} to this collection.\n * These views will render in a container placed directly in the `<body>` element.\n * The editor will detach and destroy this collection when the editor will be {@link module:core/editor/editor~Editor#destroy destroyed}.\n *\n * If you need to control the life cycle of the body collection on your own, you can create your own instance of this class.\n *\n * A body collection will render itself automatically in the DOM body element as soon as you call {@link ~BodyCollection#attachToDom}.\n * If you create multiple body collections, this class will create a special wrapper element in the DOM to limit the number of\n * elements created directly in the body and remove it when the last body collection will be\n * {@link ~BodyCollection#detachFromDom detached}.\n */\nexport default class BodyCollection extends ViewCollection {\n\t/**\n\t * The {@link module:core/editor/editor~Editor#locale editor's locale} instance.\n\t * See the view {@link module:ui/view~View#locale locale} property.\n\t */\n\tpublic readonly locale: Locale;\n\n\t/**\n\t * The element holding elements of the body region.\n\t */\n\tprivate _bodyCollectionContainer?: HTMLElement;\n\n\t/**\n\t * Creates a new instance of the {@link module:ui/editorui/bodycollection~BodyCollection}.\n\t *\n\t * @param locale The {@link module:core/editor/editor~Editor editor's locale} instance.\n\t * @param initialItems The initial items of the collection.\n\t */\n\tconstructor( locale: Locale, initialItems: Iterable<View> = [] ) {\n\t\tsuper( initialItems );\n\n\t\tthis.locale = locale;\n\t}\n\n\t/**\n\t * The element holding elements of the body region.\n\t */\n\tpublic get bodyCollectionContainer(): HTMLElement | undefined {\n\t\treturn this._bodyCollectionContainer;\n\t}\n\n\t/**\n\t * Attaches the body collection to the DOM body element. You need to execute this method to render the content of\n\t * the body collection.\n\t */\n\tpublic attachToDom(): void {\n\t\tthis._bodyCollectionContainer = new Template( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-reset_all',\n\t\t\t\t\t'ck-body',\n\t\t\t\t\t'ck-rounded-corners'\n\t\t\t\t],\n\t\t\t\tdir: this.locale.uiLanguageDirection\n\t\t\t},\n\t\t\tchildren: this\n\t\t} ).render() as HTMLElement;\n\n\t\tlet wrapper = document.querySelector( '.ck-body-wrapper' );\n\n\t\tif ( !wrapper ) {\n\t\t\twrapper = createElement( document, 'div', { class: 'ck-body-wrapper' } );\n\t\t\tdocument.body.appendChild( wrapper );\n\t\t}\n\n\t\twrapper.appendChild( this._bodyCollectionContainer );\n\t}\n\n\t/**\n\t * Detaches the collection from the DOM structure. Use this method when you do not need to use the body collection\n\t * anymore to clean-up the DOM structure.\n\t */\n\tpublic detachFromDom(): void {\n\t\tsuper.destroy();\n\n\t\tif ( this._bodyCollectionContainer ) {\n\t\t\tthis._bodyCollectionContainer.remove();\n\t\t}\n\n\t\tconst wrapper = document.querySelector( '.ck-body-wrapper' );\n\n\t\tif ( wrapper && wrapper.childElementCount == 0 ) {\n\t\t\twrapper.remove();\n\t\t}\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./icon.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* global DOMParser */\n\n/**\n * @module ui/icon/iconview\n */\n\nimport View from '../view';\n\nimport type { ObservableChangeEvent } from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/icon/icon.css';\n\n/**\n * The icon view class.\n */\nexport default class IconView extends View {\n\t/**\n\t * The SVG source of the icon.\n\t *\n\t * @observable\n\t */\n\tdeclare public content: string | undefined;\n\n\t/**\n\t * This attribute specifies the boundaries to which the\n\t * icon content should stretch.\n\t *\n\t * @observable\n\t * @default '0 0 20 20'\n\t */\n\tdeclare public viewBox: string;\n\n\t/**\n\t * The fill color of the child `path.ck-icon__fill`.\n\t *\n\t * @observable\n\t * @default ''\n\t */\n\tdeclare public fillColor: string;\n\n\t/**\n\t * When set true (default), all parts of the icon inherit the fill color from the CSS `color` property of the\n\t * icon's DOM parent.\n\t *\n\t * This effectively makes the icon monochromatic and allows it to change its fill color dynamically, for instance,\n\t * when a {@link module:ui/button/buttonview~ButtonView} displays an icon and it switches between different states\n\t * (pushed, hovered, etc.) the icon will follow along.\n\t *\n\t * **Note**: For the monochromatic icon to render properly, it must be made up of shapes that can be filled\n\t * with color instead of, for instance, paths with strokes. Be sure to use the *outline stroke* tool\n\t * (the name could be different in your vector graphics editor) before exporting your icon. Also, remove any\n\t * excess `fill=\"...\"` attributes that could break the color inheritance.\n\t *\n\t * **Note**: If you want to preserve the original look of your icon and disable dynamic color inheritance,\n\t * set this flag to `false`.\n\t *\n\t * @observable\n\t * @default true\n\t */\n\tdeclare public isColorInherited: boolean;\n\n\t/**\n\t * A list of presentational attributes that can be set on the `<svg>` element and should be preserved\n\t * when the icon {@link module:ui/icon/iconview~IconView#content content} is loaded.\n\t *\n\t * See the [specification](https://www.w3.org/TR/SVG/styling.html#TermPresentationAttribute) to learn more.\n\t */\n\tprivate static presentationalAttributeNames: Array<string> = [\n\t\t'alignment-baseline', 'baseline-shift', 'clip-path', 'clip-rule', 'color', 'color-interpolation',\n\t\t'color-interpolation-filters', 'color-rendering', 'cursor', 'direction', 'display', 'dominant-baseline', 'fill', 'fill-opacity',\n\t\t'fill-rule', 'filter', 'flood-color', 'flood-opacity', 'font-family', 'font-size', 'font-size-adjust', 'font-stretch', 'font-style',\n\t\t'font-variant', 'font-weight', 'image-rendering', 'letter-spacing', 'lighting-color', 'marker-end', 'marker-mid', 'marker-start',\n\t\t'mask', 'opacity', 'overflow', 'paint-order', 'pointer-events', 'shape-rendering', 'stop-color', 'stop-opacity', 'stroke',\n\t\t'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke-width',\n\t\t'text-anchor', 'text-decoration', 'text-overflow', 'text-rendering', 'transform', 'unicode-bidi', 'vector-effect',\n\t\t'visibility', 'white-space', 'word-spacing', 'writing-mode'\n\t];\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor() {\n\t\tsuper();\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.set( 'content', '' );\n\t\tthis.set( 'viewBox', '0 0 20 20' );\n\t\tthis.set( 'fillColor', '' );\n\t\tthis.set( 'isColorInherited', true );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'svg',\n\t\t\tns: 'http://www.w3.org/2000/svg',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-icon',\n\n\t\t\t\t\t// Exclude icon internals from the CSS reset to allow rich (non-monochromatic) icons\n\t\t\t\t\t// (https://github.com/ckeditor/ckeditor5/issues/12599).\n\t\t\t\t\t'ck-reset_all-excluded',\n\n\t\t\t\t\t// The class to remove the dynamic color inheritance is toggleable\n\t\t\t\t\t// (https://github.com/ckeditor/ckeditor5/issues/12599).\n\t\t\t\t\tbind.if( 'isColorInherited', 'ck-icon_inherit-color' )\n\t\t\t\t],\n\t\t\t\tviewBox: bind.to( 'viewBox' )\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis._updateXMLContent();\n\t\tthis._colorFillPaths();\n\n\t\t// This is a hack for lack of innerHTML binding.\n\t\t// See: https://github.com/ckeditor/ckeditor5-ui/issues/99.\n\t\tthis.on<ObservableChangeEvent>( 'change:content', () => {\n\t\t\tthis._updateXMLContent();\n\t\t\tthis._colorFillPaths();\n\t\t} );\n\n\t\tthis.on<ObservableChangeEvent>( 'change:fillColor', () => {\n\t\t\tthis._colorFillPaths();\n\t\t} );\n\t}\n\n\t/**\n\t * Updates the {@link #element} with the value of {@link #content}.\n\t */\n\tprivate _updateXMLContent() {\n\t\tif ( this.content ) {\n\t\t\tconst parsed = new DOMParser().parseFromString( this.content.trim(), 'image/svg+xml' );\n\t\t\tconst svg = parsed.querySelector( 'svg' )!;\n\t\t\tconst viewBox = svg.getAttribute( 'viewBox' );\n\n\t\t\tif ( viewBox ) {\n\t\t\t\tthis.viewBox = viewBox;\n\t\t\t}\n\n\t\t\t// Preserve presentational attributes of the <svg> element from the source.\n\t\t\t// They can affect rendering of the entire icon (https://github.com/ckeditor/ckeditor5/issues/12597).\n\t\t\tfor ( const { name, value } of Array.from( svg.attributes ) ) {\n\t\t\t\tif ( IconView.presentationalAttributeNames.includes( name ) ) {\n\t\t\t\t\tthis.element!.setAttribute( name, value );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\twhile ( this.element!.firstChild ) {\n\t\t\t\tthis.element!.removeChild( this.element!.firstChild );\n\t\t\t}\n\n\t\t\twhile ( svg.childNodes.length > 0 ) {\n\t\t\t\tthis.element!.appendChild( svg.childNodes[ 0 ] );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Fills all child `path.ck-icon__fill` with the `#fillColor`.\n\t */\n\tprivate _colorFillPaths() {\n\t\tif ( this.fillColor ) {\n\t\t\tthis.element!.querySelectorAll( '.ck-icon__fill' ).forEach( path => {\n\t\t\t\t( path as HTMLElement ).style.fill = this.fillColor;\n\t\t\t} );\n\t\t}\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./button.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/button/buttonview\n */\n\nimport View from '../view';\nimport IconView from '../icon/iconview';\n\nimport type { TemplateDefinition } from '../template';\nimport type ViewCollection from '../viewcollection';\nimport type { default as Button, ButtonExecuteEvent } from './button';\n\nimport {\n\tenv,\n\tgetEnvKeystrokeText,\n\tuid,\n\tdelay,\n\ttype Locale,\n\ttype DelayedFunc\n} from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/button/button.css';\n\n/**\n * The button view class.\n *\n * ```ts\n * const view = new ButtonView();\n *\n * view.set( {\n * \tlabel: 'A button',\n * \tkeystroke: 'Ctrl+B',\n * \ttooltip: true,\n * \twithText: true\n * } );\n *\n * view.render();\n *\n * document.body.append( view.element );\n * ```\n */\nexport default class ButtonView extends View<HTMLButtonElement> implements Button {\n\t/**\n\t * Collection of the child views inside of the button {@link #element}.\n\t */\n\tpublic readonly children: ViewCollection;\n\n\t/**\n\t * Label of the button view. It is configurable using the {@link #label label attribute}.\n\t */\n\tpublic readonly labelView: View;\n\n\t/**\n\t * The icon view of the button. Will be added to {@link #children} when the\n\t * {@link #icon icon attribute} is defined.\n\t */\n\tpublic readonly iconView: IconView;\n\n\t/**\n\t * A view displaying the keystroke of the button next to the {@link #labelView label}.\n\t * Added to {@link #children} when the {@link #withKeystroke `withKeystroke` attribute}\n\t * is defined.\n\t */\n\tpublic readonly keystrokeView: View;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public class: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public labelStyle: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public icon: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public isEnabled: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public isOn: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public isVisible: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public isToggleable: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public keystroke: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public label: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public tabindex: number;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public tooltip: Button[ 'tooltip' ];\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public tooltipPosition: Button[ 'tooltipPosition' ];\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public type: Button[ 'type' ];\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public withText: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public withKeystroke: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public role: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public ariaChecked: boolean | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public ariaLabel?: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public ariaLabelledBy: string | undefined;\n\n\t/**\n\t * Tooltip of the button bound to the template.\n\t *\n\t * @see #tooltip\n\t * @see module:ui/button/buttonview~ButtonView#_getTooltipString\n\t * @internal\n\t * @observable\n\t */\n\tdeclare public _tooltipString: string;\n\n\t/**\n\t * Delayed focus function for focus handling in Safari.\n\t */\n\tprivate _focusDelayed: DelayedFunc<() => void> | null = null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\t\tconst ariaLabelUid = uid();\n\n\t\t// Implement the Button interface.\n\t\tthis.set( 'ariaChecked', undefined );\n\t\tthis.set( 'ariaLabel', undefined );\n\t\tthis.set( 'ariaLabelledBy', `ck-editor__aria-label_${ ariaLabelUid }` );\n\t\tthis.set( 'class', undefined );\n\t\tthis.set( 'labelStyle', undefined );\n\t\tthis.set( 'icon', undefined );\n\t\tthis.set( 'isEnabled', true );\n\t\tthis.set( 'isOn', false );\n\t\tthis.set( 'isVisible', true );\n\t\tthis.set( 'isToggleable', false );\n\t\tthis.set( 'keystroke', undefined );\n\t\tthis.set( 'label', undefined );\n\t\tthis.set( 'role', undefined );\n\t\tthis.set( 'tabindex', -1 );\n\t\tthis.set( 'tooltip', false );\n\t\tthis.set( 'tooltipPosition', 's' );\n\t\tthis.set( 'type', 'button' );\n\t\tthis.set( 'withText', false );\n\t\tthis.set( 'withKeystroke', false );\n\n\t\tthis.children = this.createCollection();\n\t\tthis.labelView = this._createLabelView();\n\n\t\tthis.iconView = new IconView();\n\t\tthis.iconView.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: 'ck-button__icon'\n\t\t\t}\n\t\t} );\n\n\t\tthis.keystrokeView = this._createKeystrokeView();\n\n\t\tthis.bind( '_tooltipString' ).to(\n\t\t\tthis, 'tooltip',\n\t\t\tthis, 'label',\n\t\t\tthis, 'keystroke',\n\t\t\tthis._getTooltipString.bind( this )\n\t\t);\n\n\t\tconst template: TemplateDefinition & { on: object } = {\n\t\t\ttag: 'button',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-button',\n\t\t\t\t\tbind.to( 'class' ),\n\t\t\t\t\tbind.if( 'isEnabled', 'ck-disabled', value => !value ),\n\t\t\t\t\tbind.if( 'isVisible', 'ck-hidden', value => !value ),\n\t\t\t\t\tbind.to( 'isOn', value => value ? 'ck-on' : 'ck-off' ),\n\t\t\t\t\tbind.if( 'withText', 'ck-button_with-text' ),\n\t\t\t\t\tbind.if( 'withKeystroke', 'ck-button_with-keystroke' )\n\t\t\t\t],\n\t\t\t\trole: bind.to( 'role' ),\n\t\t\t\ttype: bind.to( 'type', value => value ? value : 'button' ),\n\t\t\t\ttabindex: bind.to( 'tabindex' ),\n\t\t\t\t'aria-label': bind.to( 'ariaLabel' ),\n\t\t\t\t'aria-labelledby': bind.to( 'ariaLabelledBy' ),\n\t\t\t\t'aria-disabled': bind.if( 'isEnabled', true, value => !value ),\n\t\t\t\t'aria-checked': bind.to( 'isOn' ),\n\t\t\t\t'aria-pressed': bind.to( 'isOn', value => this.isToggleable ? String( !!value ) : false ),\n\t\t\t\t'data-cke-tooltip-text': bind.to( '_tooltipString' ),\n\t\t\t\t'data-cke-tooltip-position': bind.to( 'tooltipPosition' )\n\t\t\t},\n\n\t\t\tchildren: this.children,\n\n\t\t\ton: {\n\t\t\t\tclick: bind.to( evt => {\n\t\t\t\t\t// We can't make the button disabled using the disabled attribute, because it won't be focusable.\n\t\t\t\t\t// Though, shouldn't this condition be moved to the button controller?\n\t\t\t\t\tif ( this.isEnabled ) {\n\t\t\t\t\t\tthis.fire<ButtonExecuteEvent>( 'execute' );\n\t\t\t\t\t} else {\n\t\t\t\t\t\t// Prevent the default when button is disabled, to block e.g.\n\t\t\t\t\t\t// automatic form submitting. See ckeditor/ckeditor5-link#74.\n\t\t\t\t\t\tevt.preventDefault();\n\t\t\t\t\t}\n\t\t\t\t} )\n\t\t\t}\n\t\t};\n\n\t\t// On Safari we have to force the focus on a button on click as it's the only browser\n\t\t// that doesn't do that automatically. See #12115.\n\t\tif ( env.isSafari ) {\n\t\t\tif ( !this._focusDelayed ) {\n\t\t\t\tthis._focusDelayed = delay( () => this.focus(), 0 );\n\t\t\t}\n\n\t\t\ttemplate.on.mousedown = bind.to( () => {\n\t\t\t\tthis._focusDelayed!();\n\t\t\t} );\n\n\t\t\ttemplate.on.mouseup = bind.to( () => {\n\t\t\t\tthis._focusDelayed!.cancel();\n\t\t\t} );\n\t\t}\n\n\t\tthis.setTemplate( template );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tif ( this.icon ) {\n\t\t\tthis.iconView.bind( 'content' ).to( this, 'icon' );\n\t\t\tthis.children.add( this.iconView );\n\t\t}\n\n\t\tthis.children.add( this.labelView );\n\n\t\tif ( this.withKeystroke && this.keystroke ) {\n\t\t\tthis.children.add( this.keystrokeView );\n\t\t}\n\t}\n\n\t/**\n\t * Focuses the {@link #element} of the button.\n\t */\n\tpublic focus(): void {\n\t\tthis.element!.focus();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tif ( this._focusDelayed ) {\n\t\t\tthis._focusDelayed.cancel();\n\t\t}\n\n\t\tsuper.destroy();\n\t}\n\n\t/**\n\t * Creates a label view instance and binds it with button attributes.\n\t */\n\tprivate _createLabelView() {\n\t\tconst labelView = new View();\n\t\tconst bind = this.bindTemplate;\n\n\t\tlabelView.setTemplate( {\n\t\t\ttag: 'span',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-button__label'\n\t\t\t\t],\n\t\t\t\tstyle: bind.to( 'labelStyle' ),\n\t\t\t\tid: this.ariaLabelledBy\n\t\t\t},\n\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttext: bind.to( 'label' )\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\n\t\treturn labelView;\n\t}\n\n\t/**\n\t * Creates a view that displays a keystroke next to a {@link #labelView label }\n\t * and binds it with button attributes.\n\t */\n\tprivate _createKeystrokeView() {\n\t\tconst keystrokeView = new View();\n\n\t\tkeystrokeView.setTemplate( {\n\t\t\ttag: 'span',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-button__keystroke'\n\t\t\t\t]\n\t\t\t},\n\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttext: this.bindTemplate.to( 'keystroke', text => getEnvKeystrokeText( text! ) )\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\n\t\treturn keystrokeView;\n\t}\n\n\t/**\n\t * Gets the text for the tooltip from the combination of\n\t * {@link #tooltip}, {@link #label} and {@link #keystroke} attributes.\n\t *\n\t * @see #tooltip\n\t * @see #_tooltipString\n\t * @param tooltip Button tooltip.\n\t * @param label Button label.\n\t * @param keystroke Button keystroke.\n\t */\n\tprivate _getTooltipString(\n\t\ttooltip: Button[ 'tooltip' ],\n\t\tlabel: string | undefined,\n\t\tkeystroke: string | undefined\n\t): string {\n\t\tif ( tooltip ) {\n\t\t\tif ( typeof tooltip == 'string' ) {\n\t\t\t\treturn tooltip;\n\t\t\t} else {\n\t\t\t\tif ( keystroke ) {\n\t\t\t\t\tkeystroke = getEnvKeystrokeText( keystroke );\n\t\t\t\t}\n\n\t\t\t\tif ( tooltip instanceof Function ) {\n\t\t\t\t\treturn tooltip( label!, keystroke );\n\t\t\t\t} else {\n\t\t\t\t\treturn `${ label }${ keystroke ? ` (${ keystroke })` : '' }`;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn '';\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./switchbutton.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/button/switchbuttonview\n */\n\nimport View from '../view';\nimport ButtonView from './buttonview';\n\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/button/switchbutton.css';\n\n/**\n * The switch button view class.\n *\n * ```ts\n * const view = new SwitchButtonView();\n *\n * view.set( {\n * \twithText: true,\n * \tlabel: 'Switch me!'\n * } );\n *\n * view.render();\n *\n * document.body.append( view.element );\n * ```\n */\nexport default class SwitchButtonView extends ButtonView {\n\t/**\n\t * The toggle switch of the button.\n\t */\n\tpublic readonly toggleSwitchView: View;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.isToggleable = true;\n\t\tthis.toggleSwitchView = this._createToggleView();\n\n\t\tthis.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: 'ck-switchbutton'\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.children.add( this.toggleSwitchView );\n\t}\n\n\t/**\n\t * Creates a toggle child view.\n\t */\n\tprivate _createToggleView() {\n\t\tconst toggleSwitchView = new View();\n\n\t\ttoggleSwitchView.setTemplate( {\n\t\t\ttag: 'span',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-button__toggle'\n\t\t\t\t]\n\t\t\t},\n\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttag: 'span',\n\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: [\n\t\t\t\t\t\t\t'ck',\n\t\t\t\t\t\t\t'ck-button__toggle__inner'\n\t\t\t\t\t\t]\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\n\t\treturn toggleSwitchView;\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./colorgrid.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./label.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/label/labelview\n */\n\nimport View from '../view';\n\nimport { uid, type Locale } from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/label/label.css';\n\n/**\n * The label view class.\n */\nexport default class LabelView extends View {\n\t/**\n\t * An unique id of the label. It can be used by other UI components to reference\n\t * the label, for instance, using the `aria-describedby` DOM attribute.\n\t */\n\tpublic readonly id: string;\n\n\t/**\n\t * The text of the label.\n\t *\n\t * @observable\n\t */\n\tdeclare public text: string | undefined;\n\n\t/**\n\t * The `for` attribute of the label (i.e. to pair with an `<input>` element).\n\t *\n\t * @observable\n\t */\n\tdeclare public for: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.set( 'text', undefined );\n\t\tthis.set( 'for', undefined );\n\n\t\tthis.id = `ck-editor__label_${ uid() }`;\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'label',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-label'\n\t\t\t\t],\n\t\t\t\tid: this.id,\n\t\t\t\tfor: bind.to( 'for' )\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttext: bind.to( 'text' )\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./labeledfieldview.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/labeledfield/labeledfieldview\n */\n\nimport View from '../view';\nimport LabelView from '../label/labelview';\n\nimport type { FocusableView } from '../focuscycler';\nimport type ViewCollection from '../viewcollection';\n\nimport { uid, type Locale } from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/labeledfield/labeledfieldview.css';\n\n/**\n * The labeled field view class. It can be used to enhance any view with the following features:\n *\n * * a label,\n * * (optional) an error message,\n * * (optional) an info (status) text,\n *\n * all bound logically by proper DOM attributes for UX and accessibility. It also provides an interface\n * (e.g. observable properties) that allows controlling those additional features.\n *\n * The constructor of this class requires a callback that returns a view to be labeled. The callback\n * is called with unique ids that allow binding of DOM properties:\n *\n * ```ts\n * const labeledInputView = new LabeledFieldView( locale, ( labeledFieldView, viewUid, statusUid ) => {\n * \tconst inputView = new InputTextView( labeledFieldView.locale );\n *\n * \tinputView.set( {\n * \t\tid: viewUid,\n * \t\tariaDescribedById: statusUid\n * \t} );\n *\n * \tinputView.bind( 'isReadOnly' ).to( labeledFieldView, 'isEnabled', value => !value );\n * \tinputView.bind( 'hasError' ).to( labeledFieldView, 'errorText', value => !!value );\n *\n * \treturn inputView;\n * } );\n *\n * labeledInputView.label = 'User name';\n * labeledInputView.infoText = 'Full name like for instance, John Doe.';\n * labeledInputView.render();\n *\n * document.body.append( labeledInputView.element );\n * ```\n *\n * See {@link module:ui/labeledfield/utils} to discover readytouse labeled input helpers for common\n * UI components.\n */\nexport default class LabeledFieldView<TFieldView extends FocusableView = FocusableView> extends View {\n\t/**\n\t * The field view that gets labeled.\n\t */\n\tpublic readonly fieldView: TFieldView;\n\n\t/**\n\t * The label view instance that describes the entire view.\n\t */\n\tpublic readonly labelView: LabelView;\n\n\t/**\n\t * The status view for the {@link #fieldView}. It displays {@link #errorText} and\n\t * {@link #infoText}.\n\t */\n\tpublic readonly statusView: View;\n\n\t/**\n\t * A collection of children of the internal wrapper element. Allows inserting additional DOM elements (views) next to\n\t * the {@link #fieldView} for easy styling (e.g. positioning).\n\t *\n\t * By default, the collection contains {@link #fieldView} and {@link #labelView}.\n\t */\n\tpublic readonly fieldWrapperChildren: ViewCollection;\n\n\t/**\n\t * The text of the label.\n\t *\n\t * @observable\n\t */\n\tdeclare public label: string | undefined;\n\n\t/**\n\t * Controls whether the component is in read-only mode.\n\t *\n\t * @observable\n\t */\n\tdeclare public isEnabled: boolean;\n\n\t/**\n\t * An observable flag set to `true` when {@link #fieldView} is empty (`false` otherwise).\n\t *\n\t * @readonly\n\t * @observable\n\t * @default true\n\t */\n\tdeclare public isEmpty: boolean;\n\n\t/**\n\t * An observable flag set to `true` when {@link #fieldView} is currently focused by\n\t * the user (`false` otherwise).\n\t *\n\t * @readonly\n\t * @observable\n\t * @default false\n\t */\n\tdeclare public isFocused: boolean;\n\n\t/**\n\t * The validation error text. When set, it will be displayed\n\t * next to the {@link #fieldView} as a typical validation error message.\n\t * Set it to `null` to hide the message.\n\t *\n\t * **Note:** Setting this property to anything but `null` will automatically\n\t * make the `hasError` of the {@link #fieldView} `true`.\n\t *\n\t * @observable\n\t */\n\tdeclare public errorText: string | null;\n\n\t/**\n\t * The additional information text displayed next to the {@link #fieldView} which can\n\t * be used to inform the user about its purpose, provide help or hints.\n\t *\n\t * Set it to `null` to hide the message.\n\t *\n\t * **Note:** This text will be displayed in the same place as {@link #errorText} but the\n\t * latter always takes precedence: if the {@link #errorText} is set, it replaces\n\t * {@link #infoText}.\n\t *\n\t * @observable\n\t * @default null\n\t */\n\tdeclare public infoText: string | null;\n\n\t/**\n\t * (Optional) The additional CSS class set on the dropdown {@link #element}.\n\t *\n\t * @observable\n\t */\n\tdeclare public class: string | undefined;\n\n\t/**\n\t * The content of the `placeholder` attribute of the {@link #fieldView}.\n\t *\n\t * @observable\n\t */\n\tdeclare public placeholder: string | undefined;\n\n\t/**\n\t * The combined status text made of {@link #errorText} and {@link #infoText}.\n\t * Note that when present, {@link #errorText} always takes precedence in the\n\t * status.\n\t *\n\t * @see #errorText\n\t * @see #infoText\n\t * @see #statusView\n\t * @private\n\t * @observable\n\t */\n\tdeclare public _statusText: string | null;\n\n\t/**\n\t * Creates an instance of the labeled field view class using a provided creator function\n\t * that provides the view to be labeled.\n\t *\n\t * @param locale The locale instance.\n\t * @param viewCreator A function that returns a {@link module:ui/view~View}\n\t * that will be labeled. The following arguments are passed to the creator function:\n\t *\n\t * * an instance of the `LabeledFieldView` to allow binding observable properties,\n\t * * an UID string that connects the {@link #labelView label} and the labeled field view in DOM,\n\t * * an UID string that connects the {@link #statusView status} and the labeled field view in DOM.\n\t */\n\tconstructor(\n\t\tlocale: Locale | undefined,\n\t\tviewCreator: ( labeledFieldView: LabeledFieldView, viewUid: string, statusUid: string ) => TFieldView\n\t) {\n\t\tsuper( locale );\n\n\t\tconst viewUid = `ck-labeled-field-view-${ uid() }`;\n\t\tconst statusUid = `ck-labeled-field-view-status-${ uid() }`;\n\n\t\tthis.fieldView = viewCreator( this, viewUid, statusUid );\n\n\t\tthis.set( 'label', undefined );\n\t\tthis.set( 'isEnabled', true );\n\t\tthis.set( 'isEmpty', true );\n\t\tthis.set( 'isFocused', false );\n\t\tthis.set( 'errorText', null );\n\t\tthis.set( 'infoText', null );\n\t\tthis.set( 'class', undefined );\n\t\tthis.set( 'placeholder', undefined );\n\n\t\tthis.labelView = this._createLabelView( viewUid );\n\t\tthis.statusView = this._createStatusView( statusUid );\n\t\tthis.fieldWrapperChildren = this.createCollection( [ this.fieldView, this.labelView ] );\n\n\t\tthis.bind( '_statusText' ).to(\n\t\t\tthis, 'errorText',\n\t\t\tthis, 'infoText',\n\t\t\t( errorText, infoText ) => errorText || infoText\n\t\t);\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-labeled-field-view',\n\t\t\t\t\tbind.to( 'class' ),\n\t\t\t\t\tbind.if( 'isEnabled', 'ck-disabled', value => !value ),\n\t\t\t\t\tbind.if( 'isEmpty', 'ck-labeled-field-view_empty' ),\n\t\t\t\t\tbind.if( 'isFocused', 'ck-labeled-field-view_focused' ),\n\t\t\t\t\tbind.if( 'placeholder', 'ck-labeled-field-view_placeholder' ),\n\t\t\t\t\tbind.if( 'errorText', 'ck-error' )\n\t\t\t\t]\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttag: 'div',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: [\n\t\t\t\t\t\t\t'ck',\n\t\t\t\t\t\t\t'ck-labeled-field-view__input-wrapper'\n\t\t\t\t\t\t]\n\t\t\t\t\t},\n\t\t\t\t\tchildren: this.fieldWrapperChildren\n\t\t\t\t},\n\t\t\t\tthis.statusView\n\t\t\t]\n\t\t} );\n\t}\n\n\t/**\n\t * Creates label view class instance and bind with view.\n\t *\n\t * @param id Unique id to set as labelView#for attribute.\n\t */\n\tprivate _createLabelView( id: string ): LabelView {\n\t\tconst labelView = new LabelView( this.locale );\n\n\t\tlabelView.for = id;\n\t\tlabelView.bind( 'text' ).to( this, 'label' );\n\n\t\treturn labelView;\n\t}\n\n\t/**\n\t * Creates the status view instance. It displays {@link #errorText} and {@link #infoText}\n\t * next to the {@link #fieldView}. See {@link #_statusText}.\n\t *\n\t * @param statusUid Unique id of the status, shared with the {@link #fieldView view's}\n\t * `aria-describedby` attribute.\n\t */\n\tprivate _createStatusView( statusUid: string ): View {\n\t\tconst statusView = new View( this.locale );\n\t\tconst bind = this.bindTemplate;\n\n\t\tstatusView.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-labeled-field-view__status',\n\t\t\t\t\tbind.if( 'errorText', 'ck-labeled-field-view__status_error' ),\n\t\t\t\t\tbind.if( '_statusText', 'ck-hidden', value => !value )\n\t\t\t\t],\n\t\t\t\tid: statusUid,\n\t\t\t\trole: bind.if( 'errorText', 'alert' )\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttext: bind.to( '_statusText' )\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\n\t\treturn statusView;\n\t}\n\n\t/**\n\t * Focuses the {@link #fieldView}.\n\t */\n\tpublic focus(): void {\n\t\tthis.fieldView.focus();\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./input.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/input/inputview\n */\n\nimport View from '../view';\n\nimport {\n\tFocusTracker,\n\ttype Locale,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/input/input.css';\n\n/**\n * The base input view class.\n */\nexport default class InputView extends View<HTMLInputElement> {\n\t/**\n\t * Stores information about the editor UI focus and propagates it so various plugins and components\n\t * are unified as a focus group.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * The value of the input.\n\t *\n\t * @observable\n\t */\n\tdeclare public value: string | undefined;\n\n\t/**\n\t * The `id` attribute of the input (i.e. to pair with a `<label>` element).\n\t *\n\t * @observable\n\t */\n\tdeclare public id: string | undefined;\n\n\t/**\n\t * The `placeholder` attribute of the input.\n\t *\n\t * @observable\n\t */\n\tdeclare public placeholder: string | undefined;\n\n\t/**\n\t * Controls whether the input view is in read-only mode.\n\t *\n\t * @observable\n\t */\n\tdeclare public isReadOnly: boolean;\n\n\t/**\n\t * Set to `true` when the field has some error. Usually controlled via\n\t * {@link module:ui/labeledinput/labeledinputview~LabeledInputView#errorText}.\n\t *\n\t * @observable\n\t */\n\tdeclare public hasError: boolean;\n\n\t/**\n\t * The `id` of the element describing this field, e.g. when it has\n\t * some error; it helps screen readers read the error text.\n\t *\n\t * @observable\n\t */\n\tdeclare public ariaDescribedById: string | undefined;\n\n\t/**\n\t * An observable flag set to `true` when the input is currently focused by the user.\n\t * Set to `false` otherwise.\n\t *\n\t * @readonly\n\t * @observable\n\t * @default false\n\t */\n\tdeclare public isFocused: boolean;\n\n\t/**\n\t * An observable flag set to `true` when the input contains no text, i.e.\n\t * when {@link #value} is `''`, `null`, or `false`.\n\t *\n\t * @readonly\n\t * @observable\n\t * @default true\n\t */\n\tdeclare public isEmpty: boolean;\n\n\t/**\n\t * Corresponds to the `inputmode` DOM attribute. Can be `text`, `numeric`, `decimal`, etc.\n\t *\n\t * @observable\n\t * @default 'text'\n\t */\n\tdeclare public inputMode: string;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.set( 'value', undefined );\n\t\tthis.set( 'id', undefined );\n\t\tthis.set( 'placeholder', undefined );\n\t\tthis.set( 'isReadOnly', false );\n\t\tthis.set( 'hasError', false );\n\t\tthis.set( 'ariaDescribedById', undefined );\n\n\t\tthis.focusTracker = new FocusTracker();\n\n\t\tthis.bind( 'isFocused' ).to( this.focusTracker );\n\t\tthis.set( 'isEmpty', true );\n\t\tthis.set( 'inputMode', 'text' );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'input',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-input',\n\t\t\t\t\tbind.if( 'isFocused', 'ck-input_focused' ),\n\t\t\t\t\tbind.if( 'isEmpty', 'ck-input-text_empty' ),\n\t\t\t\t\tbind.if( 'hasError', 'ck-error' )\n\t\t\t\t],\n\t\t\t\tid: bind.to( 'id' ),\n\t\t\t\tplaceholder: bind.to( 'placeholder' ),\n\t\t\t\treadonly: bind.to( 'isReadOnly' ),\n\t\t\t\tinputmode: bind.to( 'inputMode' ),\n\t\t\t\t'aria-invalid': bind.if( 'hasError', true ),\n\t\t\t\t'aria-describedby': bind.to( 'ariaDescribedById' )\n\t\t\t},\n\t\t\ton: {\n\t\t\t\tinput: bind.to( ( ...args ) => {\n\t\t\t\t\tthis.fire( 'input', ...args );\n\t\t\t\t\tthis._updateIsEmpty();\n\t\t\t\t} ),\n\t\t\t\tchange: bind.to( this._updateIsEmpty.bind( this ) )\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.focusTracker.add( this.element! );\n\n\t\tthis._setDomElementValue( this.value );\n\t\tthis._updateIsEmpty();\n\n\t\t// Bind `this.value` to the DOM element's value.\n\t\t// We cannot use `value` DOM attribute because removing it on Edge does not clear the DOM element's value property.\n\t\tthis.on<ObservableChangeEvent>( 'change:value', ( evt, name, value ) => {\n\t\t\tthis._setDomElementValue( value );\n\t\t\tthis._updateIsEmpty();\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.focusTracker.destroy();\n\t}\n\n\t/**\n\t * Moves the focus to the input and selects the value.\n\t */\n\tpublic select(): void {\n\t\tthis.element!.select();\n\t}\n\n\t/**\n\t * Focuses the input.\n\t */\n\tpublic focus(): void {\n\t\tthis.element!.focus();\n\t}\n\n\t/**\n\t * Updates the {@link #isEmpty} property value on demand.\n\t */\n\tprivate _updateIsEmpty() {\n\t\tthis.isEmpty = isInputElementEmpty( this.element! );\n\t}\n\n\t/**\n\t * Sets the `value` property of the {@link #element DOM element} on demand.\n\t */\n\tprivate _setDomElementValue( value: any ) {\n\t\tthis.element!.value = ( !value && value !== 0 ) ? '' : value;\n\t}\n}\n\nfunction isInputElementEmpty( domElement: HTMLInputElement ) {\n\treturn !domElement.value;\n}\n\n/**\n * Fired when the user types in the input. Corresponds to the native\n * DOM `input` event.\n *\n * @eventName ~InputView#input\n */\nexport type InputViewInputEvent = {\n\tname: 'input';\n\targs: [ InputEvent ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/inputtext/inputtextview\n */\n\nimport InputView from '../input/inputview';\n\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The text input view class.\n */\nexport default class InputTextView extends InputView {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\ttype: 'text',\n\t\t\t\tclass: [\n\t\t\t\t\t'ck-input-text'\n\t\t\t\t]\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/dropdown/dropdownpanelview\n */\n\nimport View from '../view';\nimport type ViewCollection from '../viewcollection';\nimport type DropdownPanelFocusable from './dropdownpanelfocusable';\n\nimport { logWarning, type Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The dropdown panel view class.\n *\n * See {@link module:ui/dropdown/dropdownview~DropdownView} to learn about the common usage.\n */\nexport default class DropdownPanelView extends View implements DropdownPanelFocusable {\n\t/**\n\t * Collection of the child views in this panel.\n\t *\n\t * A common child type is the {@link module:ui/list/listview~ListView} and {@link module:ui/toolbar/toolbarview~ToolbarView}.\n\t * See {@link module:ui/dropdown/utils~addListToDropdown} and\n\t * {@link module:ui/dropdown/utils~addToolbarToDropdown} to learn more about child views of dropdowns.\n\t */\n\tpublic readonly children: ViewCollection;\n\n\t/**\n\t * Controls whether the panel is visible.\n\t *\n\t * @observable\n\t */\n\tdeclare public isVisible: boolean;\n\n\t/**\n\t * The position of the panel, relative to the parent.\n\t *\n\t * This property is reflected in the CSS class set to {@link #element} that controls\n\t * the position of the panel.\n\t *\n\t * @observable\n\t * @default 'se'\n\t */\n\tdeclare public position: PanelPosition;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.set( 'isVisible', false );\n\t\tthis.set( 'position', 'se' );\n\n\t\tthis.children = this.createCollection();\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-reset',\n\t\t\t\t\t'ck-dropdown__panel',\n\t\t\t\t\tbind.to( 'position', value => `ck-dropdown__panel_${ value }` ),\n\t\t\t\t\tbind.if( 'isVisible', 'ck-dropdown__panel-visible' )\n\t\t\t\t],\n\t\t\t\ttabindex: '-1'\n\t\t\t},\n\n\t\t\tchildren: this.children,\n\n\t\t\ton: {\n\t\t\t\t// Drag and drop in the panel should not break the selection in the editor.\n\t\t\t\t// https://github.com/ckeditor/ckeditor5-ui/issues/228\n\t\t\t\tselectstart: bind.to( evt => {\n\t\t\t\t\tif ( ( evt.target as HTMLElement ).tagName.toLocaleLowerCase() === 'input' ) {\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\n\t\t\t\t\tevt.preventDefault();\n\t\t\t\t} )\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Focuses the first view in the {@link #children} collection.\n\t *\n\t * See also {@link module:ui/dropdown/dropdownpanelfocusable~DropdownPanelFocusable}.\n\t */\n\tpublic focus(): void {\n\t\tif ( this.children.length ) {\n\t\t\tconst firstChild: any = this.children.first;\n\n\t\t\tif ( typeof firstChild.focus === 'function' ) {\n\t\t\t\tfirstChild.focus();\n\t\t\t} else {\n\t\t\t\t/**\n\t\t\t\t * The child view of a dropdown could not be focused because it is missing the `focus()` method.\n\t\t\t\t *\n\t\t\t\t * This warning appears when a dropdown {@link module:ui/dropdown/dropdownview~DropdownView#isOpen gets open} and it\n\t\t\t\t * attempts to focus the {@link module:ui/dropdown/dropdownpanelview~DropdownPanelView#children first child} of its panel\n\t\t\t\t * but the child does not implement the\n\t\t\t\t * {@link module:ui/dropdown/dropdownpanelfocusable~DropdownPanelFocusable focusable interface}.\n\t\t\t\t *\n\t\t\t\t * Focusing the content of a dropdown on open greatly improves the accessibility. Please make sure the view instance\n\t\t\t\t * provides the `focus()` method for the best user experience.\n\t\t\t\t *\n\t\t\t\t * @error ui-dropdown-panel-focus-child-missing-focus\n\t\t\t\t * @param childView\n\t\t\t\t * @param dropdownPanel\n\t\t\t\t */\n\t\t\t\tlogWarning( 'ui-dropdown-panel-focus-child-missing-focus', { childView: this.children.first, dropdownPanel: this } );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Focuses the view element or last item in view collection on opening dropdown's panel.\n\t *\n\t * See also {@link module:ui/dropdown/dropdownpanelfocusable~DropdownPanelFocusable}.\n\t */\n\tpublic focusLast(): void {\n\t\tif ( this.children.length ) {\n\t\t\tconst lastChild: any = this.children.last;\n\n\t\t\tif ( typeof lastChild.focusLast === 'function' ) {\n\t\t\t\tlastChild.focusLast();\n\t\t\t} else {\n\t\t\t\tlastChild.focus();\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * The position of the panel, relative to the parent.\n */\nexport type PanelPosition = 's' | 'se' | 'sw' | 'sme' | 'smw' | 'n' | 'ne' | 'nw' | 'nme' | 'nmw';\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./dropdown.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/dropdown/dropdownview\n */\n\nimport View from '../view';\n\nimport type { default as DropdownButton, DropdownButtonOpenEvent } from './button/dropdownbutton';\nimport type { default as DropdownPanelView, PanelPosition } from './dropdownpanelview';\nimport type { FocusableView } from '../focuscycler';\nimport type ListView from '../list/listview';\nimport type ToolbarView from '../toolbar/toolbarview';\n\nimport {\n\tKeystrokeHandler,\n\tFocusTracker,\n\tgetOptimalPosition,\n\ttype Locale,\n\ttype ObservableChangeEvent,\n\ttype PositioningFunction\n} from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/dropdown/dropdown.css';\n\n/**\n * The dropdown view class. It manages the dropdown button and dropdown panel.\n *\n * In most cases, the easiest way to create a dropdown is by using the {@link module:ui/dropdown/utils~createDropdown}\n * util:\n *\n * ```ts\n * const dropdown = createDropdown( locale );\n *\n * // Configure dropdown's button properties:\n * dropdown.buttonView.set( {\n * \tlabel: 'A dropdown',\n * \twithText: true\n * } );\n *\n * dropdown.render();\n *\n * dropdown.panelView.element.textContent = 'Content of the panel';\n *\n * // Will render a dropdown with a panel containing a \"Content of the panel\" text.\n * document.body.appendChild( dropdown.element );\n * ```\n *\n * If you want to add a richer content to the dropdown panel, you can use the {@link module:ui/dropdown/utils~addListToDropdown}\n * and {@link module:ui/dropdown/utils~addToolbarToDropdown} helpers. See more examples in\n * {@link module:ui/dropdown/utils~createDropdown} documentation.\n *\n * If you want to create a completely custom dropdown, then you can compose it manually:\n *\n * ```ts\n * const button = new DropdownButtonView( locale );\n * const panel = new DropdownPanelView( locale );\n * const dropdown = new DropdownView( locale, button, panel );\n *\n * button.set( {\n * \tlabel: 'A dropdown',\n * \twithText: true\n * } );\n *\n * dropdown.render();\n *\n * panel.element.textContent = 'Content of the panel';\n *\n * // Will render a dropdown with a panel containing a \"Content of the panel\" text.\n * document.body.appendChild( dropdown.element );\n * ```\n *\n * However, dropdown created this way will contain little behavior. You will need to implement handlers for actions\n * such as {@link module:ui/bindings/clickoutsidehandler~clickOutsideHandler clicking outside an open dropdown}\n * (which should close it) and support for arrow keys inside the panel. Therefore, unless you really know what\n * you do and you really need to do it, it is recommended to use the {@link module:ui/dropdown/utils~createDropdown} helper.\n */\nexport default class DropdownView extends View<HTMLDivElement> {\n\t/**\n\t * Button of the dropdown view. Clicking the button opens the {@link #panelView}.\n\t */\n\tpublic readonly buttonView: DropdownButton & FocusableView;\n\n\t/**\n\t * Panel of the dropdown. It opens when the {@link #buttonView} is\n\t * {@link module:ui/button/button~Button#event:execute executed} (i.e. clicked).\n\t *\n\t * Child views can be added to the panel's `children` collection:\n\t *\n\t * ```ts\n\t * dropdown.panelView.children.add( childView );\n\t * ```\n\t *\n\t * See {@link module:ui/dropdown/dropdownpanelview~DropdownPanelView#children} and\n\t * {@link module:ui/viewcollection~ViewCollection#add}.\n\t */\n\tpublic readonly panelView: DropdownPanelView;\n\n\t/**\n\t * Tracks information about the DOM focus in the dropdown.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * Instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}. It manages\n\t * keystrokes of the dropdown:\n\t *\n\t * * <kbd>▼</kbd> opens the dropdown,\n\t * * <kbd>◀</kbd> and <kbd>Esc</kbd> closes the dropdown.\n\t */\n\tpublic readonly keystrokes: KeystrokeHandler;\n\n\t/**\n\t * A child {@link module:ui/list/listview~ListView list view} of the dropdown located\n\t * in its {@link module:ui/dropdown/dropdownview~DropdownView#panelView panel}.\n\t *\n\t * **Note**: Only supported when dropdown has list view added using {@link module:ui/dropdown/utils~addListToDropdown}.\n\t */\n\tpublic listView?: ListView;\n\n\t/**\n\t * A child toolbar of the dropdown located in the\n\t * {@link module:ui/dropdown/dropdownview~DropdownView#panelView panel}.\n\t *\n\t * **Note**: Only supported when dropdown has list view added using {@link module:ui/dropdown/utils~addToolbarToDropdown}.\n\t */\n\tpublic toolbarView?: ToolbarView;\n\n\t/**\n\t * Controls whether the dropdown view is open, i.e. shows or hides the {@link #panelView panel}.\n\t *\n\t * **Note**: When the dropdown gets open, it will attempt to call `focus()` on the first child of its {@link #panelView}.\n\t * See {@link module:ui/dropdown/utils~addToolbarToDropdown}, {@link module:ui/dropdown/utils~addListToDropdown}, and\n\t * {@link module:ui/dropdown/utils~focusChildOnDropdownOpen} to learn more about focus management in dropdowns.\n\t *\n\t * @observable\n\t */\n\tdeclare public isOpen: boolean;\n\n\t/**\n\t * Controls whether the dropdown is enabled, i.e. it can be clicked and execute an action.\n\t *\n\t * See {@link module:ui/button/buttonview~ButtonView#isEnabled}.\n\t *\n\t * @observable\n\t */\n\tdeclare public isEnabled: boolean;\n\n\t/**\n\t * (Optional) The additional CSS class set on the dropdown {@link #element}.\n\t *\n\t * @observable\n\t */\n\tdeclare public class: string | undefined;\n\n\t/**\n\t * (Optional) The `id` attribute of the dropdown (i.e. to pair with a `<label>` element).\n\t *\n\t * @observable\n\t */\n\tdeclare public id: string | undefined;\n\n\t/**\n\t * The position of the panel, relative to the dropdown.\n\t *\n\t * **Note**: When `'auto'`, the panel will use one of the remaining positions to stay\n\t * in the viewport, visible to the user. The positions correspond directly to\n\t * {@link module:ui/dropdown/dropdownview~DropdownView.defaultPanelPositions default panel positions}.\n\t *\n\t * **Note**: This value has an impact on the\n\t * {@link module:ui/dropdown/dropdownpanelview~DropdownPanelView#position} property\n\t * each time the panel becomes {@link #isOpen open}.\n\t *\n\t * @observable\n\t * @default 'auto'\n\t */\n\tdeclare public panelPosition: PanelPosition | 'auto';\n\n\t/**\n\t * @observable\n\t */\n\tdeclare public ariaDescribedById: string | undefined;\n\n\t/**\n\t * Creates an instance of the dropdown.\n\t *\n\t * Also see {@link #render}.\n\t *\n\t * @param locale The localization services instance.\n\t */\n\tconstructor(\n\t\tlocale: Locale | undefined,\n\t\tbuttonView: DropdownButton & FocusableView,\n\t\tpanelView: DropdownPanelView\n\t) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.buttonView = buttonView;\n\t\tthis.panelView = panelView;\n\n\t\tthis.set( 'isOpen', false );\n\t\tthis.set( 'isEnabled', true );\n\t\tthis.set( 'class', undefined );\n\t\tthis.set( 'id', undefined );\n\t\tthis.set( 'panelPosition', 'auto' );\n\n\t\tthis.keystrokes = new KeystrokeHandler();\n\t\tthis.focusTracker = new FocusTracker();\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-dropdown',\n\t\t\t\t\tbind.to( 'class' ),\n\t\t\t\t\tbind.if( 'isEnabled', 'ck-disabled', value => !value )\n\t\t\t\t],\n\t\t\t\tid: bind.to( 'id' ),\n\t\t\t\t'aria-describedby': bind.to( 'ariaDescribedById' )\n\t\t\t},\n\n\t\t\tchildren: [\n\t\t\t\tbuttonView,\n\t\t\t\tpanelView\n\t\t\t]\n\t\t} );\n\n\t\tbuttonView.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck-dropdown__button'\n\t\t\t\t],\n\t\t\t\t'data-cke-tooltip-disabled': bind.to( 'isOpen' )\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.focusTracker.add( this.buttonView.element! );\n\t\tthis.focusTracker.add( this.panelView.element! );\n\n\t\t// Toggle the dropdown when its button has been clicked.\n\t\tthis.listenTo<DropdownButtonOpenEvent>( this.buttonView, 'open', () => {\n\t\t\tthis.isOpen = !this.isOpen;\n\t\t} );\n\n\t\t// Toggle the visibility of the panel when the dropdown becomes open.\n\t\tthis.panelView.bind( 'isVisible' ).to( this, 'isOpen' );\n\n\t\t// Let the dropdown control the position of the panel. The position must\n\t\t// be updated every time the dropdown is open.\n\t\tthis.on<ObservableChangeEvent<boolean>>( 'change:isOpen', ( evt, name, isOpen ) => {\n\t\t\tif ( !isOpen ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// If \"auto\", find the best position of the panel to fit into the viewport.\n\t\t\t// Otherwise, simply assign the static position.\n\t\t\tif ( this.panelPosition === 'auto' ) {\n\t\t\t\tthis.panelView.position = DropdownView._getOptimalPosition( {\n\t\t\t\t\telement: this.panelView.element!,\n\t\t\t\t\ttarget: this.buttonView.element!,\n\t\t\t\t\tfitInViewport: true,\n\t\t\t\t\tpositions: this._panelPositions\n\t\t\t\t} ).name as PanelPosition;\n\t\t\t} else {\n\t\t\t\tthis.panelView.position = this.panelPosition;\n\t\t\t}\n\t\t} );\n\n\t\t// Listen for keystrokes coming from within #element.\n\t\tthis.keystrokes.listenTo( this.element! );\n\n\t\tconst closeDropdown = ( data: unknown, cancel: () => void ) => {\n\t\t\tif ( this.isOpen ) {\n\t\t\t\tthis.isOpen = false;\n\t\t\t\tcancel();\n\t\t\t}\n\t\t};\n\n\t\t// Open the dropdown panel using the arrow down key, just like with return or space.\n\t\tthis.keystrokes.set( 'arrowdown', ( data, cancel ) => {\n\t\t\t// Don't open if the dropdown is disabled or already open.\n\t\t\tif ( this.buttonView.isEnabled && !this.isOpen ) {\n\t\t\t\tthis.isOpen = true;\n\t\t\t\tcancel();\n\t\t\t}\n\t\t} );\n\n\t\t// Block the right arrow key (until nested dropdowns are implemented).\n\t\tthis.keystrokes.set( 'arrowright', ( data, cancel ) => {\n\t\t\tif ( this.isOpen ) {\n\t\t\t\tcancel();\n\t\t\t}\n\t\t} );\n\n\t\t// Close the dropdown using the arrow left/escape key.\n\t\tthis.keystrokes.set( 'arrowleft', closeDropdown );\n\t\tthis.keystrokes.set( 'esc', closeDropdown );\n\t}\n\n\t/**\n\t * Focuses the {@link #buttonView}.\n\t */\n\tpublic focus(): void {\n\t\tthis.buttonView.focus();\n\t}\n\n\t/**\n\t * Returns {@link #panelView panel} positions to be used by the\n\t * {@link module:utils/dom/position~getOptimalPosition `getOptimalPosition()`}\n\t * utility considering the direction of the language the UI of the editor is displayed in.\n\t */\n\tprivate get _panelPositions(): Array<PositioningFunction> {\n\t\tconst {\n\t\t\tsouth, north,\n\t\t\tsouthEast, southWest,\n\t\t\tnorthEast, northWest,\n\t\t\tsouthMiddleEast, southMiddleWest,\n\t\t\tnorthMiddleEast, northMiddleWest\n\t\t} = DropdownView.defaultPanelPositions;\n\n\t\tif ( this.locale!.uiLanguageDirection !== 'rtl' ) {\n\t\t\treturn [\n\t\t\t\tsouthEast, southWest, southMiddleEast, southMiddleWest, south,\n\t\t\t\tnorthEast, northWest, northMiddleEast, northMiddleWest, north\n\t\t\t];\n\t\t} else {\n\t\t\treturn [\n\t\t\t\tsouthWest, southEast, southMiddleWest, southMiddleEast, south,\n\t\t\t\tnorthWest, northEast, northMiddleWest, northMiddleEast, north\n\t\t\t];\n\t\t}\n\t}\n\n\t/**\n\t * A set of positioning functions used by the dropdown view to determine\n\t * the optimal position (i.e. fitting into the browser viewport) of its\n\t * {@link module:ui/dropdown/dropdownview~DropdownView#panelView panel} when\n\t * {@link module:ui/dropdown/dropdownview~DropdownView#panelPosition} is set to 'auto'`.\n\t *\n\t * The available positioning functions are as follow:\n\t *\n\t * **South**\n\t *\n\t * * `south`\n\t *\n\t * ```\n\t *\t\t\t[ Button ]\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southEast`\n\t *\n\t * ```\n\t *\t\t[ Button ]\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southWest`\n\t *\n\t * ```\n\t *\t\t [ Button ]\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southMiddleEast`\n\t *\n\t * ```\n\t *\t\t [ Button ]\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southMiddleWest`\n\t *\n\t * ```\n\t *\t\t [ Button ]\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * **North**\n\t *\n\t * * `north`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t *\t\t [ Button ]\n\t * ```\n\t *\n\t * * `northEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t *\t\t[ Button ]\n\t * ```\n\t *\n\t * * `northWest`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t *\t\t [ Button ]\n\t * ```\n\t *\n\t * * `northMiddleEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t *\t\t [ Button ]\n\t * ```\n\t *\n\t * * `northMiddleWest`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Panel |\n\t *\t\t+-----------------+\n\t *\t\t [ Button ]\n\t * ```\n\t *\n\t * Positioning functions are compatible with {@link module:utils/dom/position~Position}.\n\t *\n\t * The name that position function returns will be reflected in dropdown panel's class that\n\t * controls its placement. See {@link module:ui/dropdown/dropdownview~DropdownView#panelPosition}\n\t * to learn more.\n\t */\n\tpublic static defaultPanelPositions: Record<string, PositioningFunction> = {\n\t\tsouth: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.bottom,\n\t\t\t\tleft: buttonRect.left - ( panelRect.width - buttonRect.width ) / 2,\n\t\t\t\tname: 's'\n\t\t\t};\n\t\t},\n\t\tsouthEast: buttonRect => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.bottom,\n\t\t\t\tleft: buttonRect.left,\n\t\t\t\tname: 'se'\n\t\t\t};\n\t\t},\n\t\tsouthWest: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.bottom,\n\t\t\t\tleft: buttonRect.left - panelRect.width + buttonRect.width,\n\t\t\t\tname: 'sw'\n\t\t\t};\n\t\t},\n\t\tsouthMiddleEast: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.bottom,\n\t\t\t\tleft: buttonRect.left - ( panelRect.width - buttonRect.width ) / 4,\n\t\t\t\tname: 'sme'\n\t\t\t};\n\t\t},\n\t\tsouthMiddleWest: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.bottom,\n\t\t\t\tleft: buttonRect.left - ( panelRect.width - buttonRect.width ) * 3 / 4,\n\t\t\t\tname: 'smw'\n\t\t\t};\n\t\t},\n\t\tnorth: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.top - panelRect.height,\n\t\t\t\tleft: buttonRect.left - ( panelRect.width - buttonRect.width ) / 2,\n\t\t\t\tname: 'n'\n\t\t\t};\n\t\t},\n\t\tnorthEast: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.top - panelRect.height,\n\t\t\t\tleft: buttonRect.left,\n\t\t\t\tname: 'ne'\n\t\t\t};\n\t\t},\n\t\tnorthWest: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.top - panelRect.height,\n\t\t\t\tleft: buttonRect.left - panelRect.width + buttonRect.width,\n\t\t\t\tname: 'nw'\n\t\t\t};\n\t\t},\n\t\tnorthMiddleEast: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.top - panelRect.height,\n\t\t\t\tleft: buttonRect.left - ( panelRect.width - buttonRect.width ) / 4,\n\t\t\t\tname: 'nme'\n\t\t\t};\n\t\t},\n\t\tnorthMiddleWest: ( buttonRect, panelRect ) => {\n\t\t\treturn {\n\t\t\t\ttop: buttonRect.top - panelRect.height,\n\t\t\t\tleft: buttonRect.left - ( panelRect.width - buttonRect.width ) * 3 / 4,\n\t\t\t\tname: 'nmw'\n\t\t\t};\n\t\t}\n\t};\n\n\t/**\n\t * A function used to calculate the optimal position for the dropdown panel.\n\t */\n\tprivate static _getOptimalPosition = getOptimalPosition;\n}\n\n/**\n * Fired when the toolbar button or list item is executed.\n *\n * For {@link ~DropdownView#listView} It fires when a child of some {@link module:ui/list/listitemview~ListItemView}\n * fired `execute`.\n *\n * For {@link ~DropdownView#toolbarView} It fires when one of the buttons has been\n * {@link module:ui/button/button~Button#event:execute executed}.\n *\n * **Note**: Only supported when dropdown has list view added using {@link module:ui/dropdown/utils~addListToDropdown}\n * or {@link module:ui/dropdown/utils~addToolbarToDropdown}.\n *\n * @eventName ~DropdownView#execute\n */\nexport type DropdownViewEvent = {\n\tname: 'execute';\n\targs: [];\n};\n","export default \"<svg viewBox=\\\"0 0 10 10\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M.941 4.523a.75.75 0 1 1 1.06-1.06l3.006 3.005 3.005-3.005a.75.75 0 1 1 1.06 1.06l-3.549 3.55a.75.75 0 0 1-1.168-.136L.941 4.523z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/dropdown/button/dropdownbuttonview\n */\n\nimport ButtonView from '../../button/buttonview';\nimport type DropdownButton from './dropdownbutton';\nimport IconView from '../../icon/iconview';\n\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\nimport dropdownArrowIcon from '../../../theme/icons/dropdown-arrow.svg';\n\n/**\n * The default dropdown button view class.\n *\n * ```ts\n * const view = new DropdownButtonView();\n *\n * view.set( {\n * \tlabel: 'A button',\n * \tkeystroke: 'Ctrl+B',\n * \ttooltip: true\n * } );\n *\n * view.render();\n *\n * document.body.append( view.element );\n * ```\n *\n * Also see the {@link module:ui/dropdown/utils~createDropdown `createDropdown()` util}.\n */\nexport default class DropdownButtonView extends ButtonView implements DropdownButton {\n\t/**\n\t * An icon that displays arrow to indicate a dropdown button.\n\t */\n\tpublic readonly arrowView: IconView;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.arrowView = this._createArrowView();\n\n\t\tthis.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\t'aria-haspopup': true,\n\t\t\t\t'aria-expanded': this.bindTemplate.to( 'isOn', value => String( value ) )\n\t\t\t}\n\t\t} );\n\n\t\t// The DropdownButton interface expects the open event upon which will open the dropdown.\n\t\tthis.delegate( 'execute' ).to( this, 'open' );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.children.add( this.arrowView );\n\t}\n\n\t/**\n\t * Creates a {@link module:ui/icon/iconview~IconView} instance as {@link #arrowView}.\n\t */\n\tprivate _createArrowView() {\n\t\tconst arrowView = new IconView();\n\n\t\tarrowView.content = dropdownArrowIcon;\n\n\t\tarrowView.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: 'ck-dropdown__arrow'\n\t\t\t}\n\t\t} );\n\n\t\treturn arrowView;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/focuscycler\n */\n\nimport {\n\tisVisible,\n\ttype ArrayOrItem,\n\ttype FocusTracker,\n\ttype KeystrokeHandler\n} from '@ckeditor/ckeditor5-utils';\n\nimport type View from './view';\nimport type ViewCollection from './viewcollection';\n\n/**\n * A utility class that helps cycling over focusable {@link module:ui/view~View views} in a\n * {@link module:ui/viewcollection~ViewCollection} when the focus is tracked by the\n * {@link module:utils/focustracker~FocusTracker} instance. It helps implementing keyboard\n * navigation in HTML forms, toolbars, lists and the like.\n *\n * To work properly it requires:\n * * a collection of focusable (HTML `tabindex` attribute) views that implement the `focus()` method,\n * * an associated focus tracker to determine which view is focused.\n *\n * A simple cycler setup can look like this:\n *\n * ```ts\n * const focusables = new ViewCollection();\n * const focusTracker = new FocusTracker();\n *\n * // Add focusable views to the focus tracker.\n * focusTracker.add( ... );\n * ```\n *\n * Then, the cycler can be used manually:\n *\n * ```ts\n * const cycler = new FocusCycler( { focusables, focusTracker } );\n *\n * // Will focus the first focusable view in #focusables.\n * cycler.focusFirst();\n *\n * // Will log the next focusable item in #focusables.\n * console.log( cycler.next );\n * ```\n *\n * Alternatively, it can work side by side with the {@link module:utils/keystrokehandler~KeystrokeHandler}:\n *\n * ```ts\n * const keystrokeHandler = new KeystrokeHandler();\n *\n * // Activate the keystroke handler.\n * keystrokeHandler.listenTo( sourceOfEvents );\n *\n * const cycler = new FocusCycler( {\n * \tfocusables, focusTracker, keystrokeHandler,\n * \tactions: {\n * \t\t// When arrowup of arrowleft is detected by the #keystrokeHandler,\n * \t\t// focusPrevious() will be called on the cycler.\n * \t\tfocusPrevious: [ 'arrowup', 'arrowleft' ],\n * \t}\n * } );\n * ```\n *\n * Check out the {@glink framework/deep-dive/ui/focus-tracking \"Deep dive into focus tracking\"} guide to learn more.\n */\nexport default class FocusCycler {\n\t/**\n\t * A {@link module:ui/view~View view} collection that the cycler operates on.\n\t */\n\tpublic readonly focusables: ViewCollection;\n\n\t/**\n\t * A focus tracker instance that the cycler uses to determine the current focus\n\t * state in {@link #focusables}.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * An instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}\n\t * which can respond to certain keystrokes and cycle the focus.\n\t */\n\tpublic readonly keystrokeHandler?: KeystrokeHandler;\n\n\t/**\n\t * Actions that the cycler can take when a keystroke is pressed. Requires\n\t * `options.keystrokeHandler` to be passed and working. When an action is\n\t * performed, `preventDefault` and `stopPropagation` will be called on the event\n\t * the keystroke fired in the DOM.\n\t *\n\t * ```ts\n\t * actions: {\n\t * \t// Will call #focusPrevious() when arrowleft or arrowup is pressed.\n\t * \tfocusPrevious: [ 'arrowleft', 'arrowup' ],\n\t *\n\t * \t// Will call #focusNext() when arrowdown is pressed.\n\t * \tfocusNext: 'arrowdown'\n\t * }\n\t * ```\n\t */\n\tpublic readonly actions?: FocusCyclerActions;\n\n\t/**\n\t * Creates an instance of the focus cycler utility.\n\t *\n\t * @param options Configuration options.\n\t */\n\tconstructor( options: {\n\t\tfocusables: ViewCollection;\n\t\tfocusTracker: FocusTracker;\n\t\tkeystrokeHandler?: KeystrokeHandler;\n\t\tactions?: FocusCyclerActions;\n\t} ) {\n\t\tthis.focusables = options.focusables;\n\t\tthis.focusTracker = options.focusTracker;\n\t\tthis.keystrokeHandler = options.keystrokeHandler;\n\t\tthis.actions = options.actions;\n\n\t\tif ( options.actions && options.keystrokeHandler ) {\n\t\t\tfor ( const methodName in options.actions ) {\n\t\t\t\tlet actions = options.actions[ methodName as keyof FocusCyclerActions ]!;\n\n\t\t\t\tif ( typeof actions == 'string' ) {\n\t\t\t\t\tactions = [ actions ];\n\t\t\t\t}\n\n\t\t\t\tfor ( const keystroke of actions ) {\n\t\t\t\t\toptions.keystrokeHandler.set( keystroke, ( data, cancel ) => {\n\t\t\t\t\t\tthis[ methodName as keyof FocusCyclerActions ]();\n\t\t\t\t\t\tcancel();\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Returns the first focusable view in {@link #focusables}.\n\t * Returns `null` if there is none.\n\t *\n\t * **Note**: Hidden views (e.g. with `display: none`) are ignored.\n\t */\n\tpublic get first(): FocusableView | null {\n\t\treturn ( this.focusables.find( isFocusable ) || null ) as FocusableView | null;\n\t}\n\n\t/**\n\t * Returns the last focusable view in {@link #focusables}.\n\t * Returns `null` if there is none.\n\t *\n\t * **Note**: Hidden views (e.g. with `display: none`) are ignored.\n\t */\n\tpublic get last(): FocusableView | null {\n\t\treturn ( this.focusables.filter( isFocusable ).slice( -1 )[ 0 ] || null ) as FocusableView | null;\n\t}\n\n\t/**\n\t * Returns the next focusable view in {@link #focusables} based on {@link #current}.\n\t * Returns `null` if there is none.\n\t *\n\t * **Note**: Hidden views (e.g. with `display: none`) are ignored.\n\t */\n\tpublic get next(): FocusableView | null {\n\t\treturn this._getFocusableItem( 1 );\n\t}\n\n\t/**\n\t * Returns the previous focusable view in {@link #focusables} based on {@link #current}.\n\t * Returns `null` if there is none.\n\t *\n\t * **Note**: Hidden views (e.g. with `display: none`) are ignored.\n\t */\n\tpublic get previous(): FocusableView | null {\n\t\treturn this._getFocusableItem( -1 );\n\t}\n\n\t/**\n\t * An index of the view in the {@link #focusables} which is focused according\n\t * to {@link #focusTracker}. Returns `null` when there is no such view.\n\t */\n\tpublic get current(): number | null {\n\t\tlet index: number | null = null;\n\n\t\t// There's no focused view in the focusables.\n\t\tif ( this.focusTracker.focusedElement === null ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tthis.focusables.find( ( view, viewIndex ) => {\n\t\t\tconst focused = view.element === this.focusTracker.focusedElement;\n\n\t\t\tif ( focused ) {\n\t\t\t\tindex = viewIndex;\n\t\t\t}\n\n\t\t\treturn focused;\n\t\t} );\n\n\t\treturn index;\n\t}\n\n\t/**\n\t * Focuses the {@link #first} item in {@link #focusables}.\n\t *\n\t * **Note**: Hidden views (e.g. with `display: none`) are ignored.\n\t */\n\tpublic focusFirst(): void {\n\t\tthis._focus( this.first );\n\t}\n\n\t/**\n\t * Focuses the {@link #last} item in {@link #focusables}.\n\t *\n\t * **Note**: Hidden views (e.g. with `display: none`) are ignored.\n\t */\n\tpublic focusLast(): void {\n\t\tthis._focus( this.last );\n\t}\n\n\t/**\n\t * Focuses the {@link #next} item in {@link #focusables}.\n\t *\n\t * **Note**: Hidden views (e.g. with `display: none`) are ignored.\n\t */\n\tpublic focusNext(): void {\n\t\tthis._focus( this.next );\n\t}\n\n\t/**\n\t * Focuses the {@link #previous} item in {@link #focusables}.\n\t *\n\t * **Note**: Hidden views (e.g. with `display: none`) are ignored.\n\t */\n\tpublic focusPrevious(): void {\n\t\tthis._focus( this.previous );\n\t}\n\n\t/**\n\t * Focuses the given view if it exists.\n\t */\n\tprivate _focus( view: FocusableView | null ) {\n\t\tif ( view ) {\n\t\t\tview.focus();\n\t\t}\n\t}\n\n\t/**\n\t * Returns the next or previous focusable view in {@link #focusables} with respect\n\t * to {@link #current}.\n\t *\n\t * @param step Either `1` for checking forward from {@link #current} or `-1` for checking backwards.\n\t */\n\tprivate _getFocusableItem( step: 1 | -1 ): FocusableView | null {\n\t\t// Cache for speed.\n\t\tconst current = this.current;\n\t\tconst collectionLength = this.focusables.length;\n\n\t\tif ( !collectionLength ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// Start from the beginning if no view is focused.\n\t\t// https://github.com/ckeditor/ckeditor5-ui/issues/206\n\t\tif ( current === null ) {\n\t\t\treturn this[ step === 1 ? 'first' : 'last' ];\n\t\t}\n\n\t\t// Cycle in both directions.\n\t\tlet index = ( current + collectionLength + step ) % collectionLength;\n\n\t\tdo {\n\t\t\tconst view = this.focusables.get( index )!;\n\n\t\t\tif ( isFocusable( view ) ) {\n\t\t\t\treturn view as FocusableView;\n\t\t\t}\n\n\t\t\t// Cycle in both directions.\n\t\t\tindex = ( index + collectionLength + step ) % collectionLength;\n\t\t} while ( index !== current );\n\n\t\treturn null;\n\t}\n}\n\nexport type FocusableView = View & { focus(): void };\n\nexport interface FocusCyclerActions {\n\tfocusFirst?: ArrayOrItem<string>;\n\tfocusLast?: ArrayOrItem<string>;\n\tfocusNext?: ArrayOrItem<string>;\n\tfocusPrevious?: ArrayOrItem<string>;\n}\n\n/**\n * Checks whether a view is focusable.\n *\n * @param view A view to be checked.\n */\nfunction isFocusable( view: View & { focus?: unknown } ) {\n\treturn !!( view.focus && isVisible( view.element ) );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/toolbar/toolbarseparatorview\n */\n\nimport View from '../view';\n\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The toolbar separator view class.\n */\nexport default class ToolbarSeparatorView extends View {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'span',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-toolbar__separator'\n\t\t\t\t]\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/toolbar/toolbarlinebreakview\n */\n\nimport View from '../view';\n\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The toolbar line break view class.\n */\nexport default class ToolbarLineBreakView extends View {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'span',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-toolbar__line-break'\n\t\t\t\t]\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport type { ToolbarConfig, ToolbarConfigItem } from '@ckeditor/ckeditor5-core';\n\n/**\n * @module ui/toolbar/normalizetoolbarconfig\n */\n\n/**\n * Normalizes the toolbar configuration (`config.toolbar`), which:\n *\n * * may be defined as an `Array`:\n *\n * ```\n * toolbar: [ 'heading', 'bold', 'italic', 'link', ... ]\n * ```\n *\n * * or an `Object`:\n *\n * ```\n * toolbar: {\n * \titems: [ 'heading', 'bold', 'italic', 'link', ... ],\n * \tremoveItems: [ 'bold' ],\n * \t...\n * }\n * ```\n *\n * * or may not be defined at all (`undefined`)\n *\n * and returns it in the object form.\n *\n * @param config The value of `config.toolbar`.\n * @returns A normalized toolbar config object.\n */\nexport default function normalizeToolbarConfig(\n\tconfig: ToolbarConfig | undefined\n): {\n\titems: Array<ToolbarConfigItem>;\n\tremoveItems: Array<string>;\n\tshouldNotGroupWhenFull?: boolean;\n} {\n\tif ( Array.isArray( config ) ) {\n\t\treturn {\n\t\t\titems: config,\n\t\t\tremoveItems: []\n\t\t};\n\t}\n\n\tif ( !config ) {\n\t\treturn {\n\t\t\titems: [],\n\t\t\tremoveItems: []\n\t\t};\n\t}\n\n\treturn Object.assign( {\n\t\titems: [],\n\t\tremoveItems: []\n\t}, config );\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./toolbar.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/toolbar/toolbarview\n */\n\nimport View from '../view';\nimport FocusCycler from '../focuscycler';\nimport ToolbarSeparatorView from './toolbarseparatorview';\nimport ToolbarLineBreakView from './toolbarlinebreakview';\nimport preventDefault from '../bindings/preventdefault';\nimport { createDropdown, addToolbarToDropdown } from '../dropdown/utils';\nimport normalizeToolbarConfig from './normalizetoolbarconfig';\n\nimport type ComponentFactory from '../componentfactory';\nimport type ViewCollection from '../viewcollection';\nimport type DropdownView from '../dropdown/dropdownview';\nimport type DropdownPanelFocusable from '../dropdown/dropdownpanelfocusable';\n\nimport {\n\tFocusTracker,\n\tKeystrokeHandler,\n\tRect,\n\tResizeObserver,\n\tglobal,\n\tisVisible,\n\tlogWarning,\n\ttype CollectionAddEvent,\n\ttype CollectionChangeEvent,\n\ttype CollectionRemoveEvent,\n\ttype Locale,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport {\n\ticons,\n\ttype ToolbarConfig,\n\ttype ToolbarConfigItem\n} from '@ckeditor/ckeditor5-core';\n\nimport { isObject } from 'lodash-es';\n\nimport '../../theme/components/toolbar/toolbar.css';\n\nconst { threeVerticalDots } = icons;\n\nconst NESTED_TOOLBAR_ICONS: Record<string, string | undefined> = {\n\talignLeft: icons.alignLeft,\n\tbold: icons.bold,\n\timportExport: icons.importExport,\n\tparagraph: icons.paragraph,\n\tplus: icons.plus,\n\ttext: icons.text,\n\tthreeVerticalDots: icons.threeVerticalDots\n};\n\n/**\n * The toolbar view class.\n */\nexport default class ToolbarView extends View implements DropdownPanelFocusable {\n\t/**\n\t * A reference to the options object passed to the constructor.\n\t */\n\tpublic readonly options: ToolbarOptions;\n\n\t/**\n\t * A collection of toolbar items (buttons, dropdowns, etc.).\n\t */\n\tpublic readonly items: ViewCollection;\n\n\t/**\n\t * Tracks information about the DOM focus in the toolbar.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * An instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}\n\t * to handle keyboard navigation in the toolbar.\n\t */\n\tpublic readonly keystrokes: KeystrokeHandler;\n\n\t/**\n\t * A (child) view containing {@link #items toolbar items}.\n\t */\n\tpublic readonly itemsView: ItemsView;\n\n\t/**\n\t * A toplevel collection aggregating building blocks of the toolbar.\n\t *\n\t *\t┌───────────────── ToolbarView ─────────────────┐\n\t *\t| ┌──────────────── #children ────────────────┐ |\n\t *\t| | ┌──────────── #itemsView ───────────┐ | |\n\t *\t| | | [ item1 ] [ item2 ] ... [ itemN ] | | |\n\t *\t| | └──────────────────────────────────-┘ | |\n\t *\t| └───────────────────────────────────────────┘ |\n\t *\t└───────────────────────────────────────────────┘\n\t *\n\t * By default, it contains the {@link #itemsView} but it can be extended with additional\n\t * UI elements when necessary.\n\t */\n\tpublic readonly children: ViewCollection;\n\n\t/**\n\t * A collection of {@link #items} that take part in the focus cycling\n\t * (i.e. navigation using the keyboard). Usually, it contains a subset of {@link #items} with\n\t * some optional UI elements that also belong to the toolbar and should be focusable\n\t * by the user.\n\t */\n\tpublic readonly focusables: ViewCollection;\n\n\tdeclare public locale: Locale;\n\n\t/**\n\t * Label used by assistive technologies to describe this toolbar element.\n\t *\n\t * @observable\n\t * @default 'Editor toolbar'\n\t */\n\tdeclare public ariaLabel: string;\n\n\t/**\n\t * The maximum width of the toolbar element.\n\t *\n\t * **Note**: When set to a specific value (e.g. `'200px'`), the value will affect the behavior of the\n\t * {@link module:ui/toolbar/toolbarview~ToolbarOptions#shouldGroupWhenFull}\n\t * option by changing the number of {@link #items} that will be displayed in the toolbar at a time.\n\t *\n\t * @observable\n\t * @default 'auto'\n\t */\n\tdeclare public maxWidth: string;\n\n\t/**\n\t * An additional CSS class added to the {@link #element}.\n\t *\n\t * @observable\n\t * @member {String} #class\n\t */\n\tdeclare public class: string | undefined;\n\n\t/**\n\t * When set true, makes the toolbar look compact with {@link #element}.\n\t *\n\t * @observable\n\t * @default false\n\t */\n\tdeclare public isCompact: boolean;\n\n\t/**\n\t * Controls the orientation of toolbar items. Only available when\n\t * {@link module:ui/toolbar/toolbarview~ToolbarOptions#shouldGroupWhenFull dynamic items grouping}\n\t * is **disabled**.\n\t *\n\t * @observable\n\t */\n\tdeclare public isVertical: boolean;\n\n\t/**\n\t * Helps cycling over {@link #focusables focusable items} in the toolbar.\n\t */\n\tprivate readonly _focusCycler: FocusCycler;\n\n\t/**\n\t * An instance of the active toolbar behavior that shapes its look and functionality.\n\t *\n\t * See {@link module:ui/toolbar/toolbarview~ToolbarBehavior} to learn more.\n\t */\n\tprivate readonly _behavior: ToolbarBehavior;\n\n\t/**\n\t * Creates an instance of the {@link module:ui/toolbar/toolbarview~ToolbarView} class.\n\t *\n\t * Also see {@link #render}.\n\t *\n\t * @param locale The localization services instance.\n\t * @param options Configuration options of the toolbar.\n\t */\n\tconstructor( locale: Locale, options?: ToolbarOptions ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\t\tconst t = this.t!;\n\n\t\tthis.options = options || {};\n\n\t\tthis.set( 'ariaLabel', t( 'Editor toolbar' ) );\n\t\tthis.set( 'maxWidth', 'auto' );\n\n\t\tthis.items = this.createCollection();\n\t\tthis.focusTracker = new FocusTracker();\n\t\tthis.keystrokes = new KeystrokeHandler();\n\n\t\tthis.set( 'class', undefined );\n\t\tthis.set( 'isCompact', false );\n\n\t\tthis.itemsView = new ItemsView( locale );\n\t\tthis.children = this.createCollection();\n\t\tthis.children.add( this.itemsView );\n\t\tthis.focusables = this.createCollection();\n\n\t\tconst isRtl = locale.uiLanguageDirection === 'rtl';\n\n\t\tthis._focusCycler = new FocusCycler( {\n\t\t\tfocusables: this.focusables,\n\t\t\tfocusTracker: this.focusTracker,\n\t\t\tkeystrokeHandler: this.keystrokes,\n\t\t\tactions: {\n\t\t\t\t// Navigate toolbar items backwards using the arrow[left,up] keys.\n\t\t\t\tfocusPrevious: [ isRtl ? 'arrowright' : 'arrowleft', 'arrowup' ],\n\n\t\t\t\t// Navigate toolbar items forwards using the arrow[right,down] keys.\n\t\t\t\tfocusNext: [ isRtl ? 'arrowleft' : 'arrowright', 'arrowdown' ]\n\t\t\t}\n\t\t} );\n\n\t\tconst classes = [\n\t\t\t'ck',\n\t\t\t'ck-toolbar',\n\t\t\tbind.to( 'class' ),\n\t\t\tbind.if( 'isCompact', 'ck-toolbar_compact' )\n\t\t];\n\n\t\tif ( this.options.shouldGroupWhenFull && this.options.isFloating ) {\n\t\t\tclasses.push( 'ck-toolbar_floating' );\n\t\t}\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: classes,\n\t\t\t\trole: 'toolbar',\n\t\t\t\t'aria-label': bind.to( 'ariaLabel' ),\n\t\t\t\tstyle: {\n\t\t\t\t\tmaxWidth: bind.to( 'maxWidth' )\n\t\t\t\t},\n\t\t\t\ttabindex: -1\n\t\t\t},\n\n\t\t\tchildren: this.children,\n\n\t\t\ton: {\n\t\t\t\t// https://github.com/ckeditor/ckeditor5-ui/issues/206\n\t\t\t\tmousedown: preventDefault( this )\n\t\t\t}\n\t\t} );\n\n\t\tthis._behavior = this.options.shouldGroupWhenFull ? new DynamicGrouping( this ) : new StaticLayout( this );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.focusTracker.add( this.element! );\n\n\t\t// Children added before rendering should be known to the #focusTracker.\n\t\tfor ( const item of this.items ) {\n\t\t\tthis.focusTracker.add( item.element! );\n\t\t}\n\n\t\tthis.items.on<CollectionAddEvent<View>>( 'add', ( evt, item ) => {\n\t\t\tthis.focusTracker.add( item.element! );\n\t\t} );\n\n\t\tthis.items.on<CollectionRemoveEvent<View>>( 'remove', ( evt, item ) => {\n\t\t\tthis.focusTracker.remove( item.element! );\n\t\t} );\n\n\t\t// Start listening for the keystrokes coming from #element.\n\t\tthis.keystrokes.listenTo( this.element! );\n\n\t\tthis._behavior.render( this );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tthis._behavior.destroy();\n\t\tthis.focusTracker.destroy();\n\t\tthis.keystrokes.destroy();\n\n\t\treturn super.destroy();\n\t}\n\n\t/**\n\t * Focuses the first focusable in {@link #focusables}.\n\t */\n\tpublic focus(): void {\n\t\tthis._focusCycler.focusFirst();\n\t}\n\n\t/**\n\t * Focuses the last focusable in {@link #focusables}.\n\t */\n\tpublic focusLast(): void {\n\t\tthis._focusCycler.focusLast();\n\t}\n\n\t/**\n\t * A utility that expands the plain toolbar configuration into\n\t * {@link module:ui/toolbar/toolbarview~ToolbarView#items} using a given component factory.\n\t *\n\t * @param itemsOrConfig The toolbar items or the entire toolbar configuration object.\n\t * @param factory A factory producing toolbar items.\n\t * @param removeItems An array of items names to be removed from the configuration. When present, applies\n\t * to this toolbar and all nested ones as well.\n\t */\n\tpublic fillFromConfig(\n\t\titemsOrConfig: ToolbarConfig | undefined,\n\t\tfactory: ComponentFactory,\n\t\tremoveItems?: Array<string>\n\t): void {\n\t\tthis.items.addMany( this._buildItemsFromConfig( itemsOrConfig, factory, removeItems ) );\n\t}\n\n\t/**\n\t * A utility that expands the plain toolbar configuration into a list of view items using a given component factory.\n\t *\n\t * @param itemsOrConfig The toolbar items or the entire toolbar configuration object.\n\t * @param factory A factory producing toolbar items.\n\t * @param removeItems An array of items names to be removed from the configuration. When present, applies\n\t * to this toolbar and all nested ones as well.\n\t */\n\tprivate _buildItemsFromConfig(\n\t\titemsOrConfig: ToolbarConfig | undefined,\n\t\tfactory: ComponentFactory,\n\t\tremoveItems?: Array<string>\n\t): Array<View> {\n\t\tconst config = normalizeToolbarConfig( itemsOrConfig );\n\t\tconst normalizedRemoveItems = removeItems || config.removeItems;\n\t\tconst itemsToAdd = this._cleanItemsConfiguration( config.items, factory, normalizedRemoveItems )\n\t\t\t.map( item => {\n\t\t\t\tif ( isObject( item ) ) {\n\t\t\t\t\treturn this._createNestedToolbarDropdown( item, factory, normalizedRemoveItems );\n\t\t\t\t} else if ( item === '|' ) {\n\t\t\t\t\treturn new ToolbarSeparatorView();\n\t\t\t\t} else if ( item === '-' ) {\n\t\t\t\t\treturn new ToolbarLineBreakView();\n\t\t\t\t}\n\n\t\t\t\treturn factory.create( item );\n\t\t\t} )\n\t\t\t.filter( ( item ): item is View => !!item );\n\n\t\treturn itemsToAdd;\n\t}\n\n\t/**\n\t * Cleans up the {@link module:ui/toolbar/toolbarview~ToolbarView#items} of the toolbar by removing unwanted items and\n\t * duplicated (obsolete) separators or line breaks.\n\t *\n\t * @param items The toolbar items configuration.\n\t * @param factory A factory producing toolbar items.\n\t * @param removeItems An array of items names to be removed from the configuration.\n\t * @returns Items after the clean-up.\n\t */\n\tprivate _cleanItemsConfiguration(\n\t\titems: Array<ToolbarConfigItem>,\n\t\tfactory: ComponentFactory,\n\t\tremoveItems: Array<string>\n\t) {\n\t\tconst filteredItems = items\n\t\t\t.filter( ( item, idx, items ) => {\n\t\t\t\tif ( item === '|' ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\t// Items listed in `config.removeItems` should not be added to the toolbar.\n\t\t\t\tif ( removeItems.indexOf( item as any ) !== -1 ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\tif ( item === '-' ) {\n\t\t\t\t\t// The toolbar line breaks must not be rendered when toolbar grouping is enabled.\n\t\t\t\t\t// (https://github.com/ckeditor/ckeditor5/issues/8582)\n\t\t\t\t\tif ( this.options.shouldGroupWhenFull ) {\n\t\t\t\t\t\t/**\n\t\t\t\t\t\t * The toolbar multiline breaks (`-` items) only work when the automatic button grouping\n\t\t\t\t\t\t * is disabled in the toolbar configuration.\n\t\t\t\t\t\t * To do this, set the `shouldNotGroupWhenFull` option to `true` in the editor configuration:\n\t\t\t\t\t\t *\n\t\t\t\t\t\t * ```ts\n\t\t\t\t\t\t * const config = {\n\t\t\t\t\t\t * \ttoolbar: {\n\t\t\t\t\t\t * \t\titems: [ ... ],\n\t\t\t\t\t\t * \t\tshouldNotGroupWhenFull: true\n\t\t\t\t\t\t * \t}\n\t\t\t\t\t\t * }\n\t\t\t\t\t\t * ```\n\t\t\t\t\t\t *\n\t\t\t\t\t\t * Learn more about {@link module:core/editor/editorconfig~EditorConfig#toolbar toolbar configuration}.\n\t\t\t\t\t\t *\n\t\t\t\t\t\t * @error toolbarview-line-break-ignored-when-grouping-items\n\t\t\t\t\t\t */\n\t\t\t\t\t\tlogWarning( 'toolbarview-line-break-ignored-when-grouping-items', items );\n\n\t\t\t\t\t\treturn false;\n\t\t\t\t\t}\n\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\t// For the items that cannot be instantiated we are sending warning message. We also filter them out.\n\t\t\t\tif ( !isObject( item ) && !factory.has( item ) ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * There was a problem processing the configuration of the toolbar. The item with the given\n\t\t\t\t\t * name does not exist so it was omitted when rendering the toolbar.\n\t\t\t\t\t *\n\t\t\t\t\t * This warning usually shows up when the {@link module:core/plugin~Plugin} which is supposed\n\t\t\t\t\t * to provide a toolbar item has not been loaded or there is a typo in the configuration.\n\t\t\t\t\t *\n\t\t\t\t\t * Make sure the plugin responsible for this toolbar item is loaded and the toolbar configuration\n\t\t\t\t\t * is correct, e.g. {@link module:basic-styles/bold~Bold} is loaded for the `'bold'` toolbar item.\n\t\t\t\t\t *\n\t\t\t\t\t * You can use the following snippet to retrieve all available toolbar items:\n\t\t\t\t\t *\n\t\t\t\t\t * ```ts\n\t\t\t\t\t * Array.from( editor.ui.componentFactory.names() );\n\t\t\t\t\t * ```\n\t\t\t\t\t *\n\t\t\t\t\t * @error toolbarview-item-unavailable\n\t\t\t\t\t * @param item The name of the component or nested toolbar definition.\n\t\t\t\t\t */\n\t\t\t\t\tlogWarning( 'toolbarview-item-unavailable', { item } );\n\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\treturn true;\n\t\t\t} );\n\n\t\treturn this._cleanSeparatorsAndLineBreaks( filteredItems );\n\t}\n\n\t/**\n\t * Remove leading, trailing, and duplicated separators (`-` and `|`).\n\t *\n\t * @returns Toolbar items after the separator and line break clean-up.\n\t */\n\tprivate _cleanSeparatorsAndLineBreaks( items: Array<ToolbarConfigItem> ) {\n\t\tconst nonSeparatorPredicate = ( item: ToolbarConfigItem ) => ( item !== '-' && item !== '|' );\n\t\tconst count = items.length;\n\n\t\t// Find an index of the first item that is not a separator.\n\t\tconst firstCommandItemIndex = items.findIndex( nonSeparatorPredicate );\n\n\t\t// Items include separators only. There is no point in displaying them.\n\t\tif ( firstCommandItemIndex === -1 ) {\n\t\t\treturn [];\n\t\t}\n\n\t\t// Search from the end of the list, then convert found index back to the original direction.\n\t\tconst lastCommandItemIndex = count - items\n\t\t\t.slice()\n\t\t\t.reverse()\n\t\t\t.findIndex( nonSeparatorPredicate );\n\n\t\treturn items\n\t\t\t// Return items without the leading and trailing separators.\n\t\t\t.slice( firstCommandItemIndex, lastCommandItemIndex )\n\t\t\t// Remove duplicated separators.\n\t\t\t.filter( ( name, idx, items ) => {\n\t\t\t\t// Filter only separators.\n\t\t\t\tif ( nonSeparatorPredicate( name ) ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\tconst isDuplicated = idx > 0 && items[ idx - 1 ] === name;\n\n\t\t\t\treturn !isDuplicated;\n\t\t\t} );\n\t}\n\n\t/**\n\t * Creates a user-defined dropdown containing a toolbar with items.\n\t *\n\t * @param definition A definition of the nested toolbar dropdown.\n\t * @param definition.label A label of the dropdown.\n\t * @param definition.icon An icon of the drop-down. One of 'bold', 'plus', 'text', 'importExport', 'alignLeft',\n\t * 'paragraph' or an SVG string. When `false` is passed, no icon will be used.\n\t * @param definition.withText When set `true`, the label of the dropdown will be visible. See\n\t * {@link module:ui/button/buttonview~ButtonView#withText} to learn more.\n\t * @param definition.tooltip A tooltip of the dropdown button. See\n\t * {@link module:ui/button/buttonview~ButtonView#tooltip} to learn more. Defaults to `true`.\n\t * @param componentFactory Component factory used to create items\n\t * of the nested toolbar.\n\t */\n\tprivate _createNestedToolbarDropdown(\n\t\tdefinition: Exclude<ToolbarConfigItem, string>,\n\t\tcomponentFactory: ComponentFactory,\n\t\tremoveItems: Array<string>\n\t) {\n\t\tlet { label, icon, items, tooltip = true, withText = false } = definition;\n\n\t\titems = this._cleanItemsConfiguration( items, componentFactory, removeItems );\n\n\t\t// There is no point in rendering a dropdown without items.\n\t\tif ( !items.length ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst locale = this.locale;\n\t\tconst dropdownView = createDropdown( locale );\n\n\t\tif ( !label ) {\n\t\t\t/**\n\t\t\t * A dropdown definition in the toolbar configuration is missing a text label.\n\t\t\t *\n\t\t\t * Without a label, the dropdown becomes inaccessible to users relying on assistive technologies.\n\t\t\t * Make sure the `label` property is set in your drop-down configuration:\n\t\t\t *\n\t\t\t * ```json\n \t\t\t * {\n \t\t\t * \tlabel: 'A human-readable label',\n\t\t\t * \ticon: '...',\n\t\t\t * \titems: [ ... ]\n \t\t\t * },\n\t\t\t * ```\n\t\t\t *\n\t\t\t * Learn more about {@link module:core/editor/editorconfig~EditorConfig#toolbar toolbar configuration}.\n\t\t\t *\n\t\t\t * @error toolbarview-nested-toolbar-dropdown-missing-label\n\t\t\t */\n\t\t\tlogWarning( 'toolbarview-nested-toolbar-dropdown-missing-label', definition );\n\t\t}\n\n\t\tdropdownView.class = 'ck-toolbar__nested-toolbar-dropdown';\n\t\tdropdownView.buttonView.set( {\n\t\t\tlabel,\n\t\t\ttooltip,\n\t\t\twithText: !!withText\n\t\t} );\n\n\t\t// Allow disabling icon by passing false.\n\t\tif ( icon !== false ) {\n\t\t\t// A pre-defined icon picked by name, SVG string, a fallback (default) icon.\n\t\t\tdropdownView.buttonView.icon = NESTED_TOOLBAR_ICONS[ icon! ] || icon || threeVerticalDots;\n\t\t}\n\t\t// If the icon is disabled, display the label automatically.\n\t\telse {\n\t\t\tdropdownView.buttonView.withText = true;\n\t\t}\n\n\t\taddToolbarToDropdown( dropdownView, () => (\n\t\t\tdropdownView.toolbarView!._buildItemsFromConfig( items, componentFactory, removeItems )\n\t\t) );\n\n\t\treturn dropdownView;\n\t}\n}\n\n/**\n * Fired when some toolbar {@link ~ToolbarView#items} were grouped or ungrouped as a result of some change\n * in the toolbar geometry.\n *\n * **Note**: This event is always fired **once** regardless of the number of items that were be\n * grouped or ungrouped at a time.\n *\n * **Note**: This event is fired only if the items grouping functionality was enabled in\n * the first place (see {@link module:ui/toolbar/toolbarview~ToolbarOptions#shouldGroupWhenFull}).\n *\n * @eventName ~ToolbarView#groupedItemsUpdate\n */\nexport type ToolbarViewGroupedItemsUpdateEvent = {\n\tname: 'groupedItemsUpdate';\n\targs: [];\n};\n\n/**\n * An inner block of the {@link module:ui/toolbar/toolbarview~ToolbarView} hosting its\n * {@link module:ui/toolbar/toolbarview~ToolbarView#items}.\n */\nclass ItemsView extends View {\n\t/**\n\t * A collection of items (buttons, dropdowns, etc.).\n\t */\n\tpublic readonly children: ViewCollection;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.children = this.createCollection();\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-toolbar__items'\n\t\t\t\t]\n\t\t\t},\n\t\t\tchildren: this.children\n\t\t} );\n\t}\n}\n\n/**\n * A toolbar behavior that makes it static and unresponsive to the changes of the environment.\n * At the same time, it also makes it possible to display a toolbar with a vertical layout\n * using the {@link module:ui/toolbar/toolbarview~ToolbarView#isVertical} property.\n */\nclass StaticLayout implements ToolbarBehavior {\n\t/**\n\t * Creates an instance of the {@link module:ui/toolbar/toolbarview~StaticLayout} toolbar\n\t * behavior.\n\t *\n\t * @param view An instance of the toolbar that this behavior is added to.\n\t */\n\tconstructor( view: ToolbarView ) {\n\t\tconst bind = view.bindTemplate;\n\n\t\t// Static toolbar can be vertical when needed.\n\t\tview.set( 'isVertical', false );\n\n\t\t// 1:1 passthrough binding, all ToolbarView#items are visible.\n\t\tview.itemsView.children.bindTo( view.items ).using( item => item );\n\n\t\t// 1:1 passthrough binding, all ToolbarView#items are focusable.\n\t\tview.focusables.bindTo( view.items ).using( item => item );\n\n\t\tview.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t// When vertical, the toolbar has an additional CSS class.\n\t\t\t\t\tbind.if( 'isVertical', 'ck-toolbar_vertical' )\n\t\t\t\t]\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic render(): void {}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic destroy(): void {}\n}\n\n/**\n * A toolbar behavior that makes the items respond to changes in the geometry.\n *\n * In a nutshell, it groups {@link module:ui/toolbar/toolbarview~ToolbarView#items}\n * that do not fit visually into a single row of the toolbar (due to limited space).\n * Items that do not fit are aggregated in a dropdown displayed at the end of the toolbar.\n *\n * ```\n *\t┌──────────────────────────────────────── ToolbarView ──────────────────────────────────────────┐\n *\t| ┌─────────────────────────────────────── #children ─────────────────────────────────────────┐ |\n *\t| | ┌─────── #itemsView ────────┐ ┌──────────────────────┐ ┌── #groupedItemsDropdown ───┐ | |\n *\t| | | #ungroupedItems | | ToolbarSeparatorView | | #groupedItems | | |\n *\t| | └──────────────────────────-┘ └──────────────────────┘ └────────────────────────────┘ | |\n *\t| | \\---------- only when toolbar items overflow -------/ | |\n *\t| └───────────────────────────────────────────────────────────────────────────────────────────┘ |\n *\t└───────────────────────────────────────────────────────────────────────────────────────────────┘\n * ```\n */\nclass DynamicGrouping implements ToolbarBehavior {\n\t/**\n\t * A toolbar view this behavior belongs to.\n\t */\n\tpublic readonly view: ToolbarView;\n\n\t/**\n\t * A collection of toolbar children.\n\t */\n\tpublic readonly viewChildren: ViewCollection;\n\n\t/**\n\t * A collection of focusable toolbar elements.\n\t */\n\tpublic readonly viewFocusables: ViewCollection;\n\n\t/**\n\t * A view containing toolbar items.\n\t */\n\tpublic readonly viewItemsView: ItemsView;\n\n\t/**\n\t * Toolbar focus tracker.\n\t */\n\tpublic readonly viewFocusTracker: FocusTracker;\n\n\t/**\n\t * Toolbar locale.\n\t */\n\tpublic readonly viewLocale: Locale;\n\n\t/**\n\t * A subset of toolbar {@link module:ui/toolbar/toolbarview~ToolbarView#items}.\n\t * Aggregates items that fit into a single row of the toolbar and were not {@link #groupedItems grouped}\n\t * into a {@link #groupedItemsDropdown dropdown}. Items of this collection are displayed in the\n\t * {@link module:ui/toolbar/toolbarview~ToolbarView#itemsView}.\n\t *\n\t * When none of the {@link module:ui/toolbar/toolbarview~ToolbarView#items} were grouped, it\n\t * matches the {@link module:ui/toolbar/toolbarview~ToolbarView#items} collection in size and order.\n\t */\n\tpublic readonly ungroupedItems: ViewCollection;\n\n\t/**\n\t * A subset of toolbar {@link module:ui/toolbar/toolbarview~ToolbarView#items}.\n\t * A collection of the toolbar items that do not fit into a single row of the toolbar.\n\t * Grouped items are displayed in a dedicated {@link #groupedItemsDropdown dropdown}.\n\t *\n\t * When none of the {@link module:ui/toolbar/toolbarview~ToolbarView#items} were grouped,\n\t * this collection is empty.\n\t */\n\tpublic readonly groupedItems: ViewCollection;\n\n\t/**\n\t * The dropdown that aggregates {@link #groupedItems grouped items} that do not fit into a single\n\t * row of the toolbar. It is displayed on demand as the last of\n\t * {@link module:ui/toolbar/toolbarview~ToolbarView#children toolbar children} and offers another\n\t * (nested) toolbar which displays items that would normally overflow.\n\t */\n\tpublic readonly groupedItemsDropdown: DropdownView;\n\n\t/**\n\t * An instance of the resize observer that helps dynamically determine the geometry of the toolbar\n\t * and manage items that do not fit into a single row.\n\t *\n\t * **Note:** Created in {@link #_enableGroupingOnResize}.\n\t *\n\t * @readonly\n\t */\n\tpublic resizeObserver: ResizeObserver | null = null;\n\n\t/**\n\t * A cached value of the horizontal padding style used by {@link #_updateGrouping}\n\t * to manage the {@link module:ui/toolbar/toolbarview~ToolbarView#items} that do not fit into\n\t * a single toolbar line. This value can be reused between updates because it is unlikely that\n\t * the padding will change and reusing `Window.getComputedStyle()` is expensive.\n\t *\n\t * @readonly\n\t */\n\tpublic cachedPadding: number | null = null;\n\n\t/**\n\t * A flag indicating that an items grouping update has been queued (e.g. due to the toolbar being visible)\n\t * and should be executed immediately the next time the toolbar shows up.\n\t *\n\t * @readonly\n\t */\n\tpublic shouldUpdateGroupingOnNextResize: boolean = false;\n\n\t/**\n\t * Toolbar element.\n\t *\n\t * @readonly\n\t */\n\tpublic viewElement: HTMLElement | null | undefined;\n\n\t/**\n\t * Creates an instance of the {@link module:ui/toolbar/toolbarview~DynamicGrouping} toolbar\n\t * behavior.\n\t *\n\t * @param view An instance of the toolbar that this behavior is added to.\n\t */\n\tconstructor( view: ToolbarView ) {\n\t\tthis.view = view;\n\n\t\tthis.viewChildren = view.children;\n\t\tthis.viewFocusables = view.focusables;\n\t\tthis.viewItemsView = view.itemsView;\n\t\tthis.viewFocusTracker = view.focusTracker;\n\t\tthis.viewLocale = view.locale;\n\n\t\tthis.ungroupedItems = view.createCollection();\n\t\tthis.groupedItems = view.createCollection();\n\t\tthis.groupedItemsDropdown = this._createGroupedItemsDropdown();\n\n\t\t// Only those items that were not grouped are visible to the user.\n\t\tview.itemsView.children.bindTo( this.ungroupedItems ).using( item => item );\n\n\t\t// Make sure all #items visible in the main space of the toolbar are \"focuscycleable\".\n\t\tthis.ungroupedItems.on<CollectionChangeEvent>( 'change', this._updateFocusCycleableItems.bind( this ) );\n\n\t\t// Make sure the #groupedItemsDropdown is also included in cycling when it appears.\n\t\tview.children.on<CollectionChangeEvent>( 'change', this._updateFocusCycleableItems.bind( this ) );\n\n\t\t// ToolbarView#items is dynamic. When an item is added or removed, it should be automatically\n\t\t// represented in either grouped or ungrouped items at the right index.\n\t\t// In other words #items == concat( #ungroupedItems, #groupedItems )\n\t\t// (in length and order).\n\t\tview.items.on<CollectionChangeEvent<View>>( 'change', ( evt, changeData ) => {\n\t\t\tconst index = changeData.index;\n\t\t\tconst added = Array.from( changeData.added );\n\n\t\t\t// Removing.\n\t\t\tfor ( const removedItem of changeData.removed ) {\n\t\t\t\tif ( index >= this.ungroupedItems.length ) {\n\t\t\t\t\tthis.groupedItems.remove( removedItem );\n\t\t\t\t} else {\n\t\t\t\t\tthis.ungroupedItems.remove( removedItem );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Adding.\n\t\t\tfor ( let currentIndex = index; currentIndex < index + added.length; currentIndex++ ) {\n\t\t\t\tconst addedItem = added[ currentIndex - index ];\n\n\t\t\t\tif ( currentIndex > this.ungroupedItems.length ) {\n\t\t\t\t\tthis.groupedItems.add( addedItem, currentIndex - this.ungroupedItems.length );\n\t\t\t\t} else {\n\t\t\t\t\tthis.ungroupedItems.add( addedItem, currentIndex );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// When new ungrouped items join in and land in #ungroupedItems, there's a chance it causes\n\t\t\t// the toolbar to overflow.\n\t\t\t// Consequently if removed from grouped or ungrouped items, there is a chance\n\t\t\t// some new space is available and we could do some ungrouping.\n\t\t\tthis._updateGrouping();\n\t\t} );\n\n\t\tview.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t// To group items dynamically, the toolbar needs a dedicated CSS class.\n\t\t\t\t\t'ck-toolbar_grouping'\n\t\t\t\t]\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Enables dynamic items grouping based on the dimensions of the toolbar.\n\t *\n\t * @param view An instance of the toolbar that this behavior is added to.\n\t */\n\tpublic render( view: ToolbarView ): void {\n\t\tthis.viewElement = view.element;\n\n\t\tthis._enableGroupingOnResize();\n\t\tthis._enableGroupingOnMaxWidthChange( view );\n\t}\n\n\t/**\n\t * Cleans up the internals used by this behavior.\n\t */\n\tpublic destroy(): void {\n\t\t// The dropdown may not be in ToolbarView#children at the moment of toolbar destruction\n\t\t// so let's make sure it's actually destroyed along with the toolbar.\n\t\tthis.groupedItemsDropdown.destroy();\n\n\t\tthis.resizeObserver!.destroy();\n\t}\n\n\t/**\n\t * When called, it will check if any of the {@link #ungroupedItems} do not fit into a single row of the toolbar,\n\t * and it will move them to the {@link #groupedItems} when it happens.\n\t *\n\t * At the same time, it will also check if there is enough space in the toolbar for the first of the\n\t * {@link #groupedItems} to be returned back to {@link #ungroupedItems} and still fit into a single row\n\t * without the toolbar wrapping.\n\t */\n\tprivate _updateGrouping() {\n\t\t// Do no groupingrelated geometry analysis when the toolbar is detached from visible DOM,\n\t\t// for instance before #render(), or after render but without a parent or a parent detached\n\t\t// from DOM. DOMRects won't work anyway and there will be tons of warning in the console and\n\t\t// nothing else. This happens, for instance, when the toolbar is detached from DOM and\n\t\t// some logic adds or removes its #items.\n\t\tif ( !this.viewElement!.ownerDocument.body.contains( this.viewElement! ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Do not update grouping when the element is invisible. Such toolbar has DOMRect filled with zeros\n\t\t// and that would cause all items to be grouped. Instead, queue the grouping so it runs next time\n\t\t// the toolbar is visible (the next ResizeObserver callback execution). This is handy because\n\t\t// the grouping could be caused by increasing the #maxWidth when the toolbar was invisible and the next\n\t\t// time it shows up, some items could actually be ungrouped (https://github.com/ckeditor/ckeditor5/issues/6575).\n\t\tif ( !isVisible( this.viewElement ) ) {\n\t\t\tthis.shouldUpdateGroupingOnNextResize = true;\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Remember how many items were initially grouped so at the it is possible to figure out if the number\n\t\t// of grouped items has changed. If the number has changed, geometry of the toolbar has also changed.\n\t\tconst initialGroupedItemsCount = this.groupedItems.length;\n\t\tlet wereItemsGrouped;\n\n\t\t// Group #items as long as some wrap to the next row. This will happen, for instance,\n\t\t// when the toolbar is getting narrow and there is not enough space to display all items in\n\t\t// a single row.\n\t\twhile ( this._areItemsOverflowing ) {\n\t\t\tthis._groupLastItem();\n\n\t\t\twereItemsGrouped = true;\n\t\t}\n\n\t\t// If none were grouped now but there were some items already grouped before,\n\t\t// then, what the hell, maybe let's see if some of them can be ungrouped. This happens when,\n\t\t// for instance, the toolbar is stretching and there's more space in it than before.\n\t\tif ( !wereItemsGrouped && this.groupedItems.length ) {\n\t\t\t// Ungroup items as long as none are overflowing or there are none to ungroup left.\n\t\t\twhile ( this.groupedItems.length && !this._areItemsOverflowing ) {\n\t\t\t\tthis._ungroupFirstItem();\n\t\t\t}\n\n\t\t\t// If the ungrouping ended up with some item wrapping to the next row,\n\t\t\t// put it back to the group toolbar (\"undo the last ungroup\"). We don't know whether\n\t\t\t// an item will wrap or not until we ungroup it (that's a DOM/CSS thing) so this\n\t\t\t// cleanup is vital for the algorithm.\n\t\t\tif ( this._areItemsOverflowing ) {\n\t\t\t\tthis._groupLastItem();\n\t\t\t}\n\t\t}\n\n\t\tif ( this.groupedItems.length !== initialGroupedItemsCount ) {\n\t\t\tthis.view.fire<ToolbarViewGroupedItemsUpdateEvent>( 'groupedItemsUpdate' );\n\t\t}\n\t}\n\n\t/**\n\t * Returns `true` when {@link module:ui/toolbar/toolbarview~ToolbarView#element} children visually overflow,\n\t * for instance if the toolbar is narrower than its members. Returns `false` otherwise.\n\t */\n\tprivate get _areItemsOverflowing() {\n\t\t// An empty toolbar cannot overflow.\n\t\tif ( !this.ungroupedItems.length ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst element = this.viewElement;\n\t\tconst uiLanguageDirection = this.viewLocale.uiLanguageDirection;\n\t\tconst lastChildRect = new Rect( element!.lastChild as any );\n\t\tconst toolbarRect = new Rect( element! );\n\n\t\tif ( !this.cachedPadding ) {\n\t\t\tconst computedStyle = global.window.getComputedStyle( element! );\n\t\t\tconst paddingProperty = uiLanguageDirection === 'ltr' ? 'paddingRight' : 'paddingLeft';\n\n\t\t\t// parseInt() is essential because of quirky floating point numbers logic and DOM.\n\t\t\t// If the padding turned out too big because of that, the grouped items dropdown would\n\t\t\t// always look (from the Rect perspective) like it overflows (while it's not).\n\t\t\tthis.cachedPadding = Number.parseInt( computedStyle[ paddingProperty ] );\n\t\t}\n\n\t\tif ( uiLanguageDirection === 'ltr' ) {\n\t\t\treturn lastChildRect.right > toolbarRect.right - this.cachedPadding;\n\t\t} else {\n\t\t\treturn lastChildRect.left < toolbarRect.left + this.cachedPadding;\n\t\t}\n\t}\n\n\t/**\n\t * Enables the functionality that prevents {@link #ungroupedItems} from overflowing (wrapping to the next row)\n\t * upon resize when there is little space available. Instead, the toolbar items are moved to the\n\t * {@link #groupedItems} collection and displayed in a dropdown at the end of the row (which has its own nested toolbar).\n\t *\n\t * When called, the toolbar will automatically analyze the location of its {@link #ungroupedItems} and \"group\"\n\t * them in the dropdown if necessary. It will also observe the browser window for size changes in\n\t * the future and respond to them by grouping more items or reverting already grouped back, depending\n\t * on the visual space available.\n\t */\n\tprivate _enableGroupingOnResize() {\n\t\tlet previousWidth: number | undefined;\n\n\t\t// TODO: Consider debounce.\n\t\tthis.resizeObserver = new ResizeObserver( this.viewElement!, entry => {\n\t\t\tif ( !previousWidth || previousWidth !== entry.contentRect.width || this.shouldUpdateGroupingOnNextResize ) {\n\t\t\t\tthis.shouldUpdateGroupingOnNextResize = false;\n\n\t\t\t\tthis._updateGrouping();\n\n\t\t\t\tpreviousWidth = entry.contentRect.width;\n\t\t\t}\n\t\t} );\n\n\t\tthis._updateGrouping();\n\t}\n\n\t/**\n\t * Enables the grouping functionality, just like {@link #_enableGroupingOnResize} but the difference is that\n\t * it listens to the changes of {@link module:ui/toolbar/toolbarview~ToolbarView#maxWidth} instead.\n\t */\n\tprivate _enableGroupingOnMaxWidthChange( view: View ) {\n\t\tview.on<ObservableChangeEvent>( 'change:maxWidth', () => {\n\t\t\tthis._updateGrouping();\n\t\t} );\n\t}\n\n\t/**\n\t * When called, it will remove the last item from {@link #ungroupedItems} and move it back\n\t * to the {@link #groupedItems} collection.\n\t *\n\t * The opposite of {@link #_ungroupFirstItem}.\n\t */\n\tprivate _groupLastItem() {\n\t\tif ( !this.groupedItems.length ) {\n\t\t\tthis.viewChildren.add( new ToolbarSeparatorView() );\n\t\t\tthis.viewChildren.add( this.groupedItemsDropdown );\n\t\t\tthis.viewFocusTracker.add( this.groupedItemsDropdown.element! );\n\t\t}\n\n\t\tthis.groupedItems.add( this.ungroupedItems.remove( this.ungroupedItems.last! ), 0 );\n\t}\n\n\t/**\n\t * Moves the very first item belonging to {@link #groupedItems} back\n\t * to the {@link #ungroupedItems} collection.\n\t *\n\t * The opposite of {@link #_groupLastItem}.\n\t */\n\tprivate _ungroupFirstItem() {\n\t\tthis.ungroupedItems.add( this.groupedItems.remove( this.groupedItems.first! ) );\n\n\t\tif ( !this.groupedItems.length ) {\n\t\t\tthis.viewChildren.remove( this.groupedItemsDropdown );\n\t\t\tthis.viewChildren.remove( this.viewChildren.last! );\n\t\t\tthis.viewFocusTracker.remove( this.groupedItemsDropdown.element! );\n\t\t}\n\t}\n\n\t/**\n\t * Creates the {@link #groupedItemsDropdown} that hosts the members of the {@link #groupedItems}\n\t * collection when there is not enough space in the toolbar to display all items in a single row.\n\t */\n\tprivate _createGroupedItemsDropdown() {\n\t\tconst locale = this.viewLocale;\n\t\tconst t = locale.t;\n\t\tconst dropdown = createDropdown( locale );\n\n\t\tdropdown.class = 'ck-toolbar__grouped-dropdown';\n\n\t\t// Make sure the dropdown never sticks out to the left/right. It should be under the main toolbar.\n\t\t// (https://github.com/ckeditor/ckeditor5/issues/5608)\n\t\tdropdown.panelPosition = locale.uiLanguageDirection === 'ltr' ? 'sw' : 'se';\n\n\t\taddToolbarToDropdown( dropdown, this.groupedItems );\n\n\t\tdropdown.buttonView.set( {\n\t\t\tlabel: t( 'Show more items' ),\n\t\t\ttooltip: true,\n\t\t\ttooltipPosition: locale.uiLanguageDirection === 'rtl' ? 'se' : 'sw',\n\t\t\ticon: threeVerticalDots\n\t\t} );\n\n\t\treturn dropdown;\n\t}\n\n\t/**\n\t * Updates the {@link module:ui/toolbar/toolbarview~ToolbarView#focusables focuscycleable items}\n\t * collection so it represents the uptodate state of the UI from the perspective of the user.\n\t *\n\t * For instance, the {@link #groupedItemsDropdown} can show up and hide but when it is visible,\n\t * it must be subject to focus cycling in the toolbar.\n\t *\n\t * See the {@link module:ui/toolbar/toolbarview~ToolbarView#focusables collection} documentation\n\t * to learn more about the purpose of this method.\n\t */\n\tprivate _updateFocusCycleableItems() {\n\t\tthis.viewFocusables.clear();\n\n\t\tthis.ungroupedItems.map( item => {\n\t\t\tthis.viewFocusables.add( item );\n\t\t} );\n\n\t\tif ( this.groupedItems.length ) {\n\t\t\tthis.viewFocusables.add( this.groupedItemsDropdown );\n\t\t}\n\t}\n}\n\n/**\n * Options passed to the {@link module:ui/toolbar/toolbarview~ToolbarView#constructor} of the toolbar.\n */\nexport interface ToolbarOptions {\n\n\t/**\n\t * When set to `true`, the toolbar will automatically group {@link module:ui/toolbar/toolbarview~ToolbarView#items} that\n\t * would normally wrap to the next line when there is not enough space to display them in a single row, for\n\t * instance, if the parent container of the toolbar is narrow. For toolbars in absolutely positioned containers\n\t * without width restrictions also the {@link module:ui/toolbar/toolbarview~ToolbarOptions#isFloating} option is required to be `true`.\n\t *\n\t * See also: {@link module:ui/toolbar/toolbarview~ToolbarView#maxWidth}.\n\t */\n\tshouldGroupWhenFull?: boolean;\n\n\t/**\n\t * This option should be enabled for toolbars in absolutely positioned containers without width restrictions\n\t * to enable automatic {@link module:ui/toolbar/toolbarview~ToolbarView#items} grouping.\n\t * When this option is set to `true`, the items will stop wrapping to the next line\n\t * and together with {@link module:ui/toolbar/toolbarview~ToolbarOptions#shouldGroupWhenFull},\n\t * this will allow grouping them when there is not enough space in a single row.\n\t */\n\tisFloating?: boolean;\n}\n\n/**\n * A class interface defining the behavior of the {@link module:ui/toolbar/toolbarview~ToolbarView}.\n *\n * Toolbar behaviors extend its look and functionality and have an impact on the\n * {@link module:ui/toolbar/toolbarview~ToolbarView#element} template or\n * {@link module:ui/toolbar/toolbarview~ToolbarView#render rendering}. They can be enabled\n * conditionally, e.g. depending on the configuration of the toolbar.\n */\nexport interface ToolbarBehavior {\n\n\t/**\n\t * A method called after the toolbar has been {@link module:ui/toolbar/toolbarview~ToolbarView#render rendered}.\n\t * It can be used to, for example, customize the behavior of the toolbar when its\n\t * {@link module:ui/toolbar/toolbarview~ToolbarView#element} is available.\n\t *\n\t * @param view An instance of the toolbar being rendered.\n\t */\n\trender( view: ToolbarView ): void;\n\n\t/**\n\t * A method called after the toolbar has been {@link module:ui/toolbar/toolbarview~ToolbarView#destroy destroyed}.\n\t * It allows cleaning up after the toolbar behavior, for instance, this is the right place to detach\n\t * event listeners, free up references, etc.\n\t */\n\tdestroy(): void;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/bindings/preventdefault\n */\n\nimport type { ListenerBinding } from '../template';\nimport type View from '../view';\n\n/**\n * A helper which executes a native `Event.preventDefault()` if the target of an event equals the\n * {@link module:ui/view~View#element element of the view}. It shortens the definition of a\n * {@link module:ui/view~View#template template}.\n *\n * ```ts\n * // In a class extending View.\n * import preventDefault from '@ckeditor/ckeditor5-ui/src/bindings/preventdefault';\n *\n * // ...\n *\n * this.setTemplate( {\n * \ttag: 'div',\n *\n * \ton: {\n * \t\t// Prevent the default mousedown action on this view.\n * \t\tmousedown: preventDefault( this )\n * \t}\n * } );\n * ```\n *\n * @param view View instance that defines the template.\n */\nexport default function preventDefault( view: View ): ListenerBinding {\n\treturn view.bindTemplate.to( evt => {\n\t\tif ( evt.target === view.element ) {\n\t\t\tevt.preventDefault();\n\t\t}\n\t} );\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./list.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/list/listview\n */\n\nimport View from '../view';\nimport FocusCycler from '../focuscycler';\n\nimport type ListItemView from './listitemview';\nimport type DropdownPanelFocusable from '../dropdown/dropdownpanelfocusable';\nimport type ViewCollection from '../viewcollection';\n\nimport {\n\tFocusTracker,\n\tKeystrokeHandler,\n\ttype CollectionAddEvent,\n\ttype CollectionRemoveEvent,\n\ttype Locale\n} from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/list/list.css';\n\n/**\n * The list view class.\n */\nexport default class ListView extends View<HTMLUListElement> implements DropdownPanelFocusable {\n\t/**\n\t * Collection of the child list views.\n\t */\n\tpublic readonly items: ViewCollection;\n\n\t/**\n\t * Tracks information about DOM focus in the list.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * Instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}.\n\t */\n\tpublic readonly keystrokes: KeystrokeHandler;\n\n\t/**\n\t * Label used by assistive technologies to describe this list element.\n\t *\n\t * @observable\n\t */\n\tdeclare public ariaLabel: string | undefined;\n\n\t/**\n\t * The property reflected by the `role` DOM attribute to be used by assistive technologies.\n\t *\n\t * @observable\n\t */\n\tdeclare public role: string | undefined;\n\n\t/**\n\t * Helps cycling over focusable {@link #items} in the list.\n\t */\n\tprivate readonly _focusCycler: FocusCycler;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.items = this.createCollection();\n\t\tthis.focusTracker = new FocusTracker();\n\t\tthis.keystrokes = new KeystrokeHandler();\n\n\t\tthis._focusCycler = new FocusCycler( {\n\t\t\tfocusables: this.items,\n\t\t\tfocusTracker: this.focusTracker,\n\t\t\tkeystrokeHandler: this.keystrokes,\n\t\t\tactions: {\n\t\t\t\t// Navigate list items backwards using the arrowup key.\n\t\t\t\tfocusPrevious: 'arrowup',\n\n\t\t\t\t// Navigate toolbar items forwards using the arrowdown key.\n\t\t\t\tfocusNext: 'arrowdown'\n\t\t\t}\n\t\t} );\n\n\t\tthis.set( 'ariaLabel', undefined );\n\t\tthis.set( 'role', undefined );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'ul',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-reset',\n\t\t\t\t\t'ck-list'\n\t\t\t\t],\n\t\t\t\trole: bind.to( 'role' ),\n\t\t\t\t'aria-label': bind.to( 'ariaLabel' )\n\t\t\t},\n\n\t\t\tchildren: this.items\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\t// Items added before rendering should be known to the #focusTracker.\n\t\tfor ( const item of this.items ) {\n\t\t\tthis.focusTracker.add( item.element! );\n\t\t}\n\n\t\tthis.items.on<CollectionAddEvent<ListItemView>>( 'add', ( evt, item ) => {\n\t\t\tthis.focusTracker.add( item.element! );\n\t\t} );\n\n\t\tthis.items.on<CollectionRemoveEvent<ListItemView>>( 'remove', ( evt, item ) => {\n\t\t\tthis.focusTracker.remove( item.element! );\n\t\t} );\n\n\t\t// Start listening for the keystrokes coming from #element.\n\t\tthis.keystrokes.listenTo( this.element! );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.focusTracker.destroy();\n\t\tthis.keystrokes.destroy();\n\t}\n\n\t/**\n\t * Focuses the first focusable in {@link #items}.\n\t */\n\tpublic focus(): void {\n\t\tthis._focusCycler.focusFirst();\n\t}\n\n\t/**\n\t * Focuses the last focusable in {@link #items}.\n\t */\n\tpublic focusLast(): void {\n\t\tthis._focusCycler.focusLast();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/list/listitemview\n */\n\nimport View from '../view';\n\nimport type { FocusableView } from '../focuscycler';\nimport type ViewCollection from '../viewcollection';\n\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The list item view class.\n */\nexport default class ListItemView extends View {\n\t/**\n\t * Collection of the child views inside of the list item {@link #element}.\n\t */\n\tpublic readonly children: ViewCollection;\n\n\t/**\n\t * Controls whether the item view is visible. Visible by default, list items are hidden\n\t * using a CSS class.\n\t *\n\t * @observable\n\t * @default true\n\t */\n\tdeclare public isVisible: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.set( 'isVisible', true );\n\n\t\tthis.children = this.createCollection();\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'li',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-list__item',\n\t\t\t\t\tbind.if( 'isVisible', 'ck-hidden', value => !value )\n\t\t\t\t],\n\t\t\t\trole: 'presentation'\n\t\t\t},\n\n\t\t\tchildren: this.children\n\t\t} );\n\t}\n\n\t/**\n\t * Focuses the list item.\n\t */\n\tpublic focus(): void {\n\t\t( this.children.first as FocusableView ).focus();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/list/listseparatorview\n */\n\nimport View from '../view';\n\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The list separator view class.\n */\nexport default class ListSeparatorView extends View {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'li',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-list__separator'\n\t\t\t\t]\n\t\t\t}\n\t\t} );\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./splitbutton.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/dropdown/button/splitbuttonview\n */\n\nimport View from '../../view';\nimport ButtonView from '../../button/buttonview';\nimport type ViewCollection from '../../viewcollection';\nimport type Button from '../../button/button';\nimport type DropdownButton from './dropdownbutton';\n\nimport {\n\tKeystrokeHandler,\n\tFocusTracker,\n\ttype Locale\n} from '@ckeditor/ckeditor5-utils';\n\nimport dropdownArrowIcon from '../../../theme/icons/dropdown-arrow.svg';\n\nimport '../../../theme/components/dropdown/splitbutton.css';\n\n/**\n * The split button view class.\n *\n * ```ts\n * const view = new SplitButtonView();\n *\n * view.set( {\n * \tlabel: 'A button',\n * \tkeystroke: 'Ctrl+B',\n * \ttooltip: true\n * } );\n *\n * view.render();\n *\n * document.body.append( view.element );\n * ```\n *\n * Also see the {@link module:ui/dropdown/utils~createDropdown `createDropdown()` util}.\n */\nexport default class SplitButtonView extends View<HTMLDivElement> implements DropdownButton {\n\t/**\n\t * Collection of the child views inside of the split button {@link #element}.\n\t */\n\tpublic readonly children: ViewCollection;\n\n\t/**\n\t * A main button of split button.\n\t */\n\tpublic readonly actionView: ButtonView;\n\n\t/**\n\t * A secondary button of split button that opens dropdown.\n\t */\n\tpublic readonly arrowView: ButtonView;\n\n\t/**\n\t * Instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}. It manages\n\t * keystrokes of the split button:\n\t *\n\t * * <kbd>▶</kbd> moves focus to arrow view when action view is focused,\n\t * * <kbd>◀</kbd> moves focus to action view when arrow view is focused.\n\t */\n\tpublic readonly keystrokes: KeystrokeHandler;\n\n\t/**\n\t * Tracks information about DOM focus in the dropdown.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public label: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public keystroke: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public tooltip: Button[ 'tooltip' ];\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public tooltipPosition: Button[ 'tooltipPosition' ];\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public type: Button[ 'type' ];\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public isOn: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public isEnabled: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public isVisible: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public isToggleable: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public withText: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public withKeystroke: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public icon: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public tabindex: number;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public class: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public labelStyle: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public role: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public ariaChecked: boolean | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public ariaLabel?: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tdeclare public ariaLabelledBy: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\t// Implement the Button interface.\n\t\tthis.set( 'class', undefined );\n\t\tthis.set( 'labelStyle', undefined );\n\t\tthis.set( 'icon', undefined );\n\t\tthis.set( 'isEnabled', true );\n\t\tthis.set( 'isOn', false );\n\t\tthis.set( 'isToggleable', false );\n\t\tthis.set( 'isVisible', true );\n\t\tthis.set( 'keystroke', undefined );\n\t\tthis.set( 'withKeystroke', false );\n\t\tthis.set( 'label', undefined );\n\t\tthis.set( 'tabindex', -1 );\n\t\tthis.set( 'tooltip', false );\n\t\tthis.set( 'tooltipPosition', 's' );\n\t\tthis.set( 'type', 'button' );\n\t\tthis.set( 'withText', false );\n\n\t\tthis.children = this.createCollection();\n\t\tthis.actionView = this._createActionView();\n\t\tthis.arrowView = this._createArrowView();\n\t\tthis.keystrokes = new KeystrokeHandler();\n\t\tthis.focusTracker = new FocusTracker();\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-splitbutton',\n\t\t\t\t\tbind.to( 'class' ),\n\t\t\t\t\tbind.if( 'isVisible', 'ck-hidden', value => !value ),\n\t\t\t\t\tthis.arrowView.bindTemplate.if( 'isOn', 'ck-splitbutton_open' )\n\t\t\t\t]\n\t\t\t},\n\n\t\t\tchildren: this.children\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.children.add( this.actionView );\n\t\tthis.children.add( this.arrowView );\n\n\t\tthis.focusTracker.add( this.actionView.element! );\n\t\tthis.focusTracker.add( this.arrowView.element! );\n\n\t\tthis.keystrokes.listenTo( this.element! );\n\n\t\t// Overrides toolbar focus cycling behavior.\n\t\tthis.keystrokes.set( 'arrowright', ( evt, cancel ) => {\n\t\t\tif ( this.focusTracker.focusedElement === this.actionView.element ) {\n\t\t\t\tthis.arrowView.focus();\n\n\t\t\t\tcancel();\n\t\t\t}\n\t\t} );\n\n\t\t// Overrides toolbar focus cycling behavior.\n\t\tthis.keystrokes.set( 'arrowleft', ( evt, cancel ) => {\n\t\t\tif ( this.focusTracker.focusedElement === this.arrowView.element ) {\n\t\t\t\tthis.actionView.focus();\n\n\t\t\t\tcancel();\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.focusTracker.destroy();\n\t\tthis.keystrokes.destroy();\n\t}\n\n\t/**\n\t * Focuses the {@link module:ui/button/buttonview~ButtonView#element} of the action part of split button.\n\t */\n\tpublic focus(): void {\n\t\tthis.actionView.focus();\n\t}\n\n\t/**\n\t * Creates a {@link module:ui/button/buttonview~ButtonView} instance as {@link #actionView} and binds it with main split button\n\t * attributes.\n\t */\n\tprivate _createActionView() {\n\t\tconst actionView = new ButtonView();\n\n\t\tactionView.bind(\n\t\t\t'icon',\n\t\t\t'isEnabled',\n\t\t\t'isOn',\n\t\t\t'isToggleable',\n\t\t\t'keystroke',\n\t\t\t'label',\n\t\t\t'tabindex',\n\t\t\t'tooltip',\n\t\t\t'tooltipPosition',\n\t\t\t'type',\n\t\t\t'withText'\n\t\t).to( this );\n\n\t\tactionView.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: 'ck-splitbutton__action'\n\t\t\t}\n\t\t} );\n\n\t\tactionView.delegate( 'execute' ).to( this );\n\n\t\treturn actionView;\n\t}\n\n\t/**\n\t * Creates a {@link module:ui/button/buttonview~ButtonView} instance as {@link #arrowView} and binds it with main split button\n\t * attributes.\n\t */\n\tprivate _createArrowView() {\n\t\tconst arrowView = new ButtonView();\n\t\tconst bind = arrowView.bindTemplate;\n\n\t\tarrowView.icon = dropdownArrowIcon;\n\n\t\tarrowView.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck-splitbutton__arrow'\n\t\t\t\t],\n\t\t\t\t'data-cke-tooltip-disabled': bind.to( 'isOn' ),\n\t\t\t\t'aria-haspopup': true,\n\t\t\t\t'aria-expanded': bind.to( 'isOn', value => String( value ) )\n\t\t\t}\n\t\t} );\n\n\t\tarrowView.bind( 'isEnabled' ).to( this );\n\t\tarrowView.bind( 'label' ).to( this );\n\t\tarrowView.bind( 'tooltip' ).to( this );\n\n\t\tarrowView.delegate( 'execute' ).to( this, 'open' );\n\n\t\treturn arrowView;\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./toolbardropdown.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./listdropdown.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/dropdown/utils\n */\n\nimport DropdownPanelView from './dropdownpanelview';\nimport DropdownView from './dropdownview';\nimport DropdownButtonView from './button/dropdownbuttonview';\nimport ToolbarView from '../toolbar/toolbarview';\nimport ListView from '../list/listview';\nimport ListItemView from '../list/listitemview';\nimport ListSeparatorView from '../list/listseparatorview';\nimport ButtonView from '../button/buttonview';\nimport SplitButtonView from './button/splitbuttonview';\nimport SwitchButtonView from '../button/switchbuttonview';\nimport ViewCollection from '../viewcollection';\n\nimport clickOutsideHandler from '../bindings/clickoutsidehandler';\n\nimport type { default as View, UIViewRenderEvent } from '../view';\nimport type { ButtonExecuteEvent } from '../button/button';\nimport type Model from '../model';\nimport type DropdownButton from './button/dropdownbutton';\nimport type { FocusableView } from '../focuscycler';\nimport type { FalsyValue } from '../template';\n\nimport {\n\tglobal,\n\tpriorities,\n\tlogWarning,\n\ttype Collection,\n\ttype Locale,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/dropdown/toolbardropdown.css';\nimport '../../theme/components/dropdown/listdropdown.css';\n\n/**\n * A helper for creating dropdowns. It creates an instance of a {@link module:ui/dropdown/dropdownview~DropdownView dropdown},\n * with a {@link module:ui/dropdown/button/dropdownbutton~DropdownButton button},\n * {@link module:ui/dropdown/dropdownpanelview~DropdownPanelView panel} and all standard dropdown's behaviors.\n *\n * # Creating dropdowns\n *\n * By default, the default {@link module:ui/dropdown/button/dropdownbuttonview~DropdownButtonView} class is used as\n * definition of the button:\n *\n * ```ts\n * const dropdown = createDropdown( model );\n *\n * // Configure dropdown's button properties:\n * dropdown.buttonView.set( {\n * \tlabel: 'A dropdown',\n * \twithText: true\n * } );\n *\n * dropdown.render();\n *\n * // Will render a dropdown labeled \"A dropdown\" with an empty panel.\n * document.body.appendChild( dropdown.element );\n * ```\n *\n * You can also provide other button views (they need to implement the\n * {@link module:ui/dropdown/button/dropdownbutton~DropdownButton} interface). For instance, you can use\n * {@link module:ui/dropdown/button/splitbuttonview~SplitButtonView} to create a dropdown with a split button.\n *\n * ```ts\n * const dropdown = createDropdown( locale, SplitButtonView );\n *\n * // Configure dropdown's button properties:\n * dropdown.buttonView.set( {\n * \tlabel: 'A dropdown',\n * \twithText: true\n * } );\n *\n * dropdown.buttonView.on( 'execute', () => {\n * \t// Add the behavior of the \"action part\" of the split button.\n * \t// Split button consists of the \"action part\" and \"arrow part\".\n * \t// The arrow opens the dropdown while the action part can have some other behavior.\n * } );\n *\n * dropdown.render();\n *\n * // Will render a dropdown labeled \"A dropdown\" with an empty panel.\n * document.body.appendChild( dropdown.element );\n * ```\n *\n * # Adding content to the dropdown's panel\n *\n * The content of the panel can be inserted directly into the `dropdown.panelView.element`:\n *\n * ```ts\n * dropdown.panelView.element.textContent = 'Content of the panel';\n * ```\n *\n * However, most of the time you will want to add there either a {@link module:ui/list/listview~ListView list of options}\n * or a list of buttons (i.e. a {@link module:ui/toolbar/toolbarview~ToolbarView toolbar}).\n * To simplify the task, you can use, respectively, {@link module:ui/dropdown/utils~addListToDropdown} or\n * {@link module:ui/dropdown/utils~addToolbarToDropdown} utils.\n *\n * @param locale The locale instance.\n * @param ButtonClass The dropdown button view class. Needs to implement the\n * {@link module:ui/dropdown/button/dropdownbutton~DropdownButton} interface.\n * @returns The dropdown view instance.\n */\nexport function createDropdown(\n\tlocale: Locale | undefined,\n\tButtonClass: new ( locale?: Locale ) => DropdownButton & FocusableView = DropdownButtonView\n): DropdownView {\n\tconst buttonView = new ButtonClass( locale );\n\n\tconst panelView = new DropdownPanelView( locale );\n\tconst dropdownView = new DropdownView( locale, buttonView, panelView );\n\n\tbuttonView.bind( 'isEnabled' ).to( dropdownView );\n\n\tif ( buttonView instanceof SplitButtonView ) {\n\t\tbuttonView.arrowView.bind( 'isOn' ).to( dropdownView, 'isOpen' );\n\t} else {\n\t\tbuttonView.bind( 'isOn' ).to( dropdownView, 'isOpen' );\n\t}\n\n\taddDefaultBehavior( dropdownView );\n\n\treturn dropdownView;\n}\n\n/**\n * Adds an instance of {@link module:ui/toolbar/toolbarview~ToolbarView} to a dropdown.\n *\n * ```ts\n * const buttonsCreator = () => {\n * \tconst buttons = [];\n *\n * \t// Either create a new ButtonView instance or create existing.\n * \tbuttons.push( new ButtonView() );\n * \tbuttons.push( editor.ui.componentFactory.create( 'someButton' ) );\n * };\n *\n * const dropdown = createDropdown( locale );\n *\n * addToolbarToDropdown( dropdown, buttonsCreator, { isVertical: true } );\n *\n * // Will render a vertical button dropdown labeled \"A button dropdown\"\n * // with a button group in the panel containing two buttons.\n * // Buttons inside the dropdown will be created on first dropdown panel open.\n * dropdown.render()\n * document.body.appendChild( dropdown.element );\n * ```\n *\n * **Note:** To improve the accessibility, you can tell the dropdown to focus the first active button of the toolbar when the dropdown\n * {@link module:ui/dropdown/dropdownview~DropdownView#isOpen gets open}. See the documentation of `options` to learn more.\n *\n * **Note:** Toolbar view will be created on first open of the dropdown.\n *\n * See {@link module:ui/dropdown/utils~createDropdown} and {@link module:ui/toolbar/toolbarview~ToolbarView}.\n *\n * @param dropdownView A dropdown instance to which `ToolbarView` will be added.\n * @param options.enableActiveItemFocusOnDropdownOpen When set `true`, the focus will automatically move to the first\n * active {@link module:ui/toolbar/toolbarview~ToolbarView#items item} of the toolbar upon\n * {@link module:ui/dropdown/dropdownview~DropdownView#isOpen opening} the dropdown. Active items are those with the `isOn` property set\n * `true` (for instance {@link module:ui/button/buttonview~ButtonView buttons}). If no active items is found, the toolbar will be focused\n * as a whole resulting in the focus moving to its first focusable item (default behavior of\n * {@link module:ui/dropdown/dropdownview~DropdownView}).\n * @param options.ariaLabel Label used by assistive technologies to describe toolbar element.\n * @param options.maxWidth The maximum width of the toolbar element.\n * Details: {@link module:ui/toolbar/toolbarview~ToolbarView#maxWidth}.\n * @param options.class An additional CSS class added to the toolbar element.\n * @param options.isCompact When set true, makes the toolbar look compact with toolbar element.\n * @param options.isVertical Controls the orientation of toolbar items.\n */\nexport function addToolbarToDropdown(\n\tdropdownView: DropdownView,\n\tbuttonsOrCallback: Array<View> | ViewCollection | ( () => Array<View> | ViewCollection ),\n\toptions: {\n\t\tenableActiveItemFocusOnDropdownOpen?: boolean;\n\t\tariaLabel?: string;\n\t\tmaxWidth?: string;\n\t\tclass?: string;\n\t\tisCompact?: boolean;\n\t\tisVertical?: boolean;\n\t} = {}\n): void {\n\tdropdownView.extendTemplate( {\n\t\tattributes: {\n\t\t\tclass: [ 'ck-toolbar-dropdown' ]\n\t\t}\n\t} );\n\n\tif ( dropdownView.isOpen ) {\n\t\taddToolbarToOpenDropdown( dropdownView, buttonsOrCallback, options );\n\t} else {\n\t\tdropdownView.once(\n\t\t\t'change:isOpen',\n\t\t\t() => addToolbarToOpenDropdown( dropdownView, buttonsOrCallback, options ),\n\t\t\t{ priority: 'highest' }\n\t\t);\n\t}\n\n\tif ( options.enableActiveItemFocusOnDropdownOpen ) {\n\t\t// Accessibility: Focus the first active button in the toolbar when the dropdown gets open.\n\t\tfocusChildOnDropdownOpen( dropdownView, () => dropdownView.toolbarView!.items.find( ( item: any ) => item.isOn ) );\n\t}\n}\n\n/**\n * Adds an instance of {@link module:ui/toolbar/toolbarview~ToolbarView} to a dropdown.\n */\nfunction addToolbarToOpenDropdown(\n\tdropdownView: DropdownView,\n\tbuttonsOrCallback: Array<View> | ViewCollection | ( () => Array<View> | ViewCollection ),\n\toptions: {\n\t\tariaLabel?: string;\n\t\tmaxWidth?: string;\n\t\tclass?: string;\n\t\tisCompact?: boolean;\n\t\tisVertical?: boolean;\n\t}\n): void {\n\tconst locale = dropdownView.locale!;\n\tconst t = locale.t;\n\n\tconst toolbarView = dropdownView.toolbarView = new ToolbarView( locale );\n\tconst buttons = typeof buttonsOrCallback == 'function' ? buttonsOrCallback() : buttonsOrCallback;\n\n\ttoolbarView.ariaLabel = options.ariaLabel || t( 'Dropdown toolbar' );\n\n\tif ( options.maxWidth ) {\n\t\ttoolbarView.maxWidth = options.maxWidth;\n\t}\n\n\tif ( options.class ) {\n\t\ttoolbarView.class = options.class;\n\t}\n\n\tif ( options.isCompact ) {\n\t\ttoolbarView.isCompact = options.isCompact;\n\t}\n\n\tif ( options.isVertical ) {\n\t\ttoolbarView.isVertical = true;\n\t}\n\n\tif ( buttons instanceof ViewCollection ) {\n\t\ttoolbarView.items.bindTo( buttons ).using( item => item );\n\t} else {\n\t\ttoolbarView.items.addMany( buttons );\n\t}\n\n\tdropdownView.panelView.children.add( toolbarView );\n\ttoolbarView.items.delegate( 'execute' ).to( dropdownView );\n}\n\n/**\n * Adds an instance of {@link module:ui/list/listview~ListView} to a dropdown.\n *\n * ```ts\n * const items = new Collection();\n *\n * items.add( {\n * \ttype: 'button',\n * \tmodel: new Model( {\n * \t\twithText: true,\n * \t\tlabel: 'First item',\n * \t\tlabelStyle: 'color: red'\n * \t} )\n * } );\n *\n * items.add( {\n * \t type: 'button',\n * \t model: new Model( {\n * \t\twithText: true,\n * \t\tlabel: 'Second item',\n * \t\tlabelStyle: 'color: green',\n * \t\tclass: 'foo'\n * \t} )\n * } );\n *\n * const dropdown = createDropdown( locale );\n *\n * addListToDropdown( dropdown, items );\n *\n * // Will render a dropdown with a list in the panel containing two items.\n * dropdown.render()\n * document.body.appendChild( dropdown.element );\n * ```\n *\n * The `items` collection passed to this methods controls the presence and attributes of respective\n * {@link module:ui/list/listitemview~ListItemView list items}.\n *\n * **Note:** To improve the accessibility, when a list is added to the dropdown using this helper the dropdown will automatically attempt\n * to focus the first active item (a host to a {@link module:ui/button/buttonview~ButtonView} with\n * {@link module:ui/button/buttonview~ButtonView#isOn} set `true`) or the very first item when none are active.\n *\n * **Note:** List view will be created on first open of the dropdown.\n *\n * See {@link module:ui/dropdown/utils~createDropdown} and {@link module:list/list~List}.\n *\n * @param dropdownView A dropdown instance to which `ListVIew` will be added.\n * @param itemsOrCallback A collection of the list item definitions or a callback returning a list item definitions to populate the list.\n * @param options.ariaLabel Label used by assistive technologies to describe list element.\n * @param options.role Will be reflected by the `role` DOM attribute in `ListVIew` and used by assistive technologies.\n */\nexport function addListToDropdown(\n\tdropdownView: DropdownView,\n\titemsOrCallback: Collection<ListDropdownItemDefinition> | ( () => Collection<ListDropdownItemDefinition> ),\n\toptions: {\n\t\tariaLabel?: string;\n\t\trole?: string;\n\t} = {}\n): void {\n\tif ( dropdownView.isOpen ) {\n\t\taddListToOpenDropdown( dropdownView, itemsOrCallback, options );\n\t} else {\n\t\tdropdownView.once(\n\t\t\t'change:isOpen',\n\t\t\t() => addListToOpenDropdown( dropdownView, itemsOrCallback, options ),\n\t\t\t{ priority: 'highest' }\n\t\t);\n\t}\n\n\t// Accessibility: Focus the first active button in the list when the dropdown gets open.\n\tfocusChildOnDropdownOpen( dropdownView, () => dropdownView.listView!.items.find( item => {\n\t\tif ( item instanceof ListItemView ) {\n\t\t\treturn ( item.children.first as any ).isOn;\n\t\t}\n\n\t\treturn false;\n\t} ) );\n}\n\n/**\n * Adds an instance of {@link module:ui/list/listview~ListView} to a dropdown.\n */\nfunction addListToOpenDropdown(\n\tdropdownView: DropdownView,\n\titemsOrCallback: Collection<ListDropdownItemDefinition> | ( () => Collection<ListDropdownItemDefinition> ),\n\toptions: {\n\t\tariaLabel?: string;\n\t\trole?: string;\n\t}\n): void {\n\tconst locale = dropdownView.locale;\n\n\tconst listView = dropdownView.listView = new ListView( locale );\n\tconst items = typeof itemsOrCallback == 'function' ? itemsOrCallback() : itemsOrCallback;\n\n\tlistView.ariaLabel = options.ariaLabel;\n\tlistView.role = options.role;\n\n\tlistView.items.bindTo( items ).using( def => {\n\t\tif ( def.type === 'separator' ) {\n\t\t\treturn new ListSeparatorView( locale );\n\t\t} else if ( def.type === 'button' || def.type === 'switchbutton' ) {\n\t\t\tconst listItemView = new ListItemView( locale );\n\t\t\tlet buttonView;\n\n\t\t\tif ( def.type === 'button' ) {\n\t\t\t\tbuttonView = new ButtonView( locale );\n\t\t\t} else {\n\t\t\t\tbuttonView = new SwitchButtonView( locale );\n\t\t\t}\n\n\t\t\t// Bind all model properties to the button view.\n\t\t\tbuttonView.bind( ...Object.keys( def.model ) as Array<keyof ButtonView> ).to( def.model );\n\t\t\tbuttonView.delegate( 'execute' ).to( listItemView );\n\n\t\t\tlistItemView.children.add( buttonView );\n\n\t\t\treturn listItemView;\n\t\t}\n\n\t\treturn null;\n\t} );\n\n\tdropdownView.panelView.children.add( listView );\n\n\tlistView.items.delegate( 'execute' ).to( dropdownView );\n}\n\n/**\n * A helper to be used on an existing {@link module:ui/dropdown/dropdownview~DropdownView} that focuses\n * a specific child in DOM when the dropdown {@link module:ui/dropdown/dropdownview~DropdownView#isOpen gets open}.\n *\n * @param dropdownView A dropdown instance to which the focus behavior will be added.\n * @param childSelectorCallback A callback executed when the dropdown gets open. It should return a {@link module:ui/view~View}\n * instance (child of {@link module:ui/dropdown/dropdownview~DropdownView#panelView}) that will get focused or a falsy value.\n * If falsy value is returned, a default behavior of the dropdown will engage focusing the first focusable child in\n * the {@link module:ui/dropdown/dropdownview~DropdownView#panelView}.\n */\nexport function focusChildOnDropdownOpen(\n\tdropdownView: DropdownView,\n\tchildSelectorCallback: () => View | FalsyValue\n): void {\n\tdropdownView.on<ObservableChangeEvent>( 'change:isOpen', () => {\n\t\tif ( !dropdownView.isOpen ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst childToFocus: ( View & { focus?: () => void } ) | FalsyValue = childSelectorCallback();\n\n\t\tif ( !childToFocus ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( typeof childToFocus.focus === 'function' ) {\n\t\t\tchildToFocus.focus();\n\t\t} else {\n\t\t\t/**\n\t\t\t * The child view of a {@link module:ui/dropdown/dropdownview~DropdownView dropdown} is missing the `focus()` method\n\t\t\t * and could not be focused when the dropdown got {@link module:ui/dropdown/dropdownview~DropdownView#isOpen open}.\n\t\t\t *\n\t\t\t * Making the content of a dropdown focusable in this case greatly improves the accessibility. Please make the view instance\n\t\t\t * implements the {@link module:ui/dropdown/dropdownpanelfocusable~DropdownPanelFocusable focusable interface} for the best user\n\t\t\t * experience.\n\t\t\t *\n\t\t\t * @error ui-dropdown-focus-child-on-open-child-missing-focus\n\t\t\t * @param {module:ui/view~View} view\n\t\t\t */\n\t\t\tlogWarning( 'ui-dropdown-focus-child-on-open-child-missing-focus', { view: childToFocus } );\n\t\t}\n\n\t// * Let the panel show up first (do not focus an invisible element).\n\t// * Execute after focusDropdownPanelOnOpen(). See focusDropdownPanelOnOpen() to learn more.\n\t}, { priority: priorities.low - 10 } );\n}\n\n/**\n * Add a set of default behaviors to dropdown view.\n */\nfunction addDefaultBehavior( dropdownView: DropdownView ) {\n\tcloseDropdownOnClickOutside( dropdownView );\n\tcloseDropdownOnExecute( dropdownView );\n\tcloseDropdownOnBlur( dropdownView );\n\tfocusDropdownContentsOnArrows( dropdownView );\n\tfocusDropdownButtonOnClose( dropdownView );\n\tfocusDropdownPanelOnOpen( dropdownView );\n}\n\n/**\n * Adds a behavior to a dropdownView that closes opened dropdown when user clicks outside the dropdown.\n */\nfunction closeDropdownOnClickOutside( dropdownView: DropdownView ) {\n\tdropdownView.on<UIViewRenderEvent>( 'render', () => {\n\t\tclickOutsideHandler( {\n\t\t\temitter: dropdownView,\n\t\t\tactivator: () => dropdownView.isOpen,\n\t\t\tcallback: () => {\n\t\t\t\tdropdownView.isOpen = false;\n\t\t\t},\n\t\t\tcontextElements: () => [\n\t\t\t\tdropdownView.element!,\n\t\t\t\t...( dropdownView.focusTracker._elements as Set<HTMLElement> )\n\t\t\t]\n\t\t} );\n\t} );\n}\n\n/**\n * Adds a behavior to a dropdownView that closes the dropdown view on \"execute\" event.\n */\nfunction closeDropdownOnExecute( dropdownView: DropdownView ) {\n\t// Close the dropdown when one of the list items has been executed.\n\tdropdownView.on<ButtonExecuteEvent>( 'execute', evt => {\n\t\t// Toggling a switch button view should not close the dropdown.\n\t\tif ( evt.source instanceof SwitchButtonView ) {\n\t\t\treturn;\n\t\t}\n\n\t\tdropdownView.isOpen = false;\n\t} );\n}\n\n/**\n * Adds a behavior to a dropdown view that closes opened dropdown when it loses focus.\n */\nfunction closeDropdownOnBlur( dropdownView: DropdownView ) {\n\tdropdownView.focusTracker.on<ObservableChangeEvent<boolean>>( 'change:isFocused', ( evt, name, isFocused ) => {\n\t\tif ( dropdownView.isOpen && !isFocused ) {\n\t\t\tdropdownView.isOpen = false;\n\t\t}\n\t} );\n}\n\n/**\n * Adds a behavior to a dropdownView that focuses the dropdown's panel view contents on keystrokes.\n */\nfunction focusDropdownContentsOnArrows( dropdownView: DropdownView ) {\n\t// If the dropdown panel is already open, the arrow down key should focus the first child of the #panelView.\n\tdropdownView.keystrokes.set( 'arrowdown', ( data, cancel ) => {\n\t\tif ( dropdownView.isOpen ) {\n\t\t\tdropdownView.panelView.focus();\n\t\t\tcancel();\n\t\t}\n\t} );\n\n\t// If the dropdown panel is already open, the arrow up key should focus the last child of the #panelView.\n\tdropdownView.keystrokes.set( 'arrowup', ( data, cancel ) => {\n\t\tif ( dropdownView.isOpen ) {\n\t\t\tdropdownView.panelView.focusLast();\n\t\t\tcancel();\n\t\t}\n\t} );\n}\n\n/**\n * Adds a behavior that focuses the #buttonView when the dropdown was closed but focus was within the #panelView element.\n * This makes sure the focus is never lost.\n */\nfunction focusDropdownButtonOnClose( dropdownView: DropdownView ) {\n\tdropdownView.on<ObservableChangeEvent<boolean>>( 'change:isOpen', ( evt, name, isOpen ) => {\n\t\tif ( isOpen ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst element = dropdownView.panelView.element;\n\n\t\t// If the dropdown was closed, move the focus back to the button (#12125).\n\t\t// Don't touch the focus, if it moved somewhere else (e.g. moved to the editing root on #execute) (#12178).\n\t\t// Note: Don't use the state of the DropdownView#focusTracker here. It fires #blur with the timeout.\n\t\tif ( element && element.contains( global.document.activeElement ) ) {\n\t\t\tdropdownView.buttonView.focus();\n\t\t}\n\t} );\n}\n\n/**\n * Adds a behavior that focuses the #panelView when dropdown gets open (accessibility).\n */\nfunction focusDropdownPanelOnOpen( dropdownView: DropdownView ) {\n\tdropdownView.on<ObservableChangeEvent<boolean>>( 'change:isOpen', ( evt, name, isOpen ) => {\n\t\tif ( !isOpen ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Focus the first item in the dropdown when the dropdown opened.\n\t\tdropdownView.panelView.focus();\n\n\t// * Let the panel show up first (do not focus an invisible element).\n\t// * Also, execute before focusChildOnDropdownOpen() to make sure this helper does not break the\n\t// focus of a specific child by kicking in too late and resetting the focus in the panel.\n\t}, { priority: 'low' } );\n}\n\n/**\n * A definition of the list item used by the {@link module:ui/dropdown/utils~addListToDropdown}\n * utility.\n */\nexport type ListDropdownItemDefinition = ListDropdownSeparatorDefinition | ListDropdownButtonDefinition;\n\n/**\n * A definition of the 'separator' list item.\n */\nexport type ListDropdownSeparatorDefinition = {\n\ttype: 'separator';\n};\n\n/**\n * A definition of the 'button' or 'switchbutton' list item.\n */\nexport type ListDropdownButtonDefinition = {\n\ttype: 'button' | 'switchbutton';\n\n\t/**\n\t * Model of the item. Its properties fuel the newly created list item (or its children, depending on the `type`).\n\t */\n\tmodel: Model;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/labeledfield/utils\n */\n\nimport InputTextView from '../inputtext/inputtextview';\nimport InputNumberView from '../inputnumber/inputnumberview';\nimport { createDropdown } from '../dropdown/utils';\n\nimport type LabeledFieldView from './labeledfieldview';\nimport type DropdownView from '../dropdown/dropdownview';\nimport type { InputViewInputEvent } from '../input/inputview';\n\n/**\n * A helper for creating labeled inputs.\n *\n * It creates an instance of a {@link module:ui/inputtext/inputtextview~InputTextView input text} that is\n * logically related to a {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView labeled view} in DOM.\n *\n * The helper does the following:\n *\n * * It sets input's `id` and `ariaDescribedById` attributes.\n * * It binds input's `isReadOnly` to the labeled view.\n * * It binds input's `hasError` to the labeled view.\n * * It enables a logic that cleans up the error when user starts typing in the input.\n *\n * Usage:\n *\n * ```ts\n * const labeledInputView = new LabeledFieldView( locale, createLabeledInputText );\n * console.log( labeledInputView.fieldView ); // A text input instance.\n * ```\n *\n * @param labeledFieldView The instance of the labeled field view.\n * @param viewUid An UID string that allows DOM logical connection between the\n * {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView#labelView labeled view's label} and the input.\n * @param statusUid An UID string that allows DOM logical connection between the\n * {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView#statusView labeled view's status} and the input.\n * @returns The input text view instance.\n */\nexport function createLabeledInputText(\n\tlabeledFieldView: LabeledFieldView,\n\tviewUid: string,\n\tstatusUid: string\n): InputTextView {\n\tconst inputView = new InputTextView( labeledFieldView.locale );\n\n\tinputView.set( {\n\t\tid: viewUid,\n\t\tariaDescribedById: statusUid\n\t} );\n\n\tinputView.bind( 'isReadOnly' ).to( labeledFieldView, 'isEnabled', value => !value );\n\tinputView.bind( 'hasError' ).to( labeledFieldView, 'errorText', value => !!value );\n\n\tinputView.on<InputViewInputEvent>( 'input', () => {\n\t\t// UX: Make the error text disappear and disable the error indicator as the user\n\t\t// starts fixing the errors.\n\t\tlabeledFieldView.errorText = null;\n\t} );\n\n\tlabeledFieldView.bind( 'isEmpty', 'isFocused', 'placeholder' ).to( inputView );\n\n\treturn inputView;\n}\n\n/**\n * A helper for creating labeled number inputs.\n *\n * It creates an instance of a {@link module:ui/inputnumber/inputnumberview~InputNumberView input number} that is\n * logically related to a {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView labeled view} in DOM.\n *\n * The helper does the following:\n *\n * * It sets input's `id` and `ariaDescribedById` attributes.\n * * It binds input's `isReadOnly` to the labeled view.\n * * It binds input's `hasError` to the labeled view.\n * * It enables a logic that cleans up the error when user starts typing in the input.\n *\n * Usage:\n *\n * ```ts\n * const labeledInputView = new LabeledFieldView( locale, createLabeledInputNumber );\n * console.log( labeledInputView.fieldView ); // A number input instance.\n * ```\n *\n * @param labeledFieldView The instance of the labeled field view.\n * @param viewUid An UID string that allows DOM logical connection between the\n * {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView#labelView labeled view's label} and the input.\n * @param statusUid An UID string that allows DOM logical connection between the\n * {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView#statusView labeled view's status} and the input.\n * @returns The input number view instance.\n */\nexport function createLabeledInputNumber(\n\tlabeledFieldView: LabeledFieldView,\n\tviewUid: string,\n\tstatusUid: string\n): InputNumberView {\n\tconst inputView = new InputNumberView( labeledFieldView.locale );\n\n\tinputView.set( {\n\t\tid: viewUid,\n\t\tariaDescribedById: statusUid,\n\t\tinputMode: 'numeric'\n\t} );\n\n\tinputView.bind( 'isReadOnly' ).to( labeledFieldView, 'isEnabled', value => !value );\n\tinputView.bind( 'hasError' ).to( labeledFieldView, 'errorText', value => !!value );\n\n\tinputView.on<InputViewInputEvent>( 'input', () => {\n\t\t// UX: Make the error text disappear and disable the error indicator as the user\n\t\t// starts fixing the errors.\n\t\tlabeledFieldView.errorText = null;\n\t} );\n\n\tlabeledFieldView.bind( 'isEmpty', 'isFocused', 'placeholder' ).to( inputView );\n\n\treturn inputView;\n}\n\n/**\n * A helper for creating labeled dropdowns.\n *\n * It creates an instance of a {@link module:ui/dropdown/dropdownview~DropdownView dropdown} that is\n * logically related to a {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView labeled field view}.\n *\n * The helper does the following:\n *\n * * It sets dropdown's `id` and `ariaDescribedById` attributes.\n * * It binds input's `isEnabled` to the labeled view.\n *\n * Usage:\n *\n * ```ts\n * const labeledInputView = new LabeledFieldView( locale, createLabeledDropdown );\n * console.log( labeledInputView.fieldView ); // A dropdown instance.\n * ```\n *\n * @param labeledFieldView The instance of the labeled field view.\n * @param viewUid An UID string that allows DOM logical connection between the\n * {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView#labelView labeled view label} and the dropdown.\n * @param statusUid An UID string that allows DOM logical connection between the\n * {@link module:ui/labeledfield/labeledfieldview~LabeledFieldView#statusView labeled view status} and the dropdown.\n * @returns The dropdown view instance.\n */\nexport function createLabeledDropdown(\n\tlabeledFieldView: LabeledFieldView,\n\tviewUid: string,\n\tstatusUid: string\n): DropdownView {\n\tconst dropdownView = createDropdown( labeledFieldView.locale );\n\n\tdropdownView.set( {\n\t\tid: viewUid,\n\t\tariaDescribedById: statusUid\n\t} );\n\n\tdropdownView.bind( 'isEnabled' ).to( labeledFieldView );\n\n\treturn dropdownView;\n}\n","// Clamps a value between an upper and lower bound.\n// We use ternary operators because it makes the minified code\n// 2 times shorter then `Math.min(Math.max(a,b),c)`\nexport const clamp = (number, min = 0, max = 1) => {\n return number > max ? max : number < min ? min : number;\n};\nexport const round = (number, digits = 0, base = Math.pow(10, digits)) => {\n return Math.round(base * number) / base;\n};\n//# sourceMappingURL=math.js.map","import { round } from './math.js';\n/**\n * Valid CSS <angle> units.\n * https://developer.mozilla.org/en-US/docs/Web/CSS/angle\n */\nconst angleUnits = {\n grad: 360 / 400,\n turn: 360,\n rad: 360 / (Math.PI * 2)\n};\nexport const hexToHsva = (hex) => rgbaToHsva(hexToRgba(hex));\nexport const hexToRgba = (hex) => {\n if (hex[0] === '#')\n hex = hex.substring(1);\n if (hex.length < 6) {\n return {\n r: parseInt(hex[0] + hex[0], 16),\n g: parseInt(hex[1] + hex[1], 16),\n b: parseInt(hex[2] + hex[2], 16),\n a: hex.length === 4 ? round(parseInt(hex[3] + hex[3], 16) / 255, 2) : 1\n };\n }\n return {\n r: parseInt(hex.substring(0, 2), 16),\n g: parseInt(hex.substring(2, 4), 16),\n b: parseInt(hex.substring(4, 6), 16),\n a: hex.length === 8 ? round(parseInt(hex.substring(6, 8), 16) / 255, 2) : 1\n };\n};\nexport const parseHue = (value, unit = 'deg') => {\n return Number(value) * (angleUnits[unit] || 1);\n};\nexport const hslaStringToHsva = (hslString) => {\n const matcher = /hsla?\\(?\\s*(-?\\d*\\.?\\d+)(deg|rad|grad|turn)?[,\\s]+(-?\\d*\\.?\\d+)%?[,\\s]+(-?\\d*\\.?\\d+)%?,?\\s*[/\\s]*(-?\\d*\\.?\\d+)?(%)?\\s*\\)?/i;\n const match = matcher.exec(hslString);\n if (!match)\n return { h: 0, s: 0, v: 0, a: 1 };\n return hslaToHsva({\n h: parseHue(match[1], match[2]),\n s: Number(match[3]),\n l: Number(match[4]),\n a: match[5] === undefined ? 1 : Number(match[5]) / (match[6] ? 100 : 1)\n });\n};\nexport const hslStringToHsva = hslaStringToHsva;\nexport const hslaToHsva = ({ h, s, l, a }) => {\n s *= (l < 50 ? l : 100 - l) / 100;\n return {\n h: h,\n s: s > 0 ? ((2 * s) / (l + s)) * 100 : 0,\n v: l + s,\n a\n };\n};\nexport const hsvaToHex = (hsva) => rgbaToHex(hsvaToRgba(hsva));\nexport const hsvaToHsla = ({ h, s, v, a }) => {\n const hh = ((200 - s) * v) / 100;\n return {\n h: round(h),\n s: round(hh > 0 && hh < 200 ? ((s * v) / 100 / (hh <= 100 ? hh : 200 - hh)) * 100 : 0),\n l: round(hh / 2),\n a: round(a, 2)\n };\n};\nexport const hsvaToHsvString = (hsva) => {\n const { h, s, v } = roundHsva(hsva);\n return `hsv(${h}, ${s}%, ${v}%)`;\n};\nexport const hsvaToHsvaString = (hsva) => {\n const { h, s, v, a } = roundHsva(hsva);\n return `hsva(${h}, ${s}%, ${v}%, ${a})`;\n};\nexport const hsvaToHslString = (hsva) => {\n const { h, s, l } = hsvaToHsla(hsva);\n return `hsl(${h}, ${s}%, ${l}%)`;\n};\nexport const hsvaToHslaString = (hsva) => {\n const { h, s, l, a } = hsvaToHsla(hsva);\n return `hsla(${h}, ${s}%, ${l}%, ${a})`;\n};\nexport const hsvaToRgba = ({ h, s, v, a }) => {\n h = (h / 360) * 6;\n s = s / 100;\n v = v / 100;\n const hh = Math.floor(h), b = v * (1 - s), c = v * (1 - (h - hh) * s), d = v * (1 - (1 - h + hh) * s), module = hh % 6;\n return {\n r: round([v, c, b, b, d, v][module] * 255),\n g: round([d, v, v, c, b, b][module] * 255),\n b: round([b, b, d, v, v, c][module] * 255),\n a: round(a, 2)\n };\n};\nexport const hsvaToRgbString = (hsva) => {\n const { r, g, b } = hsvaToRgba(hsva);\n return `rgb(${r}, ${g}, ${b})`;\n};\nexport const hsvaToRgbaString = (hsva) => {\n const { r, g, b, a } = hsvaToRgba(hsva);\n return `rgba(${r}, ${g}, ${b}, ${a})`;\n};\nexport const hsvaStringToHsva = (hsvString) => {\n const matcher = /hsva?\\(?\\s*(-?\\d*\\.?\\d+)(deg|rad|grad|turn)?[,\\s]+(-?\\d*\\.?\\d+)%?[,\\s]+(-?\\d*\\.?\\d+)%?,?\\s*[/\\s]*(-?\\d*\\.?\\d+)?(%)?\\s*\\)?/i;\n const match = matcher.exec(hsvString);\n if (!match)\n return { h: 0, s: 0, v: 0, a: 1 };\n return roundHsva({\n h: parseHue(match[1], match[2]),\n s: Number(match[3]),\n v: Number(match[4]),\n a: match[5] === undefined ? 1 : Number(match[5]) / (match[6] ? 100 : 1)\n });\n};\nexport const hsvStringToHsva = hsvaStringToHsva;\nexport const rgbaStringToHsva = (rgbaString) => {\n const matcher = /rgba?\\(?\\s*(-?\\d*\\.?\\d+)(%)?[,\\s]+(-?\\d*\\.?\\d+)(%)?[,\\s]+(-?\\d*\\.?\\d+)(%)?,?\\s*[/\\s]*(-?\\d*\\.?\\d+)?(%)?\\s*\\)?/i;\n const match = matcher.exec(rgbaString);\n if (!match)\n return { h: 0, s: 0, v: 0, a: 1 };\n return rgbaToHsva({\n r: Number(match[1]) / (match[2] ? 100 / 255 : 1),\n g: Number(match[3]) / (match[4] ? 100 / 255 : 1),\n b: Number(match[5]) / (match[6] ? 100 / 255 : 1),\n a: match[7] === undefined ? 1 : Number(match[7]) / (match[8] ? 100 : 1)\n });\n};\nexport const rgbStringToHsva = rgbaStringToHsva;\nconst format = (number) => {\n const hex = number.toString(16);\n return hex.length < 2 ? '0' + hex : hex;\n};\nexport const rgbaToHex = ({ r, g, b, a }) => {\n const alphaHex = a < 1 ? format(round(a * 255)) : '';\n return '#' + format(r) + format(g) + format(b) + alphaHex;\n};\nexport const rgbaToHsva = ({ r, g, b, a }) => {\n const max = Math.max(r, g, b);\n const delta = max - Math.min(r, g, b);\n // prettier-ignore\n const hh = delta\n ? max === r\n ? (g - b) / delta\n : max === g\n ? 2 + (b - r) / delta\n : 4 + (r - g) / delta\n : 0;\n return {\n h: round(60 * (hh < 0 ? hh + 6 : hh)),\n s: round(max ? (delta / max) * 100 : 0),\n v: round((max / 255) * 100),\n a\n };\n};\nexport const roundHsva = (hsva) => ({\n h: round(hsva.h),\n s: round(hsva.s),\n v: round(hsva.v),\n a: round(hsva.a, 2)\n});\nexport const rgbaToRgb = ({ r, g, b }) => ({ r, g, b });\nexport const hslaToHsl = ({ h, s, l }) => ({ h, s, l });\nexport const hsvaToHsv = (hsva) => {\n const { h, s, v } = roundHsva(hsva);\n return { h, s, v };\n};\n//# sourceMappingURL=convert.js.map","import { hexToRgba } from './convert.js';\nexport const equalColorObjects = (first, second) => {\n if (first === second)\n return true;\n for (const prop in first) {\n // The following allows for a type-safe calling of this function (first & second have to be HSL, HSV, or RGB)\n // with type-unsafe iterating over object keys. TS does not allow this without an index (`[key: string]: number`)\n // on an object to define how iteration is normally done. To ensure extra keys are not allowed on our types,\n // we must cast our object to unknown (as RGB demands `r` be a key, while `Record<string, x>` does not care if\n // there is or not), and then as a type TS can iterate over.\n if (first[prop] !==\n second[prop])\n return false;\n }\n return true;\n};\nexport const equalColorString = (first, second) => {\n return first.replace(/\\s/g, '') === second.replace(/\\s/g, '');\n};\nexport const equalHex = (first, second) => {\n if (first.toLowerCase() === second.toLowerCase())\n return true;\n // To compare colors like `#FFF` and `ffffff` we convert them into RGB objects\n return equalColorObjects(hexToRgba(first), hexToRgba(second));\n};\n//# sourceMappingURL=compare.js.map","const cache = {};\nexport const tpl = (html) => {\n let template = cache[html];\n if (!template) {\n template = document.createElement('template');\n template.innerHTML = html;\n cache[html] = template;\n }\n return template;\n};\nexport const fire = (target, type, detail) => {\n target.dispatchEvent(new CustomEvent(type, {\n bubbles: true,\n detail\n }));\n};\n//# sourceMappingURL=dom.js.map","import { fire, tpl } from '../utils/dom.js';\nimport { clamp } from '../utils/math.js';\nlet hasTouched = false;\n// Check if an event was triggered by touch\nconst isTouch = (e) => 'touches' in e;\n// Prevent mobile browsers from handling mouse events (conflicting with touch ones).\n// If we detected a touch interaction before, we prefer reacting to touch events only.\nconst isValid = (event) => {\n if (hasTouched && !isTouch(event))\n return false;\n if (!hasTouched)\n hasTouched = isTouch(event);\n return true;\n};\nconst pointerMove = (target, event) => {\n const pointer = isTouch(event) ? event.touches[0] : event;\n const rect = target.el.getBoundingClientRect();\n fire(target.el, 'move', target.getMove({\n x: clamp((pointer.pageX - (rect.left + window.pageXOffset)) / rect.width),\n y: clamp((pointer.pageY - (rect.top + window.pageYOffset)) / rect.height)\n }));\n};\nconst keyMove = (target, event) => {\n // We use `keyCode` instead of `key` to reduce the size of the library.\n const keyCode = event.keyCode;\n // Ignore all keys except arrow ones, Page Up, Page Down, Home and End.\n if (keyCode > 40 || (target.xy && keyCode < 37) || keyCode < 33)\n return;\n // Do not scroll page by keys when color picker element has focus.\n event.preventDefault();\n // Send relative offset to the parent component.\n fire(target.el, 'move', target.getMove({\n x: keyCode === 39 // Arrow Right\n ? 0.01\n : keyCode === 37 // Arrow Left\n ? -0.01\n : keyCode === 34 // Page Down\n ? 0.05\n : keyCode === 33 // Page Up\n ? -0.05\n : keyCode === 35 // End\n ? 1\n : keyCode === 36 // Home\n ? -1\n : 0,\n y: keyCode === 40 // Arrow down\n ? 0.01\n : keyCode === 38 // Arrow Up\n ? -0.01\n : 0\n }, true));\n};\nexport class Slider {\n constructor(root, part, aria, xy) {\n const template = tpl(`<div role=\"slider\" tabindex=\"0\" part=\"${part}\" ${aria}><div part=\"${part}-pointer\"></div></div>`);\n root.appendChild(template.content.cloneNode(true));\n const el = root.querySelector(`[part=${part}]`);\n el.addEventListener('mousedown', this);\n el.addEventListener('touchstart', this);\n el.addEventListener('keydown', this);\n this.el = el;\n this.xy = xy;\n this.nodes = [el.firstChild, el];\n }\n set dragging(state) {\n const toggleEvent = state ? document.addEventListener : document.removeEventListener;\n toggleEvent(hasTouched ? 'touchmove' : 'mousemove', this);\n toggleEvent(hasTouched ? 'touchend' : 'mouseup', this);\n }\n handleEvent(event) {\n switch (event.type) {\n case 'mousedown':\n case 'touchstart':\n event.preventDefault();\n // event.button is 0 in mousedown for left button activation\n if (!isValid(event) || (!hasTouched && event.button != 0))\n return;\n this.el.focus();\n pointerMove(this, event);\n this.dragging = true;\n break;\n case 'mousemove':\n case 'touchmove':\n event.preventDefault();\n pointerMove(this, event);\n break;\n case 'mouseup':\n case 'touchend':\n this.dragging = false;\n break;\n case 'keydown':\n keyMove(this, event);\n break;\n }\n }\n style(styles) {\n styles.forEach((style, i) => {\n for (const p in style) {\n this.nodes[i].style.setProperty(p, style[p]);\n }\n });\n }\n}\n//# sourceMappingURL=slider.js.map","import { Slider } from './slider.js';\nimport { hsvaToHslString } from '../utils/convert.js';\nimport { clamp, round } from '../utils/math.js';\nexport class Hue extends Slider {\n constructor(root) {\n super(root, 'hue', 'aria-label=\"Hue\" aria-valuemin=\"0\" aria-valuemax=\"360\"', false);\n }\n update({ h }) {\n this.h = h;\n this.style([\n {\n left: `${(h / 360) * 100}%`,\n color: hsvaToHslString({ h, s: 100, v: 100, a: 1 })\n }\n ]);\n this.el.setAttribute('aria-valuenow', `${round(h)}`);\n }\n getMove(offset, key) {\n // Hue measured in degrees of the color circle ranging from 0 to 360\n return { h: key ? clamp(this.h + offset.x * 360, 0, 360) : 360 * offset.x };\n }\n}\n//# sourceMappingURL=hue.js.map","import { Slider } from './slider.js';\nimport { hsvaToHslString } from '../utils/convert.js';\nimport { clamp, round } from '../utils/math.js';\nexport class Saturation extends Slider {\n constructor(root) {\n super(root, 'saturation', 'aria-label=\"Color\"', true);\n }\n update(hsva) {\n this.hsva = hsva;\n this.style([\n {\n top: `${100 - hsva.v}%`,\n left: `${hsva.s}%`,\n color: hsvaToHslString(hsva)\n },\n {\n 'background-color': hsvaToHslString({ h: hsva.h, s: 100, v: 100, a: 1 })\n }\n ]);\n this.el.setAttribute('aria-valuetext', `Saturation ${round(hsva.s)}%, Brightness ${round(hsva.v)}%`);\n }\n getMove(offset, key) {\n // Saturation and brightness always fit into [0, 100] range\n return {\n s: key ? clamp(this.hsva.s + offset.x * 100, 0, 100) : offset.x * 100,\n v: key ? clamp(this.hsva.v - offset.y * 100, 0, 100) : Math.round(100 - offset.y * 100)\n };\n }\n}\n//# sourceMappingURL=saturation.js.map","export default `:host{display:flex;flex-direction:column;position:relative;width:200px;height:200px;user-select:none;-webkit-user-select:none;cursor:default}:host([hidden]){display:none!important}[role=slider]{position:relative;touch-action:none;user-select:none;-webkit-user-select:none;outline:0}[role=slider]:last-child{border-radius:0 0 8px 8px}[part$=pointer]{position:absolute;z-index:1;box-sizing:border-box;width:28px;height:28px;display:flex;place-content:center center;transform:translate(-50%,-50%);background-color:#fff;border:2px solid #fff;border-radius:50%;box-shadow:0 2px 4px rgba(0,0,0,.2)}[part$=pointer]::after{content:\"\";width:100%;height:100%;border-radius:inherit;background-color:currentColor}[role=slider]:focus [part$=pointer]{transform:translate(-50%,-50%) scale(1.1)}`;\n//# sourceMappingURL=color-picker.js.map","import { equalColorObjects } from '../utils/compare.js';\nimport { fire, tpl } from '../utils/dom.js';\nimport { Hue } from './hue.js';\nimport { Saturation } from './saturation.js';\nimport css from '../styles/color-picker.js';\nimport hueCss from '../styles/hue.js';\nimport saturationCss from '../styles/saturation.js';\nconst $isSame = Symbol('same');\nconst $color = Symbol('color');\nconst $hsva = Symbol('hsva');\nconst $update = Symbol('update');\nconst $parts = Symbol('parts');\nexport const $css = Symbol('css');\nexport const $sliders = Symbol('sliders');\nexport class ColorPicker extends HTMLElement {\n static get observedAttributes() {\n return ['color'];\n }\n get [$css]() {\n return [css, hueCss, saturationCss];\n }\n get [$sliders]() {\n return [Saturation, Hue];\n }\n get color() {\n return this[$color];\n }\n set color(newColor) {\n if (!this[$isSame](newColor)) {\n const newHsva = this.colorModel.toHsva(newColor);\n this[$update](newHsva);\n this[$color] = newColor;\n }\n }\n constructor() {\n super();\n const template = tpl(`<style>${this[$css].join('')}</style>`);\n const root = this.attachShadow({ mode: 'open' });\n root.appendChild(template.content.cloneNode(true));\n root.addEventListener('move', this);\n this[$parts] = this[$sliders].map((slider) => new slider(root));\n }\n connectedCallback() {\n // A user may set a property on an _instance_ of an element,\n // before its prototype has been connected to this class.\n // If so, we need to run it through the proper class setter.\n if (this.hasOwnProperty('color')) {\n const value = this.color;\n delete this['color'];\n this.color = value;\n }\n else if (!this.color) {\n this.color = this.colorModel.defaultColor;\n }\n }\n attributeChangedCallback(_attr, _oldVal, newVal) {\n const color = this.colorModel.fromAttr(newVal);\n if (!this[$isSame](color)) {\n this.color = color;\n }\n }\n handleEvent(event) {\n // Merge the current HSV color object with updated params.\n const oldHsva = this[$hsva];\n const newHsva = { ...oldHsva, ...event.detail };\n this[$update](newHsva);\n let newColor;\n if (!equalColorObjects(newHsva, oldHsva) &&\n !this[$isSame]((newColor = this.colorModel.fromHsva(newHsva)))) {\n this[$color] = newColor;\n fire(this, 'color-changed', { value: newColor });\n }\n }\n [$isSame](color) {\n return this.color && this.colorModel.equal(color, this.color);\n }\n [$update](hsva) {\n this[$hsva] = hsva;\n this[$parts].forEach((part) => part.update(hsva));\n }\n}\n//# sourceMappingURL=color-picker.js.map","export default `[part=hue]{flex:0 0 24px;background:linear-gradient(to right,red 0,#ff0 17%,#0f0 33%,#0ff 50%,#00f 67%,#f0f 83%,red 100%)}[part=hue-pointer]{top:50%;z-index:2}`;\n//# sourceMappingURL=hue.js.map","export default `[part=saturation]{flex-grow:1;border-color:transparent;border-bottom:12px solid #000;border-radius:8px 8px 0 0;background-image:linear-gradient(to top,#000,transparent),linear-gradient(to right,#fff,rgba(255,255,255,0));box-shadow:inset 0 0 0 1px rgba(0,0,0,.05)}[part=saturation-pointer]{z-index:3}`;\n//# sourceMappingURL=saturation.js.map","import { ColorPicker } from '../components/color-picker.js';\nimport { hexToHsva, hsvaToHex } from '../utils/convert.js';\nimport { equalHex } from '../utils/compare.js';\nconst colorModel = {\n defaultColor: '#000',\n toHsva: hexToHsva,\n fromHsva: ({ h, s, v }) => hsvaToHex({ h, s, v, a: 1 }),\n equal: equalHex,\n fromAttr: (color) => color\n};\nexport class HexBase extends ColorPicker {\n get colorModel() {\n return colorModel;\n }\n}\n//# sourceMappingURL=hex.js.map","import { HexBase } from './lib/entrypoints/hex.js';\n/**\n * A color picker custom element that uses HEX format.\n *\n * @element hex-color-picker\n *\n * @prop {string} color - Selected color in HEX format.\n * @attr {string} color - Selected color in HEX format.\n *\n * @fires color-changed - Event fired when color property changes.\n *\n * @csspart hue - A hue selector container.\n * @csspart saturation - A saturation selector container\n * @csspart hue-pointer - A hue pointer element.\n * @csspart saturation-pointer - A saturation pointer element.\n */\nexport class HexColorPicker extends HexBase {\n}\ncustomElements.define('hex-color-picker', HexColorPicker);\n//# sourceMappingURL=hex-color-picker.js.map","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./colorpicker.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/colorselector/documentcolorcollection\n */\n\nimport type { ColorDefinition } from '../colorgrid/colorgridview';\nimport {\n\tCollection,\n\tObservableMixin,\n\ttype CollectionAddEvent,\n\ttype CollectionRemoveEvent,\n\ttype CollectionChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\n/**\n * A collection to store document colors. It enforces colors to be unique.\n */\nexport default class DocumentColorCollection extends ObservableMixin( Collection<ColorDefinition> ) {\n\t/**\n\t * Indicates whether the document color collection is empty.\n\t *\n\t * @observable\n\t */\n\tdeclare public readonly isEmpty: boolean;\n\n\tconstructor( options?: any ) {\n\t\tsuper( options );\n\n\t\tthis.set( 'isEmpty', true );\n\n\t\tthis.on( 'change', () => {\n\t\t\tthis.set( 'isEmpty', this.length === 0 );\n\t\t} );\n\t}\n\n\t/**\n\t * Adds a color to the document color collection.\n\t *\n\t * This method ensures that no color duplicates are inserted (compared using\n\t * the color value of the {@link module:ui/colorgrid/colorgridview~ColorDefinition}).\n\t *\n\t * If the item does not have an ID, it will be automatically generated and set on the item.\n\t *\n\t * @param index The position of the item in the collection. The item is pushed to the collection when `index` is not specified.\n\t * @fires add\n\t * @fires change\n\t */\n\tpublic override add( item: ColorDefinition, index?: number ): this {\n\t\tif ( this.find( element => element.color === item.color ) ) {\n\t\t\t// No duplicates are allowed.\n\t\t\treturn this;\n\t\t}\n\n\t\treturn super.add( item, index );\n\t}\n\n\t/**\n\t * Checks if an object with given colors is present in the document color collection.\n\t */\n\tpublic hasColor( color: string ): boolean {\n\t\treturn !!this.find( item => item.color === color );\n\t}\n}\n\n/**\n * Fired when the collection was changed due to adding or removing items.\n *\n * @eventName ~DocumentColorCollection#change\n * @param data Changed items.\n */\nexport type DocumentColorCollectionChangeEvent = CollectionChangeEvent;\n\n/**\n * Fired when an item is added to the collection.\n *\n * @eventName ~DocumentColorCollection#add\n * @param item The added item.\n * @param index An index where the addition occurred.\n */\nexport type DocumentColorCollectionAddEvent = CollectionAddEvent;\n\n/**\n * Fired when an item is removed from the collection.\n *\n * @eventName ~DocumentColorCollection#remove\n * @param item The removed item.\n * @param index Index from which item was removed.\n */\nexport type DocumentColorCollectionRemoveEvent = CollectionRemoveEvent;\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./colorselector.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/componentfactory\n */\n\nimport { CKEditorError, type Locale } from '@ckeditor/ckeditor5-utils';\nimport type { Editor } from '@ckeditor/ckeditor5-core';\n\nimport type View from './view';\n\n/**\n * A helper class implementing the UI component ({@link module:ui/view~View view}) factory.\n *\n * It allows functions producing specific UI components to be registered under their unique names\n * in the factory. A registered component can be then instantiated by providing its name.\n * Note that the names are case insensitive.\n *\n * ```ts\n * // The editor provides localization tools for the factory.\n * const factory = new ComponentFactory( editor );\n *\n * factory.add( 'foo', locale => new FooView( locale ) );\n * factory.add( 'bar', locale => new BarView( locale ) );\n *\n * // An instance of FooView.\n * const fooInstance = factory.create( 'foo' );\n *\n * // Names are case insensitive so this is also allowed:\n * const barInstance = factory.create( 'Bar' );\n * ```\n *\n * The {@link module:core/editor/editor~Editor#locale editor locale} is passed to the factory\n * function when {@link module:ui/componentfactory~ComponentFactory#create} is called.\n */\nexport default class ComponentFactory {\n\t/**\n\t * The editor instance that the factory belongs to.\n\t */\n\tpublic readonly editor: Editor;\n\n\t/**\n\t * Registered component factories.\n\t */\n\tprivate readonly _components = new Map<string, { originalName: string; callback: ( locale: Locale ) => View }>();\n\n\t/**\n\t * Creates an instance of the factory.\n\t *\n\t * @param editor The editor instance.\n\t */\n\tconstructor( editor: Editor ) {\n\t\tthis.editor = editor;\n\t}\n\n\t/**\n\t * Returns an iterator of registered component names. Names are returned in lower case.\n\t */\n\tpublic* names(): IterableIterator<string> {\n\t\tfor ( const value of this._components.values() ) {\n\t\t\tyield value.originalName;\n\t\t}\n\t}\n\n\t/**\n\t * Registers a component factory function that will be used by the\n\t * {@link #create create} method and called with the\n\t * {@link module:core/editor/editor~Editor#locale editor locale} as an argument,\n\t * allowing localization of the {@link module:ui/view~View view}.\n\t *\n\t * @param name The name of the component.\n\t * @param callback The callback that returns the component.\n\t */\n\tpublic add( name: string, callback: ( locale: Locale ) => View ): void {\n\t\tthis._components.set( getNormalized( name ), { callback, originalName: name } );\n\t}\n\n\t/**\n\t * Creates an instance of a component registered in the factory under a specific name.\n\t *\n\t * When called, the {@link module:core/editor/editor~Editor#locale editor locale} is passed to\n\t * the previously {@link #add added} factory function, allowing localization of the\n\t * {@link module:ui/view~View view}.\n\t *\n\t * @param name The name of the component.\n\t * @returns The instantiated component view.\n\t */\n\tpublic create( name: string ): View {\n\t\tif ( !this.has( name ) ) {\n\t\t\t/**\n\t\t\t * The required component is not registered in the component factory. Please make sure\n\t\t\t * the provided name is correct and the component has been correctly\n\t\t\t * {@link module:ui/componentfactory~ComponentFactory#add added} to the factory.\n\t\t\t *\n\t\t\t * @error componentfactory-item-missing\n\t\t\t * @param name The name of the missing component.\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'componentfactory-item-missing',\n\t\t\t\tthis,\n\t\t\t\t{ name }\n\t\t\t);\n\t\t}\n\n\t\treturn this._components.get( getNormalized( name ) )!.callback( this.editor.locale );\n\t}\n\n\t/**\n\t * Checks if a component of a given name is registered in the factory.\n\t *\n\t * @param name The name of the component.\n\t */\n\tpublic has( name: string ): boolean {\n\t\treturn this._components.has( getNormalized( name ) );\n\t}\n}\n\n/**\n * Ensures that the component name used as the key in the internal map is in lower case.\n */\nfunction getNormalized( name: unknown ) {\n\treturn String( name ).toLowerCase();\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./balloonpanel.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/panel/balloon/balloonpanelview\n */\n\nimport View from '../../view';\nimport type ViewCollection from '../../viewcollection';\n\nimport {\n\tgetOptimalPosition,\n\tglobal,\n\tisRange,\n\ttoUnit,\n\ttype Locale,\n\ttype ObservableChangeEvent,\n\ttype PositionOptions,\n\ttype PositioningFunction,\n\ttype Rect\n} from '@ckeditor/ckeditor5-utils';\n\nimport { isElement } from 'lodash-es';\n\nimport '../../../theme/components/panel/balloonpanel.css';\n\nconst toPx = toUnit( 'px' );\nconst defaultLimiterElement = global.document.body;\n\n/**\n * The balloon panel view class.\n *\n * A floating container which can\n * {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView#pin pin} to any\n * {@link module:utils/dom/position~Options#target target} in the DOM and remain in that position\n * e.g. when the web page is scrolled.\n *\n * The balloon panel can be used to display contextual, non-blocking UI like forms, toolbars and\n * the like in its {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView#content} view\n * collection.\n *\n * There is a number of {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView.defaultPositions}\n * that the balloon can use, automatically switching from one to another when the viewport space becomes\n * scarce to keep the balloon visible to the user as long as it is possible. The balloon will also\n * accept any custom position set provided by the user compatible with the\n * {@link module:utils/dom/position~Options options}.\n *\n * ```ts\n * const panel = new BalloonPanelView( locale );\n * const childView = new ChildView();\n * const positions = BalloonPanelView.defaultPositions;\n *\n * panel.render();\n *\n * // Add a child view to the panel's content collection.\n * panel.content.add( childView );\n *\n * // Start pinning the panel to an element with the \"target\" id DOM.\n * // The balloon will remain pinned until unpin() is called.\n * panel.pin( {\n * \ttarget: document.querySelector( '#target' ),\n * \tpositions: [\n * \t\tpositions.northArrowSouth,\n * \t\tpositions.southArrowNorth\n * \t]\n * } );\n * ```\n */\nexport default class BalloonPanelView extends View {\n\t/**\n\t * A collection of the child views that creates the balloon panel contents.\n\t */\n\tpublic readonly content: ViewCollection;\n\n\t/**\n\t * The absolute top position of the balloon panel in pixels.\n\t *\n\t * @observable\n\t * @default 0\n\t */\n\tdeclare public top: number;\n\n\t/**\n\t * The absolute left position of the balloon panel in pixels.\n\t *\n\t * @observable\n\t * @default 0\n\t */\n\tdeclare public left: number;\n\n\t/**\n\t * The balloon panel's current position. The position name is reflected in the CSS class set\n\t * to the balloon, i.e. `.ck-balloon-panel_arrow_nw` for the \"arrow_nw\" position. The class\n\t * controls the minor aspects of the balloon's visual appearance like the placement\n\t * of an {@link #withArrow arrow}. To support a new position, an additional CSS must be created.\n\t *\n\t * Default position names correspond with\n\t * {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView.defaultPositions}.\n\t *\n\t * See the {@link #attachTo} and {@link #pin} methods to learn about custom balloon positions.\n\t *\n\t * @observable\n\t * @default 'arrow_nw'\n\t */\n\tdeclare public position: 'arrow_nw' | 'arrow_ne' | 'arrow_sw' | 'arrow_se';\n\n\t/**\n\t * Controls whether the balloon panel is visible or not.\n\t *\n\t * @observable\n\t * @default false\n\t */\n\tdeclare public isVisible: boolean;\n\n\t/**\n\t * Controls whether the balloon panel has an arrow. The presence of the arrow\n\t * is reflected in the `ck-balloon-panel_with-arrow` CSS class.\n\t *\n\t * @observable\n\t * @default true\n\t */\n\tdeclare public withArrow: boolean;\n\n\t/**\n\t * An additional CSS class added to the {@link #element}.\n\t *\n\t * @observable\n\t */\n\tdeclare public class: string | undefined;\n\n\t/**\n\t * A callback that starts pinning the panel when {@link #isVisible} gets\n\t * `true`. Used by {@link #pin}.\n\t *\n\t * @private\n\t */\n\tprivate _pinWhenIsVisibleCallback: ( () => void ) | null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.set( 'top', 0 );\n\t\tthis.set( 'left', 0 );\n\t\tthis.set( 'position', 'arrow_nw' );\n\t\tthis.set( 'isVisible', false );\n\t\tthis.set( 'withArrow', true );\n\t\tthis.set( 'class', undefined );\n\n\t\tthis._pinWhenIsVisibleCallback = null;\n\t\tthis.content = this.createCollection();\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-balloon-panel',\n\t\t\t\t\tbind.to( 'position', value => `ck-balloon-panel_${ value }` ),\n\t\t\t\t\tbind.if( 'isVisible', 'ck-balloon-panel_visible' ),\n\t\t\t\t\tbind.if( 'withArrow', 'ck-balloon-panel_with-arrow' ),\n\t\t\t\t\tbind.to( 'class' )\n\t\t\t\t],\n\n\t\t\t\tstyle: {\n\t\t\t\t\ttop: bind.to( 'top', toPx ),\n\t\t\t\t\tleft: bind.to( 'left', toPx )\n\t\t\t\t}\n\t\t\t},\n\n\t\t\tchildren: this.content\n\t\t} );\n\t}\n\n\t/**\n\t * Shows the panel.\n\t *\n\t * See {@link #isVisible}.\n\t */\n\tpublic show(): void {\n\t\tthis.isVisible = true;\n\t}\n\n\t/**\n\t * Hides the panel.\n\t *\n\t * See {@link #isVisible}.\n\t */\n\tpublic hide(): void {\n\t\tthis.isVisible = false;\n\t}\n\n\t/**\n\t * Attaches the panel to a specified {@link module:utils/dom/position~Options#target} with a\n\t * smart positioning heuristics that chooses from available positions to make sure the panel\n\t * is visible to the user i.e. within the limits of the viewport.\n\t *\n\t * This method accepts configuration {@link module:utils/dom/position~Options options}\n\t * to set the `target`, optional `limiter` and `positions` the balloon should choose from.\n\t *\n\t * ```ts\n\t * const panel = new BalloonPanelView( locale );\n\t * const positions = BalloonPanelView.defaultPositions;\n\t *\n\t * panel.render();\n\t *\n\t * // Attach the panel to an element with the \"target\" id DOM.\n\t * panel.attachTo( {\n\t * \ttarget: document.querySelector( '#target' ),\n\t * \tpositions: [\n\t * \t\tpositions.northArrowSouth,\n\t * \t\tpositions.southArrowNorth\n\t * \t]\n\t * } );\n\t * ```\n\t *\n\t * **Note**: Attaching the panel will also automatically {@link #show} it.\n\t *\n\t * **Note**: An attached panel will not follow its target when the window is scrolled or resized.\n\t * See the {@link #pin} method for a more permanent positioning strategy.\n\t *\n\t * @param options Positioning options compatible with {@link module:utils/dom/position~getOptimalPosition}.\n\t * Default `positions` array is {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView.defaultPositions}.\n\t */\n\tpublic attachTo( options: Partial<PositionOptions> ): void {\n\t\tthis.show();\n\n\t\tconst defaultPositions = BalloonPanelView.defaultPositions;\n\t\tconst positionOptions = Object.assign( {}, {\n\t\t\telement: this.element,\n\t\t\tpositions: [\n\t\t\t\tdefaultPositions.southArrowNorth,\n\t\t\t\tdefaultPositions.southArrowNorthMiddleWest,\n\t\t\t\tdefaultPositions.southArrowNorthMiddleEast,\n\t\t\t\tdefaultPositions.southArrowNorthWest,\n\t\t\t\tdefaultPositions.southArrowNorthEast,\n\t\t\t\tdefaultPositions.northArrowSouth,\n\t\t\t\tdefaultPositions.northArrowSouthMiddleWest,\n\t\t\t\tdefaultPositions.northArrowSouthMiddleEast,\n\t\t\t\tdefaultPositions.northArrowSouthWest,\n\t\t\t\tdefaultPositions.northArrowSouthEast,\n\t\t\t\tdefaultPositions.viewportStickyNorth\n\t\t\t],\n\t\t\tlimiter: defaultLimiterElement,\n\t\t\tfitInViewport: true\n\t\t}, options ) as PositionOptions;\n\n\t\tconst optimalPosition = BalloonPanelView._getOptimalPosition( positionOptions );\n\n\t\t// Usually browsers make some problems with super accurate values like 104.345px\n\t\t// so it is better to use int values.\n\t\tconst left = parseInt( optimalPosition.left as any );\n\t\tconst top = parseInt( optimalPosition.top as any );\n\n\t\tconst position = optimalPosition.name as this[ 'position' ];\n\t\tconst config: { withArrow?: boolean } = optimalPosition.config || {};\n\t\tconst { withArrow = true } = config;\n\n\t\tthis.top = top;\n\t\tthis.left = left;\n\t\tthis.position = position;\n\t\tthis.withArrow = withArrow;\n\t}\n\n\t/**\n\t * Works the same way as the {@link #attachTo} method except that the position of the panel is\n\t * continuously updated when:\n\t *\n\t * * any ancestor of the {@link module:utils/dom/position~Options#target}\n\t * or {@link module:utils/dom/position~Options#limiter} is scrolled,\n\t * * the browser window gets resized or scrolled.\n\t *\n\t * Thanks to that, the panel always sticks to the {@link module:utils/dom/position~Options#target}\n\t * and is immune to the changing environment.\n\t *\n\t * ```ts\n\t * const panel = new BalloonPanelView( locale );\n\t * const positions = BalloonPanelView.defaultPositions;\n\t *\n\t * panel.render();\n\t *\n\t * // Pin the panel to an element with the \"target\" id DOM.\n\t * panel.pin( {\n\t * \ttarget: document.querySelector( '#target' ),\n\t * \tpositions: [\n\t * \t\tpositions.northArrowSouth,\n\t * \t\tpositions.southArrowNorth\n\t * \t]\n\t * } );\n\t * ```\n\t *\n\t * To leave the pinned state, use the {@link #unpin} method.\n\t *\n\t * **Note**: Pinning the panel will also automatically {@link #show} it.\n\t *\n\t * @param options Positioning options compatible with {@link module:utils/dom/position~getOptimalPosition}.\n\t * Default `positions` array is {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView.defaultPositions}.\n\t */\n\tpublic pin( options: Partial<PositionOptions> ): void {\n\t\tthis.unpin();\n\n\t\tthis._pinWhenIsVisibleCallback = () => {\n\t\t\tif ( this.isVisible ) {\n\t\t\t\tthis._startPinning( options );\n\t\t\t} else {\n\t\t\t\tthis._stopPinning();\n\t\t\t}\n\t\t};\n\n\t\tthis._startPinning( options );\n\n\t\t// Control the state of the listeners depending on whether the panel is visible\n\t\t// or not.\n\t\t// TODO: Use on() (https://github.com/ckeditor/ckeditor5-utils/issues/144).\n\t\tthis.listenTo<ObservableChangeEvent>( this, 'change:isVisible', this._pinWhenIsVisibleCallback );\n\t}\n\n\t/**\n\t * Stops pinning the panel, as set up by {@link #pin}.\n\t */\n\tpublic unpin(): void {\n\t\tif ( this._pinWhenIsVisibleCallback ) {\n\t\t\t// Deactivate listeners attached by pin().\n\t\t\tthis._stopPinning();\n\n\t\t\t// Deactivate the panel pin() control logic.\n\t\t\t// TODO: Use off() (https://github.com/ckeditor/ckeditor5-utils/issues/144).\n\t\t\tthis.stopListening( this, 'change:isVisible', this._pinWhenIsVisibleCallback );\n\n\t\t\tthis._pinWhenIsVisibleCallback = null;\n\n\t\t\tthis.hide();\n\t\t}\n\t}\n\n\t/**\n\t * Starts managing the pinned state of the panel. See {@link #pin}.\n\t *\n\t * @param options Positioning options compatible with {@link module:utils/dom/position~getOptimalPosition}.\n\t */\n\tprivate _startPinning( options: Partial<PositionOptions> ) {\n\t\tthis.attachTo( options );\n\n\t\tconst targetElement = getDomElement( options.target );\n\t\tconst limiterElement = options.limiter ? getDomElement( options.limiter ) : defaultLimiterElement;\n\n\t\t// Then we need to listen on scroll event of eny element in the document.\n\t\tthis.listenTo( global.document, 'scroll', ( evt, domEvt ) => {\n\t\t\tconst scrollTarget = domEvt.target as Element;\n\n\t\t\t// The position needs to be updated if the positioning target is within the scrolled element.\n\t\t\tconst isWithinScrollTarget = targetElement && scrollTarget.contains( targetElement );\n\n\t\t\t// The position needs to be updated if the positioning limiter is within the scrolled element.\n\t\t\tconst isLimiterWithinScrollTarget = limiterElement && scrollTarget.contains( limiterElement );\n\n\t\t\t// The positioning target and/or limiter can be a Rect, object etc..\n\t\t\t// There's no way to optimize the listener then.\n\t\t\tif ( isWithinScrollTarget || isLimiterWithinScrollTarget || !targetElement || !limiterElement ) {\n\t\t\t\tthis.attachTo( options );\n\t\t\t}\n\t\t}, { useCapture: true } );\n\n\t\t// We need to listen on window resize event and update position.\n\t\tthis.listenTo( global.window, 'resize', () => {\n\t\t\tthis.attachTo( options );\n\t\t} );\n\t}\n\n\t/**\n\t * Stops managing the pinned state of the panel. See {@link #pin}.\n\t */\n\tprivate _stopPinning(): void {\n\t\tthis.stopListening( global.document, 'scroll' );\n\t\tthis.stopListening( global.window, 'resize' );\n\t}\n\n\t/**\n\t * A side offset of the arrow tip from the edge of the balloon. Controlled by CSS.\n\t *\n\t * ```\n\t *\t\t ┌───────────────────────┐\n\t *\t\t │ │\n\t *\t\t │ Balloon │\n\t *\t\t │ Content │\n\t *\t\t │ │\n\t *\t\t └──+ +───────────────┘\n\t *\t\t | \\ /\n\t *\t\t | \\/\n\t *\t\t>┼─────┼< ─────────────────────── side offset\n\t *\n\t * ```\n\t *\n\t * @default 25\n\t */\n\tpublic static arrowSideOffset = 25;\n\n\t/**\n\t * A height offset of the arrow from the edge of the balloon. Controlled by CSS.\n\t *\n\t * ```\n\t *\t\t ┌───────────────────────┐\n\t *\t\t │ │\n\t *\t\t │ Balloon │\n\t *\t\t │ Content │ -- arrow height offset\n\t *\t\t │ │ V\n\t *\t\t └──+ +───────────────┘ --- ─┼───────\n\t *\t\t \\ / │\n\t *\t\t \\/ │\n\t *\t\t────────────────────────────────┼───────\n\t *\t\t ^\n\t *\n\t *\n\t *\t\t>┼────┼< arrow height offset\n\t *\t\t │ │\n\t *\t\t │ ┌────────────────────────┐\n\t *\t\t │ │ │\n\t *\t\t │ │\n\t *\t\t │ Balloon │\n\t *\t\t │ ╲ Content │\n\t *\t\t │ ╲ │\n\t *\t\t │ │ │\n\t *\t\t │ └────────────────────────┘\n\t * ```\n\t *\n\t * @default 10\n\t*/\n\tpublic static arrowHeightOffset = 10;\n\n\t/**\n\t * A vertical offset of the balloon panel from the edge of the viewport if sticky.\n\t * It helps in accessing toolbar buttons underneath the balloon panel.\n\t *\n\t * ```\n\t *\t\t ┌───────────────────────────────────────────────────┐\n\t *\t\t │ Target │\n\t *\t\t │ │\n\t *\t\t │ /── vertical offset │\n\t *\t\t┌─────────────────────────────V─────────────────────────┐\n\t *\t\t│ Toolbar ┌─────────────┐ │\n\t *\t\t├────────────────────│ Balloon │────────────────────┤\n\t *\t\t│ │ └─────────────┘ │ │\n\t *\t\t│ │ │ │\n\t *\t\t│ │ │ │\n\t *\t\t│ │ │ │\n\t *\t\t│ └───────────────────────────────────────────────────┘ │\n\t *\t\t│ Viewport │\n\t *\t\t└───────────────────────────────────────────────────────┘\n\t * ```\n\t *\n\t * @default 20\n\t */\n\tpublic static stickyVerticalOffset = 20;\n\n\t/**\n\t * Function used to calculate the optimal position for the balloon.\n\t */\n\tprivate static _getOptimalPosition = getOptimalPosition;\n\n\t/**\n\t * A default set of positioning functions used by the balloon panel view\n\t * when attaching using the {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView#attachTo} method.\n\t *\n\t * The available positioning functions are as follows:\n\t *\n\t * **North west**\n\t *\n\t * * `northWestArrowSouthWest`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * * `northWestArrowSouthMiddleWest`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * * `northWestArrowSouth`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * * `northWestArrowSouthMiddleEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * * `northWestArrowSouthEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * **North**\n\t *\n\t * * `northArrowSouthWest`\n\t *\n\t * ```\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t *\t\t V\n\t *\t\t[ Target ]\n\t * ```\n\t *\n\t * * `northArrowSouthMiddleWest`\n\t *\n\t * ```\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t *\t\t V\n\t *\t\t[ Target ]\n\t * ```\n\t * * `northArrowSouth`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * * `northArrowSouthMiddleEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * * `northArrowSouthEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * **North east**\n\t *\n\t * * `northEastArrowSouthWest`\n\t *\n\t * ```\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t *\t\t V\n\t *\t\t[ Target ]\n\t * ```\n\t *\n\t * * `northEastArrowSouthMiddleWest`\n\t *\n\t * ```\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t *\t\t V\n\t *\t\t[ Target ]\n\t * ```\n\t *\n\t * * `northEastArrowSouth`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t[ Target ]\n\t * ```\n\t *\n\t * * `northEastArrowSouthMiddleEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * * `northEastArrowSouthEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t *\t\t V\n\t *\t\t [ Target ]\n\t * ```\n\t *\n\t * **South**\n\t *\n\t * * `southArrowNorthWest`\n\t *\n\t * ```\n\t *\t\t[ Target ]\n\t *\t\t ^\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t * ```\n\t *\n\t * * `southArrowNorthMiddleWest`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t * ```\n\t *\n\t * * `southArrowNorth`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southArrowNorthMiddleEast`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t * ```\n\t *\n\t * * `southArrowNorthEast`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * **South west**\n\t *\n\t * * `southWestArrowNorthWest`\n\t *\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southWestArrowNorthMiddleWest`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t * ```\n\t *\n\t * * `southWestArrowNorth`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southWestArrowNorthMiddleEast`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southWestArrowNorthEast`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * **South east**\n\t *\n\t * * `southEastArrowNorthWest`\n\t *\n\t * ```\n\t *\t\t[ Target ]\n\t *\t\t ^\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t * ```\n\t *\n\t * * `southEastArrowNorthMiddleWest`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t +-----------------+\n\t *\t\t | Balloon |\n\t *\t\t +-----------------+\n\t * ```\n\t *\n\t * * `southEastArrowNorth`\n\t *\n\t * ```\n\t *\t\t[ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southEastArrowNorthMiddleEast`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * * `southEastArrowNorthEast`\n\t *\n\t * ```\n\t *\t\t [ Target ]\n\t *\t\t ^\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * **West**\n\t *\n\t * * `westArrowEast`\n\t *\n\t * ```\n\t *\t\t+-----------------+\n\t *\t\t| Balloon |>[ Target ]\n\t *\t\t+-----------------+\n\t * ```\n\t *\n\t * **East**\n\t *\n\t * * `eastArrowWest`\n\t *\n\t * ```\n\t *\t\t +-----------------+\n\t *\t\t[ Target ]<| Balloon |\n\t *\t\t +-----------------+\n\t * ```\n\t *\n\t * **Sticky**\n\t *\n\t * * `viewportStickyNorth`\n\t *\n\t * ```\n\t *\t\t +---------------------------+\n\t *\t\t | [ Target ] |\n\t *\t\t | |\n\t *\t\t+-----------------------------------+\n\t *\t\t| | +-----------------+ | |\n\t *\t\t| | | Balloon | | |\n\t *\t\t| | +-----------------+ | |\n\t *\t\t| | | |\n\t *\t\t| | | |\n\t *\t\t| | | |\n\t *\t\t| | | |\n\t *\t\t| +---------------------------+ |\n\t *\t\t| Viewport |\n\t *\t\t+-----------------------------------+\n\t * ```\n\t *\n\t * See {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView#attachTo}.\n\t *\n\t * Positioning functions must be compatible with {@link module:utils/dom/position~Position}.\n\t *\n\t * Default positioning functions with customized offsets can be generated using\n\t * {@link module:ui/panel/balloon/balloonpanelview~generatePositions}.\n\t *\n\t * The name that the position function returns will be reflected in the balloon panel's class that\n\t * controls the placement of the \"arrow\". See {@link #position} to learn more.\n\t */\n\tpublic static defaultPositions = generatePositions();\n}\n\n/**\n * Returns the DOM element for given object or null, if there is none,\n * e.g. when the passed object is a Rect instance or so.\n */\nfunction getDomElement( object: any ): HTMLElement | null {\n\tif ( isElement( object ) ) {\n\t\treturn object;\n\t}\n\n\tif ( isRange( object ) ) {\n\t\treturn object.commonAncestorContainer as any;\n\t}\n\n\tif ( typeof object == 'function' ) {\n\t\treturn getDomElement( object() );\n\t}\n\n\treturn null;\n}\n\n/**\n * Returns available {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView}\n * {@link module:utils/dom/position~PositioningFunction positioning functions} adjusted by the specific offsets.\n *\n * @internal\n * @param options Options to generate positions. If not specified, this helper will simply return\n * {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView.defaultPositions}.\n * @param options.sideOffset A custom side offset (in pixels) of each position. If\n * not specified, {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView.arrowSideOffset the default value}\n * will be used.\n * @param options.heightOffset A custom height offset (in pixels) of each position. If\n * not specified, {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView.arrowHeightOffset the default value}\n * will be used.\n * @param options.stickyVerticalOffset A custom offset (in pixels) of the `viewportStickyNorth` positioning function.\n * If not specified, {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView.stickyVerticalOffset the default value}\n * will be used.\n * @param options.config Additional configuration of the balloon balloon panel view.\n * Currently only {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView#withArrow} is supported. Learn more\n * about {@link module:utils/dom/position~PositioningFunction positioning functions}.\n */\nexport function generatePositions( options: {\n\tsideOffset?: number;\n\theightOffset?: number;\n\tstickyVerticalOffset?: number;\n\tconfig?: object;\n} = {} ): Record<string, PositioningFunction> {\n\tconst {\n\t\tsideOffset = BalloonPanelView.arrowSideOffset,\n\t\theightOffset = BalloonPanelView.arrowHeightOffset,\n\t\tstickyVerticalOffset = BalloonPanelView.stickyVerticalOffset,\n\t\tconfig\n\t} = options;\n\n\treturn {\n\t\t// ------- North west\n\n\t\tnorthWestArrowSouthWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left - sideOffset,\n\t\t\tname: 'arrow_sw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthWestArrowSouthMiddleWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left - ( balloonRect.width * .25 ) - sideOffset,\n\t\t\tname: 'arrow_smw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthWestArrowSouth: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left - balloonRect.width / 2,\n\t\t\tname: 'arrow_s',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthWestArrowSouthMiddleEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left - ( balloonRect.width * .75 ) + sideOffset,\n\t\t\tname: 'arrow_sme',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthWestArrowSouthEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left - balloonRect.width + sideOffset,\n\t\t\tname: 'arrow_se',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\t// ------- North\n\n\t\tnorthArrowSouthWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - sideOffset,\n\t\t\tname: 'arrow_sw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthArrowSouthMiddleWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - ( balloonRect.width * .25 ) - sideOffset,\n\t\t\tname: 'arrow_smw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthArrowSouth: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - balloonRect.width / 2,\n\t\t\tname: 'arrow_s',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthArrowSouthMiddleEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - ( balloonRect.width * .75 ) + sideOffset,\n\t\t\tname: 'arrow_sme',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthArrowSouthEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - balloonRect.width + sideOffset,\n\t\t\tname: 'arrow_se',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\t// ------- North east\n\n\t\tnorthEastArrowSouthWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.right - sideOffset,\n\t\t\tname: 'arrow_sw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthEastArrowSouthMiddleWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.right - ( balloonRect.width * .25 ) - sideOffset,\n\t\t\tname: 'arrow_smw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthEastArrowSouth: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.right - balloonRect.width / 2,\n\t\t\tname: 'arrow_s',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthEastArrowSouthMiddleEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.right - ( balloonRect.width * .75 ) + sideOffset,\n\t\t\tname: 'arrow_sme',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tnorthEastArrowSouthEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getNorthTop( targetRect, balloonRect ),\n\t\t\tleft: targetRect.right - balloonRect.width + sideOffset,\n\t\t\tname: 'arrow_se',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\t// ------- South west\n\n\t\tsouthWestArrowNorthWest: targetRect => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left - sideOffset,\n\t\t\tname: 'arrow_nw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthWestArrowNorthMiddleWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left - ( balloonRect.width * .25 ) - sideOffset,\n\t\t\tname: 'arrow_nmw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthWestArrowNorth: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left - balloonRect.width / 2,\n\t\t\tname: 'arrow_n',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthWestArrowNorthMiddleEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left - ( balloonRect.width * .75 ) + sideOffset,\n\t\t\tname: 'arrow_nme',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthWestArrowNorthEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left - balloonRect.width + sideOffset,\n\t\t\tname: 'arrow_ne',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\t// ------- South\n\n\t\tsouthArrowNorthWest: targetRect => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - sideOffset,\n\t\t\tname: 'arrow_nw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthArrowNorthMiddleWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - ( balloonRect.width * 0.25 ) - sideOffset,\n\t\t\tname: 'arrow_nmw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthArrowNorth: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - balloonRect.width / 2,\n\t\t\tname: 'arrow_n',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthArrowNorthMiddleEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - ( balloonRect.width * 0.75 ) + sideOffset,\n\t\t\tname: 'arrow_nme',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthArrowNorthEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.left + targetRect.width / 2 - balloonRect.width + sideOffset,\n\t\t\tname: 'arrow_ne',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\t// ------- South east\n\n\t\tsouthEastArrowNorthWest: targetRect => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.right - sideOffset,\n\t\t\tname: 'arrow_nw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthEastArrowNorthMiddleWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.right - ( balloonRect.width * .25 ) - sideOffset,\n\t\t\tname: 'arrow_nmw',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthEastArrowNorth: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.right - balloonRect.width / 2,\n\t\t\tname: 'arrow_n',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthEastArrowNorthMiddleEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.right - ( balloonRect.width * .75 ) + sideOffset,\n\t\t\tname: 'arrow_nme',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\tsouthEastArrowNorthEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: getSouthTop( targetRect ),\n\t\t\tleft: targetRect.right - balloonRect.width + sideOffset,\n\t\t\tname: 'arrow_ne',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\t// ------- West\n\n\t\twestArrowEast: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: targetRect.top + targetRect.height / 2 - balloonRect.height / 2,\n\t\t\tleft: targetRect.left - balloonRect.width - heightOffset,\n\t\t\tname: 'arrow_e',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\t// ------- East\n\n\t\teastArrowWest: ( targetRect, balloonRect ) => ( {\n\t\t\ttop: targetRect.top + targetRect.height / 2 - balloonRect.height / 2,\n\t\t\tleft: targetRect.right + heightOffset,\n\t\t\tname: 'arrow_w',\n\t\t\t...( config && { config } )\n\t\t} ),\n\n\t\t// ------- Sticky\n\n\t\tviewportStickyNorth: ( targetRect, balloonRect, viewportRect ) => {\n\t\t\tif ( !targetRect.getIntersection( viewportRect! ) ) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\treturn {\n\t\t\t\ttop: viewportRect!.top + stickyVerticalOffset,\n\t\t\t\tleft: targetRect.left + targetRect.width / 2 - balloonRect.width / 2,\n\t\t\t\tname: 'arrowless',\n\t\t\t\tconfig: {\n\t\t\t\t\twithArrow: false,\n\t\t\t\t\t...config\n\t\t\t\t}\n\t\t\t};\n\t\t}\n\t};\n\n\t/**\n\t * Returns the top coordinate for positions starting with `north*`.\n\t *\n\t * @param targetRect A rect of the target.\n\t * @param balloonRect A rect of the balloon.\n\t */\n\tfunction getNorthTop( targetRect: Rect, balloonRect: Rect ) {\n\t\treturn targetRect.top - balloonRect.height - heightOffset;\n\t}\n\n\t/**\n\t * Returns the top coordinate for positions starting with `south*`.\n\t *\n\t * @param targetRect A rect of the target.\n\t */\n\tfunction getSouthTop( targetRect: Rect ) {\n\t\treturn targetRect.bottom + heightOffset;\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./tooltip.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/tooltipmanager\n */\n\nimport View from './view';\nimport BalloonPanelView, { generatePositions } from './panel/balloon/balloonpanelview';\nimport type { EditorUIUpdateEvent } from './editorui/editorui';\n\nimport {\n\tDomEmitterMixin,\n\tResizeObserver,\n\tfirst,\n\tglobal,\n\tisVisible,\n\ttype EventInfo,\n\ttype PositioningFunction\n} from '@ckeditor/ckeditor5-utils';\n\nimport type { Editor } from '@ckeditor/ckeditor5-core';\n\nimport { isElement, debounce, type DebouncedFunc } from 'lodash-es';\n\nimport '../theme/components/tooltip/tooltip.css';\n\nconst BALLOON_CLASS = 'ck-tooltip';\n\n/**\n * A tooltip manager class for the UI of the editor.\n *\n * **Note**: Most likely you do not have to use the `TooltipManager` API listed below in order to display tooltips. Popular\n * {@glink framework/architecture/ui-library UI components} support tooltips out-of-the-box via observable properties\n * (see {@link module:ui/button/buttonview~ButtonView#tooltip} and {@link module:ui/button/buttonview~ButtonView#tooltipPosition}).\n *\n * # Displaying tooltips\n *\n * To display a tooltip, set `data-cke-tooltip-text` attribute on any DOM element:\n *\n * ```ts\n * domElement.dataset.ckeTooltipText = 'My tooltip';\n * ```\n *\n * The tooltip will show up whenever the user moves the mouse over the element or the element gets focus in DOM.\n *\n * # Positioning tooltips\n *\n * To change the position of the tooltip, use the `data-cke-tooltip-position` attribute (`s`, `se`, `sw`, `n`, `e`, or `w`):\n *\n * ```ts\n * domElement.dataset.ckeTooltipText = 'Tooltip to the north';\n * domElement.dataset.ckeTooltipPosition = 'n';\n * ```\n *\n * # Disabling tooltips\n *\n * In order to disable the tooltip temporarily, use the `data-cke-tooltip-disabled` attribute:\n *\n * ```ts\n * domElement.dataset.ckeTooltipText = 'Disabled. For now.';\n * domElement.dataset.ckeTooltipDisabled = 'true';\n * ```\n *\n * # Styling tooltips\n *\n * By default, the tooltip has `.ck-tooltip` class and its text inner `.ck-tooltip__text`.\n *\n * If your tooltip requires custom styling, using `data-cke-tooltip-class` attribute will add additional class to the balloon\n * displaying the tooltip:\n *\n * ```ts\n * domElement.dataset.ckeTooltipText = 'Tooltip with a red text';\n * domElement.dataset.ckeTooltipClass = 'my-class';\n * ```\n *\n * ```css\n * .ck.ck-tooltip.my-class { color: red }\n * ```\n *\n * **Note**: This class is a singleton. All editor instances re-use the same instance loaded by\n * {@link module:ui/editorui/editorui~EditorUI} of the first editor.\n */\nexport default class TooltipManager extends DomEmitterMixin() {\n\t/**\n\t * The view rendering text of the tooltip.\n\t */\n\tpublic readonly tooltipTextView!: View & { text: string };\n\n\t/**\n\t * The instance of the balloon panel that renders and positions the tooltip.\n\t */\n\tpublic readonly balloonPanelView!: BalloonPanelView;\n\n\t/**\n\t * A set of default {@link module:utils/dom/position~PositioningFunction positioning functions} used by the `TooltipManager`\n\t * to pin tooltips in different positions.\n\t */\n\tpublic static defaultBalloonPositions = generatePositions( {\n\t\theightOffset: 5,\n\t\tsideOffset: 13\n\t} );\n\n\t/**\n\t * Stores the reference to the DOM element the tooltip is attached to. `null` when there's no tooltip\n\t * in the UI.\n\t */\n\tprivate _currentElementWithTooltip: HTMLElement | null = null;\n\n\t/**\n\t * Stores the current tooltip position. `null` when there's no tooltip in the UI.\n\t */\n\tprivate _currentTooltipPosition: TooltipPosition | null = null;\n\n\t/**\n\t * An instance of the resize observer that keeps track on target element visibility,\n\t * when it hides the tooltip should also disappear.\n\t *\n\t * {@link module:core/editor/editorconfig~EditorConfig#balloonToolbar configuration}.\n\t */\n\tprivate _resizeObserver: ResizeObserver | null = null;\n\n\t/**\n\t * A debounced version of {@link #_pinTooltip}. Tooltips show with a delay to avoid flashing and\n\t * to improve the UX.\n\t */\n\tprivate _pinTooltipDebounced!: DebouncedFunc<( targetDomElement: HTMLElement, data: TooltipData ) => void>;\n\n\tprivate readonly _watchdogExcluded!: true;\n\n\t/**\n\t * A set of editors the single tooltip manager instance must listen to.\n\t * This is mostly to handle `EditorUI#update` listeners from individual editors.\n\t */\n\tprivate static _editors = new Set<Editor>();\n\n\t/**\n\t * A reference to the `TooltipManager` instance. The class is a singleton and as such,\n\t * successive attempts at creating instances should return this instance.\n\t */\n\tprivate static _instance: TooltipManager | null = null;\n\n\t/**\n\t * Creates an instance of the tooltip manager.\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper();\n\n\t\tTooltipManager._editors.add( editor );\n\n\t\t// TooltipManager must be a singleton. Multiple instances would mean multiple tooltips attached\n\t\t// to the same DOM element with data-cke-tooltip-* attributes.\n\t\tif ( TooltipManager._instance ) {\n\t\t\treturn TooltipManager._instance;\n\t\t}\n\n\t\tTooltipManager._instance = this;\n\n\t\tthis.tooltipTextView = new View( editor.locale ) as any;\n\t\tthis.tooltipTextView.set( 'text', '' );\n\t\tthis.tooltipTextView.setTemplate( {\n\t\t\ttag: 'span',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-tooltip__text'\n\t\t\t\t]\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttext: this.tooltipTextView.bindTemplate.to( 'text' )\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\n\t\tthis.balloonPanelView = new BalloonPanelView( editor.locale );\n\t\tthis.balloonPanelView.class = BALLOON_CLASS;\n\t\tthis.balloonPanelView.content.add( this.tooltipTextView );\n\n\t\tthis._pinTooltipDebounced = debounce( this._pinTooltip, 600 );\n\n\t\tthis.listenTo( global.document, 'mouseenter', this._onEnterOrFocus.bind( this ), { useCapture: true } );\n\t\tthis.listenTo( global.document, 'mouseleave', this._onLeaveOrBlur.bind( this ), { useCapture: true } );\n\n\t\tthis.listenTo( global.document, 'focus', this._onEnterOrFocus.bind( this ), { useCapture: true } );\n\t\tthis.listenTo( global.document, 'blur', this._onLeaveOrBlur.bind( this ), { useCapture: true } );\n\n\t\tthis.listenTo( global.document, 'scroll', this._onScroll.bind( this ), { useCapture: true } );\n\n\t\t// Because this class is a singleton, its only instance is shared across all editors and connects them through the reference.\n\t\t// This causes issues with the ContextWatchdog. When an error is thrown in one editor, the watchdog traverses the references\n\t\t// and (because of shared tooltip manager) figures that the error affects all editors and restarts them all.\n\t\t// This flag, excludes tooltip manager instance from the traversal and brings ContextWatchdog back to normal.\n\t\t// More in https://github.com/ckeditor/ckeditor5/issues/12292.\n\t\tthis._watchdogExcluded = true;\n\t}\n\n\t/**\n\t * Destroys the tooltip manager.\n\t *\n\t * **Note**: The manager singleton cannot be destroyed until all editors that use it are destroyed.\n\t *\n\t * @param editor The editor the manager was created for.\n\t */\n\tpublic destroy( editor: Editor ): void {\n\t\tconst editorBodyViewCollection = editor.ui.view && editor.ui.view.body;\n\n\t\tTooltipManager._editors.delete( editor );\n\t\tthis.stopListening( editor.ui );\n\n\t\t// Prevent the balloon panel from being destroyed in the EditorUI#destroy() cascade. It should be destroyed along\n\t\t// with the last editor only (https://github.com/ckeditor/ckeditor5/issues/12602).\n\t\tif ( editorBodyViewCollection && editorBodyViewCollection.has( this.balloonPanelView ) ) {\n\t\t\teditorBodyViewCollection.remove( this.balloonPanelView );\n\t\t}\n\n\t\tif ( !TooltipManager._editors.size ) {\n\t\t\tthis._unpinTooltip();\n\t\t\tthis.balloonPanelView.destroy();\n\t\t\tthis.stopListening();\n\n\t\t\tTooltipManager._instance = null;\n\t\t}\n\t}\n\n\t/**\n\t * Returns {@link #balloonPanelView} {@link module:utils/dom/position~PositioningFunction positioning functions} for a given position\n\t * name.\n\t *\n\t * @param position Name of the position (`s`, `se`, `sw`, `n`, `e`, or `w`).\n\t * @returns Positioning functions to be used by the {@link #balloonPanelView}.\n\t */\n\tpublic static getPositioningFunctions( position: TooltipPosition ): Array<PositioningFunction> {\n\t\tconst defaultPositions = TooltipManager.defaultBalloonPositions;\n\n\t\treturn {\n\t\t\t// South is most popular. We can use positioning heuristics to avoid clipping by the viewport with the sane fallback.\n\t\t\ts: [\n\t\t\t\tdefaultPositions.southArrowNorth,\n\t\t\t\tdefaultPositions.southArrowNorthEast,\n\t\t\t\tdefaultPositions.southArrowNorthWest\n\t\t\t],\n\t\t\tn: [ defaultPositions.northArrowSouth ],\n\t\t\te: [ defaultPositions.eastArrowWest ],\n\t\t\tw: [ defaultPositions.westArrowEast ],\n\t\t\tsw: [ defaultPositions.southArrowNorthEast ],\n\t\t\tse: [ defaultPositions.southArrowNorthWest ]\n\t\t}[ position ];\n\t}\n\n\t/**\n\t * Handles displaying tooltips on `mouseenter` and `focus` in DOM.\n\t *\n\t * @param evt An object containing information about the fired event.\n\t * @param domEvent The DOM event.\n\t */\n\tprivate _onEnterOrFocus( evt: unknown, { target }: any ) {\n\t\tconst elementWithTooltipAttribute = getDescendantWithTooltip( target );\n\n\t\t// Abort when there's no descendant needing tooltip.\n\t\tif ( !elementWithTooltipAttribute ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Abort to avoid flashing when, for instance:\n\t\t// * a tooltip is displayed for a focused element, then the same element gets mouseentered,\n\t\t// * a tooltip is displayed for an element via mouseenter, then the focus moves to the same element.\n\t\tif ( elementWithTooltipAttribute === this._currentElementWithTooltip ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._unpinTooltip();\n\n\t\tthis._pinTooltipDebounced( elementWithTooltipAttribute, getTooltipData( elementWithTooltipAttribute ) );\n\t}\n\n\t/**\n\t * Handles hiding tooltips on `mouseleave` and `blur` in DOM.\n\t *\n\t * @param evt An object containing information about the fired event.\n\t * @param domEvent The DOM event.\n\t */\n\tprivate _onLeaveOrBlur( evt: EventInfo, { target, relatedTarget }: any ) {\n\t\tif ( evt.name === 'mouseleave' ) {\n\t\t\t// Don't act when the event does not concern a DOM element (e.g. a mouseleave out of an entire document),\n\t\t\tif ( !isElement( target ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// If a tooltip is currently visible, don't act for a targets other than the one it is attached to.\n\t\t\t// For instance, a random mouseleave far away in the page should not unpin the tooltip that was pinned because\n\t\t\t// of a previous focus. Only leaving the same element should hide the tooltip.\n\t\t\tif ( this._currentElementWithTooltip && target !== this._currentElementWithTooltip ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst descendantWithTooltip = getDescendantWithTooltip( target );\n\t\t\tconst relatedDescendantWithTooltip = getDescendantWithTooltip( relatedTarget );\n\n\t\t\t// Unpin when the mouse was leaving element with a tooltip to a place which does not have or has a different tooltip.\n\t\t\t// Note that this should happen whether the tooltip is already visible or not, for instance, it could be invisible but queued\n\t\t\t// (debounced): it should get canceled.\n\t\t\tif ( descendantWithTooltip && descendantWithTooltip !== relatedDescendantWithTooltip ) {\n\t\t\t\tthis._unpinTooltip();\n\t\t\t}\n\t\t}\n\t\telse {\n\t\t\t// If a tooltip is currently visible, don't act for a targets other than the one it is attached to.\n\t\t\t// For instance, a random blur in the web page should not unpin the tooltip that was pinned because of a previous mouseenter.\n\t\t\tif ( this._currentElementWithTooltip && target !== this._currentElementWithTooltip ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Note that unpinning should happen whether the tooltip is already visible or not, for instance, it could be invisible but\n\t\t\t// queued (debounced): it should get canceled (e.g. quick focus then quick blur using the keyboard).\n\t\t\tthis._unpinTooltip();\n\t\t}\n\t}\n\n\t/**\n\t * Handles hiding tooltips on `scroll` in DOM.\n\t *\n\t * @param evt An object containing information about the fired event.\n\t * @param domEvent The DOM event.\n\t */\n\tprivate _onScroll( evt: unknown, { target }: any ) {\n\t\t// No tooltip, no reason to react on scroll.\n\t\tif ( !this._currentElementWithTooltip ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// When scrolling a container that has both the balloon and the current element (common ancestor), the balloon can remain\n\t\t// visible (e.g. scrolling ≤body>). Otherwise, to avoid glitches (clipping, lagging) better just hide the tooltip.\n\t\t// Also, don't do anything when scrolling an unrelated DOM element that has nothing to do with the current element and the balloon.\n\t\tif ( target.contains( this.balloonPanelView.element ) && target.contains( this._currentElementWithTooltip ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._unpinTooltip();\n\t}\n\n\t/**\n\t * Pins the tooltip to a specific DOM element.\n\t *\n\t * @param options.text Text of the tooltip to display.\n\t * @param options.position The position of the tooltip.\n\t * @param options.cssClass Additional CSS class of the balloon with the tooltip.\n\t */\n\tprivate _pinTooltip(\n\t\ttargetDomElement: HTMLElement,\n\t\t{ text, position, cssClass }: TooltipData\n\t): void {\n\t\t// Use the body collection of the first editor.\n\t\tconst bodyViewCollection = first( TooltipManager._editors.values() )!.ui.view.body;\n\n\t\tif ( !bodyViewCollection.has( this.balloonPanelView ) ) {\n\t\t\tbodyViewCollection.add( this.balloonPanelView );\n\t\t}\n\n\t\tthis.tooltipTextView.text = text;\n\n\t\tthis.balloonPanelView.pin( {\n\t\t\ttarget: targetDomElement,\n\t\t\tpositions: TooltipManager.getPositioningFunctions( position )\n\t\t} );\n\n\t\tthis._resizeObserver = new ResizeObserver( targetDomElement, () => {\n\t\t\t// The ResizeObserver will call its callback when the target element hides and the tooltip\n\t\t\t// should also disappear (https://github.com/ckeditor/ckeditor5/issues/12492).\n\t\t\tif ( !isVisible( targetDomElement ) ) {\n\t\t\t\tthis._unpinTooltip();\n\t\t\t}\n\t\t} );\n\n\t\tthis.balloonPanelView.class = [ BALLOON_CLASS, cssClass ]\n\t\t\t.filter( className => className )\n\t\t\t.join( ' ' );\n\n\t\t// Start responding to changes in editor UI or content layout. For instance, when collaborators change content\n\t\t// and a contextual toolbar attached to a content starts to move (and so should move the tooltip).\n\t\t// Note: Using low priority to let other listeners that position contextual toolbars etc. to react first.\n\t\tfor ( const editor of TooltipManager._editors ) {\n\t\t\tthis.listenTo<EditorUIUpdateEvent>( editor.ui, 'update', this._updateTooltipPosition.bind( this ), { priority: 'low' } );\n\t\t}\n\n\t\tthis._currentElementWithTooltip = targetDomElement;\n\t\tthis._currentTooltipPosition = position;\n\t}\n\n\t/**\n\t * Unpins the tooltip and cancels all queued pinning.\n\t */\n\tprivate _unpinTooltip() {\n\t\tthis._pinTooltipDebounced.cancel();\n\n\t\tthis.balloonPanelView.unpin();\n\n\t\tfor ( const editor of TooltipManager._editors ) {\n\t\t\tthis.stopListening( editor.ui, 'update' );\n\t\t}\n\n\t\tthis._currentElementWithTooltip = null;\n\t\tthis._currentTooltipPosition = null;\n\n\t\tif ( this._resizeObserver ) {\n\t\t\tthis._resizeObserver.destroy();\n\t\t}\n\t}\n\n\t/**\n\t * Updates the position of the tooltip so it stays in sync with the element it is pinned to.\n\t *\n\t * Hides the tooltip when the element is no longer visible in DOM.\n\t */\n\tprivate _updateTooltipPosition() {\n\t\t// This could happen if the tooltip was attached somewhere in a contextual content toolbar and the toolbar\n\t\t// disappeared (e.g. removed an image).\n\t\tif ( !isVisible( this._currentElementWithTooltip ) ) {\n\t\t\tthis._unpinTooltip();\n\n\t\t\treturn;\n\t\t}\n\n\t\tthis.balloonPanelView.pin( {\n\t\t\ttarget: this._currentElementWithTooltip!,\n\t\t\tpositions: TooltipManager.getPositioningFunctions( this._currentTooltipPosition! )\n\t\t} );\n\t}\n}\n\nexport type TooltipPosition = 's' | 'n' | 'e' | 'w' | 'sw' | 'se';\n\nfunction getDescendantWithTooltip( element: HTMLElement ) {\n\tif ( !isElement( element ) ) {\n\t\treturn null;\n\t}\n\n\treturn element.closest( '[data-cke-tooltip-text]:not([data-cke-tooltip-disabled])' ) as HTMLElement;\n}\n\ninterface TooltipData {\n\ttext: string;\n\tposition: TooltipPosition;\n\tcssClass: string;\n}\n\nfunction getTooltipData( element: HTMLElement ): TooltipData {\n\treturn {\n\t\ttext: element.dataset.ckeTooltipText!,\n\t\tposition: ( element.dataset.ckeTooltipPosition || 's' ) as TooltipPosition,\n\t\tcssClass: element.dataset.ckeTooltipClass || ''\n\t};\n}\n","import debounce from './debounce.js';\nimport isObject from './isObject.js';\n\n/** Error message constants. */\nvar FUNC_ERROR_TEXT = 'Expected a function';\n\n/**\n * Creates a throttled function that only invokes `func` at most once per\n * every `wait` milliseconds. The throttled function comes with a `cancel`\n * method to cancel delayed `func` invocations and a `flush` method to\n * immediately invoke them. Provide `options` to indicate whether `func`\n * should be invoked on the leading and/or trailing edge of the `wait`\n * timeout. The `func` is invoked with the last arguments provided to the\n * throttled function. Subsequent calls to the throttled function return the\n * result of the last `func` invocation.\n *\n * **Note:** If `leading` and `trailing` options are `true`, `func` is\n * invoked on the trailing edge of the timeout only if the throttled function\n * is invoked more than once during the `wait` timeout.\n *\n * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred\n * until to the next tick, similar to `setTimeout` with a timeout of `0`.\n *\n * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/)\n * for details over the differences between `_.throttle` and `_.debounce`.\n *\n * @static\n * @memberOf _\n * @since 0.1.0\n * @category Function\n * @param {Function} func The function to throttle.\n * @param {number} [wait=0] The number of milliseconds to throttle invocations to.\n * @param {Object} [options={}] The options object.\n * @param {boolean} [options.leading=true]\n * Specify invoking on the leading edge of the timeout.\n * @param {boolean} [options.trailing=true]\n * Specify invoking on the trailing edge of the timeout.\n * @returns {Function} Returns the new throttled function.\n * @example\n *\n * // Avoid excessively updating the position while scrolling.\n * jQuery(window).on('scroll', _.throttle(updatePosition, 100));\n *\n * // Invoke `renewToken` when the click event is fired, but not more than once every 5 minutes.\n * var throttled = _.throttle(renewToken, 300000, { 'trailing': false });\n * jQuery(element).on('click', throttled);\n *\n * // Cancel the trailing throttled invocation.\n * jQuery(window).on('popstate', throttled.cancel);\n */\nfunction throttle(func, wait, options) {\n var leading = true,\n trailing = true;\n\n if (typeof func != 'function') {\n throw new TypeError(FUNC_ERROR_TEXT);\n }\n if (isObject(options)) {\n leading = 'leading' in options ? !!options.leading : leading;\n trailing = 'trailing' in options ? !!options.trailing : trailing;\n }\n return debounce(func, wait, {\n 'leading': leading,\n 'maxWait': wait,\n 'trailing': trailing\n });\n}\n\nexport default throttle;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/editorui/poweredby\n */\n\nimport type { Editor, UiConfig } from '@ckeditor/ckeditor5-core';\nimport {\n\tRect,\n\tDomEmitterMixin,\n\tfindClosestScrollableAncestor,\n\tverifyLicense,\n\ttype PositionOptions,\n\ttype Locale\n} from '@ckeditor/ckeditor5-utils';\nimport BalloonPanelView from '../panel/balloon/balloonpanelview';\nimport IconView from '../icon/iconview';\nimport View from '../view';\nimport { throttle, type DebouncedFunc } from 'lodash-es';\n\nimport poweredByIcon from '../../theme/icons/project-logo.svg';\n\nconst ICON_WIDTH = 53;\nconst ICON_HEIGHT = 10;\n// ⚠ Note, whenever changing the threshold, make sure to update the docs/support/managing-ckeditor-logo.md docs\n// as this information is also mentioned there ⚠.\nconst NARROW_ROOT_HEIGHT_THRESHOLD = 50;\nconst NARROW_ROOT_WIDTH_THRESHOLD = 350;\nconst DEFAULT_LABEL = 'Powered by';\nconst OFF_THE_SCREEN_POSITION = {\n\ttop: -99999,\n\tleft: -99999,\n\tname: 'invalid',\n\tconfig: {\n\t\twithArrow: false\n\t}\n};\n\ntype PoweredByConfig = Required<UiConfig>[ 'poweredBy' ];\n\n/**\n * A helper that enables the \"powered by\" feature in the editor and renders a link to the project's\n * webpage next to the bottom of the editable element (editor root, source editing area, etc.) when the editor is focused.\n *\n * @private\n */\nexport default class PoweredBy extends DomEmitterMixin() {\n\t/**\n\t * Editor instance the helper was created for.\n\t */\n\tprivate readonly editor: Editor;\n\n\t/**\n\t * A reference to the balloon panel hosting and positioning the \"powered by\" link and logo.\n\t */\n\tprivate _balloonView: BalloonPanelView | null;\n\n\t/**\n\t * A throttled version of the {@link #_showBalloon} method meant for frequent use to avoid performance loss.\n\t */\n\tprivate _showBalloonThrottled: DebouncedFunc<() => void>;\n\n\t/**\n\t * A reference to the last editable element (root, source editing area, etc.) focused by the user.\n\t * Since the focus can move to other focusable elements in the UI, this reference allows positioning the balloon over the\n\t * right element whether the user is typing or using the UI.\n\t */\n\tprivate _lastFocusedEditableElement: HTMLElement | null;\n\n\t/**\n\t * Creates a \"powered by\" helper for a given editor. The feature is initialized on Editor#ready\n\t * event.\n\t *\n\t * @param editor\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper();\n\n\t\tthis.editor = editor;\n\t\tthis._balloonView = null;\n\t\tthis._lastFocusedEditableElement = null;\n\t\tthis._showBalloonThrottled = throttle( this._showBalloon.bind( this ), 50, { leading: true } );\n\n\t\teditor.on( 'ready', this._handleEditorReady.bind( this ) );\n\t}\n\n\t/**\n\t * Destroys the \"powered by\" helper along with its view.\n\t */\n\tpublic destroy(): void {\n\t\tconst balloon = this._balloonView;\n\n\t\tif ( balloon ) {\n\t\t\t// Balloon gets destroyed by the body collection.\n\t\t\t// The powered by view gets destroyed by the balloon.\n\t\t\tballoon.unpin();\n\t\t\tthis._balloonView = null;\n\t\t}\n\n\t\tthis._showBalloonThrottled.cancel();\n\t\tthis.stopListening();\n\t}\n\n\t/**\n\t * Enables \"powered by\" label once the editor (ui) is ready.\n\t */\n\tprivate _handleEditorReady(): void {\n\t\tconst editor = this.editor;\n\t\tconst forceVisible = !!editor.config.get( 'ui.poweredBy.forceVisible' );\n\n\t\t/* istanbul ignore next -- @preserve */\n\t\tif ( !forceVisible && verifyLicense( editor.config.get( 'licenseKey' ) ) === 'VALID' ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// No view means no body collection to append the powered by balloon to.\n\t\tif ( !editor.ui.view ) {\n\t\t\treturn;\n\t\t}\n\n\t\teditor.ui.focusTracker.on( 'change:isFocused', ( evt, data, isFocused ) => {\n\t\t\tthis._updateLastFocusedEditableElement();\n\n\t\t\tif ( isFocused ) {\n\t\t\t\tthis._showBalloon();\n\t\t\t} else {\n\t\t\t\tthis._hideBalloon();\n\t\t\t}\n\t\t} );\n\n\t\teditor.ui.focusTracker.on( 'change:focusedElement', ( evt, data, focusedElement ) => {\n\t\t\tthis._updateLastFocusedEditableElement();\n\n\t\t\tif ( focusedElement ) {\n\t\t\t\tthis._showBalloon();\n\t\t\t}\n\t\t} );\n\n\t\teditor.ui.on( 'update', () => {\n\t\t\tthis._showBalloonThrottled();\n\t\t} );\n\t}\n\n\t/**\n\t * Creates an instance of the {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView balloon panel}\n\t * with the \"powered by\" view inside ready for positioning.\n\t */\n\tprivate _createBalloonView(): void {\n\t\tconst editor = this.editor;\n\t\tconst balloon = this._balloonView = new BalloonPanelView();\n\t\tconst poweredByConfig = getNormalizedConfig( editor );\n\t\tconst view = new PoweredByView( editor.locale, poweredByConfig.label );\n\n\t\tballoon.content.add( view );\n\t\tballoon.set( {\n\t\t\tclass: 'ck-powered-by-balloon'\n\t\t} );\n\n\t\teditor.ui.view.body.add( balloon );\n\t\teditor.ui.focusTracker.add( balloon.element! );\n\n\t\tthis._balloonView = balloon;\n\t}\n\n\t/**\n\t * Attempts to display the balloon with the \"powered by\" view.\n\t */\n\tprivate _showBalloon(): void {\n\t\tif ( !this._lastFocusedEditableElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst attachOptions = getBalloonAttachOptions( this.editor, this._lastFocusedEditableElement );\n\n\t\tif ( attachOptions ) {\n\t\t\tif ( !this._balloonView ) {\n\t\t\t\tthis._createBalloonView();\n\t\t\t}\n\n\t\t\tthis._balloonView!.pin( attachOptions );\n\t\t}\n\t}\n\n\t/**\n\t * Hides the \"powered by\" balloon if already visible.\n\t */\n\tprivate _hideBalloon(): void {\n\t\tif ( this._balloonView ) {\n\t\t\tthis._balloonView!.unpin();\n\t\t}\n\t}\n\n\t/**\n\t * Updates the {@link #_lastFocusedEditableElement} based on the state of the global focus tracker.\n\t */\n\tprivate _updateLastFocusedEditableElement(): void {\n\t\tconst editor = this.editor;\n\t\tconst isFocused = editor.ui.focusTracker.isFocused;\n\t\tconst focusedElement = editor.ui.focusTracker.focusedElement! as HTMLElement;\n\n\t\tif ( !isFocused || !focusedElement ) {\n\t\t\tthis._lastFocusedEditableElement = null;\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst editableEditorElements = Array.from( editor.ui.getEditableElementsNames() ).map( name => {\n\t\t\treturn editor.ui.getEditableElement( name );\n\t\t} );\n\n\t\tif ( editableEditorElements.includes( focusedElement ) ) {\n\t\t\tthis._lastFocusedEditableElement = focusedElement;\n\t\t} else {\n\t\t\t// If it's none of the editable element, then the focus is somewhere in the UI. Let's display powered by\n\t\t\t// over the first element then.\n\t\t\tthis._lastFocusedEditableElement = editableEditorElements[ 0 ]!;\n\t\t}\n\t}\n}\n\n/**\n * A view displaying a \"powered by\" label and project logo wrapped in a link.\n */\nclass PoweredByView extends View<HTMLDivElement> {\n\t/**\n\t * Created an instance of the \"powered by\" view.\n\t *\n\t * @param locale The localization services instance.\n\t * @param label The label text.\n\t */\n\tconstructor( locale: Locale, label: string | null ) {\n\t\tsuper( locale );\n\n\t\tconst iconView = new IconView();\n\t\tconst bind = this.bindTemplate;\n\n\t\ticonView.set( {\n\t\t\tcontent: poweredByIcon,\n\t\t\tisColorInherited: false\n\t\t} );\n\n\t\ticonView.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tstyle: {\n\t\t\t\t\twidth: ICON_WIDTH + 'px',\n\t\t\t\t\theight: ICON_HEIGHT + 'px'\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [ 'ck', 'ck-powered-by' ],\n\t\t\t\t'aria-hidden': true\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttag: 'a',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\thref: 'https://ckeditor.com/?utm_source=ckeditor&' +\n\t\t\t\t\t\t\t'utm_medium=referral&utm_campaign=701Dn000000hVgmIAE_powered_by_ckeditor_logo',\n\t\t\t\t\t\ttarget: '_blank',\n\t\t\t\t\t\ttabindex: '-1'\n\t\t\t\t\t},\n\t\t\t\t\tchildren: [\n\t\t\t\t\t\t...label ? [\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\ttag: 'span',\n\t\t\t\t\t\t\t\tattributes: {\n\t\t\t\t\t\t\t\t\tclass: [ 'ck', 'ck-powered-by__label' ]\n\t\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\t\tchildren: [ label ]\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t] : [],\n\t\t\t\t\t\ticonView\n\t\t\t\t\t],\n\t\t\t\t\ton: {\n\t\t\t\t\t\tdragstart: bind.to( evt => evt.preventDefault() )\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\t}\n}\n\nfunction getBalloonAttachOptions( editor: Editor, focusedEditableElement: HTMLElement ): Partial<PositionOptions> | null {\n\tconst poweredByConfig = getNormalizedConfig( editor )!;\n\tconst positioningFunction = poweredByConfig.side === 'right' ?\n\t\tgetLowerRightCornerPosition( focusedEditableElement, poweredByConfig ) :\n\t\tgetLowerLeftCornerPosition( focusedEditableElement, poweredByConfig );\n\n\treturn {\n\t\ttarget: focusedEditableElement,\n\t\tpositions: [ positioningFunction ]\n\t};\n}\n\nfunction getLowerRightCornerPosition( focusedEditableElement: HTMLElement, config: PoweredByConfig ) {\n\treturn getLowerCornerPosition( focusedEditableElement, config, ( rootRect, balloonRect ) => {\n\t\treturn rootRect.left + rootRect.width - balloonRect.width - config.horizontalOffset;\n\t} );\n}\n\nfunction getLowerLeftCornerPosition( focusedEditableElement: HTMLElement, config: PoweredByConfig ) {\n\treturn getLowerCornerPosition( focusedEditableElement, config, rootRect => rootRect.left + config.horizontalOffset );\n}\n\nfunction getLowerCornerPosition(\n\tfocusedEditableElement: HTMLElement,\n\tconfig: PoweredByConfig,\n\tgetBalloonLeft: ( editableElementRect: Rect, balloonRect: Rect ) => number\n) {\n\treturn ( editableElementRect: Rect, balloonRect: Rect ) => {\n\t\tconst visibleEditableElementRect = editableElementRect.getVisible();\n\n\t\t// Root cropped by ancestors.\n\t\tif ( !visibleEditableElementRect ) {\n\t\t\treturn OFF_THE_SCREEN_POSITION;\n\t\t}\n\n\t\tif ( editableElementRect.width < NARROW_ROOT_WIDTH_THRESHOLD || editableElementRect.height < NARROW_ROOT_HEIGHT_THRESHOLD ) {\n\t\t\treturn OFF_THE_SCREEN_POSITION;\n\t\t}\n\n\t\tlet balloonTop;\n\n\t\tif ( config.position === 'inside' ) {\n\t\t\tballoonTop = editableElementRect.bottom - balloonRect.height;\n\t\t}\n\t\telse {\n\t\t\tballoonTop = editableElementRect.bottom - balloonRect.height / 2;\n\t\t}\n\n\t\tballoonTop -= config.verticalOffset;\n\n\t\tconst balloonLeft = getBalloonLeft( editableElementRect, balloonRect );\n\n\t\tif ( config.position === 'inside' ) {\n\t\t\tconst newBalloonRect = balloonRect.clone().moveTo( balloonLeft, balloonTop );\n\n\t\t\t// The watermark cannot be positioned in this corner because the corner is not quite visible.\n\t\t\tif ( newBalloonRect.getIntersectionArea( visibleEditableElementRect ) < newBalloonRect.getArea() ) {\n\t\t\t\treturn OFF_THE_SCREEN_POSITION;\n\t\t\t}\n\t\t}\n\t\telse {\n\t\t\tconst firstScrollableEditableElementAncestor = findClosestScrollableAncestor( focusedEditableElement );\n\n\t\t\tif ( firstScrollableEditableElementAncestor ) {\n\t\t\t\tconst firstScrollableEditableElementAncestorRect = new Rect( firstScrollableEditableElementAncestor );\n\t\t\t\tconst notVisibleVertically = visibleEditableElementRect.bottom + balloonRect.height / 2 >\n\t\t\t\tfirstScrollableEditableElementAncestorRect.bottom;\n\t\t\t\tconst notVisibleHorizontally = config.side === 'left' ?\n\t\t\t\t\teditableElementRect.left < firstScrollableEditableElementAncestorRect.left :\n\t\t\t\t\teditableElementRect.right > firstScrollableEditableElementAncestorRect.right;\n\n\t\t\t\t// The watermark cannot be positioned in this corner because the corner is \"not visible enough\".\n\t\t\t\tif ( notVisibleVertically || notVisibleHorizontally ) {\n\t\t\t\t\treturn OFF_THE_SCREEN_POSITION;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t\treturn {\n\t\t\ttop: balloonTop,\n\t\t\tleft: balloonLeft,\n\t\t\tname: `position_${ config.position }-side_${ config.side }`,\n\t\t\tconfig: {\n\t\t\t\twithArrow: false\n\t\t\t}\n\t\t};\n\t};\n}\n\nfunction getNormalizedConfig( editor: Editor ): PoweredByConfig {\n\tconst userConfig = editor.config.get( 'ui.poweredBy' );\n\tconst position = userConfig && userConfig.position || 'border';\n\n\treturn {\n\t\tposition,\n\t\tlabel: DEFAULT_LABEL,\n\t\tverticalOffset: position === 'inside' ? 5 : 0,\n\t\thorizontalOffset: 5,\n\n\t\tside: editor.locale.contentLanguageDirection === 'ltr' ? 'right' : 'left',\n\t\t...userConfig\n\t};\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/verifylicense\n */\n\nimport { releaseDate } from './version';\n\n/**\n * Possible states of the key after verification.\n */\nexport type VerifiedKeyStatus = 'VALID' | 'INVALID';\n\n/**\n * Checks whether the given string contains information that allows you to verify the license status.\n *\n * @param token The string to check.\n * @returns String that represents the state of given `token` parameter.\n */\nexport default function verifyLicense( token: string | undefined ): VerifiedKeyStatus {\n\t// This function implements naive and partial license key check mechanism,\n\t// used only to decide whether to show or hide the \"Powered by CKEditor\" logo.\n\t//\n\t// You can read the reasoning behind showing the logo to unlicensed (GPL) users\n\t// in this thread: https://github.com/ckeditor/ckeditor5/issues/14082.\n\t//\n\t// We firmly believe in the values behind creating open-source software, even when that\n\t// means keeping the license verification logic open for everyone to see.\n\t//\n\t// Please keep this code intact. Thank you for your understanding.\n\n\tfunction oldTokenCheck( token: string ): VerifiedKeyStatus {\n\t\tif ( token.length >= 40 && token.length <= 255 ) {\n\t\t\treturn 'VALID';\n\t\t} else {\n\t\t\treturn 'INVALID';\n\t\t}\n\t}\n\n\t// TODO: issue ci#3175\n\n\tif ( !token ) {\n\t\treturn 'INVALID';\n\t}\n\n\tlet decryptedData = '';\n\n\ttry {\n\t\tdecryptedData = atob( token );\n\t} catch ( e ) {\n\t\treturn 'INVALID';\n\t}\n\n\tconst splittedDecryptedData = decryptedData.split( '-' );\n\n\tconst firstElement = splittedDecryptedData[ 0 ];\n\tconst secondElement = splittedDecryptedData[ 1 ];\n\n\tif ( !secondElement ) {\n\t\treturn oldTokenCheck( token );\n\t}\n\n\ttry {\n\t\tatob( secondElement );\n\t} catch ( e ) {\n\t\ttry {\n\t\t\tatob( firstElement );\n\n\t\t\tif ( !atob( firstElement ).length ) {\n\t\t\t\treturn oldTokenCheck( token );\n\t\t\t}\n\t\t} catch ( e ) {\n\t\t\treturn oldTokenCheck( token );\n\t\t}\n\t}\n\n\tif ( firstElement.length < 40 || firstElement.length > 255 ) {\n\t\treturn 'INVALID';\n\t}\n\n\tlet decryptedSecondElement = '';\n\n\ttry {\n\t\tatob( firstElement );\n\t\tdecryptedSecondElement = atob( secondElement );\n\t} catch ( e ) {\n\t\treturn 'INVALID';\n\t}\n\n\tif ( decryptedSecondElement.length !== 8 ) {\n\t\treturn 'INVALID';\n\t}\n\n\tconst year = Number( decryptedSecondElement.substring( 0, 4 ) );\n\tconst monthIndex = Number( decryptedSecondElement.substring( 4, 6 ) ) - 1;\n\tconst day = Number( decryptedSecondElement.substring( 6, 8 ) );\n\n\tconst date = new Date( year, monthIndex, day );\n\n\tif ( date < releaseDate || isNaN( Number( date ) ) ) {\n\t\treturn 'INVALID';\n\t}\n\n\treturn 'VALID';\n}\n","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" width=\\\"53\\\" height=\\\"10\\\" viewBox=\\\"0 0 53 10\\\"><path fill=\\\"#1C2331\\\" d=\\\"M31.724 1.492a15.139 15.139 0 0 0 .045 1.16 2.434 2.434 0 0 0-.687-.34 3.68 3.68 0 0 0-1.103-.166 2.332 2.332 0 0 0-1.14.255 1.549 1.549 0 0 0-.686.87c-.15.41-.225.98-.225 1.712 0 .939.148 1.659.444 2.161.297.503.792.754 1.487.754.452.015.9-.094 1.294-.316.296-.174.557-.4.771-.669l.14.852h1.282V.007h-1.623v1.485ZM31 6.496a1.77 1.77 0 0 1-.494.061.964.964 0 0 1-.521-.127.758.758 0 0 1-.296-.466 3.984 3.984 0 0 1-.093-.992 4.208 4.208 0 0 1 .098-1.052.753.753 0 0 1 .307-.477 1.08 1.08 0 0 1 .55-.122c.233-.004.466.026.69.089l.483.144v2.553c-.11.076-.213.143-.307.2a1.73 1.73 0 0 1-.417.189ZM35.68 0l-.702.004c-.322.002-.482.168-.48.497l.004.581c.002.33.164.493.486.49l.702-.004c.322-.002.481-.167.48-.496L36.165.49c-.002-.33-.164-.493-.486-.491ZM36.145 2.313l-1.612.01.034 5.482 1.613-.01-.035-5.482ZM39.623.79 37.989.8 38 2.306l-.946.056.006 1.009.949-.006.024 2.983c.003.476.143.844.419 1.106.275.26.658.39 1.148.387.132 0 .293-.01.483-.03.19-.02.38-.046.57-.08.163-.028.324-.068.482-.119l-.183-1.095-.702.004a.664.664 0 0 1-.456-.123.553.553 0 0 1-.14-.422l-.016-2.621 1.513-.01-.006-1.064-1.514.01-.01-1.503ZM46.226 2.388c-.41-.184-.956-.274-1.636-.27-.673.004-1.215.101-1.627.29-.402.179-.72.505-.888.91-.18.419-.268.979-.264 1.68.004.688.1 1.24.285 1.655.172.404.495.724.9.894.414.18.957.268 1.63.264.68-.004 1.224-.099 1.632-.284.4-.176.714-.501.878-.905.176-.418.263-.971.258-1.658-.004-.702-.097-1.261-.28-1.677a1.696 1.696 0 0 0-.888-.9Zm-.613 3.607a.77.77 0 0 1-.337.501 1.649 1.649 0 0 1-1.317.009.776.776 0 0 1-.343-.497 4.066 4.066 0 0 1-.105-1.02 4.136 4.136 0 0 1 .092-1.03.786.786 0 0 1 .337-.507 1.59 1.59 0 0 1 1.316-.008.79.79 0 0 1 .344.502c.078.337.113.683.105 1.03.012.343-.019.685-.092 1.02ZM52.114 2.07a2.67 2.67 0 0 0-1.128.278c-.39.191-.752.437-1.072.73l-.157-.846-1.273.008.036 5.572 1.623-.01-.024-3.78c.35-.124.646-.22.887-.286.26-.075.53-.114.8-.118l.45-.003.144-1.546-.286.001ZM22.083 7.426l-1.576-2.532a2.137 2.137 0 0 0-.172-.253 1.95 1.95 0 0 0-.304-.29.138.138 0 0 1 .042-.04 1.7 1.7 0 0 0 .328-.374l1.75-2.71c.01-.015.025-.028.024-.048-.01-.01-.021-.007-.031-.007L20.49 1.17a.078.078 0 0 0-.075.045l-.868 1.384c-.23.366-.46.732-.688 1.099a.108.108 0 0 1-.112.06c-.098-.005-.196-.001-.294-.002-.018 0-.038.006-.055-.007.002-.02.002-.039.005-.058a4.6 4.6 0 0 0 .046-.701V1.203c0-.02-.009-.032-.03-.03h-.033L16.93 1.17c-.084 0-.073-.01-.073.076v6.491c-.001.018.006.028.025.027h1.494c.083 0 .072.007.072-.071v-2.19c0-.055-.003-.11-.004-.166a3.366 3.366 0 0 0-.05-.417h.06c.104 0 .209.002.313-.002a.082.082 0 0 1 .084.05c.535.913 1.07 1.824 1.607 2.736a.104.104 0 0 0 .103.062c.554-.003 1.107-.002 1.66-.002l.069-.003-.019-.032-.188-.304ZM27.112 6.555c-.005-.08-.004-.08-.082-.08h-2.414c-.053 0-.106-.003-.159-.011a.279.279 0 0 1-.246-.209.558.558 0 0 1-.022-.15c0-.382 0-.762-.002-1.143 0-.032.007-.049.042-.044h2.504c.029.003.037-.012.034-.038V3.814c0-.089.013-.078-.076-.078h-2.44c-.07 0-.062.003-.062-.06v-.837c0-.047.004-.093.013-.14a.283.283 0 0 1 .241-.246.717.717 0 0 1 .146-.011h2.484c.024.002.035-.009.036-.033l.003-.038.03-.496c.01-.183.024-.365.034-.548.005-.085.003-.087-.082-.094-.218-.018-.437-.038-.655-.05a17.845 17.845 0 0 0-.657-.026 72.994 72.994 0 0 0-1.756-.016 1.7 1.7 0 0 0-.471.064 1.286 1.286 0 0 0-.817.655c-.099.196-.149.413-.145.633v3.875c0 .072.003.144.011.216a1.27 1.27 0 0 0 .711 1.029c.228.113.48.167.734.158.757-.005 1.515.002 2.272-.042.274-.016.548-.034.82-.053.03-.002.043-.008.04-.041-.008-.104-.012-.208-.019-.312a69.964 69.964 0 0 1-.05-.768ZM16.14 7.415l-.127-1.075c-.004-.03-.014-.04-.044-.037a13.125 13.125 0 0 1-.998.073c-.336.01-.672.02-1.008.016-.116-.001-.233-.014-.347-.039a.746.746 0 0 1-.45-.262c-.075-.1-.132-.211-.167-.33a3.324 3.324 0 0 1-.126-.773 9.113 9.113 0 0 1-.015-.749c0-.285.022-.57.065-.852.023-.158.066-.312.127-.46a.728.728 0 0 1 .518-.443 1.64 1.64 0 0 1 .397-.048c.628-.001 1.255.003 1.882.05.022.001.033-.006.036-.026l.003-.031.06-.55c.019-.177.036-.355.057-.532.004-.034-.005-.046-.04-.056a5.595 5.595 0 0 0-1.213-.21 10.783 10.783 0 0 0-.708-.02c-.24-.003-.48.01-.719.041a3.477 3.477 0 0 0-.625.14 1.912 1.912 0 0 0-.807.497c-.185.2-.33.433-.424.688a4.311 4.311 0 0 0-.24 1.096c-.031.286-.045.572-.042.86-.006.43.024.86.091 1.286.04.25.104.497.193.734.098.279.26.53.473.734.214.205.473.358.756.446.344.11.702.17 1.063.177a8.505 8.505 0 0 0 1.578-.083 6.11 6.11 0 0 0 .766-.18c.03-.008.047-.023.037-.057a.157.157 0 0 1-.003-.025Z\\\"/><path fill=\\\"#AFE229\\\" d=\\\"M6.016 6.69a1.592 1.592 0 0 0-.614.21c-.23.132-.422.32-.56.546-.044.072-.287.539-.287.539l-.836 1.528.009.006c.038.025.08.046.123.063.127.046.26.07.395.073.505.023 1.011-.007 1.517-.003.29.009.58.002.869-.022a.886.886 0 0 0 .395-.116.962.962 0 0 0 .312-.286c.056-.083.114-.163.164-.249.24-.408.48-.816.718-1.226.075-.128.148-.257.222-.386l.112-.192a1.07 1.07 0 0 0 .153-.518l-1.304.023s-1.258-.005-1.388.01Z\\\"/><path fill=\\\"#771BFF\\\" d=\\\"m2.848 9.044.76-1.39.184-.352c-.124-.067-.245-.14-.367-.21-.346-.204-.706-.384-1.045-.6a.984.984 0 0 1-.244-.207c-.108-.134-.136-.294-.144-.46-.021-.409-.002-.818-.009-1.227-.003-.195 0-.39.003-.585.004-.322.153-.553.427-.713l.833-.488c.22-.13.44-.257.662-.385.05-.029.105-.052.158-.077.272-.128.519-.047.76.085l.044.028c.123.06.242.125.358.196.318.178.635.357.952.537.095.056.187.117.275.184.194.144.254.35.266.578.016.284.007.569.006.853-.001.28.004.558 0 .838.592-.003 1.259 0 1.259 0l.723-.013c-.003-.292-.007-.584-.007-.876 0-.524.015-1.048-.016-1.571-.024-.42-.135-.8-.492-1.067a5.02 5.02 0 0 0-.506-.339A400.52 400.52 0 0 0 5.94.787C5.722.664 5.513.524 5.282.423 5.255.406 5.228.388 5.2.373 4.758.126 4.305-.026 3.807.21c-.097.046-.197.087-.29.14A699.896 699.896 0 0 0 .783 1.948c-.501.294-.773.717-.778 1.31-.004.36-.009.718-.001 1.077.016.754-.017 1.508.024 2.261.016.304.07.6.269.848.127.15.279.28.448.382.622.4 1.283.734 1.92 1.11l.183.109Z\\\"/></svg>\\n\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module utils/dom/findclosestscrollableancestor\n */\n\nimport global from './global';\n\n/**\n * Returns the closest scrollable ancestor of a DOM element.\n *\n * @param domElement DOM element.\n * @returns First ancestor of `domElement` that is scrollable or null if such ancestor doesn't exist.\n */\nexport default function findClosestScrollableAncestor( domElement: HTMLElement ): HTMLElement | null {\n\tlet element = domElement.parentElement;\n\tif ( !element ) {\n\t\treturn null;\n\t}\n\n\twhile ( element.tagName != 'BODY' ) {\n\t\tconst overflow = element.style.overflowY || global.window.getComputedStyle( element ).overflowY;\n\n\t\tif ( overflow === 'auto' || overflow === 'scroll' ) {\n\t\t\tbreak;\n\t\t}\n\n\t\telement = element.parentElement;\n\n\t\tif ( !element ) {\n\t\t\treturn null;\n\t\t}\n\t}\n\n\treturn element;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/editorui/editorui\n */\n\n/* globals console */\n\nimport ComponentFactory from '../componentfactory';\nimport TooltipManager from '../tooltipmanager';\nimport PoweredBy from './poweredby';\n\nimport type EditorUIView from './editoruiview';\nimport type ToolbarView from '../toolbar/toolbarview';\nimport type { UIViewRenderEvent } from '../view';\n\nimport {\n\tObservableMixin,\n\tisVisible,\n\tFocusTracker,\n\ttype EventInfo\n} from '@ckeditor/ckeditor5-utils';\n\nimport type { Editor } from '@ckeditor/ckeditor5-core';\nimport type { ViewDocumentLayoutChangedEvent, ViewScrollToTheSelectionEvent } from '@ckeditor/ckeditor5-engine';\n\n/**\n * A class providing the minimal interface that is required to successfully bootstrap any editor UI.\n */\nexport default abstract class EditorUI extends ObservableMixin() {\n\t/**\n\t * The editor that the UI belongs to.\n\t */\n\tpublic readonly editor: Editor;\n\n\t/**\n\t * An instance of the {@link module:ui/componentfactory~ComponentFactory}, a registry used by plugins\n\t * to register factories of specific UI components.\n\t */\n\tpublic readonly componentFactory: ComponentFactory;\n\n\t/**\n\t * Stores the information about the editor UI focus and propagates it so various plugins and components\n\t * are unified as a focus group.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * Manages the tooltips displayed on mouseover and focus across the UI.\n\t */\n\tpublic readonly tooltipManager: TooltipManager;\n\n\t/**\n\t * A helper that enables the \"powered by\" feature in the editor and renders a link to the project's webpage.\n\t */\n\tpublic readonly poweredBy: PoweredBy;\n\n\t/**\n\t * Indicates the UI is ready. Set `true` after {@link #event:ready} event is fired.\n\t *\n\t * @readonly\n\t * @default false\n\t */\n\tpublic isReady: boolean = false;\n\n\tpublic abstract get view(): EditorUIView;\n\n\t/**\n\t * Stores viewport offsets from every direction.\n\t *\n\t * Viewport offset can be used to constrain balloons or other UI elements into an element smaller than the viewport.\n\t * This can be useful if there are any other absolutely positioned elements that may interfere with editor UI.\n\t *\n\t * Example `editor.ui.viewportOffset` returns:\n\t *\n\t * ```js\n\t * {\n\t * \ttop: 50,\n\t * \tright: 50,\n\t * \tbottom: 50,\n\t * \tleft: 50\n\t * }\n\t * ```\n\t *\n\t * This property can be overriden after editor already being initialized:\n\t *\n\t * ```js\n\t * editor.ui.viewportOffset = {\n\t * \ttop: 100,\n\t * \tright: 0,\n\t * \tbottom: 0,\n\t * \tleft: 0\n\t * };\n\t * ```\n\t *\n\t * @observable\n\t */\n\tpublic declare viewportOffset: {\n\t\tleft?: number;\n\t\tright?: number;\n\t\ttop?: number;\n\t\tbottom?: number;\n\t};\n\n\t/**\n\t * Stores all editable elements used by the editor instance.\n\t */\n\tprivate _editableElementsMap = new Map<string, HTMLElement>();\n\n\t/**\n\t * All available & focusable toolbars.\n\t */\n\tprivate _focusableToolbarDefinitions: Array<FocusableToolbarDefinition> = [];\n\n\t/**\n\t * Creates an instance of the editor UI class.\n\t *\n\t * @param editor The editor instance.\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper();\n\n\t\tconst editingView = editor.editing.view;\n\n\t\tthis.editor = editor;\n\t\tthis.componentFactory = new ComponentFactory( editor );\n\t\tthis.focusTracker = new FocusTracker();\n\t\tthis.tooltipManager = new TooltipManager( editor );\n\t\tthis.poweredBy = new PoweredBy( editor );\n\n\t\tthis.set( 'viewportOffset', this._readViewportOffsetFromConfig() );\n\n\t\tthis.once<EditorUIReadyEvent>( 'ready', () => {\n\t\t\tthis.isReady = true;\n\t\t} );\n\n\t\t// Informs UI components that should be refreshed after layout change.\n\t\tthis.listenTo<ViewDocumentLayoutChangedEvent>( editingView.document, 'layoutChanged', this.update.bind( this ) );\n\t\tthis.listenTo<ViewScrollToTheSelectionEvent>( editingView, 'scrollToTheSelection', this._handleScrollToTheSelection.bind( this ) );\n\n\t\tthis._initFocusTracking();\n\t}\n\n\t/**\n\t * The main (outermost) DOM element of the editor UI.\n\t *\n\t * For example, in {@link module:editor-classic/classiceditor~ClassicEditor} it is a `<div>` which\n\t * wraps the editable element and the toolbar. In {@link module:editor-inline/inlineeditor~InlineEditor}\n\t * it is the editable element itself (as there is no other wrapper). However, in\n\t * {@link module:editor-decoupled/decouplededitor~DecoupledEditor} it is set to `null` because this editor does not\n\t * come with a single \"main\" HTML element (its editable element and toolbar are separate).\n\t *\n\t * This property can be understood as a shorthand for retrieving the element that a specific editor integration\n\t * considers to be its main DOM element.\n\t */\n\tpublic get element(): HTMLElement | null {\n\t\treturn null;\n\t}\n\n\t/**\n\t * Fires the {@link module:ui/editorui/editorui~EditorUI#event:update `update`} event.\n\t *\n\t * This method should be called when the editor UI (e.g. positions of its balloons) needs to be updated due to\n\t * some environmental change which CKEditor 5 is not aware of (e.g. resize of a container in which it is used).\n\t */\n\tpublic update(): void {\n\t\tthis.fire<EditorUIUpdateEvent>( 'update' );\n\t}\n\n\t/**\n\t * Destroys the UI.\n\t */\n\tpublic destroy(): void {\n\t\tthis.stopListening();\n\n\t\tthis.focusTracker.destroy();\n\t\tthis.tooltipManager.destroy( this.editor );\n\t\tthis.poweredBy.destroy();\n\n\t\t// Cleanup the references to the CKEditor instance stored in the native editable DOM elements.\n\t\tfor ( const domElement of this._editableElementsMap.values() ) {\n\t\t\t( domElement as any ).ckeditorInstance = null;\n\t\t\tthis.editor.keystrokes.stopListening( domElement );\n\t\t}\n\n\t\tthis._editableElementsMap = new Map();\n\t\tthis._focusableToolbarDefinitions = [];\n\t}\n\n\t/**\n\t * Stores the native DOM editable element used by the editor under a unique name.\n\t *\n\t * Also, registers the element in the editor to maintain the accessibility of the UI. When the user is editing text in a focusable\n\t * editable area, they can use the <kbd>Alt</kbd> + <kbd>F10</kbd> keystroke to navigate over editor toolbars. See {@link #addToolbar}.\n\t *\n\t * @param rootName The unique name of the editable element.\n\t * @param domElement The native DOM editable element.\n\t */\n\tpublic setEditableElement( rootName: string, domElement: HTMLElement ): void {\n\t\tthis._editableElementsMap.set( rootName, domElement );\n\n\t\t// Put a reference to the CKEditor instance in the editable native DOM element.\n\t\t// It helps 3rdparty software (browser extensions, other libraries) access and recognize\n\t\t// CKEditor 5 instances (editing roots) and use their API (there is no global editor\n\t\t// instance registry).\n\t\tif ( !( domElement as any ).ckeditorInstance ) {\n\t\t\t( domElement as any ).ckeditorInstance = this.editor;\n\t\t}\n\n\t\t// Register the element, so it becomes available for Alt+F10 and Esc navigation.\n\t\tthis.focusTracker.add( domElement );\n\n\t\tconst setUpKeystrokeHandler = () => {\n\t\t\t// The editing view of the editor is already listening to keystrokes from DOM roots (see: KeyObserver).\n\t\t\t// Do not duplicate listeners.\n\t\t\tif ( this.editor.editing.view.getDomRoot( rootName ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis.editor.keystrokes.listenTo( domElement );\n\t\t};\n\n\t\t// For editable elements set by features after EditorUI is ready (e.g. source editing).\n\t\tif ( this.isReady ) {\n\t\t\tsetUpKeystrokeHandler();\n\t\t}\n\t\t// For editable elements set while the editor is being created (e.g. DOM roots).\n\t\telse {\n\t\t\tthis.once<EditorUIReadyEvent>( 'ready', setUpKeystrokeHandler );\n\t\t}\n\t}\n\n\t/**\n\t * Removes the editable from the editor UI. Removes all handlers added by {@link #setEditableElement}.\n\t *\n\t * @param rootName The name of the editable element to remove.\n\t */\n\tpublic removeEditableElement( rootName: string ): void {\n\t\tconst domElement = this._editableElementsMap.get( rootName );\n\n\t\tif ( !domElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._editableElementsMap.delete( rootName );\n\n\t\tthis.editor.keystrokes.stopListening( domElement );\n\t\tthis.focusTracker.remove( domElement );\n\n\t\t( domElement as any ).ckeditorInstance = null;\n\t}\n\n\t/**\n\t * Returns the editable editor element with the given name or null if editable does not exist.\n\t *\n\t * @param rootName The editable name.\n\t */\n\tpublic getEditableElement( rootName: string = 'main' ): HTMLElement | undefined {\n\t\treturn this._editableElementsMap.get( rootName );\n\t}\n\n\t/**\n\t * Returns array of names of all editor editable elements.\n\t */\n\tpublic getEditableElementsNames(): IterableIterator<string> {\n\t\treturn this._editableElementsMap.keys();\n\t}\n\n\t/**\n\t * Adds a toolbar to the editor UI. Used primarily to maintain the accessibility of the UI.\n\t *\n\t * Focusable toolbars can be accessed (focused) by users by pressing the <kbd>Alt</kbd> + <kbd>F10</kbd> keystroke.\n\t * Successive keystroke presses navigate over available toolbars.\n\t *\n\t * @param toolbarView A instance of the toolbar to be registered.\n\t */\n\tpublic addToolbar( toolbarView: ToolbarView, options: FocusableToolbarOptions = {} ): void {\n\t\tif ( toolbarView.isRendered ) {\n\t\t\tthis.focusTracker.add( toolbarView.element! );\n\t\t\tthis.editor.keystrokes.listenTo( toolbarView.element! );\n\t\t} else {\n\t\t\ttoolbarView.once<UIViewRenderEvent>( 'render', () => {\n\t\t\t\tthis.focusTracker.add( toolbarView.element! );\n\t\t\t\tthis.editor.keystrokes.listenTo( toolbarView.element! );\n\t\t\t} );\n\t\t}\n\n\t\tthis._focusableToolbarDefinitions.push( { toolbarView, options } );\n\t}\n\n\t/**\n\t * Stores all editable elements used by the editor instance.\n\t *\n\t * @deprecated\n\t */\n\tprotected get _editableElements(): unknown {\n\t\t/**\n\t\t * The {@link module:ui/editorui/editorui~EditorUI#_editableElements `EditorUI#_editableElements`} property has been\n\t\t * deprecated and will be removed in the near future. Please use\n\t\t * {@link module:ui/editorui/editorui~EditorUI#setEditableElement `setEditableElement()`} and\n\t\t * {@link module:ui/editorui/editorui~EditorUI#getEditableElement `getEditableElement()`} methods instead.\n\t\t *\n\t\t * @error editor-ui-deprecated-editable-elements\n\t\t * @param editorUI Editor UI instance the deprecated property belongs to.\n\t\t */\n\t\tconsole.warn(\n\t\t\t'editor-ui-deprecated-editable-elements: ' +\n\t\t\t'The EditorUI#_editableElements property has been deprecated and will be removed in the near future.',\n\t\t\t{ editorUI: this } );\n\n\t\treturn this._editableElementsMap;\n\t}\n\n\t/**\n\t * Returns viewport offsets object:\n\t *\n\t * ```js\n\t * {\n\t * \ttop: Number,\n\t * \tright: Number,\n\t * \tbottom: Number,\n\t * \tleft: Number\n\t * }\n\t * ```\n\t *\n\t * Only top property is currently supported.\n\t */\n\tprivate _readViewportOffsetFromConfig() {\n\t\tconst editor = this.editor;\n\t\tconst viewportOffsetConfig = editor.config.get( 'ui.viewportOffset' );\n\n\t\tif ( viewportOffsetConfig ) {\n\t\t\treturn viewportOffsetConfig;\n\t\t}\n\n\t\t// Not present in EditorConfig type, because it's legacy. Hence the `as` expression.\n\t\tconst legacyOffsetConfig = editor.config.get( 'toolbar.viewportTopOffset' ) as number | undefined;\n\n\t\t// Fall back to deprecated toolbar config.\n\t\tif ( legacyOffsetConfig ) {\n\t\t\t/**\n\t\t\t * The {@link module:core/editor/editorconfig~EditorConfig#toolbar `EditorConfig#toolbar.viewportTopOffset`}\n\t\t\t * property has been deprecated and will be removed in the near future. Please use\n\t\t\t * {@link module:core/editor/editorconfig~EditorConfig#ui `EditorConfig#ui.viewportOffset`} instead.\n\t\t\t *\n\t\t\t * @error editor-ui-deprecated-viewport-offset-config\n\t\t\t */\n\t\t\tconsole.warn(\n\t\t\t\t'editor-ui-deprecated-viewport-offset-config: ' +\n\t\t\t\t'The `toolbar.vieportTopOffset` configuration option is deprecated. ' +\n\t\t\t\t'It will be removed from future CKEditor versions. Use `ui.viewportOffset.top` instead.'\n\t\t\t);\n\n\t\t\treturn { top: legacyOffsetConfig };\n\t\t}\n\n\t\t// More keys to come in the future.\n\t\treturn { top: 0 };\n\t}\n\n\t/**\n\t * Starts listening for <kbd>Alt</kbd> + <kbd>F10</kbd> and <kbd>Esc</kbd> keystrokes in the context of focusable\n\t * {@link #setEditableElement editable elements} and {@link #addToolbar toolbars}\n\t * to allow users navigate across the UI.\n\t */\n\tprivate _initFocusTracking(): void {\n\t\tconst editor = this.editor;\n\t\tconst editingView = editor.editing.view;\n\n\t\tlet lastFocusedForeignElement: HTMLElement | null;\n\t\tlet candidateDefinitions: Array<FocusableToolbarDefinition>;\n\n\t\t// Focus the next focusable toolbar on <kbd>Alt</kbd> + <kbd>F10</kbd>.\n\t\teditor.keystrokes.set( 'Alt+F10', ( data, cancel ) => {\n\t\t\tconst focusedElement = this.focusTracker.focusedElement as HTMLElement;\n\n\t\t\t// Focus moved out of a DOM element that\n\t\t\t// * is not a toolbar,\n\t\t\t// * does not belong to the editing view (e.g. source editing).\n\t\t\tif (\n\t\t\t\tArray.from( this._editableElementsMap.values() ).includes( focusedElement ) &&\n\t\t\t\t!Array.from( editingView.domRoots.values() ).includes( focusedElement )\n\t\t\t) {\n\t\t\t\tlastFocusedForeignElement = focusedElement;\n\t\t\t}\n\n\t\t\tconst currentFocusedToolbarDefinition = this._getCurrentFocusedToolbarDefinition();\n\n\t\t\t// * When focusing a toolbar for the first time, set the array of definitions for successive presses of Alt+F10.\n\t\t\t// This ensures, the navigation works always the same and no pair of toolbars takes over\n\t\t\t// (e.g. image and table toolbars when a selected image is inside a cell).\n\t\t\t// * It could be that the focus went to the toolbar by clicking a toolbar item (e.g. a dropdown). In this case,\n\t\t\t// there were no candidates so they must be obtained (#12339).\n\t\t\tif ( !currentFocusedToolbarDefinition || !candidateDefinitions ) {\n\t\t\t\tcandidateDefinitions = this._getFocusableCandidateToolbarDefinitions();\n\t\t\t}\n\n\t\t\t// In a single Alt+F10 press, check all candidates but if none were focused, don't go any further.\n\t\t\t// This prevents an infinite loop.\n\t\t\tfor ( let i = 0; i < candidateDefinitions.length; i++ ) {\n\t\t\t\tconst candidateDefinition = candidateDefinitions.shift()!;\n\n\t\t\t\t// Put the first definition to the back of the array. This allows circular navigation over all toolbars\n\t\t\t\t// on successive presses of Alt+F10.\n\t\t\t\tcandidateDefinitions.push( candidateDefinition );\n\n\t\t\t\t// Don't focus the same toolbar again. If you did, this would move focus from the nth focused toolbar item back to the\n\t\t\t\t// first item as per ToolbarView#focus() if the user navigated inside the toolbar.\n\t\t\t\tif (\n\t\t\t\t\tcandidateDefinition !== currentFocusedToolbarDefinition &&\n\t\t\t\t\tthis._focusFocusableCandidateToolbar( candidateDefinition )\n\t\t\t\t) {\n\t\t\t\t\t// Clean up after a current visible toolbar when switching to the next one.\n\t\t\t\t\tif ( currentFocusedToolbarDefinition && currentFocusedToolbarDefinition.options.afterBlur ) {\n\t\t\t\t\t\tcurrentFocusedToolbarDefinition.options.afterBlur();\n\t\t\t\t\t}\n\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tcancel();\n\t\t} );\n\n\t\t// Blur the focused toolbar on <kbd>Esc</kbd> and bring the focus back to its origin.\n\t\teditor.keystrokes.set( 'Esc', ( data, cancel ) => {\n\t\t\tconst focusedToolbarDef = this._getCurrentFocusedToolbarDefinition();\n\n\t\t\tif ( !focusedToolbarDef ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Bring focus back to where it came from before focusing the toolbar:\n\t\t\t// 1. If it came from outside the engine view (e.g. source editing), move it there.\n\t\t\tif ( lastFocusedForeignElement ) {\n\t\t\t\tlastFocusedForeignElement.focus();\n\t\t\t\tlastFocusedForeignElement = null;\n\t\t\t}\n\t\t\t// 2. There are two possibilities left:\n\t\t\t// 2.1. It could be that the focus went from an editable element in the view (root or nested).\n\t\t\t// 2.2. It could be the focus went straight to the toolbar before even focusing the editing area.\n\t\t\t// In either case, just focus the view editing. The focus will land where it belongs.\n\t\t\telse {\n\t\t\t\teditor.editing.view.focus();\n\t\t\t}\n\n\t\t\t// Clean up after the toolbar if there is anything to do there.\n\t\t\tif ( focusedToolbarDef.options.afterBlur ) {\n\t\t\t\tfocusedToolbarDef.options.afterBlur();\n\t\t\t}\n\n\t\t\tcancel();\n\t\t} );\n\t}\n\n\t/**\n\t * Returns definitions of toolbars that could potentially be focused, sorted by their importance for the user.\n\t *\n\t * Focusable toolbars candidates are either:\n\t * * already visible,\n\t * * have `beforeFocus()` set in their {@link module:ui/editorui/editorui~FocusableToolbarDefinition definition} that suggests that\n\t * they might show up when called. Keep in mind that determining whether a toolbar will show up (and become focusable) is impossible\n\t * at this stage because it depends on its implementation, that in turn depends on the editing context (selection).\n\t *\n\t * **Note**: Contextual toolbars take precedence over regular toolbars.\n\t */\n\tprivate _getFocusableCandidateToolbarDefinitions(): Array<FocusableToolbarDefinition> {\n\t\tconst definitions: Array<FocusableToolbarDefinition> = [];\n\n\t\tfor ( const toolbarDef of this._focusableToolbarDefinitions ) {\n\t\t\tconst { toolbarView, options } = toolbarDef;\n\n\t\t\tif ( isVisible( toolbarView.element ) || options.beforeFocus ) {\n\t\t\t\tdefinitions.push( toolbarDef );\n\t\t\t}\n\t\t}\n\n\t\t// Contextual and already visible toolbars have higher priority. If both are true, the toolbar will always focus first.\n\t\t// For instance, a selected widget toolbar vs inline editor toolbar: both are visible but the widget toolbar is contextual.\n\t\tdefinitions.sort( ( defA, defB ) => getToolbarDefinitionWeight( defA ) - getToolbarDefinitionWeight( defB ) );\n\n\t\treturn definitions;\n\t}\n\n\t/**\n\t * Returns a definition of the toolbar that is currently visible and focused (one of its children has focus).\n\t *\n\t * `null` is returned when no toolbar is currently focused.\n\t */\n\tprivate _getCurrentFocusedToolbarDefinition(): FocusableToolbarDefinition | null {\n\t\tfor ( const definition of this._focusableToolbarDefinitions ) {\n\t\t\tif ( definition.toolbarView.element && definition.toolbarView.element.contains( this.focusTracker.focusedElement ) ) {\n\t\t\t\treturn definition;\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Focuses a focusable toolbar candidate using its definition.\n\t *\n\t * @param candidateToolbarDefinition A definition of the toolbar to focus.\n\t * @returns `true` when the toolbar candidate was focused. `false` otherwise.\n\t */\n\tprivate _focusFocusableCandidateToolbar( candidateToolbarDefinition: FocusableToolbarDefinition ): boolean {\n\t\tconst { toolbarView, options: { beforeFocus } } = candidateToolbarDefinition;\n\n\t\tif ( beforeFocus ) {\n\t\t\tbeforeFocus();\n\t\t}\n\n\t\t// If it didn't show up after beforeFocus(), it's not focusable at all.\n\t\tif ( !isVisible( toolbarView.element ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\ttoolbarView.focus();\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Provides an integration between {@link #viewportOffset} and {@link module:utils/dom/scroll~scrollViewportToShowTarget}.\n\t * It allows the UI-agnostic engine method to consider user-configured viewport offsets specific for the integration.\n\t *\n\t * @param evt The `scrollToTheSelection` event info.\n\t * @param data The payload carried by the `scrollToTheSelection` event.\n\t */\n\tprivate _handleScrollToTheSelection(\n\t\tevt: EventInfo<'scrollToTheSelection'>,\n\t\tdata: ViewScrollToTheSelectionEvent[ 'args' ][ 0 ]\n\t): void {\n\t\tconst configuredViewportOffset = {\n\t\t\ttop: 0,\n\t\t\tbottom: 0,\n\t\t\tleft: 0,\n\t\t\tright: 0,\n\t\t\t...this.viewportOffset\n\t\t};\n\n\t\tdata.viewportOffset.top += configuredViewportOffset.top;\n\t\tdata.viewportOffset.bottom += configuredViewportOffset.bottom;\n\t\tdata.viewportOffset.left += configuredViewportOffset.left;\n\t\tdata.viewportOffset.right += configuredViewportOffset.right;\n\t}\n}\n\n/**\n * Fired when the editor UI is ready.\n *\n * Fired before {@link module:engine/controller/datacontroller~DataController#event:ready}.\n *\n * @eventName ~EditorUI#ready\n */\nexport type EditorUIReadyEvent = {\n\tname: 'ready';\n\targs: [];\n};\n\n/**\n * Fired whenever the UI (all related components) should be refreshed.\n *\n * **Note:**: The event is fired after each {@link module:engine/view/document~Document#event:layoutChanged}.\n * It can also be fired manually via the {@link module:ui/editorui/editorui~EditorUI#update} method.\n *\n * @eventName ~EditorUI#update\n */\nexport type EditorUIUpdateEvent = {\n\tname: 'update';\n\targs: [];\n};\n\n/**\n * A definition of a focusable toolbar. Used by {@link module:ui/editorui/editorui~EditorUI#addToolbar}.\n */\nexport interface FocusableToolbarDefinition {\n\n\t/**\n\t * An instance of a focusable toolbar view.\n\t */\n\ttoolbarView: ToolbarView;\n\n\t/**\n\t * Options of a focusable toolbar view:\n\t *\n\t * * `isContextual`: Marks the higher priority toolbar. For example when there are 2 visible toolbars,\n\t * it allows to distinguish which toolbar should be focused first after the `alt+f10` keystroke\n\t * * `beforeFocus`: A callback executed before the `ToolbarView` gains focus upon the `Alt+F10` keystroke.\n\t * * `afterBlur`: A callback executed after `ToolbarView` loses focus upon `Esc` keystroke but before\n\t * the focus goes back to the `origin`.\n\t */\n\toptions: FocusableToolbarOptions;\n}\n\nexport interface FocusableToolbarOptions {\n\n\t/**\n\t * Set `true` if the toolbar is attached to the content of the editor. Such toolbar takes\n\t * a precedence over other toolbars when a user pressed <kbd>Alt</kbd> + <kbd>F10</kbd>.\n\t */\n\tisContextual?: boolean;\n\n\t/**\n\t * Specify a callback executed before the toolbar instance DOM element gains focus\n\t * upon the <kbd>Alt</kbd> + <kbd>F10</kbd> keystroke.\n\t */\n\tbeforeFocus?: () => void;\n\n\t/**\n\t * Specify a callback executed after the toolbar instance DOM element loses focus upon\n\t * <kbd>Esc</kbd> keystroke but before the focus goes back to the {@link ~EditorUI#setEditableElement editable element}.\n\t */\n\tafterBlur?: () => void;\n}\n\n/**\n * Returns a number (weight) for a toolbar definition. Visible toolbars have a higher priority and so do\n * contextual toolbars (displayed in the context of a content, for instance, an image toolbar).\n *\n * A standard invisible toolbar is the heaviest. A visible contextual toolbar is the lightest.\n *\n * @param toolbarDef A toolbar definition to be weighted.\n */\nfunction getToolbarDefinitionWeight( toolbarDef: FocusableToolbarDefinition ): number {\n\tconst { toolbarView, options } = toolbarDef;\n\tlet weight = 10;\n\n\t// Prioritize already visible toolbars. They should get focused first.\n\tif ( isVisible( toolbarView.element ) ) {\n\t\tweight--;\n\t}\n\n\t// Prioritize contextual toolbars. They are displayed at the selection.\n\tif ( options.isContextual ) {\n\t\tweight--;\n\t}\n\n\treturn weight;\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./editorui.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/editorui/editoruiview\n */\n\nimport View from '../view';\nimport BodyCollection from './bodycollection';\nimport type EditableUIView from '../editableui/editableuiview';\n\nimport type { Locale, LocaleTranslate } from '@ckeditor/ckeditor5-utils';\n\nimport '../../theme/components/editorui/editorui.css';\n\n/**\n * The editor UI view class. Base class for the editor main views.\n */\nexport default abstract class EditorUIView extends View {\n\t/**\n\t * Collection of the child views, detached from the DOM\n\t * structure of the editor, like panels, icons etc.\n\t */\n\tpublic readonly body: BodyCollection;\n\n\tdeclare public locale: Locale;\n\tdeclare public t: LocaleTranslate;\n\n\tpublic abstract get editable(): EditableUIView;\n\n\t/**\n\t * Creates an instance of the editor UI view class.\n\t *\n\t * @param locale The locale instance.\n\t */\n\tconstructor( locale: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.body = new BodyCollection( locale );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.body.attachToDom();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tthis.body.detachFromDom();\n\n\t\treturn super.destroy();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/editableui/editableuiview\n */\n\nimport View from '../view';\n\nimport type { View as EditingView } from '@ckeditor/ckeditor5-engine';\nimport type { Locale, ObservableChangeEvent } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The editable UI view class.\n */\nexport default class EditableUIView extends View {\n\t/**\n\t * The name of the editable UI view.\n\t */\n\tpublic name: string | null = null;\n\n\t/**\n\t * Controls whether the editable is focused, i.e. the user is typing in it.\n\t *\n\t * @observable\n\t */\n\tdeclare public isFocused: boolean;\n\n\t/**\n\t * The editing view instance the editable is related to. Editable uses the editing\n\t * view to dynamically modify its certain DOM attributes after {@link #render rendering}.\n\t *\n\t * **Note**: The DOM attributes are performed by the editing view and not UI\n\t * {@link module:ui/view~View#bindTemplate template bindings} because once rendered,\n\t * the editable DOM element must remain under the full control of the engine to work properly.\n\t */\n\tprotected _editingView: EditingView;\n\n\t/**\n\t * The element which is the main editable element (usually the one with `contentEditable=\"true\"`).\n\t */\n\tprivate _editableElement: HTMLElement | null | undefined;\n\n\t/**\n\t * Whether an external {@link #_editableElement} was passed into the constructor, which also means\n\t * the view will not render its {@link #template}.\n\t */\n\tprivate _hasExternalElement: boolean;\n\n\t/**\n\t * Creates an instance of EditableUIView class.\n\t *\n\t * @param locale The locale instance.\n\t * @param editingView The editing view instance the editable is related to.\n\t * @param editableElement The editable element. If not specified, this view\n\t * should create it. Otherwise, the existing element should be used.\n\t */\n\tconstructor(\n\t\tlocale: Locale,\n\t\teditingView: EditingView,\n\t\teditableElement?: HTMLElement\n\t) {\n\t\tsuper( locale );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-content',\n\t\t\t\t\t'ck-editor__editable',\n\t\t\t\t\t'ck-rounded-corners'\n\t\t\t\t],\n\t\t\t\tlang: locale.contentLanguage,\n\t\t\t\tdir: locale.contentLanguageDirection\n\t\t\t}\n\t\t} );\n\n\t\tthis.set( 'isFocused', false );\n\n\t\tthis._editableElement = editableElement;\n\t\tthis._hasExternalElement = !!this._editableElement;\n\t\tthis._editingView = editingView;\n\t}\n\n\t/**\n\t * Renders the view by either applying the {@link #template} to the existing\n\t * {@link module:ui/editableui/editableuiview~EditableUIView#_editableElement} or assigning {@link #element}\n\t * as {@link module:ui/editableui/editableuiview~EditableUIView#_editableElement}.\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tif ( this._hasExternalElement ) {\n\t\t\tthis.template!.apply( this.element = this._editableElement! );\n\t\t} else {\n\t\t\tthis._editableElement = this.element;\n\t\t}\n\n\t\tthis.on<ObservableChangeEvent>( 'change:isFocused', () => this._updateIsFocusedClasses() );\n\t\tthis._updateIsFocusedClasses();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tif ( this._hasExternalElement ) {\n\t\t\tthis.template!.revert( this._editableElement! );\n\t\t}\n\n\t\tsuper.destroy();\n\t}\n\n\t/**\n\t * Whether an external {@link #_editableElement} was passed into the constructor, which also means\n\t * the view will not render its {@link #template}.\n\t */\n\tpublic get hasExternalElement(): boolean {\n\t\treturn this._hasExternalElement;\n\t}\n\n\t/**\n\t * Updates the `ck-focused` and `ck-blurred` CSS classes on the {@link #element} according to\n\t * the {@link #isFocused} property value using the {@link #_editingView editing view} API.\n\t */\n\tprivate _updateIsFocusedClasses() {\n\t\tconst editingView = this._editingView;\n\n\t\tif ( editingView.isRenderingInProgress ) {\n\t\t\tupdateAfterRender( this );\n\t\t} else {\n\t\t\tupdate( this );\n\t\t}\n\n\t\tfunction update( view: EditableUIView ) {\n\t\t\teditingView.change( writer => {\n\t\t\t\tconst viewRoot = editingView.document.getRoot( view.name! )!;\n\n\t\t\t\twriter.addClass( view.isFocused ? 'ck-focused' : 'ck-blurred', viewRoot );\n\t\t\t\twriter.removeClass( view.isFocused ? 'ck-blurred' : 'ck-focused', viewRoot );\n\t\t\t} );\n\t\t}\n\n\t\t// In a case of a multi-root editor, a callback will be attached more than once (one callback for each root).\n\t\t// While executing one callback the `isRenderingInProgress` observable is changing what causes executing another\n\t\t// callback and render is called inside the already pending render.\n\t\t// We need to be sure that callback is executed only when the value has changed from `true` to `false`.\n\t\t// See https://github.com/ckeditor/ckeditor5/issues/1676.\n\t\tfunction updateAfterRender( view: EditableUIView ) {\n\t\t\teditingView.once<ObservableChangeEvent<boolean>>( 'change:isRenderingInProgress', ( evt, name, value ) => {\n\t\t\t\tif ( !value ) {\n\t\t\t\t\tupdate( view );\n\t\t\t\t} else {\n\t\t\t\t\tupdateAfterRender( view );\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/editableui/inline/inlineeditableuiview\n */\n\nimport EditableUIView from '../editableuiview';\n\nimport type { View } from '@ckeditor/ckeditor5-engine';\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The inline editable UI class implementing an inline {@link module:ui/editableui/editableuiview~EditableUIView}.\n */\nexport default class InlineEditableUIView extends EditableUIView {\n\t/**\n\t * A function that gets called with the instance of this view as an argument and should return a string that\n\t * represents the label of the editable for assistive technologies.\n\t */\n\tprivate readonly _generateLabel: ( view: InlineEditableUIView ) => string;\n\n\t/**\n\t * Creates an instance of the InlineEditableUIView class.\n\t *\n\t * @param locale The locale instance.\n\t * @param editingView The editing view instance the editable is related to.\n\t * @param editableElement The editable element. If not specified, the\n\t * {@link module:ui/editableui/editableuiview~EditableUIView}\n\t * will create it. Otherwise, the existing element will be used.\n\t * @param options Additional configuration of the view.\n\t * @param options.label A function that gets called with the instance of this view as an argument\n\t * and should return a string that represents the label of the editable for assistive technologies. If not provided,\n\t * a default label generator is used.\n\t */\n\tconstructor(\n\t\tlocale: Locale,\n\t\teditingView: View,\n\t\teditableElement?: HTMLElement,\n\t\toptions: { label?: ( view: InlineEditableUIView ) => string } = {}\n\t) {\n\t\tsuper( locale, editingView, editableElement );\n\n\t\tconst t = locale.t;\n\n\t\tthis.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\trole: 'textbox',\n\t\t\t\tclass: 'ck-editor__editable_inline'\n\t\t\t}\n\t\t} );\n\n\t\tthis._generateLabel = options.label || ( () => t( 'Editor editing area: %0', this.name! ) );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tconst editingView = this._editingView;\n\n\t\teditingView.change( writer => {\n\t\t\tconst viewRoot = editingView.document.getRoot( this.name! );\n\n\t\t\twriter.setAttribute( 'aria-label', this._generateLabel( this ), viewRoot! );\n\t\t} );\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./formheader.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/notification/notification\n */\n\n/* globals window */\n\nimport { ContextPlugin } from '@ckeditor/ckeditor5-core';\n\n/**\n * The Notification plugin.\n *\n * This plugin sends a few types of notifications: `success`, `info` and `warning`. The notifications need to be\n * handled and displayed by a plugin responsible for showing the UI of the notifications. Using this plugin for dispatching\n * notifications makes it possible to switch the notifications UI.\n *\n * Note that every unhandled and not stopped `warning` notification will be displayed as a system alert.\n * See {@link module:ui/notification/notification~Notification#showWarning}.\n */\nexport default class Notification extends ContextPlugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Notification' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\t// Each unhandled and not stopped `show:warning` event is displayed as a system alert.\n\t\tthis.on<NotificationShowWarningEvent>( 'show:warning', ( evt, data ) => {\n\t\t\twindow.alert( data.message ); // eslint-disable-line no-alert\n\t\t}, { priority: 'lowest' } );\n\t}\n\n\t/**\n\t * Shows a success notification.\n\t *\n\t * By default, it fires the {@link #event:show:success `show:success` event} with the given `data`. The event namespace can be extended\n\t * using the `data.namespace` option. For example:\n\t *\n\t * ```ts\n\t * showSuccess( 'Image is uploaded.', {\n\t * \tnamespace: 'upload:image'\n\t * } );\n\t * ```\n\t *\n\t * will fire the `show:success:upload:image` event.\n\t *\n\t * You can provide the title of the notification:\n\t *\n\t * ```ts\n\t * showSuccess( 'Image is uploaded.', {\n\t * \ttitle: 'Image upload success'\n\t * } );\n\t * ```\n\t *\n\t * @param message The content of the notification.\n\t * @param data Additional data.\n\t * @param data.namespace Additional event namespace.\n\t * @param data.title The title of the notification.\n\t */\n\tpublic showSuccess(\n\t\tmessage: string,\n\t\tdata: { namespace?: string; title?: string } = {}\n\t): void {\n\t\tthis._showNotification( {\n\t\t\tmessage,\n\t\t\ttype: 'success',\n\t\t\tnamespace: data.namespace,\n\t\t\ttitle: data.title\n\t\t} );\n\t}\n\n\t/**\n\t * Shows an information notification.\n\t *\n\t * By default, it fires the {@link #event:show:info `show:info` event} with the given `data`. The event namespace can be extended\n\t * using the `data.namespace` option. For example:\n\t *\n\t * ```ts\n\t * showInfo( 'Editor is offline.', {\n\t * \tnamespace: 'editor:status'\n\t * } );\n\t * ```\n\t *\n\t * will fire the `show:info:editor:status` event.\n\t *\n\t * You can provide the title of the notification:\n\t *\n\t * ```ts\n\t * showInfo( 'Editor is offline.', {\n\t * \ttitle: 'Network information'\n\t * } );\n\t * ```\n\t *\n\t * @param message The content of the notification.\n\t * @param data Additional data.\n\t * @param data.namespace Additional event namespace.\n\t * @param data.title The title of the notification.\n\t */\n\tpublic showInfo(\n\t\tmessage: string,\n\t\tdata: { namespace?: string; title?: string } = {}\n\t): void {\n\t\tthis._showNotification( {\n\t\t\tmessage,\n\t\t\ttype: 'info',\n\t\t\tnamespace: data.namespace,\n\t\t\ttitle: data.title\n\t\t} );\n\t}\n\n\t/**\n\t * Shows a warning notification.\n\t *\n\t * By default, it fires the {@link #event:show:warning `show:warning` event}\n\t * with the given `data`. The event namespace can be extended using the `data.namespace` option. For example:\n\t *\n\t * ```ts\n\t * showWarning( 'Image upload error.', {\n\t * \tnamespace: 'upload:image'\n\t * } );\n\t * ```\n\t *\n\t * will fire the `show:warning:upload:image` event.\n\t *\n\t * You can provide the title of the notification:\n\t *\n\t * ```ts\n\t * showWarning( 'Image upload error.', {\n\t * \ttitle: 'Upload failed'\n\t * } );\n\t * ```\n\t *\n\t * Note that each unhandled and not stopped `warning` notification will be displayed as a system alert.\n\t * The plugin responsible for displaying warnings should `stop()` the event to prevent displaying it as an alert:\n\t *\n\t * ```ts\n\t * notifications.on( 'show:warning', ( evt, data ) => {\n\t * \t// Do something with the data.\n\t *\n\t * \t// Stop this event to prevent displaying it as an alert.\n\t * \tevt.stop();\n\t * } );\n\t * ```\n\t *\n\t * You can attach many listeners to the same event and `stop()` this event in a listener with a low priority:\n\t *\n\t * ```ts\n\t * notifications.on( 'show:warning', ( evt, data ) => {\n\t * \t// Show the warning in the UI, but do not stop it.\n\t * } );\n\t *\n\t * notifications.on( 'show:warning', ( evt, data ) => {\n\t * \t// Log the warning to some error tracker.\n\t *\n\t * \t// Stop this event to prevent displaying it as an alert.\n\t * \tevt.stop();\n\t * }, { priority: 'low' } );\n\t * ```\n\t *\n\t * @param message The content of the notification.\n\t * @param data Additional data.\n\t * @param data.namespace Additional event namespace.\n\t * @param data.title The title of the notification.\n\t */\n\tpublic showWarning(\n\t\tmessage: string,\n\t\tdata: { namespace?: string; title?: string } = {}\n\t): void {\n\t\tthis._showNotification( {\n\t\t\tmessage,\n\t\t\ttype: 'warning',\n\t\t\tnamespace: data.namespace,\n\t\t\ttitle: data.title\n\t\t} );\n\t}\n\n\t/**\n\t * Fires the `show` event with the specified type, namespace and message.\n\t *\n\t * @param data The message data.\n\t * @param data.message The content of the notification.\n\t * @param data.type The type of the message.\n\t * @param data.namespace Additional event namespace.\n\t * @param data.title The title of the notification.\n\t */\n\tprivate _showNotification( data: {\n\t\tmessage: string;\n\t\ttype: NotificationEventType;\n\t\tnamespace?: string;\n\t\ttitle?: string;\n\t} ): void {\n\t\tconst event = data.namespace ?\n\t\t\t`show:${ data.type }:${ data.namespace }` as const :\n\t\t\t`show:${ data.type }` as const;\n\n\t\tthis.fire<NotificationShowTypeEvent<typeof data.type>>( event, {\n\t\t\tmessage: data.message,\n\t\t\ttype: data.type,\n\t\t\ttitle: data.title || ''\n\t\t} );\n\t}\n}\n\nexport type NotificationEventType = 'success' | 'info' | 'warning';\n\n/**\n * Fired when one of the `showSuccess()`, `showInfo()`, `showWarning()` methods is called.\n *\n * @eventName ~Notification#show\n * @param data The notification data.\n */\nexport type NotificationShowEvent = {\n\tname: 'show';\n\targs: [ data: NotificationShowEventData ];\n};\n\n/**\n * Fired when the `showSuccess()` method is called.\n *\n * @eventName ~Notification#show:success\n * @param data The notification data.\n */\nexport type NotificationShowSuccessEvent = NotificationShowTypeEvent<'success'>;\n\n/**\n * Fired when the `showInfo()` method is called.\n *\n * @eventName ~Notification#show:info\n * @param data The notification data.\n */\nexport type NotificationShowInfoEvent = NotificationShowTypeEvent<'info'>;\n\n/**\n * Fired when the `showWarning()` method is called.\n *\n * When this event is not handled or stopped by `event.stop()`, the `data.message` of this event will\n * be automatically displayed as a system alert.\n *\n * @eventName ~Notification#show:warning\n * @param data The notification data.\n */\nexport type NotificationShowWarningEvent = NotificationShowTypeEvent<'warning'>;\n\nexport type NotificationShowTypeEvent<NotificationType extends NotificationEventType> = {\n\tname: `show:${ NotificationType }` | `show:${ NotificationType }:${ string }`;\n\targs: [ data: NotificationShowEventData<NotificationType> ];\n};\n\nexport type NotificationShowEventData<NotificationType extends NotificationEventType = NotificationEventType> = {\n\n\t/**\n\t * The content of the notification.\n\t */\n\tmessage: string;\n\n\t/**\n\t * The title of the notification.\n\t */\n\ttitle: string;\n\n\t/**\n\t * The type of the notification.\n\t */\n\ttype: NotificationType;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/model\n */\n\nimport { ObservableMixin } from '@ckeditor/ckeditor5-utils';\nimport { extend } from 'lodash-es';\n\n/**\n * The base MVC model class.\n */\nexport default class Model extends ObservableMixin() {\n\t[ x: string ]: unknown;\n\n\t/**\n\t * Creates a new Model instance.\n\t *\n\t * @param attributes The model state attributes to be defined during the instance creation.\n\t * @param properties The (out of state) properties to be appended to the instance during creation.\n\t */\n\tconstructor( attributes?: Record<string, unknown>, properties?: Record<string, unknown> ) {\n\t\tsuper();\n\n\t\t// Extend this instance with the additional (out of state) properties.\n\t\tif ( properties ) {\n\t\t\textend( this, properties );\n\t\t}\n\n\t\t// Initialize the attributes.\n\t\tif ( attributes ) {\n\t\t\tthis.set( attributes );\n\t\t}\n\t}\n}\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./balloonrotator.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./fakepanel.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/panel/balloon/contextualballoon\n */\n\nimport BalloonPanelView from './balloonpanelview';\nimport View from '../../view';\nimport ButtonView from '../../button/buttonview';\nimport type { ButtonExecuteEvent } from '../../button/button';\nimport type ViewCollection from '../../viewcollection';\n\nimport { Plugin, type Editor } from '@ckeditor/ckeditor5-core';\nimport {\n\tCKEditorError,\n\tFocusTracker,\n\tRect,\n\ttoUnit,\n\ttype Locale,\n\ttype ObservableChangeEvent,\n\ttype PositionOptions\n} from '@ckeditor/ckeditor5-utils';\n\nimport prevIcon from '../../../theme/icons/previous-arrow.svg';\nimport nextIcon from '../../../theme/icons/next-arrow.svg';\n\nimport '../../../theme/components/panel/balloonrotator.css';\nimport '../../../theme/components/panel/fakepanel.css';\n\nconst toPx = toUnit( 'px' );\n\n/**\n * Provides the common contextual balloon for the editor.\n *\n * The role of this plugin is to unify the contextual balloons logic, simplify views management and help\n * avoid the unnecessary complexity of handling multiple {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView}\n * instances in the editor.\n *\n * This plugin allows for creating single or multiple panel stacks.\n *\n * Each stack may have multiple views, with the one on the top being visible. When the visible view is removed from the stack,\n * the previous view becomes visible.\n *\n * It might be useful to implement nested navigation in a balloon. For instance, a toolbar view may contain a link button.\n * When you click it, a link view (which lets you set the URL) is created and put on top of the toolbar view, so the link panel\n * is displayed. When you finish editing the link and close (remove) the link view, the toolbar view is visible again.\n *\n * However, there are cases when there are multiple independent balloons to be displayed, for instance, if the selection\n * is inside two inline comments at the same time. For such cases, you can create two independent panel stacks.\n * The contextual balloon plugin will create a navigation bar to let the users switch between these panel stacks using the \"Next\"\n * and \"Previous\" buttons.\n *\n * If there are no views in the current stack, the balloon panel will try to switch to the next stack. If there are no\n * panels in any stack, the balloon panel will be hidden.\n *\n * **Note**: To force the balloon panel to show only one view, even if there are other stacks, use the `singleViewMode=true` option\n * when {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon#add adding} a view to a panel.\n *\n * From the implementation point of view, the contextual ballon plugin is reusing a single\n * {@link module:ui/panel/balloon/balloonpanelview~BalloonPanelView} instance to display multiple contextual balloon\n * panels in the editor. It also creates a special {@link module:ui/panel/balloon/contextualballoon~RotatorView rotator view},\n * used to manage multiple panel stacks. Rotator view is a child of the balloon panel view and the parent of the specific\n * view you want to display. If there is more than one panel stack to be displayed, the rotator view will add a\n * navigation bar. If there is only one stack, the rotator view is transparent (it does not add any UI elements).\n */\nexport default class ContextualBalloon extends Plugin {\n\t/**\n\t * The {@link module:utils/dom/position~Options#limiter position limiter}\n\t * for the {@link #view balloon}, used when no `limiter` has been passed into {@link #add}\n\t * or {@link #updatePosition}.\n\t *\n\t * By default, a function that obtains the farthest DOM\n\t * {@link module:engine/view/rooteditableelement~RootEditableElement}\n\t * of the {@link module:engine/view/document~Document#selection}.\n\t */\n\tpublic positionLimiter: PositionOptions[ 'limiter' ];\n\n\tpublic visibleStack?: string;\n\n\t/**\n\t * The currently visible view or `null` when there are no views in any stack.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public visibleView: View | null;\n\n\t/**\n\t * A total number of all stacks in the balloon.\n\t *\n\t * @private\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public _numberOfStacks: number;\n\n\t/**\n\t * A flag that controls the single view mode.\n\t *\n\t * @private\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public _singleViewMode: boolean;\n\n\t/**\n\t * The map of views and their stacks.\n\t */\n\tprivate _viewToStack = new Map<View, Stack>();\n\n\t/**\n\t * The map of IDs and stacks.\n\t */\n\tprivate _idToStack = new Map<string, Stack>();\n\n\t/**\n\t * The common balloon panel view.\n\t */\n\tprivate _view: BalloonPanelView | null = null;\n\n\t/**\n\t * Rotator view embedded in the contextual balloon.\n\t * Displays the currently visible view in the balloon and provides navigation for switching stacks.\n\t */\n\tprivate _rotatorView: RotatorView | null = null;\n\n\t/**\n\t * Displays fake panels under the balloon panel view when multiple stacks are added to the balloon.\n\t */\n\tprivate _fakePanelsView: FakePanelsView | null = null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ContextualBalloon' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis.positionLimiter = () => {\n\t\t\tconst view = this.editor.editing.view;\n\t\t\tconst viewDocument = view.document;\n\t\t\tconst editableElement = viewDocument.selection.editableElement;\n\n\t\t\tif ( editableElement ) {\n\t\t\t\treturn view.domConverter.mapViewToDom( editableElement.root ) as HTMLElement;\n\t\t\t}\n\n\t\t\treturn null;\n\t\t};\n\n\t\tthis.set( 'visibleView', null );\n\t\tthis.set( '_numberOfStacks', 0 );\n\t\tthis.set( '_singleViewMode', false );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tif ( this._view ) {\n\t\t\tthis._view.destroy();\n\t\t}\n\n\t\tif ( this._rotatorView ) {\n\t\t\tthis._rotatorView.destroy();\n\t\t}\n\n\t\tif ( this._fakePanelsView ) {\n\t\t\tthis._fakePanelsView.destroy();\n\t\t}\n\t}\n\n\t/**\n\t * The common balloon panel view.\n\t */\n\tpublic get view(): BalloonPanelView {\n\t\tif ( !this._view ) {\n\t\t\tthis._createPanelView();\n\t\t}\n\n\t\treturn this._view!;\n\t}\n\n\t/**\n\t * Returns `true` when the given view is in one of the stacks. Otherwise returns `false`.\n\t */\n\tpublic hasView( view: View ): boolean {\n\t\treturn Array.from( this._viewToStack.keys() ).includes( view );\n\t}\n\n\t/**\n\t * Adds a new view to the stack and makes it visible if the current stack is visible\n\t * or it is the first view in the balloon.\n\t *\n\t * @param data The configuration of the view.\n\t * @param data.stackId The ID of the stack that the view is added to. Defaults to `'main'`.\n\t * @param data.view The content of the balloon.\n\t * @param data.position Positioning options.\n\t * @param data.balloonClassName An additional CSS class added to the {@link #view balloon} when visible.\n\t * @param data.withArrow Whether the {@link #view balloon} should be rendered with an arrow. Defaults to `true`.\n\t * @param data.singleViewMode Whether the view should be the only visible view even if other stacks were added. Defaults to `false`.\n\t */\n\tpublic add( data: ViewConfiguration ): void {\n\t\tif ( !this._view ) {\n\t\t\tthis._createPanelView();\n\t\t}\n\n\t\tif ( this.hasView( data.view ) ) {\n\t\t\t/**\n\t\t\t * Trying to add configuration of the same view more than once.\n\t\t\t *\n\t\t\t * @error contextualballoon-add-view-exist\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'contextualballoon-add-view-exist',\n\t\t\t\t[ this, data ]\n\t\t\t);\n\t\t}\n\n\t\tconst stackId = data.stackId || 'main';\n\n\t\t// If new stack is added, creates it and show view from this stack.\n\t\tif ( !this._idToStack.has( stackId ) ) {\n\t\t\tthis._idToStack.set( stackId, new Map( [ [ data.view, data ] ] ) );\n\t\t\tthis._viewToStack.set( data.view, this._idToStack.get( stackId )! );\n\t\t\tthis._numberOfStacks = this._idToStack.size;\n\n\t\t\tif ( !this._visibleStack || data.singleViewMode ) {\n\t\t\t\tthis.showStack( stackId );\n\t\t\t}\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst stack = this._idToStack.get( stackId )!;\n\n\t\tif ( data.singleViewMode ) {\n\t\t\tthis.showStack( stackId );\n\t\t}\n\n\t\t// Add new view to the stack.\n\t\tstack.set( data.view, data );\n\t\tthis._viewToStack.set( data.view, stack );\n\n\t\t// And display it if is added to the currently visible stack.\n\t\tif ( stack === this._visibleStack ) {\n\t\t\tthis._showView( data );\n\t\t}\n\t}\n\n\t/**\n\t * Removes the given view from the stack. If the removed view was visible,\n\t * the view preceding it in the stack will become visible instead.\n\t * When there is no view in the stack, the next stack will be displayed.\n\t * When there are no more stacks, the balloon will hide.\n\t *\n\t * @param view A view to be removed from the balloon.\n\t */\n\tpublic remove( view: View ): void {\n\t\tif ( !this.hasView( view ) ) {\n\t\t\t/**\n\t\t\t * Trying to remove the configuration of the view not defined in the stack.\n\t\t\t *\n\t\t\t * @error contextualballoon-remove-view-not-exist\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'contextualballoon-remove-view-not-exist',\n\t\t\t\t[ this, view ]\n\t\t\t);\n\t\t}\n\n\t\tconst stack = this._viewToStack.get( view )!;\n\n\t\tif ( this._singleViewMode && this.visibleView === view ) {\n\t\t\tthis._singleViewMode = false;\n\t\t}\n\n\t\t// When visible view will be removed we need to show a preceding view or next stack\n\t\t// if a view is the only view in the stack.\n\t\tif ( this.visibleView === view ) {\n\t\t\tif ( stack.size === 1 ) {\n\t\t\t\tif ( this._idToStack.size > 1 ) {\n\t\t\t\t\tthis._showNextStack();\n\t\t\t\t} else {\n\t\t\t\t\tthis.view.hide();\n\t\t\t\t\tthis.visibleView = null;\n\t\t\t\t\tthis._rotatorView!.hideView();\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tthis._showView( Array.from( stack.values() )[ stack.size - 2 ] );\n\t\t\t}\n\t\t}\n\n\t\tif ( stack.size === 1 ) {\n\t\t\tthis._idToStack.delete( this._getStackId( stack ) );\n\t\t\tthis._numberOfStacks = this._idToStack.size;\n\t\t} else {\n\t\t\tstack.delete( view );\n\t\t}\n\n\t\tthis._viewToStack.delete( view );\n\t}\n\n\t/**\n\t * Updates the position of the balloon using the position data of the first visible view in the stack.\n\t * When new position data is given, the position data of the currently visible view will be updated.\n\t *\n\t * @param position Position options.\n\t */\n\tpublic updatePosition( position?: Partial<PositionOptions> ): void {\n\t\tif ( position ) {\n\t\t\tthis._visibleStack.get( this.visibleView! )!.position = position;\n\t\t}\n\n\t\tthis.view.pin( this._getBalloonPosition()! );\n\t\tthis._fakePanelsView!.updatePosition();\n\t}\n\n\t/**\n\t * Shows the last view from the stack of a given ID.\n\t */\n\tpublic showStack( id: string ): void {\n\t\tthis.visibleStack = id;\n\t\tconst stack = this._idToStack.get( id );\n\n\t\tif ( !stack ) {\n\t\t\t/**\n\t\t\t * Trying to show a stack that does not exist.\n\t\t\t *\n\t\t\t * @error contextualballoon-showstack-stack-not-exist\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'contextualballoon-showstack-stack-not-exist',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\tif ( this._visibleStack === stack ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._showView( Array.from( stack.values() ).pop()! );\n\t}\n\n\t/**\n\t * Initializes view instances.\n\t */\n\tprivate _createPanelView(): void {\n\t\tthis._view = new BalloonPanelView( this.editor.locale );\n\n\t\tthis.editor.ui.view.body.add( this._view );\n\t\tthis.editor.ui.focusTracker.add( this._view.element! );\n\n\t\tthis._rotatorView = this._createRotatorView();\n\t\tthis._fakePanelsView = this._createFakePanelsView();\n\t}\n\n\t/**\n\t * Returns the stack of the currently visible view.\n\t */\n\tprivate get _visibleStack() {\n\t\treturn this._viewToStack.get( this.visibleView! )!;\n\t}\n\n\t/**\n\t * Returns the ID of the given stack.\n\t */\n\tprivate _getStackId( stack: Stack ) {\n\t\tconst entry = Array.from( this._idToStack.entries() ).find( entry => entry[ 1 ] === stack )!;\n\n\t\treturn entry[ 0 ];\n\t}\n\n\t/**\n\t * Shows the last view from the next stack.\n\t */\n\tprivate _showNextStack() {\n\t\tconst stacks = Array.from( this._idToStack.values() );\n\n\t\tlet nextIndex = stacks.indexOf( this._visibleStack ) + 1;\n\n\t\tif ( !stacks[ nextIndex ] ) {\n\t\t\tnextIndex = 0;\n\t\t}\n\n\t\tthis.showStack( this._getStackId( stacks[ nextIndex ] ) );\n\t}\n\n\t/**\n\t * Shows the last view from the previous stack.\n\t */\n\tprivate _showPrevStack() {\n\t\tconst stacks = Array.from( this._idToStack.values() );\n\n\t\tlet nextIndex = stacks.indexOf( this._visibleStack ) - 1;\n\n\t\tif ( !stacks[ nextIndex ] ) {\n\t\t\tnextIndex = stacks.length - 1;\n\t\t}\n\n\t\tthis.showStack( this._getStackId( stacks[ nextIndex ] ) );\n\t}\n\n\t/**\n\t * Creates a rotator view.\n\t */\n\tprivate _createRotatorView() {\n\t\tconst view = new RotatorView( this.editor.locale );\n\t\tconst t = this.editor.locale.t;\n\n\t\tthis.view.content.add( view );\n\n\t\t// Hide navigation when there is only a one stack & not in single view mode.\n\t\tview.bind( 'isNavigationVisible' ).to( this, '_numberOfStacks', this, '_singleViewMode', ( value, isSingleViewMode ) => {\n\t\t\treturn !isSingleViewMode && value > 1;\n\t\t} );\n\n\t\t// Update balloon position after toggling navigation.\n\t\tview.on<ObservableChangeEvent>( 'change:isNavigationVisible', () => ( this.updatePosition() ), { priority: 'low' } );\n\n\t\t// Update stacks counter value.\n\t\tview.bind( 'counter' ).to( this, 'visibleView', this, '_numberOfStacks', ( visibleView, numberOfStacks ) => {\n\t\t\tif ( numberOfStacks < 2 ) {\n\t\t\t\treturn '';\n\t\t\t}\n\n\t\t\tconst current = Array.from( this._idToStack.values() ).indexOf( this._visibleStack ) + 1;\n\n\t\t\treturn t( '%0 of %1', [ current, numberOfStacks ] );\n\t\t} );\n\n\t\tview.buttonNextView.on<ButtonExecuteEvent>( 'execute', () => {\n\t\t\t// When current view has a focus then move focus to the editable before removing it,\n\t\t\t// otherwise editor will lost focus.\n\t\t\tif ( view.focusTracker.isFocused ) {\n\t\t\t\tthis.editor.editing.view.focus();\n\t\t\t}\n\n\t\t\tthis._showNextStack();\n\t\t} );\n\n\t\tview.buttonPrevView.on<ButtonExecuteEvent>( 'execute', () => {\n\t\t\t// When current view has a focus then move focus to the editable before removing it,\n\t\t\t// otherwise editor will lost focus.\n\t\t\tif ( view.focusTracker.isFocused ) {\n\t\t\t\tthis.editor.editing.view.focus();\n\t\t\t}\n\n\t\t\tthis._showPrevStack();\n\t\t} );\n\n\t\treturn view;\n\t}\n\n\t/**\n\t * Creates a fake panels view.\n\t */\n\tprivate _createFakePanelsView() {\n\t\tconst view = new FakePanelsView( this.editor.locale, this.view );\n\n\t\tview.bind( 'numberOfPanels' ).to( this, '_numberOfStacks', this, '_singleViewMode', ( number, isSingleViewMode ) => {\n\t\t\tconst showPanels = !isSingleViewMode && number >= 2;\n\n\t\t\treturn showPanels ? Math.min( number - 1, 2 ) : 0;\n\t\t} );\n\n\t\tview.listenTo<ObservableChangeEvent>( this.view, 'change:top', () => view.updatePosition() );\n\t\tview.listenTo<ObservableChangeEvent>( this.view, 'change:left', () => view.updatePosition() );\n\n\t\tthis.editor.ui.view.body.add( view );\n\n\t\treturn view;\n\t}\n\n\t/**\n\t * Sets the view as the content of the balloon and attaches the balloon using position\n\t * options of the first view.\n\t *\n\t * @param data Configuration.\n\t * @param data.view The view to show in the balloon.\n\t * @param data.balloonClassName Additional class name which will be added to the {@link #view balloon}.\n\t * @param data.withArrow Whether the {@link #view balloon} should be rendered with an arrow.\n\t */\n\tprivate _showView( { view, balloonClassName = '', withArrow = true, singleViewMode = false }: ViewConfiguration ) {\n\t\tthis.view.class = balloonClassName;\n\t\tthis.view.withArrow = withArrow;\n\n\t\tthis._rotatorView!.showView( view );\n\t\tthis.visibleView = view;\n\t\tthis.view.pin( this._getBalloonPosition()! );\n\t\tthis._fakePanelsView!.updatePosition();\n\n\t\tif ( singleViewMode ) {\n\t\t\tthis._singleViewMode = true;\n\t\t}\n\t}\n\n\t/**\n\t * Returns position options of the last view in the stack.\n\t * This keeps the balloon in the same position when the view is changed.\n\t */\n\tprivate _getBalloonPosition() {\n\t\tlet position = Array.from( this._visibleStack.values() ).pop()!.position;\n\n\t\tif ( position ) {\n\t\t\t// Use the default limiter if none has been specified.\n\t\t\tif ( !position.limiter ) {\n\t\t\t\t// Don't modify the original options object.\n\t\t\t\tposition = Object.assign( {}, position, {\n\t\t\t\t\tlimiter: this.positionLimiter\n\t\t\t\t} );\n\t\t\t}\n\n\t\t\t// Don't modify the original options object.\n\t\t\tposition = Object.assign( {}, position, {\n\t\t\t\tviewportOffsetConfig: this.editor.ui.viewportOffset\n\t\t\t} );\n\t\t}\n\n\t\treturn position;\n\t}\n}\n\n/**\n * The configuration of the view.\n */\nexport interface ViewConfiguration {\n\n\t/**\n\t * The ID of the stack that the view is added to.\n\t *\n\t * @default 'main'\n\t */\n\tstackId?: string;\n\n\t/**\n\t * The content of the balloon.\n\t */\n\tview: View;\n\n\t/**\n\t * Positioning options.\n\t */\n\tposition?: Partial<PositionOptions>;\n\n\t/**\n\t * An additional CSS class added to the {@link #view balloon} when visible.\n\t */\n\tballoonClassName?: string;\n\n\t/**\n\t * Whether the {@link #view balloon} should be rendered with an arrow.\n\t *\n\t * @default true\n\t */\n\twithArrow?: boolean;\n\n\t/**\n\t * Whether the view should be the only visible view even if other stacks were added.\n\t *\n\t * @default false\n\t */\n\tsingleViewMode?: boolean;\n}\n\ntype Stack = Map<View, ViewConfiguration>;\n\n/**\n * Rotator view is a helper class for the {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon ContextualBalloon}.\n * It is used for displaying the last view from the current stack and providing navigation buttons for switching stacks.\n * See the {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon ContextualBalloon} documentation to learn more.\n */\nexport class RotatorView extends View {\n\t/**\n\t * Used for checking if a view is focused or not.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * Navigation button for switching the stack to the previous one.\n\t */\n\tpublic readonly buttonPrevView: ButtonView;\n\n\t/**\n\t * Navigation button for switching the stack to the next one.\n\t */\n\tpublic readonly buttonNextView: ButtonView;\n\n\t/**\n\t * A collection of the child views that creates the rotator content.\n\t */\n\tpublic readonly content: ViewCollection;\n\n\t/**\n\t * Defines whether navigation is visible or not.\n\t *\n\t * @observable\n\t */\n\tdeclare public isNavigationVisible: boolean;\n\n\t/**\n\t * @observable\n\t */\n\tdeclare public counter: string;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst t = locale.t;\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.set( 'isNavigationVisible', true );\n\n\t\tthis.focusTracker = new FocusTracker();\n\t\tthis.buttonPrevView = this._createButtonView( t( 'Previous' ), prevIcon );\n\t\tthis.buttonNextView = this._createButtonView( t( 'Next' ), nextIcon );\n\t\tthis.content = this.createCollection();\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-balloon-rotator'\n\t\t\t\t],\n\t\t\t\t'z-index': '-1'\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttag: 'div',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: [\n\t\t\t\t\t\t\t'ck-balloon-rotator__navigation',\n\t\t\t\t\t\t\tbind.to( 'isNavigationVisible', value => value ? '' : 'ck-hidden' )\n\t\t\t\t\t\t]\n\t\t\t\t\t},\n\t\t\t\t\tchildren: [\n\t\t\t\t\t\tthis.buttonPrevView,\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\ttag: 'span',\n\n\t\t\t\t\t\t\tattributes: {\n\t\t\t\t\t\t\t\tclass: [\n\t\t\t\t\t\t\t\t\t'ck-balloon-rotator__counter'\n\t\t\t\t\t\t\t\t]\n\t\t\t\t\t\t\t},\n\n\t\t\t\t\t\t\tchildren: [\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\ttext: bind.to( 'counter' )\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t]\n\t\t\t\t\t\t},\n\t\t\t\t\t\tthis.buttonNextView\n\t\t\t\t\t]\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttag: 'div',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: 'ck-balloon-rotator__content'\n\t\t\t\t\t},\n\t\t\t\t\tchildren: this.content\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.focusTracker.add( this.element! );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.focusTracker.destroy();\n\t}\n\n\t/**\n\t * Shows a given view.\n\t *\n\t * @param view The view to show.\n\t */\n\tpublic showView( view: View ): void {\n\t\tthis.hideView();\n\t\tthis.content.add( view );\n\t}\n\n\t/**\n\t * Hides the currently displayed view.\n\t */\n\tpublic hideView(): void {\n\t\tthis.content.clear();\n\t}\n\n\t/**\n\t * Creates a navigation button view.\n\t *\n\t * @param label The button label.\n\t * @param icon The button icon.\n\t */\n\tprivate _createButtonView( label: string, icon: string ) {\n\t\tconst view = new ButtonView( this.locale );\n\n\t\tview.set( {\n\t\t\tlabel,\n\t\t\ticon,\n\t\t\ttooltip: true\n\t\t} );\n\n\t\treturn view;\n\t}\n}\n\n/**\n * Displays additional layers under the balloon when multiple stacks are added to the balloon.\n */\nclass FakePanelsView extends View {\n\t/**\n\t * Collection of the child views which creates fake panel content.\n\t */\n\tpublic readonly content: ViewCollection;\n\n\t/**\n\t * Fake panels top offset.\n\t *\n\t * @observable\n\t */\n\tdeclare public top: number;\n\n\t/**\n\t * Fake panels left offset.\n\t *\n\t * @observable\n\t */\n\tdeclare public left: number;\n\n\t/**\n\t * Fake panels height.\n\t *\n\t * @observable\n\t */\n\tdeclare public height: number;\n\n\t/**\n\t * Fake panels width.\n\t *\n\t * @observable\n\t */\n\tdeclare public width: number;\n\n\t/**\n\t * Number of rendered fake panels.\n\t *\n\t * @observable\n\t */\n\tdeclare public numberOfPanels: number;\n\n\t/**\n\t * Context.\n\t */\n\tprivate readonly _balloonPanelView: BalloonPanelView;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale: Locale, balloonPanelView: BalloonPanelView ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.set( 'top', 0 );\n\t\tthis.set( 'left', 0 );\n\t\tthis.set( 'height', 0 );\n\t\tthis.set( 'width', 0 );\n\t\tthis.set( 'numberOfPanels', 0 );\n\n\t\tthis.content = this.createCollection();\n\t\tthis._balloonPanelView = balloonPanelView;\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck-fake-panel',\n\t\t\t\t\tbind.to( 'numberOfPanels', number => number ? '' : 'ck-hidden' )\n\t\t\t\t],\n\t\t\t\tstyle: {\n\t\t\t\t\ttop: bind.to( 'top', toPx ),\n\t\t\t\t\tleft: bind.to( 'left', toPx ),\n\t\t\t\t\twidth: bind.to( 'width', toPx ),\n\t\t\t\t\theight: bind.to( 'height', toPx )\n\t\t\t\t}\n\t\t\t},\n\t\t\tchildren: this.content\n\t\t} );\n\n\t\tthis.on<ObservableChangeEvent<number>>( 'change:numberOfPanels', ( evt, name, next, prev ) => {\n\t\t\tif ( next > prev ) {\n\t\t\t\tthis._addPanels( next - prev );\n\t\t\t} else {\n\t\t\t\tthis._removePanels( prev - next );\n\t\t\t}\n\n\t\t\tthis.updatePosition();\n\t\t} );\n\t}\n\n\tprivate _addPanels( number: number ) {\n\t\twhile ( number-- ) {\n\t\t\tconst view = new View();\n\n\t\t\tview.setTemplate( { tag: 'div' } );\n\n\t\t\tthis.content.add( view );\n\t\t\tthis.registerChild( view );\n\t\t}\n\t}\n\n\tprivate _removePanels( number: number ) {\n\t\twhile ( number-- ) {\n\t\t\tconst view = this.content.last!;\n\n\t\t\tthis.content.remove( view );\n\t\t\tthis.deregisterChild( view );\n\t\t\tview.destroy();\n\t\t}\n\t}\n\n\t/**\n\t * Updates coordinates of fake panels.\n\t */\n\tpublic updatePosition() {\n\t\tif ( this.numberOfPanels ) {\n\t\t\tconst { top, left } = this._balloonPanelView;\n\t\t\tconst { width, height } = new Rect( this._balloonPanelView.element! );\n\n\t\t\tObject.assign( this, { top, left, width, height } );\n\t\t}\n\t}\n}\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M11.463 5.187a.888.888 0 1 1 1.254 1.255L9.16 10l3.557 3.557a.888.888 0 1 1-1.254 1.255L7.26 10.61a.888.888 0 0 1 .16-1.382l4.043-4.042z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M8.537 14.813a.888.888 0 1 1-1.254-1.255L10.84 10 7.283 6.442a.888.888 0 1 1 1.254-1.255L12.74 9.39a.888.888 0 0 1-.16 1.382l-4.043 4.042z\\\"/></svg>\";","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./stickypanel.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/panel/sticky/stickypanelview\n */\n\nimport View from '../../view';\nimport Template from '../../template';\n\nimport type ViewCollection from '../../viewcollection';\n\nimport {\n\ttype Locale,\n\ttype ObservableChangeEvent,\n\tgetElementsIntersectionRect,\n\tgetScrollableAncestors,\n\tglobal,\n\ttoUnit,\n\tRect\n} from '@ckeditor/ckeditor5-utils';\n\n// @if CK_DEBUG_STICKYPANEL // const RectDrawer = require( '@ckeditor/ckeditor5-utils/tests/_utils/rectdrawer' ).default\n\nimport '../../../theme/components/panel/stickypanel.css';\n\nconst toPx = toUnit( 'px' );\n\n/**\n * The sticky panel view class.\n */\nexport default class StickyPanelView extends View {\n\t/**\n\t * Collection of the child views which creates balloon panel contents.\n\t */\n\tpublic readonly content: ViewCollection;\n\n\t/**\n\t * Controls whether the sticky panel should be active.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public isActive: boolean;\n\n\t/**\n\t * Controls whether the sticky panel is in the \"sticky\" state.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public isSticky: boolean;\n\n\t/**\n\t * The limiter element for the sticky panel instance. Its bounding rect limits\n\t * the \"stickyness\" of the panel, i.e. when the panel reaches the bottom\n\t * edge of the limiter, it becomes sticky to that edge and does not float\n\t * off the limiter. It is mandatory for the panel to work properly and once\n\t * set, it cannot be changed.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public limiterElement: HTMLElement | null;\n\n\t/**\n\t * The offset from the bottom edge of {@link #limiterElement}\n\t * which stops the panel from stickying any further to prevent limiter's content\n\t * from being completely covered.\n\t *\n\t * @readonly\n\t * @observable\n\t * @default 50\n\t */\n\tdeclare public limiterBottomOffset: number;\n\n\t/**\n\t * The offset from the top edge of the web browser's viewport which makes the\n\t * panel become sticky. The default value is `0`, which means the panel becomes\n\t * sticky when it's upper edge touches the top of the page viewport.\n\t *\n\t * This attribute is useful when the web page has UI elements positioned to the top\n\t * either using `position: fixed` or `position: sticky`, which would cover the\n\t * sticky panel or viceversa (depending on the `z-index` hierarchy).\n\t *\n\t * Bound to {@link module:ui/editorui/editorui~EditorUI#viewportOffset `EditorUI#viewportOffset`}.\n\t *\n\t * If {@link module:core/editor/editorconfig~EditorConfig#ui `EditorConfig#ui.viewportOffset.top`} is defined, then\n\t * it will override the default value.\n\t *\n\t * @observable\n\t * @default 0\n\t */\n\tdeclare public viewportTopOffset: number;\n\n\t/**\n\t * Controls the `margin-left` CSS style of the panel.\n\t *\n\t * @private\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public _marginLeft: string | null;\n\n\t/**\n\t * Set `true` if the sticky panel reached the bottom edge of the\n\t * {@link #limiterElement}.\n\t *\n\t * @private\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public _isStickyToTheBottomOfLimiter: boolean;\n\n\t/**\n\t * The `top` CSS position of the panel when it is sticky to the top of the viewport or scrollable\n\t * ancestors of the {@link #limiterElement}.\n\t *\n\t * @private\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public _stickyTopOffset: number | null;\n\n\t/**\n\t * The `bottom` CSS position of the panel when it is sticky to the bottom of the {@link #limiterElement}.\n\t *\n\t * @private\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public _stickyBottomOffset: number | null;\n\n\t/**\n\t * A dummy element which visually fills the space as long as the\n\t * actual panel is sticky. It prevents flickering of the UI.\n\t */\n\tprivate _contentPanelPlaceholder: HTMLElement;\n\n\t/**\n\t * The panel which accepts children into {@link #content} collection.\n\t * Also an element which is positioned when {@link #isSticky}.\n\t */\n\tprivate _contentPanel: HTMLElement;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.set( 'isActive', false );\n\t\tthis.set( 'isSticky', false );\n\t\tthis.set( 'limiterElement', null );\n\t\tthis.set( 'limiterBottomOffset', 50 );\n\t\tthis.set( 'viewportTopOffset', 0 );\n\n\t\tthis.set( '_marginLeft', null );\n\t\tthis.set( '_isStickyToTheBottomOfLimiter', false );\n\n\t\tthis.set( '_stickyTopOffset', null );\n\t\tthis.set( '_stickyBottomOffset', null );\n\n\t\tthis.content = this.createCollection();\n\n\t\tthis._contentPanelPlaceholder = new Template( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-sticky-panel__placeholder'\n\t\t\t\t],\n\t\t\t\tstyle: {\n\t\t\t\t\tdisplay: bind.to( 'isSticky', isSticky => isSticky ? 'block' : 'none' ),\n\t\t\t\t\theight: bind.to( 'isSticky', isSticky => {\n\t\t\t\t\t\treturn isSticky ? toPx( this._contentPanelRect.height ) : null;\n\t\t\t\t\t} )\n\t\t\t\t}\n\t\t\t}\n\t\t} ).render() as HTMLElement;\n\n\t\tthis._contentPanel = new Template( {\n\t\t\ttag: 'div',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-sticky-panel__content',\n\t\t\t\t\t// Toggle class of the panel when \"sticky\" state changes in the view.\n\t\t\t\t\tbind.if( 'isSticky', 'ck-sticky-panel__content_sticky' ),\n\t\t\t\t\tbind.if( '_isStickyToTheBottomOfLimiter', 'ck-sticky-panel__content_sticky_bottom-limit' )\n\t\t\t\t],\n\t\t\t\tstyle: {\n\t\t\t\t\twidth: bind.to( 'isSticky', isSticky => {\n\t\t\t\t\t\treturn isSticky ? toPx( this._contentPanelPlaceholder.getBoundingClientRect().width ) : null;\n\t\t\t\t\t} ),\n\n\t\t\t\t\ttop: bind.to( '_stickyTopOffset', value => value ? toPx( value ) : value ),\n\t\t\t\t\tbottom: bind.to( '_stickyBottomOffset', value => value ? toPx( value ) : value ),\n\n\t\t\t\t\tmarginLeft: bind.to( '_marginLeft' )\n\t\t\t\t}\n\t\t\t},\n\n\t\t\tchildren: this.content\n\t\t} ).render() as HTMLElement;\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-sticky-panel'\n\t\t\t\t]\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\tthis._contentPanelPlaceholder,\n\t\t\t\tthis._contentPanel\n\t\t\t]\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\t// Check if the panel should go into the sticky state immediately.\n\t\tthis.checkIfShouldBeSticky();\n\n\t\t// Update sticky state of the panel as the window and ancestors are being scrolled.\n\t\tthis.listenTo( global.document, 'scroll', ( evt, data ) => {\n\t\t\tthis.checkIfShouldBeSticky( data.target as HTMLElement | Document );\n\t\t}, { useCapture: true } );\n\n\t\t// Synchronize with `model.isActive` because sticking an inactive panel is pointless.\n\t\tthis.listenTo<ObservableChangeEvent>( this, 'change:isActive', () => {\n\t\t\tthis.checkIfShouldBeSticky();\n\t\t} );\n\t}\n\n\t/**\n\t * Analyzes the environment to decide whether the panel should be sticky or not.\n\t * Then handles the positioning of the panel.\n\t *\n\t * @param [scrollTarget] The element which is being scrolled.\n\t */\n\tpublic checkIfShouldBeSticky( scrollTarget?: HTMLElement | Document ): void {\n\t\t// @if CK_DEBUG_STICKYPANEL // RectDrawer.clear();\n\n\t\tif ( !this.limiterElement || !this.isActive ) {\n\t\t\tthis._unstick();\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst scrollableAncestors = getScrollableAncestors( this.limiterElement );\n\n\t\tif ( scrollTarget && !scrollableAncestors.includes( scrollTarget ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst visibleAncestorsRect = getElementsIntersectionRect( scrollableAncestors, this.viewportTopOffset );\n\t\tconst limiterRect = new Rect( this.limiterElement );\n\n\t\t// @if CK_DEBUG_STICKYPANEL // if ( visibleAncestorsRect ) {\n\t\t// @if CK_DEBUG_STICKYPANEL // \tRectDrawer.draw( visibleAncestorsRect,\n\t\t// @if CK_DEBUG_STICKYPANEL // \t\t{ outlineWidth: '3px', opacity: '.8', outlineColor: 'red', outlineOffset: '-3px' },\n\t\t// @if CK_DEBUG_STICKYPANEL // \t\t'Visible anc'\n\t\t// @if CK_DEBUG_STICKYPANEL // \t);\n\t\t// @if CK_DEBUG_STICKYPANEL // }\n\t\t// @if CK_DEBUG_STICKYPANEL //\n\t\t// @if CK_DEBUG_STICKYPANEL // RectDrawer.draw( limiterRect,\n\t\t// @if CK_DEBUG_STICKYPANEL // \t{ outlineWidth: '3px', opacity: '.8', outlineColor: 'green', outlineOffset: '-3px' },\n\t\t// @if CK_DEBUG_STICKYPANEL // \t'Limiter'\n\t\t// @if CK_DEBUG_STICKYPANEL // );\n\n\t\t// Stick the panel only if\n\t\t// * the limiter's ancestors are intersecting with each other so that some of their rects are visible,\n\t\t// * and the limiter's top edge is above the visible ancestors' top edge.\n\t\tif ( visibleAncestorsRect && limiterRect.top < visibleAncestorsRect.top ) {\n\t\t\tconst visibleLimiterRect = limiterRect.getIntersection( visibleAncestorsRect );\n\n\t\t\t// Sticky the panel only if the limiter's visible rect is at least partially visible in the\n\t\t\t// visible ancestors' rects intersection.\n\t\t\tif ( visibleLimiterRect ) {\n\t\t\t\t// @if CK_DEBUG_STICKYPANEL // RectDrawer.draw( visibleLimiterRect,\n\t\t\t\t// @if CK_DEBUG_STICKYPANEL // \t{ outlineWidth: '3px', opacity: '.8', outlineColor: 'fuchsia', outlineOffset: '-3px',\n\t\t\t\t// @if CK_DEBUG_STICKYPANEL // \t\tbackgroundColor: 'rgba(255, 0, 255, .3)' },\n\t\t\t\t// @if CK_DEBUG_STICKYPANEL // \t'Visible limiter'\n\t\t\t\t// @if CK_DEBUG_STICKYPANEL // );\n\n\t\t\t\tconst visibleAncestorsTop = visibleAncestorsRect.top;\n\n\t\t\t\t// Check if there's a change the panel can be sticky to the bottom of the limiter.\n\t\t\t\tif ( visibleAncestorsTop + this._contentPanelRect.height + this.limiterBottomOffset > visibleLimiterRect.bottom ) {\n\t\t\t\t\tconst stickyBottomOffset = Math.max( limiterRect.bottom - visibleAncestorsRect.bottom, 0 ) + this.limiterBottomOffset;\n\t\t\t\t\t// @if CK_DEBUG_STICKYPANEL // const stickyBottomOffsetRect = new Rect( {\n\t\t\t\t\t// @if CK_DEBUG_STICKYPANEL // \ttop: limiterRect.bottom - stickyBottomOffset, left: 0, right: 2000,\n\t\t\t\t\t// @if CK_DEBUG_STICKYPANEL // \tbottom: limiterRect.bottom - stickyBottomOffset, width: 2000, height: 1\n\t\t\t\t\t// @if CK_DEBUG_STICKYPANEL // } );\n\t\t\t\t\t// @if CK_DEBUG_STICKYPANEL // RectDrawer.draw( stickyBottomOffsetRect,\n\t\t\t\t\t// @if CK_DEBUG_STICKYPANEL // \t{ outlineWidth: '1px', opacity: '.8', outlineColor: 'black' },\n\t\t\t\t\t// @if CK_DEBUG_STICKYPANEL // \t'Sticky bottom offset'\n\t\t\t\t\t// @if CK_DEBUG_STICKYPANEL // );\n\n\t\t\t\t\t// Check if sticking the panel to the bottom of the limiter does not cause it to suddenly\n\t\t\t\t\t// move upwards if there's not enough space for it.\n\t\t\t\t\tif ( limiterRect.bottom - stickyBottomOffset > limiterRect.top + this._contentPanelRect.height ) {\n\t\t\t\t\t\tthis._stickToBottomOfLimiter( stickyBottomOffset );\n\t\t\t\t\t} else {\n\t\t\t\t\t\tthis._unstick();\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\tif ( this._contentPanelRect.height + this.limiterBottomOffset < limiterRect.height ) {\n\t\t\t\t\t\tthis._stickToTopOfAncestors( visibleAncestorsTop );\n\t\t\t\t\t} else {\n\t\t\t\t\t\tthis._unstick();\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tthis._unstick();\n\t\t\t}\n\t\t} else {\n\t\t\tthis._unstick();\n\t\t}\n\n\t\t// @if CK_DEBUG_STICKYPANEL // console.clear();\n\t\t// @if CK_DEBUG_STICKYPANEL // console.log( 'isSticky', this.isSticky );\n\t\t// @if CK_DEBUG_STICKYPANEL // console.log( '_isStickyToTheBottomOfLimiter', this._isStickyToTheBottomOfLimiter );\n\t\t// @if CK_DEBUG_STICKYPANEL // console.log( '_stickyTopOffset', this._stickyTopOffset );\n\t\t// @if CK_DEBUG_STICKYPANEL // console.log( '_stickyBottomOffset', this._stickyBottomOffset );\n\t}\n\n\t/**\n\t * Sticks the panel at the given CSS `top` offset.\n\t *\n\t * @private\n\t * @param topOffset\n\t */\n\tprivate _stickToTopOfAncestors( topOffset: number ) {\n\t\tthis.isSticky = true;\n\t\tthis._isStickyToTheBottomOfLimiter = false;\n\t\tthis._stickyTopOffset = topOffset;\n\t\tthis._stickyBottomOffset = null;\n\t\tthis._marginLeft = toPx( -global.window.scrollX );\n\t}\n\n\t/**\n\t * Sticks the panel at the bottom of the limiter with a given CSS `bottom` offset.\n\t *\n\t * @private\n\t * @param stickyBottomOffset\n\t */\n\tprivate _stickToBottomOfLimiter( stickyBottomOffset: number ) {\n\t\tthis.isSticky = true;\n\t\tthis._isStickyToTheBottomOfLimiter = true;\n\t\tthis._stickyTopOffset = null;\n\t\tthis._stickyBottomOffset = stickyBottomOffset;\n\t\tthis._marginLeft = toPx( -global.window.scrollX );\n\t}\n\n\t/**\n\t * Unsticks the panel putting it back to its original position.\n\t *\n\t * @private\n\t */\n\tprivate _unstick() {\n\t\tthis.isSticky = false;\n\t\tthis._isStickyToTheBottomOfLimiter = false;\n\t\tthis._stickyTopOffset = null;\n\t\tthis._stickyBottomOffset = null;\n\t\tthis._marginLeft = null;\n\t}\n\n\t/**\n\t * Returns the bounding rect of the {@link #_contentPanel}.\n\t *\n\t * @private\n\t */\n\tprivate get _contentPanelRect(): Rect {\n\t\treturn new Rect( this._contentPanel );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/toolbar/balloon/balloontoolbar\n */\n\nimport ContextualBalloon from '../../panel/balloon/contextualballoon';\nimport ToolbarView, { type ToolbarViewGroupedItemsUpdateEvent } from '../toolbarview';\nimport BalloonPanelView, { generatePositions } from '../../panel/balloon/balloonpanelview';\nimport normalizeToolbarConfig from '../normalizetoolbarconfig';\n\nimport type { EditorUIReadyEvent, EditorUIUpdateEvent } from '../../editorui/editorui';\n\nimport {\n\tPlugin,\n\ttype Editor,\n\ttype EditorReadyEvent\n} from '@ckeditor/ckeditor5-core';\n\nimport {\n\tFocusTracker,\n\tRect,\n\tResizeObserver,\n\tenv,\n\tglobal,\n\ttoUnit,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport type {\n\tDocumentSelection,\n\tDocumentSelectionChangeRangeEvent,\n\tSchema\n} from '@ckeditor/ckeditor5-engine';\n\nimport { debounce, type DebouncedFunc } from 'lodash-es';\n\nconst toPx = toUnit( 'px' );\n\n/**\n * The contextual toolbar.\n *\n * It uses the {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon contextual balloon plugin}.\n */\nexport default class BalloonToolbar extends Plugin {\n\t/**\n\t * The toolbar view displayed in the balloon.\n\t */\n\tpublic readonly toolbarView: ToolbarView;\n\n\t/**\n\t * Tracks the focus of the {@link module:ui/editorui/editorui~EditorUI#getEditableElement editable element}\n\t * and the {@link #toolbarView}. When both are blurred then the toolbar should hide.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * A cached and normalized `config.balloonToolbar` object.\n\t */\n\tprivate _balloonConfig: ReturnType<typeof normalizeToolbarConfig>;\n\n\t/**\n\t * An instance of the resize observer that allows to respond to changes in editable's geometry\n\t * so the toolbar can stay within its boundaries (and group toolbar items that do not fit).\n\t *\n\t * **Note**: Used only when `shouldNotGroupWhenFull` was **not** set in the\n\t * {@link module:core/editor/editorconfig~EditorConfig#balloonToolbar configuration}.\n\t *\n\t * **Note:** Created in {@link #init}.\n\t */\n\tprivate _resizeObserver: ResizeObserver | null = null;\n\n\t/**\n\t * The contextual balloon plugin instance.\n\t */\n\tprivate readonly _balloon: ContextualBalloon;\n\n\t/**\n\t * Fires `_selectionChangeDebounced` event using `lodash#debounce`.\n\t *\n\t * This event is an internal plugin event which is fired 200 ms after model selection last change.\n\t * This is to makes easy test debounced action without need to use `setTimeout`.\n\t *\n\t * This function is stored as a plugin property to make possible to cancel\n\t * trailing debounced invocation on destroy.\n\t */\n\tprivate readonly _fireSelectionChangeDebounced: DebouncedFunc<() => void>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'BalloonToolbar' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ContextualBalloon ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis._balloonConfig = normalizeToolbarConfig( editor.config.get( 'balloonToolbar' ) );\n\t\tthis.toolbarView = this._createToolbarView();\n\t\tthis.focusTracker = new FocusTracker();\n\n\t\t// Wait for the EditorUI#init. EditableElement is not available before.\n\t\teditor.ui.once<EditorUIReadyEvent>( 'ready', () => {\n\t\t\tthis.focusTracker.add( editor.ui.getEditableElement()! );\n\t\t\tthis.focusTracker.add( this.toolbarView.element! );\n\t\t} );\n\n\t\t// Register the toolbar so it becomes available for Alt+F10 and Esc navigation.\n\t\teditor.ui.addToolbar( this.toolbarView, {\n\t\t\tbeforeFocus: () => this.show( true ),\n\t\t\tafterBlur: () => this.hide(),\n\t\t\tisContextual: true\n\t\t} );\n\n\t\tthis._balloon = editor.plugins.get( ContextualBalloon );\n\t\tthis._fireSelectionChangeDebounced = debounce( () => this.fire( '_selectionChangeDebounced' ), 200 );\n\n\t\t// The appearance of the BalloonToolbar method is eventdriven.\n\t\t// It is possible to stop the #show event and this prevent the toolbar from showing up.\n\t\tthis.decorate( 'show' );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\n\t\t// Show/hide the toolbar on editable focus/blur.\n\t\tthis.listenTo<ObservableChangeEvent<boolean>>( this.focusTracker, 'change:isFocused', ( evt, name, isFocused ) => {\n\t\t\tconst isToolbarVisible = this._balloon.visibleView === this.toolbarView;\n\n\t\t\tif ( !isFocused && isToolbarVisible ) {\n\t\t\t\tthis.hide();\n\t\t\t} else if ( isFocused ) {\n\t\t\t\tthis.show();\n\t\t\t}\n\t\t} );\n\n\t\t// Hide the toolbar when the selection is changed by a direct change or has changed to collapsed.\n\t\tthis.listenTo<DocumentSelectionChangeRangeEvent>( selection, 'change:range', ( evt, data ) => {\n\t\t\tif ( data.directChange || selection.isCollapsed ) {\n\t\t\t\tthis.hide();\n\t\t\t}\n\n\t\t\t// Fire internal `_selectionChangeDebounced` event to use it for showing\n\t\t\t// the toolbar after the selection stops changing.\n\t\t\tthis._fireSelectionChangeDebounced();\n\t\t} );\n\n\t\t// Show the toolbar when the selection stops changing.\n\t\tthis.listenTo( this, '_selectionChangeDebounced', () => {\n\t\t\tif ( this.editor.editing.view.document.isFocused ) {\n\t\t\t\tthis.show();\n\t\t\t}\n\t\t} );\n\n\t\tif ( !this._balloonConfig.shouldNotGroupWhenFull ) {\n\t\t\tthis.listenTo<EditorReadyEvent>( editor, 'ready', () => {\n\t\t\t\tconst editableElement = editor.ui.view.editable.element!;\n\n\t\t\t\t// Set #toolbarView's max-width on the initialization and update it on the editable resize.\n\t\t\t\tthis._resizeObserver = new ResizeObserver( editableElement, entry => {\n\t\t\t\t\t// The max-width equals 90% of the editable's width for the best user experience.\n\t\t\t\t\t// The value keeps the balloon very close to the boundaries of the editable and limits the cases\n\t\t\t\t\t// when the balloon juts out from the editable element it belongs to.\n\t\t\t\t\tthis.toolbarView.maxWidth = toPx( entry.contentRect.width * .9 );\n\t\t\t\t} );\n\t\t\t} );\n\t\t}\n\n\t\t// Listen to the toolbar view and whenever it changes its geometry due to some items being\n\t\t// grouped or ungrouped, update the position of the balloon because a shorter/longer toolbar\n\t\t// means the balloon could be pointing at the wrong place. Once updated, the balloon will point\n\t\t// at the right selection in the content again.\n\t\t// https://github.com/ckeditor/ckeditor5/issues/6444\n\t\tthis.listenTo<ToolbarViewGroupedItemsUpdateEvent>( this.toolbarView, 'groupedItemsUpdate', () => {\n\t\t\tthis._updatePosition();\n\t\t} );\n\t}\n\n\t/**\n\t * Creates toolbar components based on given configuration.\n\t * This needs to be done when all plugins are ready.\n\t */\n\tpublic afterInit(): void {\n\t\tconst factory = this.editor.ui.componentFactory;\n\n\t\tthis.toolbarView.fillFromConfig( this._balloonConfig, factory );\n\t}\n\n\t/**\n\t * Creates the toolbar view instance.\n\t */\n\tprivate _createToolbarView() {\n\t\tconst t = this.editor.locale.t;\n\t\tconst shouldGroupWhenFull = !this._balloonConfig.shouldNotGroupWhenFull;\n\t\tconst toolbarView = new ToolbarView( this.editor.locale, {\n\t\t\tshouldGroupWhenFull,\n\t\t\tisFloating: true\n\t\t} );\n\n\t\ttoolbarView.ariaLabel = t( 'Editor contextual toolbar' );\n\t\ttoolbarView.render();\n\n\t\treturn toolbarView;\n\t}\n\n\t/**\n\t * Shows the toolbar and attaches it to the selection.\n\t *\n\t * Fires {@link #event:show} event which can be stopped to prevent the toolbar from showing up.\n\t *\n\t * @param showForCollapsedSelection When set `true`, the toolbar will show despite collapsed selection in the\n\t * editing view.\n\t */\n\tpublic show( showForCollapsedSelection: boolean = false ): void {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\t\tconst schema = editor.model.schema;\n\n\t\t// Do not add the toolbar to the balloon stack twice.\n\t\tif ( this._balloon.hasView( this.toolbarView ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Do not show the toolbar when the selection is collapsed.\n\t\tif ( selection.isCollapsed && !showForCollapsedSelection ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Do not show the toolbar when there is more than one range in the selection and they fully contain selectable elements.\n\t\t// See https://github.com/ckeditor/ckeditor5/issues/6443.\n\t\tif ( selectionContainsOnlyMultipleSelectables( selection, schema ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Don not show the toolbar when all components inside are disabled\n\t\t// see https://github.com/ckeditor/ckeditor5-ui/issues/269.\n\t\tif ( Array.from( this.toolbarView.items ).every( ( item: any ) => item.isEnabled !== undefined && !item.isEnabled ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Update the toolbar position when the editor ui should be refreshed.\n\t\tthis.listenTo<EditorUIUpdateEvent>( this.editor.ui, 'update', () => {\n\t\t\tthis._updatePosition();\n\t\t} );\n\n\t\t// Add the toolbar to the common editor contextual balloon.\n\t\tthis._balloon.add( {\n\t\t\tview: this.toolbarView,\n\t\t\tposition: this._getBalloonPositionData(),\n\t\t\tballoonClassName: 'ck-toolbar-container'\n\t\t} );\n\t}\n\n\t/**\n\t * Hides the toolbar.\n\t */\n\tpublic hide(): void {\n\t\tif ( this._balloon.hasView( this.toolbarView ) ) {\n\t\t\tthis.stopListening( this.editor.ui, 'update' );\n\t\t\tthis._balloon.remove( this.toolbarView );\n\t\t}\n\t}\n\n\t/**\n\t * Returns positioning options for the {@link #_balloon}. They control the way balloon is attached\n\t * to the selection.\n\t */\n\tprivate _getBalloonPositionData() {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\t\tconst viewSelection = viewDocument.selection;\n\n\t\t// Get direction of the selection.\n\t\tconst isBackward = viewDocument.selection.isBackward;\n\n\t\treturn {\n\t\t\t// Because the target for BalloonPanelView is a Rect (not DOMRange), it's geometry will stay fixed\n\t\t\t// as the window scrolls. To let the BalloonPanelView follow such Rect, is must be continuously\n\t\t\t// computed and hence, the target is defined as a function instead of a static value.\n\t\t\t// https://github.com/ckeditor/ckeditor5-ui/issues/195\n\t\t\ttarget: () => {\n\t\t\t\tconst range = isBackward ? viewSelection.getFirstRange() : viewSelection.getLastRange();\n\t\t\t\tconst rangeRects = Rect.getDomRangeRects( view.domConverter.viewRangeToDom( range! ) );\n\n\t\t\t\t// Select the proper range rect depending on the direction of the selection.\n\t\t\t\tif ( isBackward ) {\n\t\t\t\t\treturn rangeRects[ 0 ];\n\t\t\t\t} else {\n\t\t\t\t\t// Ditch the zero-width \"orphan\" rect in the next line for the forward selection if there's\n\t\t\t\t\t// another one preceding it. It is not rendered as a selection by the web browser anyway.\n\t\t\t\t\t// https://github.com/ckeditor/ckeditor5-ui/issues/308\n\t\t\t\t\tif ( rangeRects.length > 1 && rangeRects[ rangeRects.length - 1 ].width === 0 ) {\n\t\t\t\t\t\trangeRects.pop();\n\t\t\t\t\t}\n\n\t\t\t\t\treturn rangeRects[ rangeRects.length - 1 ];\n\t\t\t\t}\n\t\t\t},\n\t\t\tpositions: this._getBalloonPositions( isBackward )\n\t\t};\n\t}\n\n\t/**\n\t * Updates the position of the {@link #_balloon} to make up for changes:\n\t *\n\t * * in the geometry of the selection it is attached to (e.g. the selection moved in the viewport or expanded or shrunk),\n\t * * or the geometry of the balloon toolbar itself (e.g. the toolbar has grouped or ungrouped some items and it is shorter or longer).\n\t */\n\tprivate _updatePosition() {\n\t\tthis._balloon.updatePosition( this._getBalloonPositionData() );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.stopListening();\n\t\tthis._fireSelectionChangeDebounced.cancel();\n\t\tthis.toolbarView.destroy();\n\t\tthis.focusTracker.destroy();\n\n\t\tif ( this._resizeObserver ) {\n\t\t\tthis._resizeObserver.destroy();\n\t\t}\n\t}\n\n\t/**\n\t * Returns toolbar positions for the given direction of the selection.\n\t */\n\tprivate _getBalloonPositions( isBackward: boolean ) {\n\t\tconst isSafariIniOS = env.isSafari && env.isiOS;\n\n\t\t// https://github.com/ckeditor/ckeditor5/issues/7707\n\t\tconst positions = isSafariIniOS ? generatePositions( {\n\t\t\t// 20px when zoomed out. Less then 20px when zoomed in; the \"radius\" of the native selection handle gets\n\t\t\t// smaller as the user zooms in. No less than the default v-offset, though.\n\t\t\theightOffset: Math.max(\n\t\t\t\tBalloonPanelView.arrowHeightOffset,\n\t\t\t\tMath.round( 20 / global.window.visualViewport!.scale )\n\t\t\t)\n\t\t} ) : BalloonPanelView.defaultPositions;\n\n\t\treturn isBackward ? [\n\t\t\tpositions.northWestArrowSouth,\n\t\t\tpositions.northWestArrowSouthWest,\n\t\t\tpositions.northWestArrowSouthEast,\n\t\t\tpositions.northWestArrowSouthMiddleEast,\n\t\t\tpositions.northWestArrowSouthMiddleWest,\n\t\t\tpositions.southWestArrowNorth,\n\t\t\tpositions.southWestArrowNorthWest,\n\t\t\tpositions.southWestArrowNorthEast,\n\t\t\tpositions.southWestArrowNorthMiddleWest,\n\t\t\tpositions.southWestArrowNorthMiddleEast\n\t\t] : [\n\t\t\tpositions.southEastArrowNorth,\n\t\t\tpositions.southEastArrowNorthEast,\n\t\t\tpositions.southEastArrowNorthWest,\n\t\t\tpositions.southEastArrowNorthMiddleEast,\n\t\t\tpositions.southEastArrowNorthMiddleWest,\n\t\t\tpositions.northEastArrowSouth,\n\t\t\tpositions.northEastArrowSouthEast,\n\t\t\tpositions.northEastArrowSouthWest,\n\t\t\tpositions.northEastArrowSouthMiddleEast,\n\t\t\tpositions.northEastArrowSouthMiddleWest\n\t\t];\n\t}\n}\n\n/**\n * Returns \"true\" when the selection has multiple ranges and each range contains a selectable element\n * and nothing else.\n */\nfunction selectionContainsOnlyMultipleSelectables( selection: DocumentSelection, schema: Schema ) {\n\t// It doesn't contain multiple objects if there is only one range.\n\tif ( selection.rangeCount === 1 ) {\n\t\treturn false;\n\t}\n\n\treturn [ ...selection.getRanges() ].every( range => {\n\t\tconst element = range.getContainedElement();\n\n\t\treturn element && schema.isSelectable( element );\n\t} );\n}\n\n/**\n * This event is fired just before the toolbar shows up. Stopping this event will prevent this.\n *\n * @eventName ~BalloonToolbar#show\n */\nexport type BalloonToolbarShowEvent = {\n\tname: 'show';\n\targs: [];\n};\n","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./blocktoolbar.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/toolbar/block/blockbuttonview\n */\n\nimport ButtonView from '../../button/buttonview';\n\nimport { toUnit, type Locale } from '@ckeditor/ckeditor5-utils';\n\nimport '../../../theme/components/toolbar/blocktoolbar.css';\n\nconst toPx = toUnit( 'px' );\n\n/**\n * The block button view class.\n *\n * This view represents a button attached next to block element where the selection is anchored.\n *\n * See {@link module:ui/toolbar/block/blocktoolbar~BlockToolbar}.\n */\nexport default class BlockButtonView extends ButtonView {\n\t/**\n\t * Top offset.\n\t *\n\t * @observable\n\t */\n\tdeclare public top: number;\n\n\t/**\n\t * Left offset.\n\t *\n\t * @observable\n\t */\n\tdeclare public left: number;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\t// Hide button on init.\n\t\tthis.isVisible = false;\n\n\t\tthis.isToggleable = true;\n\n\t\tthis.set( 'top', 0 );\n\t\tthis.set( 'left', 0 );\n\n\t\tthis.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: 'ck-block-toolbar-button',\n\t\t\t\tstyle: {\n\t\t\t\t\ttop: bind.to( 'top', val => toPx( val ) ),\n\t\t\t\t\tleft: bind.to( 'left', val => toPx( val ) )\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/toolbar/block/blocktoolbar\n */\n\n/* global window */\n\nimport {\n\tPlugin,\n\ticons,\n\ttype Editor\n} from '@ckeditor/ckeditor5-core';\n\nimport {\n\tRect,\n\tResizeObserver,\n\tgetOptimalPosition,\n\ttoUnit,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport type { DocumentSelectionChangeRangeEvent } from '@ckeditor/ckeditor5-engine';\n\nimport BlockButtonView from './blockbuttonview';\nimport BalloonPanelView from '../../panel/balloon/balloonpanelview';\nimport ToolbarView from '../toolbarview';\nimport clickOutsideHandler from '../../bindings/clickoutsidehandler';\nimport normalizeToolbarConfig from '../normalizetoolbarconfig';\n\nimport type { ButtonExecuteEvent } from '../../button/button';\nimport type { EditorUIUpdateEvent } from '../../editorui/editorui';\n\nconst toPx = toUnit( 'px' );\nconst { pilcrow } = icons;\n\n/**\n * The block toolbar plugin.\n *\n * This plugin provides a button positioned next to the block of content where the selection is anchored.\n * Upon clicking the button, a dropdown providing access to editor features shows up, as configured in\n * {@link module:core/editor/editorconfig~EditorConfig#blockToolbar}.\n *\n * By default, the button is displayed next to all elements marked in {@link module:engine/model/schema~Schema}\n * as `$block` for which the toolbar provides at least one option.\n *\n * By default, the button is attached so its right boundary is touching the\n * {@link module:engine/view/editableelement~EditableElement}:\n *\n * ```\n * __ |\n * | || This is a block of content that the\n * ¯¯ | button is attached to. This is a\n * | block of content that the button is\n * | attached to.\n * ```\n *\n * The position of the button can be adjusted using the CSS `transform` property:\n *\n * ```css\n * .ck-block-toolbar-button {\n * \ttransform: translateX( -10px );\n * }\n * ```\n *\n * ```\n * __ |\n * | | | This is a block of content that the\n * ¯¯ | button is attached to. This is a\n * | block of content that the button is\n * | attached to.\n * ```\n *\n * **Note**: If you plan to run the editor in a righttoleft (RTL) language, keep in mind the button\n * will be attached to the **right** boundary of the editable area. In that case, make sure the\n * CSS position adjustment works properly by adding the following styles:\n *\n * ```css\n * .ck[dir=\"rtl\"] .ck-block-toolbar-button {\n * \ttransform: translateX( 10px );\n * }\n * ```\n */\nexport default class BlockToolbar extends Plugin {\n\t/**\n\t * The toolbar view.\n\t */\n\tpublic readonly toolbarView: ToolbarView;\n\n\t/**\n\t * The balloon panel view, containing the {@link #toolbarView}.\n\t */\n\tpublic readonly panelView: BalloonPanelView;\n\n\t/**\n\t * The button view that opens the {@link #toolbarView}.\n\t */\n\tpublic readonly buttonView: BlockButtonView;\n\n\t/**\n\t * An instance of the resize observer that allows to respond to changes in editable's geometry\n\t * so the toolbar can stay within its boundaries (and group toolbar items that do not fit).\n\t *\n\t * **Note**: Used only when `shouldNotGroupWhenFull` was **not** set in the\n\t * {@link module:core/editor/editorconfig~EditorConfig#blockToolbar configuration}.\n\t *\n\t * **Note:** Created in {@link #afterInit}.\n\t */\n\tprivate _resizeObserver: ResizeObserver | null = null;\n\n\t/**\n\t * A cached and normalized `config.blockToolbar` object.\n\t */\n\tprivate _blockToolbarConfig: ReturnType<typeof normalizeToolbarConfig>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'BlockToolbar' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis._blockToolbarConfig = normalizeToolbarConfig( this.editor.config.get( 'blockToolbar' ) );\n\t\tthis.toolbarView = this._createToolbarView();\n\t\tthis.panelView = this._createPanelView();\n\t\tthis.buttonView = this._createButtonView();\n\n\t\t// Close the #panelView upon clicking outside of the plugin UI.\n\t\tclickOutsideHandler( {\n\t\t\temitter: this.panelView,\n\t\t\tcontextElements: [ this.panelView.element!, this.buttonView.element! ],\n\t\t\tactivator: () => this.panelView.isVisible,\n\t\t\tcallback: () => this._hidePanel()\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Hides panel on a direct selection change.\n\t\tthis.listenTo<DocumentSelectionChangeRangeEvent>( editor.model.document.selection, 'change:range', ( evt, data ) => {\n\t\t\tif ( data.directChange ) {\n\t\t\t\tthis._hidePanel();\n\t\t\t}\n\t\t} );\n\n\t\tthis.listenTo<EditorUIUpdateEvent>( editor.ui, 'update', () => this._updateButton() );\n\t\t// `low` priority is used because of https://github.com/ckeditor/ckeditor5-core/issues/133.\n\t\tthis.listenTo<ObservableChangeEvent>( editor, 'change:isReadOnly', () => this._updateButton(), { priority: 'low' } );\n\t\tthis.listenTo<ObservableChangeEvent>( editor.ui.focusTracker, 'change:isFocused', () => this._updateButton() );\n\n\t\t// Reposition button on resize.\n\t\tthis.listenTo<ObservableChangeEvent<boolean>>( this.buttonView, 'change:isVisible', ( evt, name, isVisible ) => {\n\t\t\tif ( isVisible ) {\n\t\t\t\t// Keep correct position of button and panel on window#resize.\n\t\t\t\tthis.buttonView.listenTo( window, 'resize', () => this._updateButton() );\n\t\t\t} else {\n\t\t\t\t// Stop repositioning button when is hidden.\n\t\t\t\tthis.buttonView.stopListening( window, 'resize' );\n\n\t\t\t\t// Hide the panel when the button disappears.\n\t\t\t\tthis._hidePanel();\n\t\t\t}\n\t\t} );\n\n\t\t// Register the toolbar so it becomes available for Alt+F10 and Esc navigation.\n\t\teditor.ui.addToolbar( this.toolbarView, {\n\t\t\tbeforeFocus: () => this._showPanel(),\n\t\t\tafterBlur: () => this._hidePanel()\n\t\t} );\n\t}\n\n\t/**\n\t * Fills the toolbar with its items based on the configuration.\n\t *\n\t * **Note:** This needs to be done after all plugins are ready.\n\t */\n\tpublic afterInit(): void {\n\t\tthis.toolbarView.fillFromConfig( this._blockToolbarConfig, this.editor.ui.componentFactory );\n\n\t\t// Hide panel before executing each button in the panel.\n\t\tfor ( const item of this.toolbarView.items ) {\n\t\t\titem.on<ButtonExecuteEvent>( 'execute', () => this._hidePanel( true ), { priority: 'high' } );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\t// Destroy created UI components as they are not automatically destroyed (see ckeditor5#1341).\n\t\tthis.panelView.destroy();\n\t\tthis.buttonView.destroy();\n\t\tthis.toolbarView.destroy();\n\n\t\tif ( this._resizeObserver ) {\n\t\t\tthis._resizeObserver.destroy();\n\t\t}\n\t}\n\n\t/**\n\t * Creates the {@link #toolbarView}.\n\t */\n\tprivate _createToolbarView(): ToolbarView {\n\t\tconst t = this.editor.locale.t;\n\t\tconst shouldGroupWhenFull = !this._blockToolbarConfig.shouldNotGroupWhenFull;\n\t\tconst toolbarView = new ToolbarView( this.editor.locale, {\n\t\t\tshouldGroupWhenFull,\n\t\t\tisFloating: true\n\t\t} );\n\n\t\ttoolbarView.ariaLabel = t( 'Editor block content toolbar' );\n\n\t\treturn toolbarView;\n\t}\n\n\t/**\n\t * Creates the {@link #panelView}.\n\t */\n\tprivate _createPanelView(): BalloonPanelView {\n\t\tconst editor = this.editor;\n\t\tconst panelView = new BalloonPanelView( editor.locale );\n\n\t\tpanelView.content.add( this.toolbarView );\n\t\tpanelView.class = 'ck-toolbar-container';\n\t\teditor.ui.view.body.add( panelView );\n\t\teditor.ui.focusTracker.add( panelView.element! );\n\n\t\t// Close #panelView on `Esc` press.\n\t\tthis.toolbarView.keystrokes.set( 'Esc', ( evt, cancel ) => {\n\t\t\tthis._hidePanel( true );\n\t\t\tcancel();\n\t\t} );\n\n\t\treturn panelView;\n\t}\n\n\t/**\n\t * Creates the {@link #buttonView}.\n\t */\n\tprivate _createButtonView(): BlockButtonView {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst buttonView = new BlockButtonView( editor.locale );\n\n\t\tbuttonView.set( {\n\t\t\tlabel: t( 'Edit block' ),\n\t\t\ticon: pilcrow,\n\t\t\twithText: false\n\t\t} );\n\n\t\t// Bind the panelView observable properties to the buttonView.\n\t\tbuttonView.bind( 'isOn' ).to( this.panelView, 'isVisible' );\n\t\tbuttonView.bind( 'tooltip' ).to( this.panelView, 'isVisible', isVisible => !isVisible );\n\n\t\t// Toggle the panelView upon buttonView#execute.\n\t\tthis.listenTo<ButtonExecuteEvent>( buttonView, 'execute', () => {\n\t\t\tif ( !this.panelView.isVisible ) {\n\t\t\t\tthis._showPanel();\n\t\t\t} else {\n\t\t\t\tthis._hidePanel( true );\n\t\t\t}\n\t\t} );\n\n\t\teditor.ui.view.body.add( buttonView );\n\t\teditor.ui.focusTracker.add( buttonView.element! );\n\n\t\treturn buttonView;\n\t}\n\n\t/**\n\t * Shows or hides the button.\n\t * When all the conditions for displaying the button are matched, it shows the button. Hides otherwise.\n\t */\n\tprivate _updateButton() {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst view = editor.editing.view;\n\n\t\t// Hides the button when the editor is not focused.\n\t\tif ( !editor.ui.focusTracker.isFocused ) {\n\t\t\tthis._hideButton();\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Hides the button when the selection is in non-editable place.\n\t\tif ( !editor.model.canEditAt( editor.model.document.selection ) ) {\n\t\t\tthis._hideButton();\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Get the first selected block, button will be attached to this element.\n\t\tconst modelTarget = Array.from( model.document.selection.getSelectedBlocks() )[ 0 ];\n\n\t\t// Hides the button when there is no enabled item in toolbar for the current block element.\n\t\tif ( !modelTarget || Array.from( this.toolbarView.items ).every( ( item: any ) => !item.isEnabled ) ) {\n\t\t\tthis._hideButton();\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Get DOM target element.\n\t\tconst domTarget = view.domConverter.mapViewToDom( editor.editing.mapper.toViewElement( modelTarget )! );\n\n\t\t// Show block button.\n\t\tthis.buttonView.isVisible = true;\n\n\t\t// Make sure that the block toolbar panel is resized properly.\n\t\tthis._setupToolbarResize();\n\n\t\t// Attach block button to target DOM element.\n\t\tthis._attachButtonToElement( domTarget as any );\n\n\t\t// When panel is opened then refresh it position to be properly aligned with block button.\n\t\tif ( this.panelView.isVisible ) {\n\t\t\tthis._showPanel();\n\t\t}\n\t}\n\n\t/**\n\t * Hides the button.\n\t */\n\tprivate _hideButton() {\n\t\tthis.buttonView.isVisible = false;\n\t}\n\n\t/**\n\t * Shows the {@link #toolbarView} attached to the {@link #buttonView}.\n\t * If the toolbar is already visible, then it simply repositions it.\n\t */\n\tprivate _showPanel() {\n\t\t// Usually, the only way to show the toolbar is by pressing the block button. It makes it impossible for\n\t\t// the toolbar to show up when the button is invisible (feature does not make sense for the selection then).\n\t\t// The toolbar navigation using Alt+F10 does not access the button but shows the panel directly using this method.\n\t\t// So we need to check whether this is possible first.\n\t\tif ( !this.buttonView.isVisible ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst wasVisible = this.panelView.isVisible;\n\n\t\t// So here's the thing: If there was no initial panelView#show() or these two were in different order, the toolbar\n\t\t// positioning will break in RTL editors. Weird, right? What you show know is that the toolbar\n\t\t// grouping works thanks to:\n\t\t//\n\t\t// * the ResizeObserver, which kicks in as soon as the toolbar shows up in DOM (becomes visible again).\n\t\t// * the observable ToolbarView#maxWidth, which triggers re-grouping when changed.\n\t\t//\n\t\t// Here are the possible scenarios:\n\t\t//\n\t\t// 1. (WRONG ❌) If the #maxWidth is set when the toolbar is invisible, it won't affect item grouping (no DOMRects, no grouping).\n\t\t// Then, when panelView.pin() is called, the position of the toolbar will be calculated for the old\n\t\t// items grouping state, and when finally ResizeObserver kicks in (hey, the toolbar is visible now, right?)\n\t\t// it will group/ungroup some items and the length of the toolbar will change. But since in RTL the toolbar\n\t\t// is attached on the right side and the positioning uses CSS \"left\", it will result in the toolbar shifting\n\t\t// to the left and being displayed in the wrong place.\n\t\t// 2. (WRONG ❌) If the panelView.pin() is called first and #maxWidth set next, then basically the story repeats. The balloon\n\t\t// calculates the position for the old toolbar grouping state, then the toolbar re-groups items and because\n\t\t// it is positioned using CSS \"left\" it will move.\n\t\t// 3. (RIGHT ✅) We show the panel first (the toolbar does re-grouping but it does not matter), then the #maxWidth\n\t\t// is set allowing the toolbar to re-group again and finally panelView.pin() does the positioning when the\n\t\t// items grouping state is stable and final.\n\t\t//\n\t\t// https://github.com/ckeditor/ckeditor5/issues/6449, https://github.com/ckeditor/ckeditor5/issues/6575\n\t\tthis.panelView.show();\n\n\t\tconst editableElement = this._getSelectedEditableElement();\n\n\t\tthis.toolbarView.maxWidth = this._getToolbarMaxWidth( editableElement );\n\n\t\tthis.panelView.pin( {\n\t\t\ttarget: this.buttonView.element!,\n\t\t\tlimiter: editableElement\n\t\t} );\n\n\t\tif ( !wasVisible ) {\n\t\t\t( this.toolbarView.items.get( 0 ) as any ).focus();\n\t\t}\n\t}\n\n\t/**\n\t * Returns currently selected editable, based on the model selection.\n\t */\n\tprivate _getSelectedEditableElement(): HTMLElement {\n\t\tconst selectedModelRootName = this.editor.model.document.selection.getFirstRange()!.root.rootName!;\n\n\t\treturn this.editor.ui.getEditableElement( selectedModelRootName )!;\n\t}\n\n\t/**\n\t * Hides the {@link #toolbarView}.\n\t *\n\t * @param focusEditable When `true`, the editable will be focused after hiding the panel.\n\t */\n\tprivate _hidePanel( focusEditable?: boolean ) {\n\t\tthis.panelView.isVisible = false;\n\n\t\tif ( focusEditable ) {\n\t\t\tthis.editor.editing.view.focus();\n\t\t}\n\t}\n\n\t/**\n\t * Attaches the {@link #buttonView} to the target block of content.\n\t *\n\t * @param targetElement Target element.\n\t */\n\tprivate _attachButtonToElement( targetElement: HTMLElement ) {\n\t\tconst contentStyles = window.getComputedStyle( targetElement );\n\n\t\tconst editableRect = new Rect( this._getSelectedEditableElement() );\n\t\tconst contentPaddingTop = parseInt( contentStyles.paddingTop, 10 );\n\t\t// When line height is not an integer then treat it as \"normal\".\n\t\t// MDN says that 'normal' == ~1.2 on desktop browsers.\n\t\tconst contentLineHeight = parseInt( contentStyles.lineHeight, 10 ) || parseInt( contentStyles.fontSize, 10 ) * 1.2;\n\n\t\tconst position = getOptimalPosition( {\n\t\t\telement: this.buttonView.element!,\n\t\t\ttarget: targetElement,\n\t\t\tpositions: [\n\t\t\t\t( contentRect, buttonRect ) => {\n\t\t\t\t\tlet left;\n\n\t\t\t\t\tif ( this.editor.locale.uiLanguageDirection === 'ltr' ) {\n\t\t\t\t\t\tleft = editableRect.left - buttonRect.width;\n\t\t\t\t\t} else {\n\t\t\t\t\t\tleft = editableRect.right;\n\t\t\t\t\t}\n\n\t\t\t\t\treturn {\n\t\t\t\t\t\ttop: contentRect.top + contentPaddingTop + ( contentLineHeight - buttonRect.height ) / 2,\n\t\t\t\t\t\tleft\n\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\n\t\tthis.buttonView.top = position.top;\n\t\tthis.buttonView.left = position.left;\n\t}\n\n\t/**\n\t * Creates a resize observer that observes selected editable and resizes the toolbar panel accordingly.\n\t */\n\tprivate _setupToolbarResize() {\n\t\tconst editableElement = this._getSelectedEditableElement();\n\n\t\t// Do this only if the automatic grouping is turned on.\n\t\tif ( !this._blockToolbarConfig.shouldNotGroupWhenFull ) {\n\t\t\t// If resize observer is attached to a different editable than currently selected editable, re-attach it.\n\t\t\tif ( this._resizeObserver && this._resizeObserver.element !== editableElement ) {\n\t\t\t\tthis._resizeObserver.destroy();\n\t\t\t\tthis._resizeObserver = null;\n\t\t\t}\n\n\t\t\tif ( !this._resizeObserver ) {\n\t\t\t\tthis._resizeObserver = new ResizeObserver( editableElement, () => {\n\t\t\t\t\tthis.toolbarView.maxWidth = this._getToolbarMaxWidth( editableElement );\n\t\t\t\t} );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Gets the {@link #toolbarView} max-width, based on given `editableElement` width plus the distance between the farthest\n\t * edge of the {@link #buttonView} and the editable.\n\t *\n\t * @returns A maximum width that toolbar can have, in pixels.\n\t */\n\tprivate _getToolbarMaxWidth( editableElement: HTMLElement ) {\n\t\tconst editableRect = new Rect( editableElement );\n\t\tconst buttonRect = new Rect( this.buttonView.element! );\n\t\tconst isRTL = this.editor.locale.uiLanguageDirection === 'rtl';\n\t\tconst offset = isRTL ? ( buttonRect.left - editableRect.right ) + buttonRect.width : editableRect.left - buttonRect.left;\n\n\t\treturn toPx( editableRect.width + offset );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module watchdog/watchdog\n */\n\n/* globals window */\n\n// eslint-disable-next-line ckeditor5-rules/no-cross-package-imports\nimport type { CKEditorError } from 'ckeditor5/src/utils';\nimport type { EditorWatchdogRestartEvent } from './editorwatchdog';\nimport type { ContextWatchdogItemErrorEvent, ContextWatchdogItemRestartEvent } from './contextwatchdog';\n\n/**\n * An abstract watchdog class that handles most of the error handling process and the state of the underlying component.\n *\n * See the {@glink features/watchdog Watchdog feature guide} to learn the rationale behind it and how to use it.\n *\n * @internal\n */\nexport default abstract class Watchdog {\n\t/**\n\t * An array of crashes saved as an object with the following properties:\n\t *\n\t * * `message`: `String`,\n\t * * `stack`: `String`,\n\t * * `date`: `Number`,\n\t * * `filename`: `String | undefined`,\n\t * * `lineno`: `Number | undefined`,\n\t * * `colno`: `Number | undefined`,\n\t */\n\tpublic readonly crashes: Array<{\n\t\tmessage: string;\n\t\tstack?: string;\n\t\tdate: number;\n\n\t\t// `evt.filename`, `evt.lineno` and `evt.colno` are available only in ErrorEvent events\n\t\tfilename?: string;\n\t\tlineno?: number;\n\t\tcolno?: number;\n\t}> = [];\n\n\t/**\n\t * Specifies the state of the item watched by the watchdog. The state can be one of the following values:\n\t *\n\t * * `initializing` &ndash; Before the first initialization, and after crashes, before the item is ready.\n\t * * `ready` &ndash; A state when the user can interact with the item.\n\t * * `crashed` &ndash; A state when an error occurs. It quickly changes to `initializing` or `crashedPermanently`\n\t * depending on how many and how frequent errors have been caught recently.\n\t * * `crashedPermanently` &ndash; A state when the watchdog stops reacting to errors and keeps the item it is watching crashed,\n\t * * `destroyed` &ndash; A state when the item is manually destroyed by the user after calling `watchdog.destroy()`.\n\t */\n\tpublic state: WatchdogState = 'initializing';\n\n\t/**\n\t * @see module:watchdog/watchdog~WatchdogConfig\n\t */\n\tprivate _crashNumberLimit: number;\n\n\t/**\n\t * Returns the result of the `Date.now()` call. It can be overridden in tests to mock time as some popular\n\t * approaches like `sinon.useFakeTimers()` do not work well with error handling.\n\t */\n\tprivate _now = Date.now;\n\n\t/**\n\t * @see module:watchdog/watchdog~WatchdogConfig\n\t */\n\tprivate _minimumNonErrorTimePeriod: number;\n\n\t/**\n\t * Checks if the event error comes from the underlying item and restarts the item.\n\t */\n\tprivate _boundErrorHandler: ( evt: ErrorEvent | PromiseRejectionEvent ) => void;\n\n\t/**\n\t * The method responsible for restarting the watched item.\n\t */\n\tprotected abstract _restart(): Promise<unknown>;\n\n\t/**\n\t * Traverses the error context and the watched item to find out whether the error should\n\t * be handled by the given item.\n\t *\n\t * @internal\n\t */\n\tpublic abstract _isErrorComingFromThisItem( error: CKEditorError ): boolean;\n\n\t/**\n\t * The watched item.\n\t *\n\t * @internal\n\t */\n\tpublic abstract get _item(): unknown;\n\n\t/**\n\t * A dictionary of event emitter listeners.\n\t */\n\tprivate _listeners: Record<string, Array<( ...args: any ) => void>>;\n\n\t/**\n\t * @param {module:watchdog/watchdog~WatchdogConfig} config The watchdog plugin configuration.\n\t */\n\tconstructor( config: WatchdogConfig ) {\n\t\tthis.crashes = [];\n\t\tthis._crashNumberLimit = typeof config.crashNumberLimit === 'number' ? config.crashNumberLimit : 3;\n\t\tthis._minimumNonErrorTimePeriod = typeof config.minimumNonErrorTimePeriod === 'number' ? config.minimumNonErrorTimePeriod : 5000;\n\n\t\tthis._boundErrorHandler = evt => {\n\t\t\t// `evt.error` is exposed by EventError while `evt.reason` is available in PromiseRejectionEvent.\n\t\t\tconst error = 'error' in evt ? evt.error : evt.reason;\n\n\t\t\t// Note that `evt.reason` might be everything that is in the promise rejection.\n\t\t\t// Similarly everything that is thrown lands in `evt.error`.\n\t\t\tif ( error instanceof Error ) {\n\t\t\t\tthis._handleError( error, evt );\n\t\t\t}\n\t\t};\n\n\t\tthis._listeners = {};\n\n\t\tif ( !( this as any )._restart ) {\n\t\t\tthrow new Error(\n\t\t\t\t'The Watchdog class was split into the abstract `Watchdog` class and the `EditorWatchdog` class. ' +\n\t\t\t\t'Please, use `EditorWatchdog` if you have used the `Watchdog` class previously.'\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * Destroys the watchdog and releases the resources.\n\t */\n\tpublic destroy(): void {\n\t\tthis._stopErrorHandling();\n\n\t\tthis._listeners = {};\n\t}\n\n\t/**\n\t * Starts listening to a specific event name by registering a callback that will be executed\n\t * whenever an event with a given name fires.\n\t *\n\t * Note that this method differs from the CKEditor 5's default `EventEmitterMixin` implementation.\n\t *\n\t * @param eventName The event name.\n\t * @param callback A callback which will be added to event listeners.\n\t */\n\tpublic on<K extends keyof EventMap>( eventName: K, callback: EventCallback<K> ): void {\n\t\tif ( !this._listeners[ eventName ] ) {\n\t\t\tthis._listeners[ eventName ] = [];\n\t\t}\n\n\t\tthis._listeners[ eventName ].push( callback );\n\t}\n\n\t/**\n\t * Stops listening to the specified event name by removing the callback from event listeners.\n\t *\n\t * Note that this method differs from the CKEditor 5's default `EventEmitterMixin` implementation.\n\t *\n\t * @param eventName The event name.\n\t * @param callback A callback which will be removed from event listeners.\n\t */\n\tpublic off( eventName: keyof EventMap, callback: unknown ): void {\n\t\tthis._listeners[ eventName ] = this._listeners[ eventName ]\n\t\t\t.filter( cb => cb !== callback );\n\t}\n\n\t/**\n\t * Fires an event with a given event name and arguments.\n\t *\n\t * Note that this method differs from the CKEditor 5's default `EventEmitterMixin` implementation.\n\t */\n\tprotected _fire<K extends keyof EventMap>( eventName: K, ...args: EventArgs<K> ): void {\n\t\tconst callbacks = this._listeners[ eventName ] || [];\n\n\t\tfor ( const callback of callbacks ) {\n\t\t\tcallback.apply( this, [ null, ...args ] );\n\t\t}\n\t}\n\n\t/**\n\t * Starts error handling by attaching global error handlers.\n\t */\n\tprotected _startErrorHandling(): void {\n\t\twindow.addEventListener( 'error', this._boundErrorHandler );\n\t\twindow.addEventListener( 'unhandledrejection', this._boundErrorHandler );\n\t}\n\n\t/**\n\t * Stops error handling by detaching global error handlers.\n\t */\n\tprotected _stopErrorHandling(): void {\n\t\twindow.removeEventListener( 'error', this._boundErrorHandler );\n\t\twindow.removeEventListener( 'unhandledrejection', this._boundErrorHandler );\n\t}\n\n\t/**\n\t * Checks if an error comes from the watched item and restarts it.\n\t * It reacts to {@link module:utils/ckeditorerror~CKEditorError `CKEditorError` errors} only.\n\t *\n\t * @fires error\n\t * @param error Error.\n\t * @param evt An error event.\n\t */\n\tprivate _handleError( error: Error, evt: ErrorEvent | PromiseRejectionEvent ): void {\n\t\t// @if CK_DEBUG // const err = error as CKEditorError;\n\t\t// @if CK_DEBUG // if ( err.is && err.is( 'CKEditorError' ) && err.context === undefined ) {\n\t\t// @if CK_DEBUG // console.warn( 'The error is missing its context and Watchdog cannot restart the proper item.' );\n\t\t// @if CK_DEBUG // }\n\n\t\tif ( this._shouldReactToError( error ) ) {\n\t\t\tthis.crashes.push( {\n\t\t\t\tmessage: error.message,\n\t\t\t\tstack: error.stack,\n\n\t\t\t\t// `evt.filename`, `evt.lineno` and `evt.colno` are available only in ErrorEvent events\n\t\t\t\tfilename: evt instanceof ErrorEvent ? evt.filename : undefined,\n\t\t\t\tlineno: evt instanceof ErrorEvent ? evt.lineno : undefined,\n\t\t\t\tcolno: evt instanceof ErrorEvent ? evt.colno : undefined,\n\t\t\t\tdate: this._now()\n\t\t\t} );\n\n\t\t\tconst causesRestart = this._shouldRestart();\n\n\t\t\tthis.state = 'crashed';\n\t\t\tthis._fire( 'stateChange' );\n\t\t\tthis._fire( 'error', { error, causesRestart } );\n\n\t\t\tif ( causesRestart ) {\n\t\t\t\tthis._restart();\n\t\t\t} else {\n\t\t\t\tthis.state = 'crashedPermanently';\n\t\t\t\tthis._fire( 'stateChange' );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Checks whether an error should be handled by the watchdog.\n\t *\n\t * @param error An error that was caught by the error handling process.\n\t */\n\tprivate _shouldReactToError( error: Error ): boolean {\n\t\treturn (\n\t\t\t( error as any ).is &&\n\t\t\t( error as any ).is( 'CKEditorError' ) &&\n\t\t\t( error as CKEditorError ).context !== undefined &&\n\n\t\t\t// In some cases the watched item should not be restarted - e.g. during the item initialization.\n\t\t\t// That's why the `null` was introduced as a correct error context which does cause restarting.\n\t\t\t( error as CKEditorError ).context !== null &&\n\n\t\t\t// Do not react to errors if the watchdog is in states other than `ready`.\n\t\t\tthis.state === 'ready' &&\n\n\t\t\tthis._isErrorComingFromThisItem( error as CKEditorError )\n\t\t);\n\t}\n\n\t/**\n\t * Checks if the watchdog should restart the underlying item.\n\t */\n\tprivate _shouldRestart(): boolean {\n\t\tif ( this.crashes.length <= this._crashNumberLimit ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tconst lastErrorTime = this.crashes[ this.crashes.length - 1 ].date;\n\t\tconst firstMeaningfulErrorTime = this.crashes[ this.crashes.length - 1 - this._crashNumberLimit ].date;\n\n\t\tconst averageNonErrorTimePeriod = ( lastErrorTime - firstMeaningfulErrorTime ) / this._crashNumberLimit;\n\n\t\treturn averageNonErrorTimePeriod > this._minimumNonErrorTimePeriod;\n\t}\n}\n\n/**\n * Fired when a new {@link module:utils/ckeditorerror~CKEditorError `CKEditorError`} error connected to the watchdog instance occurs\n * and the watchdog will react to it.\n *\n * ```ts\n * watchdog.on( 'error', ( evt, { error, causesRestart } ) => {\n * \tconsole.log( 'An error occurred.' );\n * } );\n * ```\n *\n * @eventName ~Watchdog#error\n */\nexport type WatchdogErrorEvent = {\n\tname: 'error';\n\targs: [ WatchdogErrorEventData ];\n};\n\n/**\n * The `error` event data.\n */\nexport type WatchdogErrorEventData = {\n\terror: Error;\n\tcausesRestart: boolean;\n};\n\n/**\n * Fired when the watchdog state changed.\n *\n * @eventName ~Watchdog#stateChange\n */\nexport type WatchdogStateChangeEvent = {\n\tname: 'stateChange';\n\targs: [];\n};\n\n/**\n * The map of watchdog events.\n */\nexport interface EventMap {\n\tstateChange: WatchdogStateChangeEvent;\n\terror: WatchdogErrorEvent;\n\trestart: EditorWatchdogRestartEvent;\n\titemError: ContextWatchdogItemErrorEvent;\n\titemRestart: ContextWatchdogItemRestartEvent;\n}\n\n/**\n * Utility type that gets the arguments type for the given event.\n */\nexport type EventArgs<K extends keyof EventMap> = EventMap[ K ][ 'args' ];\n\n/**\n * Utility type that gets the callback type for the given event.\n */\nexport type EventCallback<K extends keyof EventMap> = ( evt: null, ...args: EventArgs<K> ) => void;\n\n/**\n * The watchdog plugin configuration.\n */\nexport interface WatchdogConfig {\n\n\t/**\n\t * A threshold specifying the number of watched item crashes\n\t * when the watchdog stops restarting the item in case of errors.\n\t * After this limit is reached and the time between the last errors is shorter than `minimumNonErrorTimePeriod`,\n\t * the watchdog changes its state to `crashedPermanently` and it stops restarting the item. This prevents an infinite restart loop.\n\t *\n\t * @default 3\n\t */\n\tcrashNumberLimit?: number;\n\n\t/**\n\t * An average number of milliseconds between the last watched item errors\n\t * (defaults to 5000). When the period of time between errors is lower than that and the `crashNumberLimit` is also reached,\n\t * the watchdog changes its state to `crashedPermanently` and it stops restarting the item. This prevents an infinite restart loop.\n\t *\n\t * @default 5000\n\t */\n\tminimumNonErrorTimePeriod?: number;\n\n\t/**\n\t * A minimum number of milliseconds between saving the editor data internally (defaults to 5000).\n\t * Note that for large documents this might impact the editor performance.\n\t *\n\t * @default 5000\n\t */\n\tsaveInterval?: number;\n}\n\n/**\n * Specifies the state of the item watched by the watchdog.\n */\nexport type WatchdogState = 'initializing' | 'ready' | 'crashed' | 'crashedPermanently' | 'destroyed';\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module watchdog/utils/getsubnodes\n */\n\n/* globals EventTarget, Event */\n\nexport default function getSubNodes( head: unknown, excludedProperties = new Set() ): Set<unknown> {\n\tconst nodes = [ head ];\n\n\t// @if CK_DEBUG_WATCHDOG // const prevNodeMap = new Map();\n\n\t// Nodes are stored to prevent infinite looping.\n\tconst subNodes = new Set();\n\tlet nodeIndex = 0;\n\n\twhile ( nodes.length > nodeIndex ) {\n\t\t// Incrementing the iterator is much faster than changing size of the array with Array.prototype.shift().\n\t\tconst node = nodes[ nodeIndex++ ];\n\n\t\tif ( subNodes.has( node ) || !shouldNodeBeIncluded( node ) || excludedProperties.has( node ) ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tsubNodes.add( node );\n\n\t\t// Handle arrays, maps, sets, custom collections that implements `[ Symbol.iterator ]()`, etc.\n\t\tif ( Symbol.iterator in node ) {\n\t\t\t// The custom editor iterators might cause some problems if the editor is crashed.\n\t\t\ttry {\n\t\t\t\tfor ( const n of node as Iterable<unknown> ) {\n\t\t\t\t\tnodes.push( n );\n\n\t\t\t\t\t// @if CK_DEBUG_WATCHDOG // if ( !prevNodeMap.has( n ) ) {\n\t\t\t\t\t// @if CK_DEBUG_WATCHDOG // \tprevNodeMap.set( n, node );\n\t\t\t\t\t// @if CK_DEBUG_WATCHDOG // }\n\t\t\t\t}\n\t\t\t} catch ( err ) {\n\t\t\t\t// Do not log errors for broken structures\n\t\t\t\t// since we are in the error handling process already.\n\t\t\t\t// eslint-disable-line no-empty\n\t\t\t}\n\t\t} else {\n\t\t\tfor ( const key in node ) {\n\t\t\t\t// We share a reference via the protobuf library within the editors,\n\t\t\t\t// hence the shared value should be skipped. Although, it's not a perfect\n\t\t\t\t// solution since new places like that might occur in the future.\n\t\t\t\tif ( key === 'defaultValue' ) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\tnodes.push( node[ key ] );\n\n\t\t\t\t// @if CK_DEBUG_WATCHDOG // if ( !prevNodeMap.has( node[ key ] ) ) {\n\t\t\t\t// @if CK_DEBUG_WATCHDOG // \tprevNodeMap.set( node[ key ], node );\n\t\t\t\t// @if CK_DEBUG_WATCHDOG // }\n\t\t\t}\n\t\t}\n\t}\n\n\t// @if CK_DEBUG_WATCHDOG // return { subNodes, prevNodeMap } as any;\n\n\treturn subNodes;\n}\n\nfunction shouldNodeBeIncluded( node: unknown ): node is Record<string, unknown> | Iterable<unknown> {\n\tconst type = Object.prototype.toString.call( node );\n\tconst typeOfNode = typeof node;\n\n\treturn !(\n\t\ttypeOfNode === 'number' ||\n\t\ttypeOfNode === 'boolean' ||\n\t\ttypeOfNode === 'string' ||\n\t\ttypeOfNode === 'symbol' ||\n\t\ttypeOfNode === 'function' ||\n\t\ttype === '[object Date]' ||\n\t\ttype === '[object RegExp]' ||\n\t\ttype === '[object Module]' ||\n\n\t\tnode === undefined ||\n\t\tnode === null ||\n\n\t\t// This flag is meant to exclude singletons shared across editor instances. So when an error is thrown in one editor,\n\t\t// the other editors connected through the reference to the same singleton are not restarted. This is a temporary workaround\n\t\t// until a better solution is found.\n\t\t// More in https://github.com/ckeditor/ckeditor5/issues/12292.\n\t\t( node as any )._watchdogExcluded ||\n\n\t\t// Skip native DOM objects, e.g. Window, nodes, events, etc.\n\t\tnode instanceof EventTarget ||\n\t\tnode instanceof Event\n\t);\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module watchdog/utils/areconnectedthroughproperties\n */\n\n/* globals console */\n\nimport getSubNodes from './getsubnodes';\n\n/**\n * Traverses both structures to find out whether there is a reference that is shared between both structures.\n */\nexport default function areConnectedThroughProperties( target1: unknown, target2: unknown, excludedNodes = new Set() ): boolean {\n\tif ( target1 === target2 && isObject( target1 ) ) {\n\t\treturn true;\n\t}\n\n\t// @if CK_DEBUG_WATCHDOG // return checkConnectionBetweenProps( target1, target2, excludedNodes );\n\n\tconst subNodes1 = getSubNodes( target1, excludedNodes );\n\tconst subNodes2 = getSubNodes( target2, excludedNodes );\n\n\tfor ( const node of subNodes1 ) {\n\t\tif ( subNodes2.has( node ) ) {\n\t\t\treturn true;\n\t\t}\n\t}\n\n\treturn false;\n}\n\n/* istanbul ignore next -- @preserve */\n// eslint-disable-next-line\nfunction checkConnectionBetweenProps( target1: unknown, target2: unknown, excludedNodes: any ) {\n\tconst { subNodes: subNodes1, prevNodeMap: prevNodeMap1 } = getSubNodes( target1, excludedNodes.subNodes ) as any;\n\tconst { subNodes: subNodes2, prevNodeMap: prevNodeMap2 } = getSubNodes( target2, excludedNodes.subNodes ) as any;\n\n\tfor ( const sharedNode of subNodes1 ) {\n\t\tif ( subNodes2.has( sharedNode ) ) {\n\t\t\tconst connection = [];\n\n\t\t\tconnection.push( sharedNode );\n\n\t\t\tlet node = prevNodeMap1.get( sharedNode );\n\n\t\t\twhile ( node && node !== target1 ) {\n\t\t\t\tconnection.push( node );\n\t\t\t\tnode = prevNodeMap1.get( node );\n\t\t\t}\n\n\t\t\tnode = prevNodeMap2.get( sharedNode );\n\n\t\t\twhile ( node && node !== target2 ) {\n\t\t\t\tconnection.unshift( node );\n\t\t\t\tnode = prevNodeMap2.get( node );\n\t\t\t}\n\n\t\t\tconsole.log( '--------' );\n\t\t\tconsole.log( { target1 } );\n\t\t\tconsole.log( { sharedNode } );\n\t\t\tconsole.log( { target2 } );\n\t\t\tconsole.log( { connection } );\n\n\t\t\treturn true;\n\t\t}\n\t}\n\n\treturn false;\n}\n\nfunction isObject( structure: unknown ): boolean {\n\treturn typeof structure === 'object' && structure !== null;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module watchdog/editorwatchdog\n */\n\n/* globals console */\n\n// eslint-disable-next-line ckeditor5-rules/no-cross-package-imports\nimport type { CKEditorError } from 'ckeditor5/src/utils';\n\n// eslint-disable-next-line ckeditor5-rules/no-cross-package-imports\nimport type { Editor, EditorConfig, Context, EditorReadyEvent } from 'ckeditor5/src/core';\n\n// eslint-disable-next-line ckeditor5-rules/no-cross-package-imports\nimport type { RootAttributes } from '@ckeditor/ckeditor5-editor-multi-root';\n\nimport areConnectedThroughProperties from './utils/areconnectedthroughproperties';\nimport Watchdog, { type WatchdogConfig } from './watchdog';\n\nimport { throttle, cloneDeepWith, isElement, type DebouncedFunc } from 'lodash-es';\n\n// eslint-disable-next-line ckeditor5-rules/no-cross-package-imports\nimport type { Node, Text, Element, Writer } from 'ckeditor5/src/engine';\n\n/**\n * A watchdog for CKEditor 5 editors.\n *\n * See the {@glink features/watchdog Watchdog feature guide} to learn the rationale behind it and\n * how to use it.\n */\nexport default class EditorWatchdog<TEditor extends Editor = Editor> extends Watchdog {\n\t/**\n\t * The current editor instance.\n\t */\n\tprivate _editor: TEditor | null = null;\n\n\t/**\n\t * Throttled save method. The `save()` method is called the specified `saveInterval` after `throttledSave()` is called,\n\t * unless a new action happens in the meantime.\n\t */\n\tprivate _throttledSave: DebouncedFunc<() => void>;\n\n\t/**\n\t * The latest saved editor data represented as a root name -> root data object.\n\t */\n\tprivate _data?: EditorData;\n\n\t/**\n\t * The last document version.\n\t */\n\tprivate _lastDocumentVersion?: number;\n\n\t/**\n\t * The editor source element or data.\n\t */\n\tprivate _elementOrData?: HTMLElement | string | Record<string, string> | Record<string, HTMLElement>;\n\n\t/**\n\t * Specifies whether the editor was initialized using document data (`true`) or HTML elements (`false`).\n\t */\n\tprivate _initUsingData = true;\n\n\t/**\n\t * The latest record of the editor editable elements. Used to restart the editor.\n\t */\n\tprivate _editables: Record<string, HTMLElement> = {};\n\n\t/**\n\t * The editor configuration.\n\t */\n\tprivate _config?: EditorConfig;\n\n\t/**\n\t * The creation method.\n\t *\n\t * @see #setCreator\n\t */\n\tdeclare protected _creator: EditorCreatorFunction<TEditor>;\n\n\t/**\n\t * The destruction method.\n\t *\n\t * @see #setDestructor\n\t */\n\tdeclare protected _destructor: ( editor: Editor ) => Promise<unknown>;\n\n\tprivate _excludedProps?: Set<unknown>;\n\n\t/**\n\t * @param Editor The editor class.\n\t * @param watchdogConfig The watchdog plugin configuration.\n\t */\n\tconstructor( Editor: { create( ...args: any ): Promise<TEditor> } | null, watchdogConfig: WatchdogConfig = {} ) {\n\t\tsuper( watchdogConfig );\n\n\t\t// this._editorClass = Editor;\n\n\t\tthis._throttledSave = throttle(\n\t\t\tthis._save.bind( this ),\n\t\t\ttypeof watchdogConfig.saveInterval === 'number' ? watchdogConfig.saveInterval : 5000\n\t\t);\n\n\t\t// Set default creator and destructor functions:\n\t\tif ( Editor ) {\n\t\t\tthis._creator = ( ( elementOrData, config ) => Editor.create( elementOrData, config ) );\n\t\t}\n\n\t\tthis._destructor = editor => editor.destroy();\n\t}\n\n\t/**\n\t * The current editor instance.\n\t */\n\tpublic get editor(): TEditor | null {\n\t\treturn this._editor;\n\t}\n\n\t/**\n\t * @internal\n\t */\n\tpublic get _item(): TEditor | null {\n\t\treturn this._editor;\n\t}\n\n\t/**\n\t * Sets the function that is responsible for the editor creation.\n\t * It expects a function that should return a promise.\n\t *\n\t * ```ts\n\t * watchdog.setCreator( ( element, config ) => ClassicEditor.create( element, config ) );\n\t * ```\n\t */\n\tpublic setCreator( creator: EditorCreatorFunction<TEditor> ): void {\n\t\tthis._creator = creator;\n\t}\n\n\t/**\n\t * Sets the function that is responsible for the editor destruction.\n\t * Overrides the default destruction function, which destroys only the editor instance.\n\t * It expects a function that should return a promise or `undefined`.\n\t *\n\t * ```ts\n\t * watchdog.setDestructor( editor => {\n\t * \t// Do something before the editor is destroyed.\n\t *\n\t * \treturn editor\n\t * \t\t.destroy()\n\t * \t\t.then( () => {\n\t * \t\t\t// Do something after the editor is destroyed.\n\t * \t\t} );\n\t * } );\n\t * ```\n\t */\n\tpublic setDestructor( destructor: ( editor: Editor ) => Promise<unknown> ): void {\n\t\tthis._destructor = destructor;\n\t}\n\n\t/**\n\t * Restarts the editor instance. This method is called whenever an editor error occurs. It fires the `restart` event and changes\n\t * the state to `initializing`.\n\t *\n\t * @fires restart\n\t */\n\tprotected override _restart(): Promise<unknown> {\n\t\treturn Promise.resolve()\n\t\t\t.then( () => {\n\t\t\t\tthis.state = 'initializing';\n\t\t\t\tthis._fire( 'stateChange' );\n\n\t\t\t\treturn this._destroy();\n\t\t\t} )\n\t\t\t.catch( err => {\n\t\t\t\tconsole.error( 'An error happened during the editor destroying.', err );\n\t\t\t} )\n\t\t\t.then( () => {\n\t\t\t\t// Pre-process some data from the original editor config.\n\t\t\t\t// Our goal here is to make sure that the restarted editor will be reinitialized with correct set of roots.\n\t\t\t\t// We are not interested in any data set in config or in `.create()` first parameter. It will be replaced anyway.\n\t\t\t\t// But we need to set them correctly to make sure that proper roots are created.\n\t\t\t\t//\n\t\t\t\t// Since a different set of roots will be created, `lazyRoots` and `rootsAttributes` properties must be managed too.\n\n\t\t\t\t// Keys are root names, values are ''. Used when the editor was initialized by setting the first parameter to document data.\n\t\t\t\tconst existingRoots: Record<string, string> = {};\n\t\t\t\t// Keeps lazy roots. They may be different when compared to initial config if some of the roots were loaded.\n\t\t\t\tconst lazyRoots: Array<string> = [];\n\t\t\t\t// Roots attributes from the old config. Will be referred when setting new attributes.\n\t\t\t\tconst oldRootsAttributes: Record<string, RootAttributes> = this._config!.rootsAttributes || {};\n\t\t\t\t// New attributes to be set. Is filled only for roots that still exist in the document.\n\t\t\t\tconst rootsAttributes: Record<string, RootAttributes> = {};\n\n\t\t\t\t// Traverse through the roots saved when the editor crashed and set up the discussed values.\n\t\t\t\tfor ( const [ rootName, rootData ] of Object.entries( this._data!.roots ) ) {\n\t\t\t\t\tif ( rootData.isLoaded ) {\n\t\t\t\t\t\texistingRoots[ rootName ] = '';\n\t\t\t\t\t\trootsAttributes[ rootName ] = oldRootsAttributes[ rootName ] || {};\n\t\t\t\t\t} else {\n\t\t\t\t\t\tlazyRoots.push( rootName );\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tconst updatedConfig: EditorConfig = {\n\t\t\t\t\t...this._config,\n\t\t\t\t\textraPlugins: this._config!.extraPlugins || [],\n\t\t\t\t\tlazyRoots,\n\t\t\t\t\trootsAttributes,\n\t\t\t\t\t_watchdogInitialData: this._data\n\t\t\t\t};\n\n\t\t\t\t// Delete `initialData` as it is not needed. Data will be set by the watchdog based on `_watchdogInitialData`.\n\t\t\t\t// First parameter of the editor `.create()` will be used to set up initial roots.\n\t\t\t\tdelete updatedConfig.initialData;\n\n\t\t\t\tupdatedConfig.extraPlugins!.push( EditorWatchdogInitPlugin as any );\n\n\t\t\t\tif ( this._initUsingData ) {\n\t\t\t\t\treturn this.create( existingRoots, updatedConfig, updatedConfig.context );\n\t\t\t\t} else {\n\t\t\t\t\t// Set correct editables to make sure that proper roots are created and linked with DOM elements.\n\t\t\t\t\t// No need to set initial data, as it would be discarded anyway.\n\t\t\t\t\t//\n\t\t\t\t\t// If one element was initially set in `elementOrData`, then use that original element to restart the editor.\n\t\t\t\t\t// This is for compatibility purposes with single-root editor types.\n\t\t\t\t\tif ( isElement( this._elementOrData ) ) {\n\t\t\t\t\t\treturn this.create( this._elementOrData, updatedConfig, updatedConfig.context );\n\t\t\t\t\t} else {\n\t\t\t\t\t\treturn this.create( this._editables, updatedConfig, updatedConfig.context );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} )\n\t\t\t.then( () => {\n\t\t\t\tthis._fire( 'restart' );\n\t\t\t} );\n\t}\n\n\t/**\n\t * Creates the editor instance and keeps it running, using the defined creator and destructor.\n\t *\n\t * @param elementOrData The editor source element or the editor data.\n\t * @param config The editor configuration.\n\t * @param context A context for the editor.\n\t */\n\tpublic create(\n\t\telementOrData: HTMLElement | string | Record<string, string> | Record<string, HTMLElement> = this._elementOrData!,\n\t\tconfig: EditorConfig = this._config!,\n\t\tcontext?: Context\n\t): Promise<unknown> {\n\t\treturn Promise.resolve()\n\t\t\t.then( () => {\n\t\t\t\tsuper._startErrorHandling();\n\n\t\t\t\tthis._elementOrData = elementOrData;\n\n\t\t\t\t// Use document data in the first parameter of the editor `.create()` call only if it was used like this originally.\n\t\t\t\t// Use document data if a string or object with strings was passed.\n\t\t\t\tthis._initUsingData = typeof elementOrData == 'string' ||\n\t\t\t\t\t( Object.keys( elementOrData ).length > 0 && typeof Object.values( elementOrData )[ 0 ] == 'string' );\n\n\t\t\t\t// Clone configuration because it might be shared within multiple watchdog instances. Otherwise,\n\t\t\t\t// when an error occurs in one of these editors, the watchdog will restart all of them.\n\t\t\t\tthis._config = this._cloneEditorConfiguration( config ) || {};\n\n\t\t\t\tthis._config!.context = context;\n\n\t\t\t\treturn this._creator( elementOrData, this._config! );\n\t\t\t} )\n\t\t\t.then( editor => {\n\t\t\t\tthis._editor = editor;\n\n\t\t\t\teditor.model.document.on( 'change:data', this._throttledSave );\n\n\t\t\t\tthis._lastDocumentVersion = editor.model.document.version;\n\t\t\t\tthis._data = this._getData();\n\n\t\t\t\tif ( !this._initUsingData ) {\n\t\t\t\t\tthis._editables = this._getEditables();\n\t\t\t\t}\n\n\t\t\t\tthis.state = 'ready';\n\t\t\t\tthis._fire( 'stateChange' );\n\t\t\t} );\n\t}\n\n\t/**\n\t * Destroys the watchdog and the current editor instance. It fires the callback\n\t * registered in {@link #setDestructor `setDestructor()`} and uses it to destroy the editor instance.\n\t * It also sets the state to `destroyed`.\n\t */\n\tpublic override destroy(): Promise<unknown> {\n\t\treturn Promise.resolve()\n\t\t\t.then( () => {\n\t\t\t\tthis.state = 'destroyed';\n\t\t\t\tthis._fire( 'stateChange' );\n\n\t\t\t\tsuper.destroy();\n\n\t\t\t\treturn this._destroy();\n\t\t\t} );\n\t}\n\n\tprivate _destroy(): Promise<unknown> {\n\t\treturn Promise.resolve()\n\t\t\t.then( () => {\n\t\t\t\tthis._stopErrorHandling();\n\n\t\t\t\tthis._throttledSave.cancel();\n\n\t\t\t\tconst editor = this._editor;\n\n\t\t\t\tthis._editor = null;\n\n\t\t\t\t// Remove the `change:data` listener before destroying the editor.\n\t\t\t\t// Incorrectly written plugins may trigger firing `change:data` events during the editor destruction phase\n\t\t\t\t// causing the watchdog to call `editor.getData()` when some parts of editor are already destroyed.\n\t\t\t\teditor!.model.document.off( 'change:data', this._throttledSave );\n\n\t\t\t\treturn this._destructor( editor! );\n\t\t\t} );\n\t}\n\n\t/**\n\t * Saves the editor data, so it can be restored after the crash even if the data cannot be fetched at\n\t * the moment of the crash.\n\t */\n\tprivate _save(): void {\n\t\tconst version = this._editor!.model.document.version;\n\n\t\ttry {\n\t\t\tthis._data = this._getData();\n\n\t\t\tif ( !this._initUsingData ) {\n\t\t\t\tthis._editables = this._getEditables();\n\t\t\t}\n\n\t\t\tthis._lastDocumentVersion = version;\n\t\t} catch ( err ) {\n\t\t\tconsole.error(\n\t\t\t\terr,\n\t\t\t\t'An error happened during restoring editor data. ' +\n\t\t\t\t'Editor will be restored from the previously saved data.'\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * @internal\n\t */\n\tpublic _setExcludedProperties( props: Set<unknown> ): void {\n\t\tthis._excludedProps = props;\n\t}\n\n\t/**\n\t * Gets all data that is required to reinitialize editor instance.\n\t */\n\tprivate _getData(): EditorData {\n\t\tconst editor = this._editor!;\n\t\tconst roots = editor.model.document.roots.filter( root => root.isAttached() && root.rootName != '$graveyard' );\n\n\t\tconst { plugins } = editor;\n\t\t// `as any` to avoid linking from external private repo.\n\t\tconst commentsRepository = plugins.has( 'CommentsRepository' ) && plugins.get( 'CommentsRepository' ) as any;\n\t\tconst trackChanges = plugins.has( 'TrackChanges' ) && plugins.get( 'TrackChanges' ) as any;\n\n\t\tconst data: EditorData = {\n\t\t\troots: {},\n\t\t\tmarkers: {},\n\t\t\tcommentThreads: JSON.stringify( [] ),\n\t\t\tsuggestions: JSON.stringify( [] )\n\t\t};\n\n\t\troots.forEach( root => {\n\t\t\tdata.roots[ root.rootName ] = {\n\t\t\t\tcontent: JSON.stringify( Array.from( root.getChildren() ) ),\n\t\t\t\tattributes: JSON.stringify( Array.from( root.getAttributes() ) ),\n\t\t\t\tisLoaded: root._isLoaded\n\t\t\t};\n\t\t} );\n\n\t\tfor ( const marker of editor.model.markers ) {\n\t\t\tif ( !marker._affectsData ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tdata.markers[ marker.name ] = {\n\t\t\t\trangeJSON: marker.getRange().toJSON() as any,\n\t\t\t\tusingOperation: marker._managedUsingOperations,\n\t\t\t\taffectsData: marker._affectsData\n\t\t\t};\n\t\t}\n\n\t\tif ( commentsRepository ) {\n\t\t\tdata.commentThreads = JSON.stringify( commentsRepository.getCommentThreads( { toJSON: true, skipNotAttached: true } ) );\n\t\t}\n\n\t\tif ( trackChanges ) {\n\t\t\tdata.suggestions = JSON.stringify( trackChanges.getSuggestions( { toJSON: true, skipNotAttached: true } ) );\n\t\t}\n\n\t\treturn data;\n\t}\n\n\t/**\n\t * For each attached model root, returns its HTML editable element (if available).\n\t */\n\tprivate _getEditables(): Record<string, HTMLElement> {\n\t\tconst editables: Record<string, HTMLElement> = {};\n\n\t\tfor ( const rootName of this.editor!.model.document.getRootNames() ) {\n\t\t\tconst editable = this.editor!.ui.getEditableElement( rootName );\n\n\t\t\tif ( editable ) {\n\t\t\t\teditables[ rootName ] = editable;\n\t\t\t}\n\t\t}\n\n\t\treturn editables;\n\t}\n\n\t/**\n\t * Traverses the error context and the current editor to find out whether these structures are connected\n\t * to each other via properties.\n\t *\n\t * @internal\n\t */\n\tpublic _isErrorComingFromThisItem( error: CKEditorError ): boolean {\n\t\treturn areConnectedThroughProperties( this._editor, error.context, this._excludedProps );\n\t}\n\n\t/**\n\t * Clones the editor configuration.\n\t */\n\tprivate _cloneEditorConfiguration( config: EditorConfig ): EditorConfig {\n\t\treturn cloneDeepWith( config, ( value, key ) => {\n\t\t\t// Leave DOM references.\n\t\t\tif ( isElement( value ) ) {\n\t\t\t\treturn value;\n\t\t\t}\n\n\t\t\tif ( key === 'context' ) {\n\t\t\t\treturn value;\n\t\t\t}\n\t\t} );\n\t}\n}\n\n/**\n * Internal plugin that is used to stop the default editor initialization and restoring the editor state\n * based on the `editor.config._watchdogInitialData` data.\n */\nclass EditorWatchdogInitPlugin {\n\tpublic editor: Editor;\n\n\tprivate _data: EditorData;\n\n\tconstructor( editor: Editor ) {\n\t\tthis.editor = editor;\n\n\t\tthis._data = editor.config.get( '_watchdogInitialData' )!;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\t// Stops the default editor initialization and use the saved data to restore the editor state.\n\t\t// Some of data could not be initialize as a config properties. It is important to keep the data\n\t\t// in the same form as it was before the restarting.\n\t\tthis.editor.data.on( 'init', evt => {\n\t\t\tevt.stop();\n\n\t\t\tthis.editor.model.enqueueChange( { isUndoable: false }, writer => {\n\t\t\t\tthis._restoreCollaborationData();\n\t\t\t\tthis._restoreEditorData( writer );\n\t\t\t} );\n\n\t\t\tthis.editor.data.fire<EditorReadyEvent>( 'ready' );\n\n\t\t\t// Keep priority `'high' - 1` to be sure that RTC initialization will be first.\n\t\t}, { priority: 1000 - 1 } );\n\t}\n\n\t/**\n\t * Creates a model node (element or text) based on provided JSON.\n\t */\n\tprivate _createNode( writer: Writer, jsonNode: any ): Text | Element {\n\t\tif ( 'name' in jsonNode ) {\n\t\t\t// If child has name property, it is an Element.\n\t\t\tconst element = writer.createElement( jsonNode.name, jsonNode.attributes );\n\n\t\t\tif ( jsonNode.children ) {\n\t\t\t\tfor ( const child of jsonNode.children ) {\n\t\t\t\t\telement._appendChild( this._createNode( writer, child ) );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\treturn element;\n\t\t} else {\n\t\t\t// Otherwise, it is a Text node.\n\t\t\treturn writer.createText( jsonNode.data, jsonNode.attributes );\n\t\t}\n\t}\n\n\t/**\n\t * Restores the editor by setting the document data, roots attributes and markers.\n\t */\n\tprivate _restoreEditorData( writer: Writer ): void {\n\t\tconst editor = this.editor!;\n\n\t\tObject.entries( this._data!.roots ).forEach( ( [ rootName, { content, attributes } ] ) => {\n\t\t\tconst parsedNodes: Array<Node | Element> = JSON.parse( content );\n\t\t\tconst parsedAttributes: Array<[ string, unknown ]> = JSON.parse( attributes );\n\n\t\t\tconst rootElement = editor.model.document.getRoot( rootName )!;\n\n\t\t\tfor ( const [ key, value ] of parsedAttributes ) {\n\t\t\t\twriter.setAttribute( key, value, rootElement );\n\t\t\t}\n\n\t\t\tfor ( const child of parsedNodes ) {\n\t\t\t\tconst node = this._createNode( writer, child );\n\n\t\t\t\twriter.insert( node, rootElement, 'end' );\n\t\t\t}\n\t\t} );\n\n\t\tObject.entries( this._data!.markers ).forEach( ( [ markerName, markerOptions ] ) => {\n\t\t\tconst { document } = editor.model;\n\t\t\tconst {\n\t\t\t\trangeJSON: { start, end },\n\t\t\t\t...options\n\t\t\t} = markerOptions;\n\n\t\t\tconst root = document.getRoot( start.root )!;\n\t\t\tconst startPosition = writer.createPositionFromPath( root, start.path, start.stickiness );\n\t\t\tconst endPosition = writer.createPositionFromPath( root, end.path, end.stickiness );\n\n\t\t\tconst range = writer.createRange( startPosition, endPosition );\n\n\t\t\twriter.addMarker( markerName, {\n\t\t\t\trange,\n\t\t\t\t...options\n\t\t\t} );\n\t\t} );\n\t}\n\n\t/**\n\t * Restores the editor collaboration data - comment threads and suggestions.\n\t */\n\tprivate _restoreCollaborationData() {\n\t\t// `as any` to avoid linking from external private repo.\n\t\tconst parsedCommentThreads: Array<any> = JSON.parse( this._data.commentThreads );\n\t\tconst parsedSuggestions: Array<any> = JSON.parse( this._data.suggestions );\n\n\t\tparsedCommentThreads.forEach( commentThreadData => {\n\t\t\tconst channelId = this.editor.config.get( 'collaboration.channelId' )!;\n\t\t\tconst commentsRepository = this.editor!.plugins.get( 'CommentsRepository' ) as any;\n\n\t\t\tif ( commentsRepository.hasCommentThread( commentThreadData.threadId ) ) {\n\t\t\t\tconst commentThread = commentsRepository.getCommentThread( commentThreadData.threadId )!;\n\n\t\t\t\tcommentThread.remove();\n\t\t\t}\n\n\t\t\tcommentsRepository.addCommentThread( { channelId, ...commentThreadData } );\n\t\t} );\n\n\t\tparsedSuggestions.forEach( suggestionData => {\n\t\t\tconst trackChangesEditing = this.editor!.plugins.get( 'TrackChangesEditing' ) as any;\n\n\t\t\tif ( trackChangesEditing.hasSuggestion( suggestionData.id ) ) {\n\t\t\t\tconst suggestion = trackChangesEditing.getSuggestion( suggestionData.id );\n\n\t\t\t\tsuggestion.attributes = suggestionData.attributes;\n\t\t\t} else {\n\t\t\t\ttrackChangesEditing.addSuggestionData( suggestionData );\n\t\t\t}\n\t\t} );\n\t}\n}\n\nexport type EditorData = {\n\troots: Record<string, {\n\t\tcontent: string;\n\t\tattributes: string;\n\t\tisLoaded: boolean;\n\t}>;\n\tmarkers: Record<string, {\n\t\trangeJSON: { start: any; end: any };\n\t\tusingOperation: boolean;\n\t\taffectsData: boolean;\n\t}>;\n\tcommentThreads: string;\n\tsuggestions: string;\n};\n\n/**\n * Fired after the watchdog restarts the error in case of a crash.\n *\n * @eventName ~EditorWatchdog#restart\n */\nexport type EditorWatchdogRestartEvent = {\n\tname: 'restart';\n\targs: [];\n\treturn: undefined;\n};\n\nexport type EditorCreatorFunction<TEditor = Editor> = (\n\telementOrData: HTMLElement | string | Record<string, string> | Record<string, HTMLElement>,\n\tconfig: EditorConfig\n) => Promise<TEditor>;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module watchdog/contextwatchdog\n */\n\n/* globals console */\n\n// eslint-disable-next-line ckeditor5-rules/no-cross-package-imports\nimport type {\n\tContext,\n\tEditor,\n\tEditorConfig,\n\tContextConfig\n} from 'ckeditor5/src/core';\n\n// eslint-disable-next-line ckeditor5-rules/no-cross-package-imports\nimport type {\n\tArrayOrItem,\n\tCKEditorError\n} from 'ckeditor5/src/utils';\n\nimport Watchdog, { type WatchdogConfig, type WatchdogState } from './watchdog';\nimport EditorWatchdog, { type EditorCreatorFunction } from './editorwatchdog';\nimport areConnectedThroughProperties from './utils/areconnectedthroughproperties';\nimport getSubNodes from './utils/getsubnodes';\n\nconst mainQueueId = Symbol( 'MainQueueId' );\n\n/**\n * A watchdog for the {@link module:core/context~Context} class.\n *\n * See the {@glink features/watchdog Watchdog feature guide} to learn the rationale behind it and\n * how to use it.\n */\nexport default class ContextWatchdog<TContext extends Context = Context> extends Watchdog {\n\t/**\n\t * A map of internal watchdogs for added items.\n\t */\n\tprotected _watchdogs = new Map<string, EditorWatchdog>();\n\n\t/**\n\t * The watchdog configuration.\n\t */\n\tprivate readonly _watchdogConfig: WatchdogConfig;\n\n\t/**\n\t * The current context instance.\n\t */\n\tprivate _context: TContext | null = null;\n\n\t/**\n\t * Context properties (nodes/references) that are gathered during the initial context creation\n\t * and are used to distinguish the origin of an error.\n\t */\n\tprivate _contextProps = new Set<unknown>();\n\n\t/**\n\t * An action queue, which is used to handle async functions queuing.\n\t */\n\tprivate _actionQueues = new ActionQueues();\n\n\t/**\n\t * The configuration for the {@link module:core/context~Context}.\n\t */\n\tprivate _contextConfig?: ContextConfig;\n\n\t/**\n\t * The creation method.\n\t *\n\t * @see #setCreator\n\t */\n\tdeclare protected _creator: ( config: ContextConfig ) => Promise<TContext>;\n\n\t/**\n\t * The destruction method.\n\t *\n\t * @see #setDestructor\n\t */\n\tdeclare protected _destructor: ( context: Context ) => Promise<unknown>;\n\n\t/**\n\t * The watched item.\n\t */\n\tpublic _item: unknown;\n\n\t/**\n\t * The context watchdog class constructor.\n\t *\n\t * ```ts\n\t * const watchdog = new ContextWatchdog( Context );\n\t *\n\t * await watchdog.create( contextConfiguration );\n\t *\n\t * await watchdog.add( item );\n\t * ```\n\t *\n\t * See the {@glink features/watchdog Watchdog feature guide} to learn more how to use this feature.\n\t *\n\t * @param Context The {@link module:core/context~Context} class.\n\t * @param watchdogConfig The watchdog configuration.\n\t */\n\tconstructor(\n\t\tContext: { create( ...args: any ): Promise<TContext> },\n\t\twatchdogConfig: WatchdogConfig = {}\n\t) {\n\t\tsuper( watchdogConfig );\n\n\t\tthis._watchdogConfig = watchdogConfig;\n\n\t\t// Default creator and destructor.\n\t\tthis._creator = contextConfig => Context.create( contextConfig );\n\t\tthis._destructor = context => context.destroy();\n\n\t\tthis._actionQueues.onEmpty( () => {\n\t\t\tif ( this.state === 'initializing' ) {\n\t\t\t\tthis.state = 'ready';\n\t\t\t\tthis._fire( 'stateChange' );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Sets the function that is responsible for the context creation.\n\t * It expects a function that should return a promise (or `undefined`).\n\t *\n\t * ```ts\n\t * watchdog.setCreator( config => Context.create( config ) );\n\t * ```\n\t */\n\tpublic setCreator( creator: ( config: ContextConfig ) => Promise<TContext> ): void {\n\t\tthis._creator = creator;\n\t}\n\n\t/**\n\t * Sets the function that is responsible for the context destruction.\n\t * Overrides the default destruction function, which destroys only the context instance.\n\t * It expects a function that should return a promise (or `undefined`).\n\t *\n\t * ```ts\n\t * watchdog.setDestructor( context => {\n\t * \t// Do something before the context is destroyed.\n\t *\n\t * \treturn context\n\t * \t\t.destroy()\n\t * \t\t.then( () => {\n\t * \t\t\t// Do something after the context is destroyed.\n\t * \t\t} );\n\t * } );\n\t * ```\n\t */\n\tpublic setDestructor( destructor: ( context: Context ) => Promise<unknown> ): void {\n\t\tthis._destructor = destructor;\n\t}\n\n\t/**\n\t * The context instance. Keep in mind that this property might be changed when the context watchdog restarts,\n\t * so do not keep this instance internally. Always operate on the `ContextWatchdog#context` property.\n\t */\n\tpublic get context(): Context | null {\n\t\treturn this._context;\n\t}\n\n\t/**\n\t * Initializes the context watchdog. Once it is created, the watchdog takes care about\n\t * recreating the context and the provided items, and starts the error handling mechanism.\n\t *\n\t * ```ts\n\t * await watchdog.create( {\n\t * \tplugins: []\n\t * } );\n\t * ```\n\t *\n\t * @param contextConfig The context configuration. See {@link module:core/context~Context}.\n\t */\n\tpublic create( contextConfig: ContextConfig = {} ): Promise<unknown> {\n\t\treturn this._actionQueues.enqueue( mainQueueId, () => {\n\t\t\tthis._contextConfig = contextConfig;\n\n\t\t\treturn this._create();\n\t\t} );\n\t}\n\n\t/**\n\t * Returns an item instance with the given `itemId`.\n\t *\n\t * ```ts\n\t * const editor1 = watchdog.getItem( 'editor1' );\n\t * ```\n\t *\n\t * @param itemId The item ID.\n\t * @returns The item instance or `undefined` if an item with a given ID has not been found.\n\t */\n\tpublic getItem( itemId: string ): unknown {\n\t\tconst watchdog = this._getWatchdog( itemId );\n\n\t\treturn watchdog._item;\n\t}\n\n\t/**\n\t * Gets the state of the given item. See {@link #state} for a list of available states.\n\t *\n\t * ```ts\n\t * const editor1State = watchdog.getItemState( 'editor1' );\n\t * ```\n\t *\n\t * @param itemId Item ID.\n\t * @returns The state of the item.\n\t */\n\tpublic getItemState( itemId: string ): WatchdogState {\n\t\tconst watchdog = this._getWatchdog( itemId );\n\n\t\treturn watchdog.state;\n\t}\n\n\t/**\n\t * Adds items to the watchdog. Once created, instances of these items will be available using the {@link #getItem} method.\n\t *\n\t * Items can be passed together as an array of objects:\n\t *\n\t * ```ts\n\t * await watchdog.add( [ {\n\t * \tid: 'editor1',\n\t * \ttype: 'editor',\n\t * \tsourceElementOrData: document.querySelector( '#editor' ),\n\t * \tconfig: {\n\t * \t\tplugins: [ Essentials, Paragraph, Bold, Italic ],\n\t * \t\ttoolbar: [ 'bold', 'italic', 'alignment' ]\n\t * \t},\n\t * \tcreator: ( element, config ) => ClassicEditor.create( element, config )\n\t * } ] );\n\t * ```\n\t *\n\t * Or one by one as objects:\n\t *\n\t * ```ts\n\t * await watchdog.add( {\n\t * \tid: 'editor1',\n\t * \ttype: 'editor',\n\t * \tsourceElementOrData: document.querySelector( '#editor' ),\n\t * \tconfig: {\n\t * \t\tplugins: [ Essentials, Paragraph, Bold, Italic ],\n\t * \t\ttoolbar: [ 'bold', 'italic', 'alignment' ]\n\t * \t},\n\t * \tcreator: ( element, config ) => ClassicEditor.create( element, config )\n\t * ] );\n\t * ```\n\t *\n\t * Then an instance can be retrieved using the {@link #getItem} method:\n\t *\n\t * ```ts\n\t * const editor1 = watchdog.getItem( 'editor1' );\n\t * ```\n\t *\n\t * Note that this method can be called multiple times, but for performance reasons it is better\n\t * to pass all items together.\n\t *\n\t * @param itemConfigurationOrItemConfigurations An item configuration object or an array of item configurations.\n\t */\n\tpublic add( itemConfigurationOrItemConfigurations: ArrayOrItem<WatchdogItemConfiguration> ): Promise<unknown> {\n\t\tconst itemConfigurations = toArray( itemConfigurationOrItemConfigurations );\n\n\t\treturn Promise.all( itemConfigurations.map( item => {\n\t\t\treturn this._actionQueues.enqueue( item.id, () => {\n\t\t\t\tif ( this.state === 'destroyed' ) {\n\t\t\t\t\tthrow new Error( 'Cannot add items to destroyed watchdog.' );\n\t\t\t\t}\n\n\t\t\t\tif ( !this._context ) {\n\t\t\t\t\tthrow new Error( 'Context was not created yet. You should call the `ContextWatchdog#create()` method first.' );\n\t\t\t\t}\n\n\t\t\t\tlet watchdog: EditorWatchdog;\n\n\t\t\t\tif ( this._watchdogs.has( item.id ) ) {\n\t\t\t\t\tthrow new Error( `Item with the given id is already added: '${ item.id }'.` );\n\t\t\t\t}\n\n\t\t\t\tif ( item.type === 'editor' ) {\n\t\t\t\t\twatchdog = new EditorWatchdog( null, this._watchdogConfig );\n\t\t\t\t\twatchdog.setCreator( item.creator );\n\t\t\t\t\twatchdog._setExcludedProperties( this._contextProps );\n\n\t\t\t\t\tif ( item.destructor ) {\n\t\t\t\t\t\twatchdog.setDestructor( item.destructor );\n\t\t\t\t\t}\n\n\t\t\t\t\tthis._watchdogs.set( item.id, watchdog );\n\n\t\t\t\t\t// Enqueue the internal watchdog errors within the main queue.\n\t\t\t\t\t// And propagate the internal `error` events as `itemError` event.\n\t\t\t\t\twatchdog.on( 'error', ( evt, { error, causesRestart } ) => {\n\t\t\t\t\t\tthis._fire( 'itemError', { itemId: item.id, error } );\n\n\t\t\t\t\t\t// Do not enqueue the item restart action if the item will not restart.\n\t\t\t\t\t\tif ( !causesRestart ) {\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tthis._actionQueues.enqueue( item.id, () => new Promise<void>( res => {\n\t\t\t\t\t\t\tconst rethrowRestartEventOnce = () => {\n\t\t\t\t\t\t\t\twatchdog.off( 'restart', rethrowRestartEventOnce );\n\n\t\t\t\t\t\t\t\tthis._fire( 'itemRestart', { itemId: item.id } );\n\n\t\t\t\t\t\t\t\tres();\n\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\t\twatchdog.on( 'restart', rethrowRestartEventOnce );\n\t\t\t\t\t\t} ) );\n\t\t\t\t\t} );\n\n\t\t\t\t\treturn watchdog.create( item.sourceElementOrData, item.config, this._context );\n\t\t\t\t} else {\n\t\t\t\t\tthrow new Error( `Not supported item type: '${ item.type }'.` );\n\t\t\t\t}\n\t\t\t} );\n\t\t} ) );\n\t}\n\n\t/**\n\t * Removes and destroys item(s) with given ID(s).\n\t *\n\t * ```ts\n\t * await watchdog.remove( 'editor1' );\n\t * ```\n\t *\n\t * Or\n\t *\n\t * ```ts\n\t * await watchdog.remove( [ 'editor1', 'editor2' ] );\n\t * ```\n\t *\n\t * @param itemIdOrItemIds Item ID or an array of item IDs.\n\t */\n\tpublic remove( itemIdOrItemIds: ArrayOrItem<string> ): Promise<unknown> {\n\t\tconst itemIds = toArray( itemIdOrItemIds );\n\n\t\treturn Promise.all( itemIds.map( itemId => {\n\t\t\treturn this._actionQueues.enqueue( itemId, () => {\n\t\t\t\tconst watchdog = this._getWatchdog( itemId );\n\n\t\t\t\tthis._watchdogs.delete( itemId );\n\n\t\t\t\treturn watchdog.destroy();\n\t\t\t} );\n\t\t} ) );\n\t}\n\n\t/**\n\t * Destroys the context watchdog and all added items.\n\t * Once the context watchdog is destroyed, new items cannot be added.\n\t *\n\t * ```ts\n\t * await watchdog.destroy();\n\t * ```\n\t */\n\tpublic override destroy(): Promise<unknown> {\n\t\treturn this._actionQueues.enqueue( mainQueueId, () => {\n\t\t\tthis.state = 'destroyed';\n\t\t\tthis._fire( 'stateChange' );\n\n\t\t\tsuper.destroy();\n\n\t\t\treturn this._destroy();\n\t\t} );\n\t}\n\n\t/**\n\t * Restarts the context watchdog.\n\t */\n\tprotected _restart(): Promise<unknown> {\n\t\treturn this._actionQueues.enqueue( mainQueueId, () => {\n\t\t\tthis.state = 'initializing';\n\t\t\tthis._fire( 'stateChange' );\n\n\t\t\treturn this._destroy()\n\t\t\t\t.catch( err => {\n\t\t\t\t\tconsole.error( 'An error happened during destroying the context or items.', err );\n\t\t\t\t} )\n\t\t\t\t.then( () => this._create() )\n\t\t\t\t.then( () => this._fire( 'restart' ) );\n\t\t} );\n\t}\n\n\t/**\n\t * Initializes the context watchdog.\n\t */\n\tprivate _create(): Promise<unknown> {\n\t\treturn Promise.resolve()\n\t\t\t.then( () => {\n\t\t\t\tthis._startErrorHandling();\n\n\t\t\t\treturn this._creator( this._contextConfig! );\n\t\t\t} )\n\t\t\t.then( context => {\n\t\t\t\tthis._context = context;\n\t\t\t\tthis._contextProps = getSubNodes( this._context );\n\n\t\t\t\treturn Promise.all(\n\t\t\t\t\tArray.from( this._watchdogs.values() )\n\t\t\t\t\t\t.map( watchdog => {\n\t\t\t\t\t\t\twatchdog._setExcludedProperties( this._contextProps );\n\n\t\t\t\t\t\t\treturn watchdog.create( undefined, undefined, this._context! );\n\t\t\t\t\t\t} )\n\t\t\t\t);\n\t\t\t} );\n\t}\n\n\t/**\n\t * Destroys the context instance and all added items.\n\t */\n\tprivate _destroy(): Promise<unknown> {\n\t\treturn Promise.resolve()\n\t\t\t.then( () => {\n\t\t\t\tthis._stopErrorHandling();\n\n\t\t\t\tconst context = this._context!;\n\n\t\t\t\tthis._context = null;\n\t\t\t\tthis._contextProps = new Set();\n\n\t\t\t\treturn Promise.all(\n\t\t\t\t\tArray.from( this._watchdogs.values() )\n\t\t\t\t\t\t.map( watchdog => watchdog.destroy() )\n\t\t\t\t)\n\t\t\t\t\t// Context destructor destroys each editor.\n\t\t\t\t\t.then( () => this._destructor( context ) );\n\t\t\t} );\n\t}\n\n\t/**\n\t * Returns the watchdog for a given item ID.\n\t *\n\t * @param itemId Item ID.\n\t */\n\tprotected _getWatchdog( itemId: string ): Watchdog {\n\t\tconst watchdog = this._watchdogs.get( itemId );\n\n\t\tif ( !watchdog ) {\n\t\t\tthrow new Error( `Item with the given id was not registered: ${ itemId }.` );\n\t\t}\n\n\t\treturn watchdog;\n\t}\n\n\t/**\n\t * Checks whether an error comes from the context instance and not from the item instances.\n\t *\n\t * @internal\n\t */\n\tpublic _isErrorComingFromThisItem( error: CKEditorError ): boolean {\n\t\tfor ( const watchdog of this._watchdogs.values() ) {\n\t\t\tif ( watchdog._isErrorComingFromThisItem( error ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn areConnectedThroughProperties( this._context, error.context );\n\t}\n}\n\n/**\n * Fired after the watchdog restarts the context and the added items because of a crash.\n *\n * ```ts\n * watchdog.on( 'restart', () => {\n * \tconsole.log( 'The context has been restarted.' );\n * } );\n * ```\n *\n * @eventName ~ContextWatchdog#restart\n */\nexport type ContextWatchdogRestartEvent = {\n\tname: 'restart';\n\targs: [];\n\treturn: undefined;\n};\n\n/**\n * Fired when a new error occurred in one of the added items.\n *\n * ```ts\n * watchdog.on( 'itemError', ( evt, { error, itemId } ) => {\n * \tconsole.log( `An error occurred in an item with the '${ itemId }' ID.` );\n * } );\n * ```\n *\n * @eventName ~ContextWatchdog#itemError\n */\nexport type ContextWatchdogItemErrorEvent = {\n\tname: 'itemError';\n\targs: [ ContextWatchdogItemErrorEventData ];\n\treturn: undefined;\n};\n\n/**\n * The `itemError` event data.\n */\nexport type ContextWatchdogItemErrorEventData = {\n\titemId: string;\n\terror: Error;\n};\n\n/**\n * Fired after an item has been restarted.\n *\n * ```ts\n * \twatchdog.on( 'itemRestart', ( evt, { itemId } ) => {\n *\t\tconsole.log( 'An item with with the '${ itemId }' ID has been restarted.' );\n * \t} );\n * ```\n *\n * @eventName ~ContextWatchdog#itemRestart\n */\nexport type ContextWatchdogItemRestartEvent = {\n\tname: 'itemRestart';\n\targs: [ ContextWatchdogItemRestartEventData ];\n\treturn: undefined;\n};\n\n/**\n * The `itemRestart` event data.\n */\nexport type ContextWatchdogItemRestartEventData = {\n\titemId: string;\n};\n\n/**\n * Manager of action queues that allows queuing async functions.\n */\nclass ActionQueues {\n\tpublic _onEmptyCallbacks: Array<() => void> = [];\n\tpublic _queues = new Map<symbol | string | number, Promise<unknown>>();\n\tpublic _activeActions = 0;\n\n\t/**\n\t * Used to register callbacks that will be run when the queue becomes empty.\n\t *\n\t * @param onEmptyCallback A callback that will be run whenever the queue becomes empty.\n\t */\n\tpublic onEmpty( onEmptyCallback: () => void ): void {\n\t\tthis._onEmptyCallbacks.push( onEmptyCallback );\n\t}\n\n\t/**\n\t * It adds asynchronous actions (functions) to the proper queue and runs them one by one.\n\t *\n\t * @param queueId The action queue ID.\n\t * @param action A function that should be enqueued.\n\t */\n\tpublic enqueue( queueId: symbol | string | number, action: () => unknown ): Promise<unknown> {\n\t\tconst isMainAction = queueId === mainQueueId;\n\n\t\tthis._activeActions++;\n\n\t\tif ( !this._queues.get( queueId ) ) {\n\t\t\tthis._queues.set( queueId, Promise.resolve() );\n\t\t}\n\n\t\t// List all sources of actions that the current action needs to await for.\n\t\t// For the main action wait for all other actions.\n\t\t// For the item action wait only for the item queue and the main queue.\n\t\tconst awaitedActions = isMainAction ?\n\t\t\tPromise.all( this._queues.values() ) :\n\t\t\tPromise.all( [ this._queues.get( mainQueueId ), this._queues.get( queueId ) ] );\n\n\t\tconst queueWithAction = awaitedActions.then( action );\n\n\t\t// Catch all errors in the main queue to stack promises even if an error occurred in the past.\n\t\tconst nonErrorQueue = queueWithAction.catch( () => {} );\n\n\t\tthis._queues.set( queueId, nonErrorQueue );\n\n\t\treturn queueWithAction.finally( () => {\n\t\t\tthis._activeActions--;\n\n\t\t\tif ( this._queues.get( queueId ) === nonErrorQueue && this._activeActions === 0 ) {\n\t\t\t\tthis._onEmptyCallbacks.forEach( cb => cb() );\n\t\t\t}\n\t\t} );\n\t}\n}\n\n/**\n * Transforms any value to an array. If the provided value is already an array, it is returned unchanged.\n *\n * @param elementOrArray The value to transform to an array.\n * @returns An array created from data.\n */\nfunction toArray<T>( elementOrArray: ArrayOrItem<T> ): Array<T> {\n\treturn Array.isArray( elementOrArray ) ? elementOrArray : [ elementOrArray ];\n}\n\n/**\n * The watchdog item configuration interface.\n */\nexport interface WatchdogItemConfiguration {\n\n\t/**\n\t * id A unique item identificator.\n\t */\n\tid: string;\n\n\t/**\n\t * The type of the item to create. At the moment, only `'editor'` is supported.\n\t */\n\ttype: 'editor';\n\n\t/**\n\t * A function that initializes the item (the editor). The function takes editor initialization arguments\n\t * and should return a promise. For example: `( el, config ) => ClassicEditor.create( el, config )`.\n\t */\n\tcreator: EditorCreatorFunction;\n\n\t/**\n\t * A function that destroys the item instance (the editor). The function\n\t * takes an item and should return a promise. For example: `editor => editor.destroy()`\n\t */\n\tdestructor?: ( editor: Editor ) => Promise<unknown>;\n\n\t/**\n\t * The source element or data that will be passed\n\t * as the first argument to the `Editor.create()` method.\n\t */\n\tsourceElementOrData: string | HTMLElement;\n\n\t/**\n\t * An editor configuration.\n \t */\n\tconfig: EditorConfig;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module editor-balloon/ballooneditorui\n */\n\nimport {\n\ttype Editor\n} from 'ckeditor5/src/core';\n\nimport {\n\tEditorUI,\n\ttype EditorUIReadyEvent\n} from 'ckeditor5/src/ui';\n\nimport { enablePlaceholder } from 'ckeditor5/src/engine';\n\nimport type BalloonEditorUIView from './ballooneditoruiview';\n\n/**\n * The balloon editor UI class.\n */\nexport default class BalloonEditorUI extends EditorUI {\n\t/**\n\t * The main (topmost) view of the editor UI.\n\t */\n\tpublic readonly view: BalloonEditorUIView;\n\n\t/**\n\t * Creates an instance of the balloon editor UI class.\n\t *\n\t * @param editor The editor instance.\n\t * @param view The view of the UI.\n\t */\n\tconstructor( editor: Editor, view: BalloonEditorUIView ) {\n\t\tsuper( editor );\n\n\t\tthis.view = view;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override get element(): HTMLElement | null {\n\t\treturn this.view.editable.element;\n\t}\n\n\t/**\n\t * Initializes the UI.\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = this.view;\n\t\tconst editingView = editor.editing.view;\n\t\tconst editable = view.editable;\n\t\tconst editingRoot = editingView.document.getRoot()!;\n\n\t\t// The editable UI and editing root should share the same name. Then name is used\n\t\t// to recognize the particular editable, for instance in ARIA attributes.\n\t\teditable.name = editingRoot.rootName;\n\n\t\tview.render();\n\n\t\t// The editable UI element in DOM is available for sure only after the editor UI view has been rendered.\n\t\t// But it can be available earlier if a DOM element has been passed to BalloonEditor.create().\n\t\tconst editableElement = editable.element!;\n\n\t\t// Register the editable UI view in the editor. A single editor instance can aggregate multiple\n\t\t// editable areas (roots) but the balloon editor has only one.\n\t\tthis.setEditableElement( editable.name, editableElement );\n\n\t\t// Let the editable UI element respond to the changes in the global editor focus\n\t\t// tracker. It has been added to the same tracker a few lines above but, in reality, there are\n\t\t// many focusable areas in the editor, like balloons, toolbars or dropdowns and as long\n\t\t// as they have focus, the editable should act like it is focused too (although technically\n\t\t// it isn't), e.g. by setting the proper CSS class, visually announcing focus to the user.\n\t\t// Doing otherwise will result in editable focus styles disappearing, once e.g. the\n\t\t// toolbar gets focused.\n\t\teditable.bind( 'isFocused' ).to( this.focusTracker );\n\n\t\t// Bind the editable UI element to the editing view, making it an end and entrypoint\n\t\t// of the editor's engine. This is where the engine meets the UI.\n\t\teditingView.attachDomRoot( editableElement );\n\n\t\tthis._initPlaceholder();\n\t\tthis.fire<EditorUIReadyEvent>( 'ready' );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tconst view = this.view;\n\t\tconst editingView = this.editor.editing.view;\n\n\t\teditingView.detachDomRoot( view.editable.name! );\n\t\tview.destroy();\n\t}\n\n\t/**\n\t * Enable the placeholder text on the editing root.\n\t */\n\tprivate _initPlaceholder(): void {\n\t\tconst editor = this.editor;\n\t\tconst editingView = editor.editing.view;\n\t\tconst editingRoot = editingView.document.getRoot()!;\n\t\tconst placeholder = editor.config.get( 'placeholder' );\n\n\t\tif ( placeholder ) {\n\t\t\tconst placeholderText = typeof placeholder === 'string' ? placeholder : placeholder[ editingRoot.rootName ];\n\n\t\t\tif ( placeholderText ) {\n\t\t\t\teditingRoot.placeholder = placeholderText;\n\t\t\t}\n\t\t}\n\n\t\tenablePlaceholder( {\n\t\t\tview: editingView,\n\t\t\telement: editingRoot,\n\t\t\tisDirectHost: false,\n\t\t\tkeepOnFocus: true\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module editor-balloon/ballooneditoruiview\n */\n\nimport { EditorUIView, InlineEditableUIView } from 'ckeditor5/src/ui';\nimport type { Locale } from 'ckeditor5/src/utils';\nimport type { View } from 'ckeditor5/src/engine';\n\n/**\n * Contextual editor UI view. Uses the {@link module:ui/editableui/inline/inlineeditableuiview~InlineEditableUIView}.\n */\nexport default class BalloonEditorUIView extends EditorUIView {\n\t/**\n\t * Editable UI view.\n\t */\n\tpublic readonly editable: InlineEditableUIView;\n\n\t/**\n\t * Creates an instance of the balloon editor UI view.\n\t *\n\t * @param locale The {@link module:core/editor/editor~Editor#locale} instance.\n\t * @param editingView The editing view instance this view is related to.\n\t * @param editableElement The editable element. If not specified, it will be automatically created by\n\t * {@link module:ui/editableui/editableuiview~EditableUIView}. Otherwise, the given element will be used.\n\t */\n\tconstructor(\n\t\tlocale: Locale,\n\t\teditingView: View,\n\t\teditableElement?: HTMLElement\n\t) {\n\t\tsuper( locale );\n\n\t\tconst t = locale.t;\n\n\t\tthis.editable = new InlineEditableUIView( locale, editingView, editableElement, {\n\t\t\tlabel: editableView => {\n\t\t\t\treturn t( 'Rich Text Editor. Editing area: %0', editableView.name! );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.registerChild( this.editable );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module editor-balloon/ballooneditor\n */\n\nimport {\n\tEditor,\n\tContext,\n\tDataApiMixin,\n\tElementApiMixin,\n\tattachToForm,\n\tsecureSourceElement,\n\ttype EditorConfig,\n\ttype EditorReadyEvent\n} from 'ckeditor5/src/core';\n\nimport { BalloonToolbar } from 'ckeditor5/src/ui';\nimport { CKEditorError, getDataFromElement } from 'ckeditor5/src/utils';\n\nimport { ContextWatchdog, EditorWatchdog } from 'ckeditor5/src/watchdog';\n\nimport BalloonEditorUI from './ballooneditorui';\nimport BalloonEditorUIView from './ballooneditoruiview';\n\nimport { isElement as _isElement } from 'lodash-es';\n\n/**\n * The {@glink installation/getting-started/predefined-builds#balloon-editor balloon editor}\n * implementation (Medium-like editor).\n * It uses an inline editable and a toolbar based on the {@link module:ui/toolbar/balloon/balloontoolbar~BalloonToolbar}.\n * See the {@glink examples/builds/balloon-editor demo}.\n *\n * In order to create a balloon editor instance, use the static\n * {@link module:editor-balloon/ballooneditor~BalloonEditor.create `BalloonEditor.create()`} method.\n *\n * # Balloon editor and balloon build\n *\n * The balloon editor can be used directly from source (if you installed the\n * [`@ckeditor/ckeditor5-editor-balloon`](https://www.npmjs.com/package/@ckeditor/ckeditor5-editor-balloon) package)\n * but it is also available in the {@glink installation/getting-started/predefined-builds#balloon-editor balloon build}.\n *\n * {@glink installation/getting-started/predefined-builds Builds}\n * are ready-to-use editors with plugins bundled in. When using the editor from\n * source you need to take care of loading all plugins by yourself\n * (through the {@link module:core/editor/editorconfig~EditorConfig#plugins `config.plugins`} option).\n * Using the editor from source gives much better flexibility and allows easier customization.\n *\n * Read more about initializing the editor from source or as a build in\n * {@link module:editor-balloon/ballooneditor~BalloonEditor.create `BalloonEditor.create()`}.\n */\nexport default class BalloonEditor extends DataApiMixin( ElementApiMixin( Editor ) ) {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic readonly ui: BalloonEditorUI;\n\n\t/**\n\t * Creates an instance of the balloon editor.\n\t *\n\t * **Note:** do not use the constructor to create editor instances. Use the static\n\t * {@link module:editor-balloon/ballooneditor~BalloonEditor.create `BalloonEditor.create()`} method instead.\n\t *\n\t * @param sourceElementOrData The DOM element that will be the source for the created editor\n\t * (on which the editor will be initialized) or initial data for the editor. For more information see\n\t * {@link module:editor-balloon/ballooneditor~BalloonEditor.create `BalloonEditor.create()`}.\n\t * @param config The editor configuration.\n\t */\n\tprotected constructor( sourceElementOrData: HTMLElement | string, config: EditorConfig = {} ) {\n\t\t// If both `config.initialData` is set and initial data is passed as the constructor parameter, then throw.\n\t\tif ( !isElement( sourceElementOrData ) && config.initialData !== undefined ) {\n\t\t\t// Documented in core/editor/editorconfig.jsdoc.\n\t\t\t// eslint-disable-next-line ckeditor5-rules/ckeditor-error-message\n\t\t\tthrow new CKEditorError( 'editor-create-initial-data', null );\n\t\t}\n\n\t\tsuper( config );\n\n\t\tif ( this.config.get( 'initialData' ) === undefined ) {\n\t\t\tthis.config.set( 'initialData', getInitialData( sourceElementOrData ) );\n\t\t}\n\n\t\tif ( isElement( sourceElementOrData ) ) {\n\t\t\tthis.sourceElement = sourceElementOrData;\n\t\t\tsecureSourceElement( this, sourceElementOrData );\n\t\t}\n\n\t\tconst plugins = this.config.get( 'plugins' )!;\n\t\tplugins.push( BalloonToolbar );\n\n\t\tthis.config.set( 'plugins', plugins );\n\n\t\tthis.config.define( 'balloonToolbar', this.config.get( 'toolbar' ) );\n\n\t\tthis.model.document.createRoot();\n\n\t\tconst view = new BalloonEditorUIView( this.locale, this.editing.view, this.sourceElement );\n\t\tthis.ui = new BalloonEditorUI( this, view );\n\n\t\tattachToForm( this );\n\t}\n\n\t/**\n\t * Destroys the editor instance, releasing all resources used by it.\n\t *\n\t * Updates the original editor element with the data if the\n\t * {@link module:core/editor/editorconfig~EditorConfig#updateSourceElementOnDestroy `updateSourceElementOnDestroy`}\n\t * configuration option is set to `true`.\n\t */\n\tpublic override destroy(): Promise<unknown> {\n\t\t// Cache the data, then destroy.\n\t\t// It's safe to assume that the model->view conversion will not work after super.destroy().\n\t\tconst data = this.getData();\n\n\t\tthis.ui.destroy();\n\n\t\treturn super.destroy()\n\t\t\t.then( () => {\n\t\t\t\tif ( this.sourceElement ) {\n\t\t\t\t\tthis.updateSourceElement( data );\n\t\t\t\t}\n\t\t\t} );\n\t}\n\n\t/**\n\t * Creates a new balloon editor instance.\n\t *\n\t * There are three general ways how the editor can be initialized.\n\t *\n\t * # Using an existing DOM element (and loading data from it)\n\t *\n\t * You can initialize the editor using an existing DOM element:\n\t *\n\t * ```ts\n\t * BalloonEditor\n\t * \t.create( document.querySelector( '#editor' ) )\n\t * \t.then( editor => {\n\t * \t\tconsole.log( 'Editor was initialized', editor );\n\t * \t} )\n\t * \t.catch( err => {\n\t * \t\tconsole.error( err.stack );\n\t * \t} );\n\t * ```\n\t *\n\t * The element's content will be used as the editor data and the element will become the editable element.\n\t *\n\t * # Creating a detached editor\n\t *\n\t * Alternatively, you can initialize the editor by passing the initial data directly as a string.\n\t * In this case, the editor will render an element that must be inserted into the DOM for the editor to work properly:\n\t *\n\t * ```ts\n\t * BalloonEditor\n\t * \t.create( '<p>Hello world!</p>' )\n\t * \t.then( editor => {\n\t * \t\tconsole.log( 'Editor was initialized', editor );\n\t *\n\t * \t\t// Initial data was provided so the editor UI element needs to be added manually to the DOM.\n\t * \t\tdocument.body.appendChild( editor.ui.element );\n\t * \t} )\n\t * \t.catch( err => {\n\t * \t\tconsole.error( err.stack );\n\t * \t} );\n\t * ```\n\t *\n\t * This lets you dynamically append the editor to your web page whenever it is convenient for you. You may use this method if your\n\t * web page content is generated on the client side and the DOM structure is not ready at the moment when you initialize the editor.\n\t *\n\t * # Using an existing DOM element (and data provided in `config.initialData`)\n\t *\n\t * You can also mix these two ways by providing a DOM element to be used and passing the initial data through the configuration:\n\t *\n\t * ```ts\n\t * BalloonEditor\n\t * \t.create( document.querySelector( '#editor' ), {\n\t * \t\tinitialData: '<h2>Initial data</h2><p>Foo bar.</p>'\n\t * \t} )\n\t * \t.then( editor => {\n\t * \t\tconsole.log( 'Editor was initialized', editor );\n\t * \t} )\n\t * \t.catch( err => {\n\t * \t\tconsole.error( err.stack );\n\t * \t} );\n\t * ```\n\t *\n\t * This method can be used to initialize the editor on an existing element with the specified content in case if your integration\n\t * makes it difficult to set the content of the source element.\n\t *\n\t * Note that an error will be thrown if you pass the initial data both as the first parameter and also in the configuration.\n\t *\n\t * # Configuring the editor\n\t *\n\t * See the {@link module:core/editor/editorconfig~EditorConfig editor configuration documentation} to learn more about\n\t * customizing plugins, toolbar and more.\n\t *\n\t * # Using the editor from source\n\t *\n\t * The code samples listed in the previous sections of this documentation assume that you are using an\n\t * {@glink installation/getting-started/predefined-builds editor build} (for example `@ckeditor/ckeditor5-build-balloon`).\n\t *\n\t * If you want to use the balloon editor from source (`@ckeditor/ckeditor5-editor-balloon/src/ballooneditor`),\n\t * you need to define the list of\n\t * {@link module:core/editor/editorconfig~EditorConfig#plugins plugins to be initialized} and\n\t * {@link module:core/editor/editorconfig~EditorConfig#toolbar toolbar items}. Read more about using the editor from\n\t * source in the {@glink installation/advanced/alternative-setups/integrating-from-source-webpack dedicated guide}.\n\t *\n\t * @param sourceElementOrData The DOM element that will be the source for the created editor\n\t * or the editor's initial data.\n\t *\n\t * If a DOM element is passed, its content will be automatically loaded to the editor upon initialization.\n\t * The editor data will be set back to the original element once the editor is destroyed only if the\n\t * {@link module:core/editor/editorconfig~EditorConfig#updateSourceElementOnDestroy updateSourceElementOnDestroy}\n\t * option is set to `true`.\n\t *\n\t * If the initial data is passed, a detached editor will be created. In this case you need to insert it into the DOM manually.\n\t * It is available under the {@link module:editor-balloon/ballooneditorui~BalloonEditorUI#element `editor.ui.element`} property.\n\t *\n\t * @param config The editor configuration.\n\t * @returns A promise resolved once the editor is ready. The promise resolves with the created editor instance.\n\t */\n\tpublic static override create( sourceElementOrData: HTMLElement | string, config: EditorConfig = {} ): Promise<BalloonEditor> {\n\t\treturn new Promise( resolve => {\n\t\t\tif ( isElement( sourceElementOrData ) && sourceElementOrData.tagName === 'TEXTAREA' ) {\n\t\t\t\t// Documented in core/editor/editor.js\n\t\t\t\t// eslint-disable-next-line ckeditor5-rules/ckeditor-error-message\n\t\t\t\tthrow new CKEditorError( 'editor-wrong-element', null );\n\t\t\t}\n\n\t\t\tconst editor = new this( sourceElementOrData, config );\n\n\t\t\tresolve(\n\t\t\t\teditor.initPlugins()\n\t\t\t\t\t.then( () => editor.ui.init() )\n\t\t\t\t\t.then( () => editor.data.init( editor.config.get( 'initialData' )! ) )\n\t\t\t\t\t.then( () => editor.fire<EditorReadyEvent>( 'ready' ) )\n\t\t\t\t\t.then( () => editor )\n\t\t\t);\n\t\t} );\n\t}\n\n\t/**\n\t * The {@link module:core/context~Context} class.\n\t *\n\t * Exposed as static editor field for easier access in editor builds.\n\t */\n\tpublic static Context = Context;\n\n\t/**\n\t * The {@link module:watchdog/editorwatchdog~EditorWatchdog} class.\n\t *\n\t * Exposed as static editor field for easier access in editor builds.\n\t */\n\tpublic static EditorWatchdog = EditorWatchdog;\n\n\t/**\n\t * The {@link module:watchdog/contextwatchdog~ContextWatchdog} class.\n\t *\n\t * Exposed as static editor field for easier access in editor builds.\n\t */\n\tpublic static ContextWatchdog = ContextWatchdog;\n}\n\nfunction getInitialData( sourceElementOrData: HTMLElement | string ): string {\n\treturn isElement( sourceElementOrData ) ? getDataFromElement( sourceElementOrData ) : sourceElementOrData;\n}\n\nfunction isElement( value: any ): value is Element {\n\treturn _isElement( value );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals HTMLTextAreaElement */\n\n/**\n * @module utils/dom/getdatafromelement\n */\n\n/**\n * Gets data from a given source element.\n *\n * @param el The element from which the data will be retrieved.\n * @returns The data string.\n */\nexport default function getDataFromElement( el: HTMLElement ): string {\n\tif ( el instanceof HTMLTextAreaElement ) {\n\t\treturn el.value;\n\t}\n\n\treturn el.innerHTML;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/editor/utils/securesourceelement\n */\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type { default as Editor, EditorDestroyEvent } from '../editor';\nimport type { ElementApi } from './elementapimixin';\n\n/**\n * Marks the source element on which the editor was initialized. This prevents other editor instances from using this element.\n *\n * Running multiple editor instances on the same source element causes various issues and it is\n * crucial this helper is called as soon as the source element is known to prevent collisions.\n *\n * @param editor Editor instance.\n * @param sourceElement Element to bind with the editor instance.\n */\nexport default function secureSourceElement( editor: Editor, sourceElement: HTMLElement & { ckeditorInstance?: Editor } ): void {\n\tif ( sourceElement.ckeditorInstance ) {\n\t\t/**\n\t\t * A DOM element used to create the editor (e.g.\n\t\t * {@link module:editor-inline/inlineeditor~InlineEditor.create `InlineEditor.create()`})\n\t\t * has already been used to create another editor instance. Make sure each editor is\n\t\t * created with an unique DOM element.\n\t\t *\n\t\t * @error editor-source-element-already-used\n\t\t * @param element DOM element that caused the collision.\n\t\t */\n\t\tthrow new CKEditorError(\n\t\t\t'editor-source-element-already-used',\n\t\t\teditor\n\t\t);\n\t}\n\n\tsourceElement.ckeditorInstance = editor;\n\n\teditor.once<EditorDestroyEvent>( 'destroy', () => {\n\t\tdelete sourceElement.ckeditorInstance;\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module core/editor/utils/attachtoform\n */\n\nimport { isFunction } from 'lodash-es';\n\nimport { CKEditorError } from '@ckeditor/ckeditor5-utils';\n\nimport type { default as Editor, EditorDestroyEvent } from '../editor';\nimport type { ElementApi } from './elementapimixin';\n\n/**\n * Checks if the editor is initialized on a `<textarea>` element that belongs to a form. If yes, it updates the editor's element\n * content before submitting the form.\n *\n * This helper requires the {@link module:core/editor/utils/elementapimixin~ElementApi ElementApi interface}.\n *\n * @param editor Editor instance.\n */\nexport default function attachToForm( editor: Editor & ElementApi ): void {\n\tif ( !isFunction( editor.updateSourceElement ) ) {\n\t\t/**\n\t\t * The editor passed to `attachToForm()` must implement the\n\t\t * {@link module:core/editor/utils/elementapimixin~ElementApi} interface.\n\t\t *\n\t\t * @error attachtoform-missing-elementapi-interface\n\t\t */\n\t\tthrow new CKEditorError(\n\t\t\t'attachtoform-missing-elementapi-interface',\n\t\t\teditor\n\t\t);\n\t}\n\n\tconst sourceElement = editor.sourceElement;\n\n\t// Only when replacing a textarea which is inside of a form element.\n\tif ( isTextArea( sourceElement ) && sourceElement.form ) {\n\t\tlet originalSubmit: () => void;\n\t\tconst form = sourceElement.form;\n\t\tconst onSubmit = () => editor.updateSourceElement();\n\n\t\t// Replace the original form#submit() to call a custom submit function first.\n\t\t// Check if #submit is a function because the form might have an input named \"submit\".\n\t\tif ( isFunction( form.submit ) ) {\n\t\t\toriginalSubmit = form.submit;\n\n\t\t\tform.submit = () => {\n\t\t\t\tonSubmit();\n\t\t\t\toriginalSubmit.apply( form );\n\t\t\t};\n\t\t}\n\n\t\t// Update the replaced textarea with data before each form#submit event.\n\t\tform.addEventListener( 'submit', onSubmit );\n\n\t\t// Remove the submit listener and revert the original submit method on\n\t\t// editor#destroy.\n\t\teditor.on<EditorDestroyEvent>( 'destroy', () => {\n\t\t\tform.removeEventListener( 'submit', onSubmit );\n\n\t\t\tif ( originalSubmit ) {\n\t\t\t\tform.submit = originalSubmit;\n\t\t\t}\n\t\t} );\n\t}\n}\n\nfunction isTextArea( sourceElement: HTMLElement | undefined ): sourceElement is HTMLTextAreaElement {\n\treturn !!sourceElement && sourceElement.tagName.toLowerCase() === 'textarea';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/clipboardobserver\n */\n\nimport { EventInfo } from '@ckeditor/ckeditor5-utils';\n\nimport {\n\tDataTransfer,\n\tDomEventObserver,\n\ttype DomEventData,\n\ttype View,\n\ttype ViewDocumentFragment,\n\ttype ViewElement,\n\ttype ViewRange\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * Clipboard events observer.\n *\n * Fires the following events:\n *\n * * {@link module:engine/view/document~Document#event:clipboardInput},\n * * {@link module:engine/view/document~Document#event:paste},\n * * {@link module:engine/view/document~Document#event:copy},\n * * {@link module:engine/view/document~Document#event:cut},\n * * {@link module:engine/view/document~Document#event:drop},\n * * {@link module:engine/view/document~Document#event:dragover},\n * * {@link module:engine/view/document~Document#event:dragging},\n * * {@link module:engine/view/document~Document#event:dragstart},\n * * {@link module:engine/view/document~Document#event:dragend},\n * * {@link module:engine/view/document~Document#event:dragenter},\n * * {@link module:engine/view/document~Document#event:dragleave}.\n *\n * **Note**: This observer is not available by default (ckeditor5-engine does not add it on its own).\n * To make it available, it needs to be added to {@link module:engine/view/document~Document} by using\n * the {@link module:engine/view/view~View#addObserver `View#addObserver()`} method. Alternatively, you can load the\n * {@link module:clipboard/clipboard~Clipboard} plugin which adds this observer automatically (because it uses it).\n */\nexport default class ClipboardObserver extends DomEventObserver<\n\t'paste' | 'copy' | 'cut' | 'drop' | 'dragover' | 'dragstart' | 'dragend' | 'dragenter' | 'dragleave',\n\tClipboardEventData\n> {\n\tpublic readonly domEventType = [\n\t\t'paste', 'copy', 'cut', 'drop', 'dragover', 'dragstart', 'dragend', 'dragenter', 'dragleave'\n\t] as const;\n\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tconst viewDocument = this.document;\n\n\t\tthis.listenTo<ViewDocumentPasteEvent>( viewDocument, 'paste', handleInput( 'clipboardInput' ), { priority: 'low' } );\n\t\tthis.listenTo<ViewDocumentDropEvent>( viewDocument, 'drop', handleInput( 'clipboardInput' ), { priority: 'low' } );\n\t\tthis.listenTo<ViewDocumentDragOverEvent>( viewDocument, 'dragover', handleInput( 'dragging' ), { priority: 'low' } );\n\n\t\tfunction handleInput( type: 'clipboardInput' | 'dragging' ) {\n\t\t\treturn ( evt: EventInfo, data: DomEventData & ClipboardEventData ) => {\n\t\t\t\tdata.preventDefault();\n\n\t\t\t\tconst targetRanges = data.dropRange ? [ data.dropRange ] : null;\n\t\t\t\tconst eventInfo = new EventInfo( viewDocument, type );\n\n\t\t\t\tviewDocument.fire( eventInfo, {\n\t\t\t\t\tdataTransfer: data.dataTransfer,\n\t\t\t\t\tmethod: evt.name,\n\t\t\t\t\ttargetRanges,\n\t\t\t\t\ttarget: data.target,\n\t\t\t\t\tdomEvent: data.domEvent\n\t\t\t\t} );\n\n\t\t\t\t// If CKEditor handled the input, do not bubble the original event any further.\n\t\t\t\t// This helps external integrations recognize that fact and act accordingly.\n\t\t\t\t// https://github.com/ckeditor/ckeditor5-upload/issues/92\n\t\t\t\tif ( eventInfo.stop.called ) {\n\t\t\t\t\tdata.stopPropagation();\n\t\t\t\t}\n\t\t\t};\n\t\t}\n\t}\n\n\tpublic onDomEvent( domEvent: ClipboardEvent | DragEvent ): void {\n\t\tconst nativeDataTransfer = 'clipboardData' in domEvent ? domEvent.clipboardData! : domEvent.dataTransfer!;\n\t\tconst cacheFiles = domEvent.type == 'drop' || domEvent.type == 'paste';\n\n\t\tconst evtData: ClipboardEventData = {\n\t\t\tdataTransfer: new DataTransfer( nativeDataTransfer, { cacheFiles } )\n\t\t};\n\n\t\tif ( domEvent.type == 'drop' || domEvent.type == 'dragover' ) {\n\t\t\tevtData.dropRange = getDropViewRange( this.view, domEvent as DragEvent );\n\t\t}\n\n\t\tthis.fire( domEvent.type, domEvent, evtData );\n\t}\n}\n\n/**\n * The data of 'paste', 'copy', 'cut', 'drop', 'dragover', 'dragstart', 'dragend', 'dragenter' and 'dragleave' events.\n */\nexport interface ClipboardEventData {\n\n\t/**\n\t * The data transfer instance.\n\t */\n\tdataTransfer: DataTransfer;\n\n\t/**\n\t * The position into which the content is dropped.\n\t */\n\tdropRange?: ViewRange | null;\n}\n\nfunction getDropViewRange( view: View, domEvent: DragEvent & { rangeParent?: Node; rangeOffset?: number } ) {\n\tconst domDoc = ( domEvent.target as Node ).ownerDocument!;\n\tconst x = domEvent.clientX;\n\tconst y = domEvent.clientY;\n\tlet domRange;\n\n\t// Webkit & Blink.\n\tif ( domDoc.caretRangeFromPoint && domDoc.caretRangeFromPoint( x, y ) ) {\n\t\tdomRange = domDoc.caretRangeFromPoint( x, y );\n\t}\n\t// FF.\n\telse if ( domEvent.rangeParent ) {\n\t\tdomRange = domDoc.createRange();\n\t\tdomRange.setStart( domEvent.rangeParent, domEvent.rangeOffset! );\n\t\tdomRange.collapse( true );\n\t}\n\n\tif ( domRange ) {\n\t\treturn view.domConverter.domRangeToView( domRange );\n\t}\n\n\treturn null;\n}\n\n/**\n * Fired as a continuation of the {@link module:engine/view/document~Document#event:paste} and\n * {@link module:engine/view/document~Document#event:drop} events.\n *\n * It is a part of the {@glink framework/deep-dive/clipboard#input-pipeline clipboard input pipeline}.\n *\n * This event carries a `dataTransfer` object which comes from the clipboard and whose content should be processed\n * and inserted into the editor.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:clipboard/clipboardobserver~ClipboardObserver\n * @see module:clipboard/clipboard~Clipboard\n *\n * @eventName module:engine/view/document~Document#clipboardInput\n * @param data The event data.\n */\nexport type ViewDocumentClipboardInputEvent = {\n\tname: 'clipboardInput';\n\targs: [ data: DomEventData<ClipboardEvent | DragEvent> & ClipboardInputEventData ];\n};\n\n/**\n * The value of the {@link module:engine/view/document~Document#event:paste},\n * {@link module:engine/view/document~Document#event:copy} and {@link module:engine/view/document~Document#event:cut} events.\n *\n * In order to access the clipboard data, use the `dataTransfer` property.\n */\nexport interface ClipboardInputEventData {\n\n\t/**\n\t * Data transfer instance.\n\t */\n\tdataTransfer: DataTransfer;\n\n\t/**\n\t * Whether the event was triggered by a paste or a drop operation.\n\t */\n\tmethod: 'paste' | 'drop';\n\n\t/**\n\t * The tree view element representing the target.\n\t */\n\ttarget: ViewElement;\n\n\t/**\n\t * The ranges which are the target of the operation (usually into which the content should be inserted).\n\t * If the clipboard input was triggered by a paste operation, this property is not set. If by a drop operation,\n\t * then it is the drop position (which can be different than the selection at the moment of the drop).\n\t */\n\ttargetRanges: Array<ViewRange> | null;\n\n\t/**\n\t * The content of clipboard input.\n\t */\n\tcontent?: ViewDocumentFragment;\n}\n\n/**\n * Fired when the user drags the content over one of the editing roots of the editor.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:engine/view/document~Document#event:clipboardInput\n *\n * @eventName module:engine/view/document~Document#dragover\n * @param data The event data.\n */\nexport type ViewDocumentDragOverEvent = {\n\tname: 'dragover';\n\targs: [ data: DomEventData<DragEvent> & ClipboardEventData ];\n};\n\n/**\n * Fired when the user dropped the content into one of the editing roots of the editor.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:engine/view/document~Document#event:clipboardInput\n *\n * @eventName module:engine/view/document~Document#drop\n * @param data The event data.\n */\nexport type ViewDocumentDropEvent = {\n\tname: 'drop';\n\targs: [ data: DomEventData<DragEvent> & ClipboardEventData ];\n};\n\n/**\n * Fired when the user pasted the content into one of the editing roots of the editor.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:engine/view/document~Document#event:clipboardInput\n *\n * @eventName module:engine/view/document~Document#paste\n * @param {module:clipboard/clipboardobserver~ClipboardEventData} data The event data.\n */\nexport type ViewDocumentPasteEvent = {\n\tname: 'paste';\n\targs: [ data: DomEventData<ClipboardEvent> & ClipboardEventData ];\n};\n\n/**\n * Fired when the user copied the content from one of the editing roots of the editor.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:clipboard/clipboardobserver~ClipboardObserver\n *\n * @eventName module:engine/view/document~Document#copy\n * @param data The event data.\n */\nexport type ViewDocumentCopyEvent = {\n\tname: 'copy';\n\targs: [ data: DomEventData<ClipboardEvent> & ClipboardEventData ];\n};\n\n/**\n * Fired when the user cut the content from one of the editing roots of the editor.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:clipboard/clipboardobserver~ClipboardObserver\n *\n * @eventName module:engine/view/document~Document#cut\n * @param data The event data.\n */\nexport type ViewDocumentCutEvent = {\n\tname: 'cut';\n\targs: [ data: DomEventData<ClipboardEvent> & ClipboardEventData ];\n};\n\n/**\n * Fired as a continuation of the {@link module:engine/view/document~Document#event:dragover} event.\n *\n * It is a part of the {@glink framework/deep-dive/clipboard#input-pipeline clipboard input pipeline}.\n *\n * This event carries a `dataTransfer` object which comes from the clipboard and whose content should be processed\n * and inserted into the editor.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:clipboard/clipboardobserver~ClipboardObserver\n * @see module:clipboard/clipboard~Clipboard\n *\n * @eventName module:engine/view/document~Document#dragging\n * @param data The event data.\n */\nexport type ViewDocumentDraggingEvent = {\n\tname: 'dragging';\n\targs: [ data: DomEventData<DragEvent> & DraggingEventData ];\n};\n\nexport interface DraggingEventData {\n\n\t/**\n\t * The data transfer instance.\n\t */\n\tdataTransfer: DataTransfer;\n\n\t/**\n\t * Whether the event was triggered by a paste or a drop operation.\n\t */\n\tmethod: 'dragover';\n\n\t/**\n\t * The tree view element representing the target.\n\t */\n\ttarget: Element;\n\n\t/**\n\t * Ranges which are the target of the operation (usually into which the content should be inserted).\n\t * It is the drop position (which can be different than the selection at the moment of drop).\n\t */\n\ttargetRanges: Array<ViewRange> | null;\n}\n\n/**\n * Fired when the user starts dragging the content in one of the editing roots of the editor.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:engine/view/document~Document#event:clipboardInput\n *\n * @eventName module:engine/view/document~Document#dragstart\n * @param data The event data.\n */\nexport type ViewDocumentDragStartEvent = {\n\tname: 'dragstart';\n\targs: [ data: DomEventData<DragEvent> & ClipboardEventData ];\n};\n\n/**\n * Fired when the user ended dragging the content.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:engine/view/document~Document#event:clipboardInput\n *\n * @eventName module:engine/view/document~Document#dragend\n * @param data The event data.\n */\nexport type ViewDocumentDragEndEvent = {\n\tname: 'dragend';\n\targs: [ data: DomEventData<DragEvent> & ClipboardEventData ];\n};\n\n/**\n * Fired when the user drags the content into one of the editing roots of the editor.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:engine/view/document~Document#event:clipboardInput\n *\n * @eventName module:engine/view/document~Document#dragenter\n * @param data The event data.\n */\nexport type ViewDocumentDragEnterEvent = {\n\tname: 'dragenter';\n\targs: [ data: DomEventData<DragEvent> & ClipboardEventData ];\n};\n\n/**\n * Fired when the user drags the content out of one of the editing roots of the editor.\n *\n * Introduced by {@link module:clipboard/clipboardobserver~ClipboardObserver}.\n *\n * **Note**: This event is not available by default. To make it available, {@link module:clipboard/clipboardobserver~ClipboardObserver}\n * needs to be added to the {@link module:engine/view/document~Document} by using the {@link module:engine/view/view~View#addObserver}\n * method. This is usually done by the {@link module:clipboard/clipboard~Clipboard} plugin, but if for some reason it is not loaded,\n * the observer must be added manually.\n *\n * @see module:engine/view/document~Document#event:clipboardInput\n *\n * @eventName module:engine/view/document~Document#dragleave\n * @param data The event data.\n */\nexport type ViewDocumentDragLeaveEvent = {\n\tname: 'dragleave';\n\targs: [ data: DomEventData<DragEvent> & ClipboardEventData ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/utils/viewtoplaintext\n */\n\nimport type { ViewDocumentFragment, ViewElement, ViewItem } from '@ckeditor/ckeditor5-engine';\n\n// Elements which should not have empty-line padding.\n// Most `view.ContainerElement` want to be separate by new-line, but some are creating one structure\n// together (like `<li>`) so it is better to separate them by only one \"\\n\".\nconst smallPaddingElements = [ 'figcaption', 'li' ];\n\n/**\n * Converts {@link module:engine/view/item~Item view item} and all of its children to plain text.\n *\n * @param viewItem View item to convert.\n * @returns Plain text representation of `viewItem`.\n */\nexport default function viewToPlainText( viewItem: ViewItem | ViewDocumentFragment ): string {\n\tlet text = '';\n\n\tif ( viewItem.is( '$text' ) || viewItem.is( '$textProxy' ) ) {\n\t\t// If item is `Text` or `TextProxy` simple take its text data.\n\t\ttext = viewItem.data;\n\t} else if ( viewItem.is( 'element', 'img' ) && viewItem.hasAttribute( 'alt' ) ) {\n\t\t// Special case for images - use alt attribute if it is provided.\n\t\ttext = viewItem.getAttribute( 'alt' )!;\n\t} else if ( viewItem.is( 'element', 'br' ) ) {\n\t\t// A soft break should be converted into a single line break (#8045).\n\t\ttext = '\\n';\n\t} else {\n\t\t// Other elements are document fragments, attribute elements or container elements.\n\t\t// They don't have their own text value, so convert their children.\n\t\tlet prev = null;\n\n\t\tfor ( const child of ( viewItem as ViewElement | ViewDocumentFragment ).getChildren() ) {\n\t\t\tconst childText = viewToPlainText( child );\n\n\t\t\t// Separate container element children with one or more new-line characters.\n\t\t\tif ( prev && ( prev.is( 'containerElement' ) || child.is( 'containerElement' ) ) ) {\n\t\t\t\tif (\n\t\t\t\t\tsmallPaddingElements.includes( ( prev as ViewElement ).name ) ||\n\t\t\t\t\tsmallPaddingElements.includes( ( child as ViewElement ).name )\n\t\t\t\t) {\n\t\t\t\t\ttext += '\\n';\n\t\t\t\t} else {\n\t\t\t\t\ttext += '\\n\\n';\n\t\t\t\t}\n\t\t\t}\n\n\t\t\ttext += childText;\n\t\t\tprev = child;\n\t\t}\n\t}\n\n\treturn text;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/clipboardpipeline\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\n\nimport { EventInfo } from '@ckeditor/ckeditor5-utils';\n\nimport type {\n\tDataTransfer,\n\tDocumentFragment,\n\tDomEventData,\n\tRange,\n\tViewDocumentFragment,\n\tViewRange\n} from '@ckeditor/ckeditor5-engine';\n\nimport ClipboardObserver, {\n\ttype ClipboardEventData,\n\ttype ViewDocumentCopyEvent,\n\ttype ViewDocumentCutEvent,\n\ttype ViewDocumentClipboardInputEvent\n} from './clipboardobserver';\n\nimport plainTextToHtml from './utils/plaintexttohtml';\nimport normalizeClipboardHtml from './utils/normalizeclipboarddata';\nimport viewToPlainText from './utils/viewtoplaintext';\n\n// Input pipeline events overview:\n//\n// ┌──────────────────────┐ ┌──────────────────────┐\n// │ view.Document │ │ view.Document │\n// │ paste │ │ drop │\n// └───────────┬──────────┘ └───────────┬──────────┘\n// │ │\n// └────────────────┌────────────────┘\n// │\n// ┌─────────V────────┐\n// │ view.Document │ Retrieves text/html or text/plain from data.dataTransfer\n// │ clipboardInput │ and processes it to view.DocumentFragment.\n// └─────────┬────────┘\n// │\n// ┌───────────V───────────┐\n// │ ClipboardPipeline │ Converts view.DocumentFragment to model.DocumentFragment.\n// │ inputTransformation │\n// └───────────┬───────────┘\n// │\n// ┌──────────V──────────┐\n// │ ClipboardPipeline │ Calls model.insertContent().\n// │ contentInsertion │\n// └─────────────────────┘\n//\n//\n// Output pipeline events overview:\n//\n// ┌──────────────────────┐ ┌──────────────────────┐\n// │ view.Document │ │ view.Document │ Retrieves the selected model.DocumentFragment\n// │ copy │ │ cut │ and converts it to view.DocumentFragment.\n// └───────────┬──────────┘ └───────────┬──────────┘\n// │ │\n// └────────────────┌────────────────┘\n// │\n// ┌─────────V────────┐\n// │ view.Document │ Processes view.DocumentFragment to text/html and text/plain\n// │ clipboardOutput │ and stores the results in data.dataTransfer.\n// └──────────────────┘\n//\n\n/**\n * The clipboard pipeline feature. It is responsible for intercepting the `paste` and `drop` events and\n * passing the pasted content through a series of events in order to insert it into the editor's content.\n * It also handles the `cut` and `copy` events to fill the native clipboard with the serialized editor's data.\n *\n * # Input pipeline\n *\n * The behavior of the default handlers (all at a `low` priority):\n *\n * ## Event: `paste` or `drop`\n *\n * 1. Translates the event data.\n * 2. Fires the {@link module:engine/view/document~Document#event:clipboardInput `view.Document#clipboardInput`} event.\n *\n * ## Event: `view.Document#clipboardInput`\n *\n * 1. If the `data.content` event field is already set (by some listener on a higher priority), it takes this content and fires the event\n * from the last point.\n * 2. Otherwise, it retrieves `text/html` or `text/plain` from `data.dataTransfer`.\n * 3. Normalizes the raw data by applying simple filters on string data.\n * 4. Processes the raw data to {@link module:engine/view/documentfragment~DocumentFragment `view.DocumentFragment`} with the\n * {@link module:engine/controller/datacontroller~DataController#htmlProcessor `DataController#htmlProcessor`}.\n * 5. Fires the {@link module:clipboard/clipboardpipeline~ClipboardPipeline#event:inputTransformation\n * `ClipboardPipeline#inputTransformation`} event with the view document fragment in the `data.content` event field.\n *\n * ## Event: `ClipboardPipeline#inputTransformation`\n *\n * 1. Converts {@link module:engine/view/documentfragment~DocumentFragment `view.DocumentFragment`} from the `data.content` field to\n * {@link module:engine/model/documentfragment~DocumentFragment `model.DocumentFragment`}.\n * 2. Fires the {@link module:clipboard/clipboardpipeline~ClipboardPipeline#event:contentInsertion `ClipboardPipeline#contentInsertion`}\n * event with the model document fragment in the `data.content` event field.\n * **Note**: The `ClipboardPipeline#contentInsertion` event is fired within a model change block to allow other handlers\n * to run in the same block without post-fixers called in between (i.e., the selection post-fixer).\n *\n * ## Event: `ClipboardPipeline#contentInsertion`\n *\n * 1. Calls {@link module:engine/model/model~Model#insertContent `model.insertContent()`} to insert `data.content`\n * at the current selection position.\n *\n * # Output pipeline\n *\n * The behavior of the default handlers (all at a `low` priority):\n *\n * ## Event: `copy`, `cut` or `dragstart`\n *\n * 1. Retrieves the selected {@link module:engine/model/documentfragment~DocumentFragment `model.DocumentFragment`} by calling\n * {@link module:engine/model/model~Model#getSelectedContent `model#getSelectedContent()`}.\n * 2. Converts the model document fragment to {@link module:engine/view/documentfragment~DocumentFragment `view.DocumentFragment`}.\n * 3. Fires the {@link module:engine/view/document~Document#event:clipboardOutput `view.Document#clipboardOutput`} event\n * with the view document fragment in the `data.content` event field.\n *\n * ## Event: `view.Document#clipboardOutput`\n *\n * 1. Processes `data.content` to HTML and plain text with the\n * {@link module:engine/controller/datacontroller~DataController#htmlProcessor `DataController#htmlProcessor`}.\n * 2. Updates the `data.dataTransfer` data for `text/html` and `text/plain` with the processed data.\n * 3. For the `cut` method, calls {@link module:engine/model/model~Model#deleteContent `model.deleteContent()`}\n * on the current selection.\n *\n * Read more about the clipboard integration in the {@glink framework/deep-dive/clipboard clipboard deep-dive} guide.\n */\nexport default class ClipboardPipeline extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ClipboardPipeline' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\n\t\tview.addObserver( ClipboardObserver );\n\n\t\tthis._setupPasteDrop();\n\t\tthis._setupCopyCut();\n\t}\n\n\t/**\n\t * The clipboard paste pipeline.\n\t */\n\tprivate _setupPasteDrop(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\t// Pasting is disabled when selection is in non-editable place.\n\t\t// Dropping is disabled in drag and drop handler.\n\t\tthis.listenTo<ViewDocumentClipboardInputEvent>( viewDocument, 'clipboardInput', ( evt, data ) => {\n\t\t\tif ( data.method == 'paste' && !editor.model.canEditAt( editor.model.document.selection ) ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { priority: 'highest' } );\n\n\t\tthis.listenTo<ViewDocumentClipboardInputEvent>( viewDocument, 'clipboardInput', ( evt, data ) => {\n\t\t\tconst dataTransfer = data.dataTransfer;\n\t\t\tlet content: ViewDocumentFragment;\n\n\t\t\t// Some feature could already inject content in the higher priority event handler (i.e., codeBlock).\n\t\t\tif ( data.content ) {\n\t\t\t\tcontent = data.content;\n\t\t\t} else {\n\t\t\t\tlet contentData = '';\n\n\t\t\t\tif ( dataTransfer.getData( 'text/html' ) ) {\n\t\t\t\t\tcontentData = normalizeClipboardHtml( dataTransfer.getData( 'text/html' ) );\n\t\t\t\t} else if ( dataTransfer.getData( 'text/plain' ) ) {\n\t\t\t\t\tcontentData = plainTextToHtml( dataTransfer.getData( 'text/plain' ) );\n\t\t\t\t}\n\n\t\t\t\tcontent = this.editor.data.htmlProcessor.toView( contentData );\n\t\t\t}\n\n\t\t\tconst eventInfo = new EventInfo( this, 'inputTransformation' );\n\n\t\t\tthis.fire<ClipboardInputTransformationEvent>( eventInfo, {\n\t\t\t\tcontent,\n\t\t\t\tdataTransfer,\n\t\t\t\ttargetRanges: data.targetRanges,\n\t\t\t\tmethod: data.method as 'paste' | 'drop'\n\t\t\t} );\n\n\t\t\t// If CKEditor handled the input, do not bubble the original event any further.\n\t\t\t// This helps external integrations recognize this fact and act accordingly.\n\t\t\t// https://github.com/ckeditor/ckeditor5-upload/issues/92\n\t\t\tif ( eventInfo.stop.called ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\n\t\t\tview.scrollToTheSelection();\n\t\t}, { priority: 'low' } );\n\n\t\tthis.listenTo<ClipboardInputTransformationEvent>( this, 'inputTransformation', ( evt, data ) => {\n\t\t\tif ( data.content.isEmpty ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst dataController = this.editor.data;\n\n\t\t\t// Convert the pasted content into a model document fragment.\n\t\t\t// The conversion is contextual, but in this case an \"all allowed\" context is needed\n\t\t\t// and for that we use the $clipboardHolder item.\n\t\t\tconst modelFragment = dataController.toModel( data.content, '$clipboardHolder' );\n\n\t\t\tif ( modelFragment.childCount == 0 ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tevt.stop();\n\n\t\t\t// Fire content insertion event in a single change block to allow other handlers to run in the same block\n\t\t\t// without post-fixers called in between (i.e., the selection post-fixer).\n\t\t\tmodel.change( () => {\n\t\t\t\tthis.fire<ClipboardContentInsertionEvent>( 'contentInsertion', {\n\t\t\t\t\tcontent: modelFragment,\n\t\t\t\t\tmethod: data.method,\n\t\t\t\t\tdataTransfer: data.dataTransfer,\n\t\t\t\t\ttargetRanges: data.targetRanges\n\t\t\t\t} );\n\t\t\t} );\n\t\t}, { priority: 'low' } );\n\n\t\tthis.listenTo<ClipboardContentInsertionEvent>( this, 'contentInsertion', ( evt, data ) => {\n\t\t\tdata.resultRange = model.insertContent( data.content );\n\t\t}, { priority: 'low' } );\n\t}\n\n\t/**\n\t * The clipboard copy/cut pipeline.\n\t */\n\tprivate _setupCopyCut(): void {\n\t\tconst editor = this.editor;\n\t\tconst modelDocument = editor.model.document;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\tconst onCopyCut = ( evt: EventInfo<'copy' | 'cut'>, data: DomEventData<ClipboardEvent> & ClipboardEventData ) => {\n\t\t\tconst dataTransfer = data.dataTransfer;\n\n\t\t\tdata.preventDefault();\n\n\t\t\tconst content = editor.data.toView( editor.model.getSelectedContent( modelDocument.selection ) );\n\n\t\t\tviewDocument.fire<ViewDocumentClipboardOutputEvent>( 'clipboardOutput', {\n\t\t\t\tdataTransfer,\n\t\t\t\tcontent,\n\t\t\t\tmethod: evt.name\n\t\t\t} );\n\t\t};\n\n\t\tthis.listenTo<ViewDocumentCopyEvent>( viewDocument, 'copy', onCopyCut, { priority: 'low' } );\n\t\tthis.listenTo<ViewDocumentCutEvent>( viewDocument, 'cut', ( evt, data ) => {\n\t\t\t// Cutting is disabled when selection is in non-editable place.\n\t\t\t// See: https://github.com/ckeditor/ckeditor5-clipboard/issues/26.\n\t\t\tif ( !editor.model.canEditAt( editor.model.document.selection ) ) {\n\t\t\t\tdata.preventDefault();\n\t\t\t} else {\n\t\t\t\tonCopyCut( evt, data );\n\t\t\t}\n\t\t}, { priority: 'low' } );\n\n\t\tthis.listenTo<ViewDocumentClipboardOutputEvent>( viewDocument, 'clipboardOutput', ( evt, data ) => {\n\t\t\tif ( !data.content.isEmpty ) {\n\t\t\t\tdata.dataTransfer.setData( 'text/html', this.editor.data.htmlProcessor.toData( data.content ) );\n\t\t\t\tdata.dataTransfer.setData( 'text/plain', viewToPlainText( data.content ) );\n\t\t\t}\n\n\t\t\tif ( data.method == 'cut' ) {\n\t\t\t\teditor.model.deleteContent( modelDocument.selection );\n\t\t\t}\n\t\t}, { priority: 'low' } );\n\t}\n}\n\n/**\n * Fired with the `content`, `dataTransfer`, `method`, and `targetRanges` properties:\n *\n * * The `content` which comes from the clipboard (it was pasted or dropped) should be processed in order to be inserted into the editor.\n * * The `dataTransfer` object is available in case the transformation functions need access to the raw clipboard data.\n * * The `method` indicates the original DOM event (for example `'drop'` or `'paste'`).\n * * The `targetRanges` property is an array of view ranges (it is available only for `'drop'`).\n *\n * It is a part of the {@glink framework/deep-dive/clipboard#input-pipeline clipboard input pipeline}.\n *\n * **Note**: You should not stop this event if you want to change the input data. You should modify the `content` property instead.\n *\n * @see module:clipboard/clipboardobserver~ClipboardObserver\n * @see module:clipboard/clipboardpipeline~ClipboardPipeline\n *\n * @eventName ~ClipboardPipeline#inputTransformation\n * @param data The event data.\n */\nexport type ClipboardInputTransformationEvent = {\n\tname: 'inputTransformation';\n\targs: [ data: ClipboardInputTransformationData ];\n};\n\n/**\n * The data of 'inputTransformation' event.\n */\nexport interface ClipboardInputTransformationData {\n\n\t/**\n\t * The event data.\n\t * The content to be inserted into the editor. It can be modified by event listeners. Read more about the clipboard pipelines in\n\t * the {@glink framework/deep-dive/clipboard clipboard deep-dive} guide.\n\t */\n\tcontent: ViewDocumentFragment;\n\n\t/**\n\t * The data transfer instance.\n\t */\n\tdataTransfer: DataTransfer;\n\n\t/**\n\t * The target drop ranges.\n\t */\n\ttargetRanges: Array<ViewRange> | null;\n\n\t/**\n\t * Whether the event was triggered by a paste or a drop operation.\n\t */\n\tmethod: 'paste' | 'drop';\n}\n\n/**\n * Fired with the `content`, `dataTransfer`, `method`, and `targetRanges` properties:\n *\n * * The `content` which comes from the clipboard (was pasted or dropped) should be processed in order to be inserted into the editor.\n * * The `dataTransfer` object is available in case the transformation functions need access to the raw clipboard data.\n * * The `method` indicates the original DOM event (for example `'drop'` or `'paste'`).\n * * The `targetRanges` property is an array of view ranges (it is available only for `'drop'`).\n *\n * Event handlers can modify the content according to the final insertion position.\n *\n * It is a part of the {@glink framework/deep-dive/clipboard#input-pipeline clipboard input pipeline}.\n *\n * **Note**: You should not stop this event if you want to change the input data. You should modify the `content` property instead.\n *\n * @see module:clipboard/clipboardobserver~ClipboardObserver\n * @see module:clipboard/clipboardpipeline~ClipboardPipeline\n * @see module:clipboard/clipboardpipeline~ClipboardPipeline#event:inputTransformation\n *\n * @eventName ~ClipboardPipeline#contentInsertion\n * @param data The event data.\n */\nexport type ClipboardContentInsertionEvent = {\n\tname: 'contentInsertion';\n\targs: [ data: ClipboardContentInsertionData ];\n};\n\n/**\n * The data of 'contentInsertion' event.\n */\nexport interface ClipboardContentInsertionData {\n\n\t/**\n\t * The content to be inserted into the editor.\n\t * Read more about the clipboard pipelines in the {@glink framework/deep-dive/clipboard clipboard deep-dive} guide.\n\t */\n\tcontent: DocumentFragment;\n\n\t/**\n\t * Whether the event was triggered by a paste or a drop operation.\n\t */\n\tmethod: 'paste' | 'drop';\n\n\t/**\n\t * The data transfer instance.\n\t */\n\n\tdataTransfer: DataTransfer;\n\n\t/**\n\t * The target drop ranges.\n\t */\n\ttargetRanges: Array<ViewRange> | null;\n\n\t/**\n\t * The result of the `model.insertContent()` call\n\t * (inserted by the event handler at a low priority).\n\t */\n\tresultRange?: Range;\n}\n\n/**\n * Fired on {@link module:engine/view/document~Document#event:copy} and {@link module:engine/view/document~Document#event:cut}\n * with a copy of the selected content. The content can be processed before it ends up in the clipboard.\n *\n * It is a part of the {@glink framework/deep-dive/clipboard#output-pipeline clipboard output pipeline}.\n *\n * @see module:clipboard/clipboardobserver~ClipboardObserver\n * @see module:clipboard/clipboardpipeline~ClipboardPipeline\n *\n * @eventName module:engine/view/document~Document#clipboardOutput\n * @param data The event data.\n */\nexport type ViewDocumentClipboardOutputEvent = {\n\tname: 'clipboardOutput';\n\targs: [ data: ViewDocumentClipboardOutputEventData ];\n};\n\n/**\n * The value of the 'clipboardOutput' event.\n */\nexport interface ViewDocumentClipboardOutputEventData {\n\n\t/**\n\t * The data transfer instance.\n\t *\n\t * @readonly\n\t */\n\tdataTransfer: DataTransfer;\n\n\t/**\n\t * Content to be put into the clipboard. It can be modified by the event listeners.\n\t * Read more about the clipboard pipelines in the {@glink framework/deep-dive/clipboard clipboard deep-dive} guide.\n\t */\n\tcontent: ViewDocumentFragment;\n\n\t/**\n\t * Whether the event was triggered by a copy or cut operation.\n\t */\n\tmethod: 'copy' | 'cut' | 'dragstart';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/utils/normalizeclipboarddata\n */\n\n/**\n * Removes some popular browser quirks out of the clipboard data (HTML).\n * Removes all HTML comments. These are considered an internal thing and it makes little sense if they leak into the editor data.\n *\n * @param data The HTML data to normalize.\n * @returns Normalized HTML.\n */\nexport default function normalizeClipboardData( data: string ): string {\n\treturn data\n\t\t.replace( /<span(?: class=\"Apple-converted-space\"|)>(\\s+)<\\/span>/g, ( fullMatch, spaces ) => {\n\t\t\t// Handle the most popular and problematic case when even a single space becomes an nbsp;.\n\t\t\t// Decode those to normal spaces. Read more in https://github.com/ckeditor/ckeditor5-clipboard/issues/2.\n\t\t\tif ( spaces.length == 1 ) {\n\t\t\t\treturn ' ';\n\t\t\t}\n\n\t\t\treturn spaces;\n\t\t} )\n\t\t// Remove all HTML comments.\n\t\t.replace( /<!--[\\s\\S]*?-->/g, '' );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/utils/plaintexttohtml\n */\n\n/**\n * Converts plain text to its HTML-ized version.\n *\n * @param text The plain text to convert.\n * @returns HTML generated from the plain text.\n */\nexport default function plainTextToHtml( text: string ): string {\n\ttext = text\n\t\t// Encode <>.\n\t\t.replace( /</g, '&lt;' )\n\t\t.replace( />/g, '&gt;' )\n\t\t// Creates a paragraph for each double line break.\n\t\t.replace( /\\r?\\n\\r?\\n/g, '</p><p>' )\n\t\t// Creates a line break for each single line break.\n\t\t.replace( /\\r?\\n/g, '<br>' )\n\t\t// Replace tabs with four spaces.\n\t\t.replace( /\\t/g, '&nbsp;&nbsp;&nbsp;&nbsp;' )\n\t\t// Preserve trailing spaces (only the first and last one the rest is handled below).\n\t\t.replace( /^\\s/, '&nbsp;' )\n\t\t.replace( /\\s$/, '&nbsp;' )\n\t\t// Preserve other subsequent spaces now.\n\t\t.replace( /\\s\\s/g, ' &nbsp;' );\n\n\tif ( text.includes( '</p><p>' ) || text.includes( '<br>' ) ) {\n\t\t// If we created paragraphs above, add the trailing ones.\n\t\ttext = `<p>${ text }</p>`;\n\t}\n\n\t// TODO:\n\t// * What about '\\nfoo' vs ' foo'?\n\n\treturn text;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/utils/changebuffer\n */\n\nimport type {\n\tModel,\n\tBatch,\n\tDocumentChangeEvent,\n\tDocumentSelectionChangeEvent\n} from '@ckeditor/ckeditor5-engine';\n\nimport type { EventInfo } from '@ckeditor/ckeditor5-utils';\n\n/**\n * Change buffer allows to group atomic changes (like characters that have been typed) into\n * {@link module:engine/model/batch~Batch batches}.\n *\n * Batches represent single undo steps, hence changes added to one single batch are undone together.\n *\n * The buffer has a configurable limit of atomic changes that it can accommodate. After the limit was\n * exceeded (see {@link ~ChangeBuffer#input}), a new batch is created in {@link ~ChangeBuffer#batch}.\n *\n * To use the change buffer you need to let it know about the number of changes that were added to the batch:\n *\n * ```ts\n * const buffer = new ChangeBuffer( model, LIMIT );\n *\n * // Later on in your feature:\n * buffer.batch.insert( pos, insertedCharacters );\n * buffer.input( insertedCharacters.length );\n * ```\n */\nexport default class ChangeBuffer {\n\t/**\n\t * The model instance.\n\t */\n\tpublic readonly model: Model;\n\n\t/**\n\t * The maximum number of atomic changes which can be contained in one batch.\n\t */\n\tpublic readonly limit: number;\n\n\t/**\n\t * Whether the buffer is locked. A locked buffer cannot be reset unless it gets unlocked.\n\t */\n\tprivate _isLocked: boolean;\n\n\t/**\n\t * The number of atomic changes in the buffer. Once it exceeds the {@link #limit},\n\t * the {@link #batch batch} is set to a new one.\n\t */\n\tprivate _size: number;\n\n\t/**\n\t * The current batch instance.\n\t */\n\tprivate _batch: Batch | null = null;\n\n\t/**\n\t * The callback to document the change event which later needs to be removed.\n\t */\n\tprivate readonly _changeCallback: ( evt: EventInfo, batch: Batch ) => void;\n\n\t/**\n\t * The callback to document selection `change:attribute` and `change:range` events which resets the buffer.\n\t */\n\tprivate readonly _selectionChangeCallback: () => void;\n\n\t/**\n\t * Creates a new instance of the change buffer.\n\t *\n\t * @param limit The maximum number of atomic changes which can be contained in one batch.\n\t */\n\tconstructor( model: Model, limit: number = 20 ) {\n\t\tthis.model = model;\n\t\tthis._size = 0;\n\t\tthis.limit = limit;\n\t\tthis._isLocked = false;\n\n\t\t// The function to be called in order to notify the buffer about batches which appeared in the document.\n\t\t// The callback will check whether it is a new batch and in that case the buffer will be flushed.\n\t\t//\n\t\t// The reason why the buffer needs to be flushed whenever a new batch appears is that the changes added afterwards\n\t\t// should be added to a new batch. For instance, when the user types, then inserts an image, and then types again,\n\t\t// the characters typed after inserting the image should be added to a different batch than the characters typed before.\n\t\tthis._changeCallback = ( evt, batch ) => {\n\t\t\tif ( batch.isLocal && batch.isUndoable && batch !== this._batch ) {\n\t\t\t\tthis._reset( true );\n\t\t\t}\n\t\t};\n\n\t\tthis._selectionChangeCallback = () => {\n\t\t\tthis._reset();\n\t\t};\n\n\t\tthis.model.document.on<DocumentChangeEvent>( 'change', this._changeCallback );\n\n\t\tthis.model.document.selection.on<DocumentSelectionChangeEvent>( 'change:range', this._selectionChangeCallback );\n\t\tthis.model.document.selection.on<DocumentSelectionChangeEvent>( 'change:attribute', this._selectionChangeCallback );\n\t}\n\n\t/**\n\t * The current batch to which a feature should add its operations. Once the {@link #size}\n\t * is reached or exceeds the {@link #limit}, the batch is set to a new instance and the size is reset.\n\t */\n\tpublic get batch(): Batch {\n\t\tif ( !this._batch ) {\n\t\t\tthis._batch = this.model.createBatch( { isTyping: true } );\n\t\t}\n\n\t\treturn this._batch;\n\t}\n\n\t/**\n\t * The number of atomic changes in the buffer. Once it exceeds the {@link #limit},\n\t * the {@link #batch batch} is set to a new one.\n\t */\n\tpublic get size(): number {\n\t\treturn this._size;\n\t}\n\n\t/**\n\t * The input number of changes into the buffer. Once the {@link #size} is\n\t * reached or exceeds the {@link #limit}, the batch is set to a new instance and the size is reset.\n\t *\n\t * @param changeCount The number of atomic changes to input.\n\t */\n\tpublic input( changeCount: number ): void {\n\t\tthis._size += changeCount;\n\n\t\tif ( this._size >= this.limit ) {\n\t\t\tthis._reset( true );\n\t\t}\n\t}\n\n\t/**\n\t * Whether the buffer is locked. A locked buffer cannot be reset unless it gets unlocked.\n\t */\n\tpublic get isLocked(): boolean {\n\t\treturn this._isLocked;\n\t}\n\n\t/**\n\t * Locks the buffer.\n\t */\n\tpublic lock(): void {\n\t\tthis._isLocked = true;\n\t}\n\n\t/**\n\t * Unlocks the buffer.\n\t */\n\tpublic unlock(): void {\n\t\tthis._isLocked = false;\n\t}\n\n\t/**\n\t * Destroys the buffer.\n\t */\n\tpublic destroy(): void {\n\t\tthis.model.document.off( 'change', this._changeCallback );\n\t\tthis.model.document.selection.off( 'change:range', this._selectionChangeCallback );\n\t\tthis.model.document.selection.off( 'change:attribute', this._selectionChangeCallback );\n\t}\n\n\t/**\n\t * Resets the change buffer.\n\t *\n\t * @param ignoreLock Whether internal lock {@link #isLocked} should be ignored.\n\t */\n\tprivate _reset( ignoreLock: boolean = false ): void {\n\t\tif ( !this.isLocked || ignoreLock ) {\n\t\t\tthis._batch = null;\n\t\t\tthis._size = 0;\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/inserttextcommand\n */\n\nimport { Command, type Editor } from '@ckeditor/ckeditor5-core';\n\nimport ChangeBuffer from './utils/changebuffer';\n\nimport type { DocumentSelection, Range, Selection } from '@ckeditor/ckeditor5-engine';\n\n/**\n * The insert text command. Used by the {@link module:typing/input~Input input feature} to handle typing.\n */\nexport default class InsertTextCommand extends Command {\n\t/**\n\t * Typing's change buffer used to group subsequent changes into batches.\n\t */\n\tprivate readonly _buffer: ChangeBuffer;\n\n\t/**\n\t * Creates an instance of the command.\n\t *\n\t * @param undoStepSize The maximum number of atomic changes\n\t * which can be contained in one batch in the command buffer.\n\t */\n\tconstructor( editor: Editor, undoStepSize: number ) {\n\t\tsuper( editor );\n\n\t\tthis._buffer = new ChangeBuffer( editor.model, undoStepSize );\n\n\t\t// Since this command may execute on different selectable than selection, it should be checked directly in execute block.\n\t\tthis._isEnabledBasedOnSelection = false;\n\t}\n\n\t/**\n\t * The current change buffer.\n\t */\n\tpublic get buffer(): ChangeBuffer {\n\t\treturn this._buffer;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis._buffer.destroy();\n\t}\n\n\t/**\n\t * Executes the input command. It replaces the content within the given range with the given text.\n\t * Replacing is a two step process, first the content within the range is removed and then the new text is inserted\n\t * at the beginning of the range (which after the removal is a collapsed range).\n\t *\n\t * @fires execute\n\t * @param options The command options.\n\t */\n\tpublic override execute( options: InsertTextCommandOptions = {} ): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\t\tconst text = options.text || '';\n\t\tconst textInsertions = text.length;\n\n\t\tlet selection: Selection | DocumentSelection = doc.selection;\n\n\t\tif ( options.selection ) {\n\t\t\tselection = options.selection;\n\t\t} else if ( options.range ) {\n\t\t\tselection = model.createSelection( options.range );\n\t\t}\n\n\t\t// Stop executing if selectable is in non-editable place.\n\t\tif ( !model.canEditAt( selection ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst resultRange = options.resultRange;\n\n\t\tmodel.enqueueChange( this._buffer.batch, writer => {\n\t\t\tthis._buffer.lock();\n\n\t\t\tmodel.deleteContent( selection );\n\n\t\t\tif ( text ) {\n\t\t\t\tmodel.insertContent( writer.createText( text, doc.selection.getAttributes() ), selection );\n\t\t\t}\n\n\t\t\tif ( resultRange ) {\n\t\t\t\twriter.setSelection( resultRange );\n\t\t\t} else if ( !selection.is( 'documentSelection' ) ) {\n\t\t\t\twriter.setSelection( selection );\n\t\t\t}\n\n\t\t\tthis._buffer.unlock();\n\n\t\t\tthis._buffer.input( textInsertions );\n\t\t} );\n\t}\n}\n\n/**\n * Interface with parameters for executing InsertTextCommand.\n *\n * Both `range` and `selection` parameters are used for defining selection but should not be used together.\n * If both are defined, only `selection` will be considered.\n */\nexport interface InsertTextCommandOptions {\n\n\t/**\n\t * The text to be inserted.\n\t */\n\ttext?: string;\n\n\t/**\n\t * The selection in which the text is inserted.\n\t * Inserting a text into a selection deletes the current content within selection ranges. If the selection is not specified,\n\t * the current selection in the model will be used instead.\n\t */\n\tselection?: Selection | DocumentSelection;\n\n\t/**\n\t * The range in which the text is inserted. Defaults to the first range in the current selection.\n\t */\n\trange?: Range;\n\n\t/**\n\t * The range where the selection should be placed after the insertion.\n\t * If not specified, the selection will be placed right after the inserted text.\n\t */\n\tresultRange?: Range;\n}\n\nexport interface InsertTextCommandExecuteEvent {\n\tname: 'execute';\n\targs: [\n\t\tdata: [ options: InsertTextCommandOptions ]\n\t];\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/inserttextobserver\n */\n\nimport { env, EventInfo } from '@ckeditor/ckeditor5-utils';\n\nimport {\n\tDomEventData,\n\tObserver,\n\tFocusObserver,\n\ttype View,\n\ttype ViewDocumentCompositionEndEvent,\n\ttype ViewDocumentInputEvent,\n\ttype ViewDocumentSelection,\n\ttype ViewRange,\n\ttype ViewSelection\n} from '@ckeditor/ckeditor5-engine';\n\nconst TYPING_INPUT_TYPES = [\n\t// For collapsed range:\n\t//\t- This one is a regular typing (all browsers, all systems).\n\t//\t- This one is used by Chrome when typing accented letter 2nd step when the user selects the accent (Mac).\n\t// For non-collapsed range:\n\t//\t- This one is used by Chrome when typing accented letter when the selection box first appears (Mac).\n\t//\t- This one is used by Safari when accepting spell check suggestions from the context menu (Mac).\n\t'insertText',\n\n\t// This one is used by Safari when typing accented letter (Mac).\n\t// This one is used by Safari when accepting spell check suggestions from the autocorrection pop-up (Mac).\n\t'insertReplacementText'\n];\n\n/**\n * Text insertion observer introduces the {@link module:engine/view/document~Document#event:insertText} event.\n */\nexport default class InsertTextObserver extends Observer {\n\t/**\n\t * Instance of the focus observer. Insert text observer calls\n\t * {@link module:engine/view/observer/focusobserver~FocusObserver#flush} to mark the latest focus change as complete.\n\t */\n\tpublic readonly focusObserver: FocusObserver;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tthis.focusObserver = view.getObserver( FocusObserver );\n\n\t\t// On Android composition events should immediately be applied to the model. Rendering is not disabled.\n\t\t// On non-Android the model is updated only on composition end.\n\t\t// On Android we can't rely on composition start/end to update model.\n\t\tif ( env.isAndroid ) {\n\t\t\tTYPING_INPUT_TYPES.push( 'insertCompositionText' );\n\t\t}\n\n\t\tconst viewDocument = view.document;\n\n\t\tviewDocument.on<ViewDocumentInputEvent>( 'beforeinput', ( evt, data ) => {\n\t\t\tif ( !this.isEnabled ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst { data: text, targetRanges, inputType, domEvent } = data;\n\n\t\t\tif ( !TYPING_INPUT_TYPES.includes( inputType ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Mark the latest focus change as complete (we are typing in editable after the focus\n\t\t\t// so the selection is in the focused element).\n\t\t\tthis.focusObserver.flush();\n\n\t\t\tconst eventInfo = new EventInfo( viewDocument, 'insertText' );\n\n\t\t\tviewDocument.fire( eventInfo, new DomEventData( view, domEvent, {\n\t\t\t\ttext,\n\t\t\t\tselection: view.createSelection( targetRanges )\n\t\t\t} ) );\n\n\t\t\t// Stop the beforeinput event if `delete` event was stopped.\n\t\t\t// https://github.com/ckeditor/ckeditor5/issues/753\n\t\t\tif ( eventInfo.stop.called ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t} );\n\n\t\t// Note: The priority must be lower than the CompositionObserver handler to call it after the renderer is unblocked.\n\t\tviewDocument.on<ViewDocumentCompositionEndEvent>( 'compositionend', ( evt, { data, domEvent } ) => {\n\t\t\t// On Android composition events are immediately applied to the model.\n\t\t\t// On non-Android the model is updated only on composition end.\n\t\t\t// On Android we can't rely on composition start/end to update model.\n\t\t\tif ( !this.isEnabled || env.isAndroid ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// In case of aborted composition.\n\t\t\tif ( !data ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.log( `%c[InsertTextObserver]%c Fire insertText event, text: ${ JSON.stringify( data ) }`,\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight: bold; color: green;', ''\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t// How do we know where to insert the composed text?\n\t\t\t// The selection observer is blocked and the view is not updated with the composition changes.\n\t\t\t// There were three options:\n\t\t\t// - Store the selection on `compositionstart` and use it now. This wouldn't work in RTC\n\t\t\t// where the view would change and the stored selection might get incorrect.\n\t\t\t// We'd need to fallback to the current view selection anyway.\n\t\t\t// - Use the current view selection. This is a bit weird and non-intuitive because\n\t\t\t// this isn't necessarily the selection on which the user started composing.\n\t\t\t// We cannot even know whether it's still collapsed (there might be some weird\n\t\t\t// editor feature that changed it in unpredictable ways for us). But it's by far\n\t\t\t// the simplest solution and should be stable (the selection is definitely correct)\n\t\t\t// and probably mostly predictable (features usually don't modify the selection\n\t\t\t// unless called explicitly by the user).\n\t\t\t// - Try to follow it from the `beforeinput` events. This would be really complex as each\n\t\t\t// `beforeinput` would come with just the range it's changing and we'd need to calculate that.\n\t\t\t// We decided to go with the 2nd option for its simplicity and stability.\n\t\t\tviewDocument.fire( 'insertText', new DomEventData( view, domEvent, {\n\t\t\t\ttext: data,\n\t\t\t\tselection: viewDocument.selection\n\t\t\t} ) );\n\t\t}, { priority: 'lowest' } );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic observe(): void {}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic stopObserving(): void {}\n}\n\n/**\n * Event fired when the user types text, for instance presses <kbd>A</kbd> or <kbd>?</kbd> in the\n * editing view document.\n *\n * **Note**: This event will **not** fire for keystrokes such as <kbd>Delete</kbd> or <kbd>Enter</kbd>.\n * They have dedicated events, see {@link module:engine/view/document~Document#event:delete} and\n * {@link module:engine/view/document~Document#event:enter} to learn more.\n *\n * **Note**: This event is fired by the {@link module:typing/inserttextobserver~InsertTextObserver input feature}.\n *\n * @eventName module:engine/view/document~Document#insertText\n * @param data The event data.\n */\nexport type ViewDocumentInsertTextEvent = {\n\tname: 'insertText';\n\targs: [ data: InsertTextEventData ];\n};\n\nexport interface InsertTextEventData extends DomEventData {\n\n\t/**\n\t * The text to be inserted.\n\t */\n\ttext: string;\n\n\t/**\n\t * The selection into which the text should be inserted.\n\t * If not specified, the insertion should occur at the current view selection.\n\t */\n\tselection: ViewSelection | ViewDocumentSelection;\n\n\t/**\n\t * The range that view selection should be set to after insertion.\n\t */\n\tresultRange?: ViewRange;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/input\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport { env } from '@ckeditor/ckeditor5-utils';\n\nimport InsertTextCommand from './inserttextcommand';\nimport InsertTextObserver, { type ViewDocumentInsertTextEvent } from './inserttextobserver';\n\nimport type { Model } from '@ckeditor/ckeditor5-engine';\n\n/**\n * Handles text input coming from the keyboard or other input methods.\n */\nexport default class Input extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Input' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst view = editor.editing.view;\n\t\tconst modelSelection = model.document.selection;\n\n\t\tview.addObserver( InsertTextObserver );\n\n\t\t// TODO The above default configuration value should be defined using editor.config.define() once it's fixed.\n\t\tconst insertTextCommand = new InsertTextCommand( editor, editor.config.get( 'typing.undoStep' ) || 20 );\n\n\t\t// Register `insertText` command and add `input` command as an alias for backward compatibility.\n\t\teditor.commands.add( 'insertText', insertTextCommand );\n\t\teditor.commands.add( 'input', insertTextCommand );\n\n\t\tthis.listenTo<ViewDocumentInsertTextEvent>( view.document, 'insertText', ( evt, data ) => {\n\t\t\t// Rendering is disabled while composing so prevent events that will be rendered by the engine\n\t\t\t// and should not be applied by the browser.\n\t\t\tif ( !view.document.isComposing ) {\n\t\t\t\tdata.preventDefault();\n\t\t\t}\n\n\t\t\tconst { text, selection: viewSelection, resultRange: viewResultRange } = data;\n\n\t\t\t// If view selection was specified, translate it to model selection.\n\t\t\tconst modelRanges = Array.from( viewSelection.getRanges() ).map( viewRange => {\n\t\t\t\treturn editor.editing.mapper.toModelRange( viewRange );\n\t\t\t} );\n\n\t\t\tlet insertText = text;\n\n\t\t\t// Typing in English on Android is firing composition events for the whole typed word.\n\t\t\t// We need to check the target range text to only apply the difference.\n\t\t\tif ( env.isAndroid ) {\n\t\t\t\tconst selectedText = Array.from( modelRanges[ 0 ].getItems() ).reduce( ( rangeText, node ) => {\n\t\t\t\t\treturn rangeText + ( node.is( '$textProxy' ) ? node.data : '' );\n\t\t\t\t}, '' );\n\n\t\t\t\tif ( selectedText ) {\n\t\t\t\t\tif ( selectedText.length <= insertText.length ) {\n\t\t\t\t\t\tif ( insertText.startsWith( selectedText ) ) {\n\t\t\t\t\t\t\tinsertText = insertText.substring( selectedText.length );\n\t\t\t\t\t\t\t( modelRanges[ 0 ] as any ).start = modelRanges[ 0 ].start.getShiftedBy( selectedText.length );\n\t\t\t\t\t\t}\n\t\t\t\t\t} else {\n\t\t\t\t\t\tif ( selectedText.startsWith( insertText ) ) {\n\t\t\t\t\t\t\t// TODO this should be mapped as delete?\n\t\t\t\t\t\t\t( modelRanges[ 0 ] as any ).start = modelRanges[ 0 ].start.getShiftedBy( insertText.length );\n\t\t\t\t\t\t\tinsertText = '';\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst insertTextCommandData: Parameters<InsertTextCommand[ 'execute' ]>[ 0 ] = {\n\t\t\t\ttext: insertText,\n\t\t\t\tselection: model.createSelection( modelRanges )\n\t\t\t};\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.log( '%c[Input]%c Execute insertText:',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight: bold; color: green;', '',\n\t\t\t// @if CK_DEBUG_TYPING // \t\tinsertText,\n\t\t\t// @if CK_DEBUG_TYPING // \t\t`[${ modelRanges[ 0 ].start.path }]-[${ modelRanges[ 0 ].end.path }]`\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\tif ( viewResultRange ) {\n\t\t\t\tinsertTextCommandData.resultRange = editor.editing.mapper.toModelRange( viewResultRange );\n\t\t\t}\n\n\t\t\teditor.execute( 'insertText', insertTextCommandData );\n\n\t\t\tview.scrollToTheSelection();\n\t\t} );\n\n\t\tif ( env.isAndroid ) {\n\t\t\t// On Android with English keyboard, the composition starts just by putting caret\n\t\t\t// at the word end or by selecting a table column. This is not a real composition started.\n\t\t\t// Trigger delete content on first composition key pressed.\n\t\t\tthis.listenTo( view.document, 'keydown', ( evt, data ) => {\n\t\t\t\tif ( modelSelection.isCollapsed || data.keyCode != 229 || !view.document.isComposing ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconst firstPositionPath = modelSelection.getFirstPosition()!.path;\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconst lastPositionPath = modelSelection.getLastPosition()!.path;\n\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconsole.log( '%c[Input]%c KeyDown 229 -> model.deleteContent()',\n\t\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight: bold; color: green;', '',\n\t\t\t\t// @if CK_DEBUG_TYPING // \t\t`[${ firstPositionPath }]-[${ lastPositionPath }]`\n\t\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t\tdeleteSelectionContent( model, insertTextCommand );\n\t\t\t} );\n\t\t} else {\n\t\t\t// Note: The priority must precede the CompositionObserver handler to call it before\n\t\t\t// the renderer is blocked, because we want to render this change.\n\t\t\tthis.listenTo( view.document, 'compositionstart', () => {\n\t\t\t\tif ( modelSelection.isCollapsed ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconst firstPositionPath = modelSelection.getFirstPosition()!.path;\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconst lastPositionPath = modelSelection.getLastPosition()!.path;\n\n\t\t\t\t// @if CK_DEBUG_TYPING // \tconsole.log( '%c[Input]%c Composition start -> model.deleteContent()',\n\t\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight: bold; color: green;', '',\n\t\t\t\t// @if CK_DEBUG_TYPING // \t\t`[${ firstPositionPath }]-[${ lastPositionPath }]`\n\t\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\t\tdeleteSelectionContent( model, insertTextCommand );\n\t\t\t} );\n\t\t}\n\t}\n}\n\nfunction deleteSelectionContent( model: Model, insertTextCommand: InsertTextCommand ): void {\n\t// By relying on the state of the input command we allow disabling the entire input easily\n\t// by just disabling the input command. We couldve used here the delete command but that\n\t// would mean requiring the delete feature which would block loading one without the other.\n\t// We could also check the editor.isReadOnly property, but that wouldn't allow to block\n\t// the input without blocking other features.\n\tif ( !insertTextCommand.isEnabled ) {\n\t\treturn;\n\t}\n\n\tconst buffer = insertTextCommand.buffer;\n\n\tbuffer.lock();\n\n\tmodel.enqueueChange( buffer.batch, () => {\n\t\tmodel.deleteContent( model.document.selection );\n\t} );\n\n\tbuffer.unlock();\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/deletecommand\n */\n\nimport { Command, type Editor } from '@ckeditor/ckeditor5-core';\nimport { count } from '@ckeditor/ckeditor5-utils';\nimport type { DocumentSelection, Element, Selection, Writer } from '@ckeditor/ckeditor5-engine';\n\nimport ChangeBuffer from './utils/changebuffer';\n\n/**\n * The delete command. Used by the {@link module:typing/delete~Delete delete feature} to handle the <kbd>Delete</kbd> and\n * <kbd>Backspace</kbd> keys.\n */\nexport default class DeleteCommand extends Command {\n\t/**\n\t * The directionality of the delete describing in what direction it should\n\t * consume the content when the selection is collapsed.\n\t */\n\tpublic readonly direction: 'forward' | 'backward';\n\n\t/**\n\t * Delete's change buffer used to group subsequent changes into batches.\n\t */\n\tprivate readonly _buffer: ChangeBuffer;\n\n\t/**\n\t * Creates an instance of the command.\n\t *\n\t * @param direction The directionality of the delete describing in what direction it\n\t * should consume the content when the selection is collapsed.\n\t */\n\tconstructor( editor: Editor, direction: 'forward' | 'backward' ) {\n\t\tsuper( editor );\n\n\t\tthis.direction = direction;\n\t\tthis._buffer = new ChangeBuffer( editor.model, editor.config.get( 'typing.undoStep' ) );\n\n\t\t// Since this command may execute on different selectable than selection, it should be checked directly in execute block.\n\t\tthis._isEnabledBasedOnSelection = false;\n\t}\n\n\t/**\n\t * The current change buffer.\n\t */\n\tpublic get buffer(): ChangeBuffer {\n\t\treturn this._buffer;\n\t}\n\n\t/**\n\t * Executes the delete command. Depending on whether the selection is collapsed or not, deletes its content\n\t * or a piece of content in the {@link #direction defined direction}.\n\t *\n\t * @fires execute\n\t * @param options The command options.\n\t * @param options.unit See {@link module:engine/model/utils/modifyselection~modifySelection}'s options.\n\t * @param options.sequence A number describing which subsequent delete event it is without the key being released.\n\t * See the {@link module:engine/view/document~Document#event:delete} event data.\n\t * @param options.selection Selection to remove. If not set, current model selection will be used.\n\t */\n\tpublic override execute( options: {\n\t\tunit?: 'character' | 'codePoint' | 'word';\n\t\tsequence?: number;\n\t\tselection?: Selection | DocumentSelection;\n\t} = {} ): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\n\t\tmodel.enqueueChange( this._buffer.batch, writer => {\n\t\t\tthis._buffer.lock();\n\n\t\t\tconst selection = writer.createSelection( options.selection || doc.selection );\n\n\t\t\t// Don't execute command when selection is in non-editable place.\n\t\t\tif ( !model.canEditAt( selection ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst sequence = options.sequence || 1;\n\n\t\t\t// Do not replace the whole selected content if selection was collapsed.\n\t\t\t// This prevents such situation:\n\t\t\t//\n\t\t\t// <h1></h1><p>[]</p>\t--> <h1>[</h1><p>]</p> \t\t--> <p></p>\n\t\t\t// starting content\t\t--> after `modifySelection`\t--> after `deleteContent`.\n\t\t\tconst doNotResetEntireContent = selection.isCollapsed;\n\n\t\t\t// Try to extend the selection in the specified direction.\n\t\t\tif ( selection.isCollapsed ) {\n\t\t\t\tmodel.modifySelection( selection, {\n\t\t\t\t\tdirection: this.direction,\n\t\t\t\t\tunit: options.unit,\n\t\t\t\t\ttreatEmojiAsSingleUnit: true\n\t\t\t\t} );\n\t\t\t}\n\n\t\t\t// Check if deleting in an empty editor. See #61.\n\t\t\tif ( this._shouldEntireContentBeReplacedWithParagraph( sequence ) ) {\n\t\t\t\tthis._replaceEntireContentWithParagraph( writer );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Check if deleting in the first empty block.\n\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/8137.\n\t\t\tif ( this._shouldReplaceFirstBlockWithParagraph( selection, sequence ) ) {\n\t\t\t\tthis.editor.execute( 'paragraph', { selection } );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// If selection is still collapsed, then there's nothing to delete.\n\t\t\tif ( selection.isCollapsed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet changeCount = 0;\n\n\t\t\tselection.getFirstRange()!.getMinimalFlatRanges().forEach( range => {\n\t\t\t\tchangeCount += count(\n\t\t\t\t\trange.getWalker( { singleCharacters: true, ignoreElementEnd: true, shallow: true } )\n\t\t\t\t);\n\t\t\t} );\n\n\t\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t\t// @if CK_DEBUG_TYPING // \tconsole.log( '%c[DeleteCommand]%c Delete content',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight: bold; color: green;', '',\n\t\t\t// @if CK_DEBUG_TYPING // \t\t`[${ selection.getFirstPosition()!.path }]-[${ selection.getLastPosition()!.path }]`, options\n\t\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t\t// @if CK_DEBUG_TYPING // }\n\n\t\t\tmodel.deleteContent( selection, {\n\t\t\t\tdoNotResetEntireContent,\n\t\t\t\tdirection: this.direction\n\t\t\t} );\n\n\t\t\tthis._buffer.input( changeCount );\n\n\t\t\twriter.setSelection( selection );\n\n\t\t\tthis._buffer.unlock();\n\t\t} );\n\t}\n\n\t/**\n\t * If the user keeps <kbd>Backspace</kbd> or <kbd>Delete</kbd> key pressed, the content of the current\n\t * editable will be cleared. However, this will not yet lead to resetting the remaining block to a paragraph\n\t * (which happens e.g. when the user does <kbd>Ctrl</kbd> + <kbd>A</kbd>, <kbd>Backspace</kbd>).\n\t *\n\t * But, if the user pressed the key in an empty editable for the first time,\n\t * we want to replace the entire content with a paragraph if:\n\t *\n\t * * the current limit element is empty,\n\t * * the paragraph is allowed in the limit element,\n\t * * the limit doesn't already have a paragraph inside.\n\t *\n\t * See https://github.com/ckeditor/ckeditor5-typing/issues/61.\n\t *\n\t * @param sequence A number describing which subsequent delete event it is without the key being released.\n\t */\n\tprivate _shouldEntireContentBeReplacedWithParagraph( sequence: number ): boolean {\n\t\t// Does nothing if user pressed and held the \"Backspace\" or \"Delete\" key.\n\t\tif ( sequence > 1 ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\t\tconst selection = doc.selection;\n\t\tconst limitElement = model.schema.getLimitElement( selection );\n\n\t\t// If a collapsed selection contains the whole content it means that the content is empty\n\t\t// (from the user perspective).\n\t\tconst limitElementIsEmpty = selection.isCollapsed && selection.containsEntireContent( limitElement );\n\n\t\tif ( !limitElementIsEmpty ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( !model.schema.checkChild( limitElement, 'paragraph' ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst limitElementFirstChild = limitElement.getChild( 0 );\n\n\t\t// Does nothing if the limit element already contains only a paragraph.\n\t\t// We ignore the case when paragraph might have some inline elements (<p><inlineWidget>[]</inlineWidget></p>)\n\t\t// because we don't support such cases yet and it's unclear whether inlineWidget shouldn't be a limit itself.\n\t\tif ( limitElementFirstChild && limitElementFirstChild.is( 'element', 'paragraph' ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * The entire content is replaced with the paragraph. Selection is moved inside the paragraph.\n\t *\n\t * @param writer The model writer.\n\t */\n\tprivate _replaceEntireContentWithParagraph( writer: Writer ): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\t\tconst selection = doc.selection;\n\t\tconst limitElement = model.schema.getLimitElement( selection );\n\t\tconst paragraph = writer.createElement( 'paragraph' );\n\n\t\twriter.remove( writer.createRangeIn( limitElement ) );\n\t\twriter.insert( paragraph, limitElement );\n\n\t\twriter.setSelection( paragraph, 0 );\n\t}\n\n\t/**\n\t * Checks if the selection is inside an empty element that is the first child of the limit element\n\t * and should be replaced with a paragraph.\n\t *\n\t * @param selection The selection.\n\t * @param sequence A number describing which subsequent delete event it is without the key being released.\n\t */\n\tprivate _shouldReplaceFirstBlockWithParagraph( selection: Selection, sequence: number ): boolean {\n\t\tconst model = this.editor.model;\n\n\t\t// Does nothing if user pressed and held the \"Backspace\" key or it was a \"Delete\" button.\n\t\tif ( sequence > 1 || this.direction != 'backward' ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( !selection.isCollapsed ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst position = selection.getFirstPosition()!;\n\t\tconst limitElement = model.schema.getLimitElement( position );\n\t\tconst limitElementFirstChild = limitElement.getChild( 0 ) as Element;\n\n\t\t// Only elements that are direct children of the limit element can be replaced.\n\t\t// Unwrapping from a block quote should be handled in a dedicated feature.\n\t\tif ( position.parent != limitElementFirstChild ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// A block should be replaced only if it was empty.\n\t\tif ( !selection.containsEntireContent( limitElementFirstChild ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Replace with a paragraph only if it's allowed there.\n\t\tif ( !model.schema.checkChild( limitElement, 'paragraph' ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Does nothing if the limit element already contains only a paragraph.\n\t\tif ( limitElementFirstChild.name == 'paragraph' ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/deleteobserver\n */\n\nimport {\n\tenv,\n\tkeyCodes,\n\tisInsideCombinedSymbol,\n\tisInsideEmojiSequence,\n\tisInsideSurrogatePair\n} from '@ckeditor/ckeditor5-utils';\nimport {\n\tBubblingEventInfo,\n\tDomEventData,\n\tObserver,\n\ttype BubblingEvent,\n\ttype ViewDocumentInputEvent,\n\ttype ViewDocumentKeyDownEvent,\n\ttype ViewDocumentKeyUpEvent,\n\ttype ViewDocumentSelection,\n\ttype ViewSelection,\n\ttype ViewRange,\n\ttype View\n} from '@ckeditor/ckeditor5-engine';\n\nconst DELETE_CHARACTER = 'character';\nconst DELETE_WORD = 'word';\nconst DELETE_CODE_POINT = 'codePoint';\nconst DELETE_SELECTION = 'selection';\nconst DELETE_BACKWARD = 'backward';\nconst DELETE_FORWARD = 'forward';\n\ntype DeleteEventSpec = {\n\tunit: 'selection' | 'codePoint' | 'character' | 'word';\n\tdirection: 'backward' | 'forward';\n};\n\nconst DELETE_EVENT_TYPES: Record<string, DeleteEventSpec> = {\n\t// --------------------------------------- Backward delete types -----------------------------------------------------\n\n\t// This happens in Safari on Mac when some content is selected and Ctrl + K is pressed.\n\tdeleteContent: {\n\t\tunit: DELETE_SELECTION,\n\n\t\t// According to the Input Events Level 2 spec, this delete type has no direction\n\t\t// but to keep things simple, let's default to backward.\n\t\tdirection: DELETE_BACKWARD\n\t},\n\t// Chrome and Safari on Mac: Backspace or Ctrl + H\n\tdeleteContentBackward: {\n\t\t// This kind of deletions must be done on the code point-level instead of target range provided by the DOM beforeinput event.\n\t\t// Take for instance \"👨‍👩‍👧‍👧\", it equals:\n\t\t//\n\t\t//\t* [ \"👨\", \"ZERO WIDTH JOINER\", \"👩\", \"ZERO WIDTH JOINER\", \"👧\", \"ZERO WIDTH JOINER\", \"👧\" ]\n\t\t//\t* or simply \"\\u{1F468}\\u200D\\u{1F469}\\u200D\\u{1F467}\\u200D\\u{1F467}\"\n\t\t//\n\t\t// The range provided by the browser would cause the entire multi-byte grapheme to disappear while the user\n\t\t// intention when deleting backwards (\"👨‍👩‍👧‍👧[]\", then backspace) is gradual \"decomposition\" (first to \"👨‍👩‍👧‍[]\",\n\t\t// then to \"👨‍👩‍[]\", etc.).\n\t\t//\n\t\t//\t* \"👨‍👩‍👧‍👧[]\" + backward delete (by code point) -> results in \"👨‍👩‍👧[]\", removed the last \"👧\" 👍\n\t\t//\t* \"👨‍👩‍👧‍👧[]\" + backward delete (by character) -> results in \"[]\", removed the whole grapheme 👎\n\t\t//\n\t\t// Deleting by code-point is simply a better UX. See \"deleteContentForward\" to learn more.\n\t\tunit: DELETE_CODE_POINT,\n\t\tdirection: DELETE_BACKWARD\n\t},\n\t// On Mac: Option + Backspace.\n\t// On iOS: Hold the backspace for a while and the whole words will start to disappear.\n\tdeleteWordBackward: {\n\t\tunit: DELETE_WORD,\n\t\tdirection: DELETE_BACKWARD\n\t},\n\t// Safari on Mac: Cmd + Backspace\n\tdeleteHardLineBackward: {\n\t\tunit: DELETE_SELECTION,\n\t\tdirection: DELETE_BACKWARD\n\t},\n\t// Chrome on Mac: Cmd + Backspace.\n\tdeleteSoftLineBackward: {\n\t\tunit: DELETE_SELECTION,\n\t\tdirection: DELETE_BACKWARD\n\t},\n\n\t// --------------------------------------- Forward delete types -----------------------------------------------------\n\n\t// Chrome on Mac: Fn + Backspace or Ctrl + D\n\t// Safari on Mac: Ctrl + K or Ctrl + D\n\tdeleteContentForward: {\n\t\t// Unlike backward delete, this delete must be performed by character instead of by code point, which\n\t\t// provides the best UX for working with accented letters.\n\t\t// Take, for example \"b̂\" (\"\\u0062\\u0302\", or [ \"LATIN SMALL LETTER B\", \"COMBINING CIRCUMFLEX ACCENT\" ]):\n\t\t//\n\t\t//\t* \"b̂[]\" + backward delete (by code point) -> results in \"b[]\", removed the combining mark 👍\n\t\t//\t* \"[]b̂\" + forward delete (by code point) -> results in \"[]^\", a bare combining mark does that not make sense when alone 👎\n\t\t//\t* \"[]b̂\" + forward delete (by character) -> results in \"[]\", removed both \"b\" and the combining mark 👍\n\t\t//\n\t\t// See: \"deleteContentBackward\" to learn more.\n\t\tunit: DELETE_CHARACTER,\n\t\tdirection: DELETE_FORWARD\n\t},\n\t// On Mac: Fn + Option + Backspace.\n\tdeleteWordForward: {\n\t\tunit: DELETE_WORD,\n\t\tdirection: DELETE_FORWARD\n\t},\n\t// Chrome on Mac: Ctrl + K (you have to disable the Link plugin first, though, because it uses the same keystroke)\n\t// This is weird that it does not work in Safari on Mac despite being listed in the official shortcuts listing\n\t// on Apple's webpage.\n\tdeleteHardLineForward: {\n\t\tunit: DELETE_SELECTION,\n\t\tdirection: DELETE_FORWARD\n\t},\n\t// At this moment there is no known way to trigger this event type but let's keep it for the symmetry with\n\t// deleteSoftLineBackward.\n\tdeleteSoftLineForward: {\n\t\tunit: DELETE_SELECTION,\n\t\tdirection: DELETE_FORWARD\n\t}\n};\n\n/**\n * Delete observer introduces the {@link module:engine/view/document~Document#event:delete} event.\n */\nexport default class DeleteObserver extends Observer {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tconst document = view.document;\n\n\t\t// It matters how many subsequent deletions were made, e.g. when the backspace key was pressed and held\n\t\t// by the user for some time. For instance, if such scenario ocurred and the heading the selection was\n\t\t// anchored to was the only content of the editor, it will not be converted into a paragraph (the user\n\t\t// wanted to clean it up, not remove it, it's about UX). Check out the DeleteCommand implementation to learn more.\n\t\t//\n\t\t// Fun fact: Safari on Mac won't fire beforeinput for backspace in an empty heading (only content).\n\t\tlet sequence = 0;\n\n\t\tdocument.on( 'keydown', () => {\n\t\t\tsequence++;\n\t\t} );\n\n\t\tdocument.on( 'keyup', () => {\n\t\t\tsequence = 0;\n\t\t} );\n\n\t\tdocument.on<ViewDocumentInputEvent>( 'beforeinput', ( evt, data ) => {\n\t\t\tif ( !this.isEnabled ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst { targetRanges, domEvent, inputType } = data;\n\t\t\tconst deleteEventSpec = DELETE_EVENT_TYPES[ inputType ];\n\n\t\t\tif ( !deleteEventSpec ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst deleteData: Partial<DeleteEventData> = {\n\t\t\t\tdirection: deleteEventSpec.direction,\n\t\t\t\tunit: deleteEventSpec.unit,\n\t\t\t\tsequence\n\t\t\t};\n\n\t\t\tif ( deleteData.unit == DELETE_SELECTION ) {\n\t\t\t\tdeleteData.selectionToRemove = view.createSelection( targetRanges[ 0 ] );\n\t\t\t}\n\n\t\t\t// The default deletion unit for deleteContentBackward is a single code point\n\t\t\t// but if the browser provides a wider target range then we should use it.\n\t\t\tif ( inputType === 'deleteContentBackward' ) {\n\t\t\t\t// On Android, deleteContentBackward has sequence 1 by default.\n\t\t\t\tif ( env.isAndroid ) {\n\t\t\t\t\tdeleteData.sequence = 1;\n\t\t\t\t}\n\n\t\t\t\t// The beforeInput event wants more than a single character to be removed.\n\t\t\t\tif ( shouldUseTargetRanges( targetRanges ) ) {\n\t\t\t\t\tdeleteData.unit = DELETE_SELECTION;\n\t\t\t\t\tdeleteData.selectionToRemove = view.createSelection( targetRanges );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst eventInfo = new BubblingEventInfo( document, 'delete', targetRanges[ 0 ] );\n\n\t\t\tdocument.fire( eventInfo, new DomEventData( view, domEvent, deleteData ) );\n\n\t\t\t// Stop the beforeinput event if `delete` event was stopped.\n\t\t\t// https://github.com/ckeditor/ckeditor5/issues/753\n\t\t\tif ( eventInfo.stop.called ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t} );\n\n\t\t// TODO: to be removed when https://bugs.chromium.org/p/chromium/issues/detail?id=1365311 is solved.\n\t\tif ( env.isBlink ) {\n\t\t\tenableChromeWorkaround( this );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic observe(): void {}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic stopObserving(): void {}\n}\n\n/**\n * Event fired when the user tries to delete content (e.g. presses <kbd>Delete</kbd> or <kbd>Backspace</kbd>).\n *\n * Note: This event is fired by the {@link module:typing/deleteobserver~DeleteObserver delete observer}\n * (usually registered by the {@link module:typing/delete~Delete delete feature}).\n *\n * @eventName module:engine/view/document~Document#delete\n * @param data The event data.\n */\nexport type ViewDocumentDeleteEvent = BubblingEvent<{\n\tname: 'delete';\n\targs: [ data: DeleteEventData ];\n}>;\n\nexport interface DeleteEventData extends DomEventData<InputEvent> {\n\n\t/**\n\t * The direction in which the deletion should happen.\n\t */\n\tdirection: 'backward' | 'forward';\n\n\t/**\n\t * The \"amount\" of content that should be deleted.\n\t */\n\tunit: 'selection' | 'codePoint' | 'character' | 'word';\n\n\t/**\n\t * A number describing which subsequent delete event it is without the key being released.\n\t * If it's 2 or more it means that the key was pressed and hold.\n\t */\n\tsequence: number;\n\n\t/**\n\t * View selection which content should be removed. If not set,\n\t * current selection should be used.\n\t */\n\tselectionToRemove?: ViewSelection | ViewDocumentSelection;\n}\n\n/**\n * Enables workaround for the issue https://github.com/ckeditor/ckeditor5/issues/11904.\n */\nfunction enableChromeWorkaround( observer: DeleteObserver ) {\n\tconst view = observer.view;\n\tconst document = view.document;\n\n\tlet pressedKeyCode: number | null = null;\n\tlet beforeInputReceived = false;\n\n\tdocument.on<ViewDocumentKeyDownEvent>( 'keydown', ( evt, { keyCode } ) => {\n\t\tpressedKeyCode = keyCode;\n\t\tbeforeInputReceived = false;\n\t} );\n\n\tdocument.on<ViewDocumentKeyUpEvent>( 'keyup', ( evt, { keyCode, domEvent } ) => {\n\t\tconst selection = document.selection;\n\t\tconst shouldFireDeleteEvent = observer.isEnabled &&\n\t\t\tkeyCode == pressedKeyCode &&\n\t\t\tisDeleteKeyCode( keyCode ) &&\n\t\t\t!selection.isCollapsed &&\n\t\t\t!beforeInputReceived;\n\n\t\tpressedKeyCode = null;\n\n\t\tif ( shouldFireDeleteEvent ) {\n\t\t\tconst targetRange = selection.getFirstRange()!;\n\t\t\tconst eventInfo = new BubblingEventInfo( document, 'delete', targetRange );\n\t\t\tconst deleteData: Partial<DeleteEventData> = {\n\t\t\t\tunit: DELETE_SELECTION,\n\t\t\t\tdirection: getDeleteDirection( keyCode ),\n\t\t\t\tselectionToRemove: selection\n\t\t\t};\n\n\t\t\tdocument.fire( eventInfo, new DomEventData( view, domEvent, deleteData ) );\n\t\t}\n\t} );\n\n\tdocument.on<ViewDocumentInputEvent>( 'beforeinput', ( evt, { inputType } ) => {\n\t\tconst deleteEventSpec = DELETE_EVENT_TYPES[ inputType ];\n\t\tconst isMatchingBeforeInput = isDeleteKeyCode( pressedKeyCode ) &&\n\t\t\tdeleteEventSpec &&\n\t\t\tdeleteEventSpec.direction == getDeleteDirection( pressedKeyCode );\n\n\t\tif ( isMatchingBeforeInput ) {\n\t\t\tbeforeInputReceived = true;\n\t\t}\n\t}, { priority: 'high' } );\n\n\tdocument.on<ViewDocumentInputEvent>( 'beforeinput', ( evt, { inputType, data } ) => {\n\t\tconst shouldIgnoreBeforeInput = pressedKeyCode == keyCodes.delete &&\n\t\t\tinputType == 'insertText' &&\n\t\t\tdata == '\\x7f'; // Delete character :P\n\n\t\tif ( shouldIgnoreBeforeInput ) {\n\t\t\tevt.stop();\n\t\t}\n\t}, { priority: 'high' } );\n\n\tfunction isDeleteKeyCode( keyCode: number | null ): boolean {\n\t\treturn keyCode == keyCodes.backspace || keyCode == keyCodes.delete;\n\t}\n\n\tfunction getDeleteDirection( keyCode: number | null ): 'backward' | 'forward' {\n\t\treturn keyCode == keyCodes.backspace ? DELETE_BACKWARD : DELETE_FORWARD;\n\t}\n}\n\n/**\n * Verifies whether the given target ranges cover more than a single character and should be used instead of a single code-point deletion.\n */\nfunction shouldUseTargetRanges( targetRanges: Array<ViewRange> ): boolean {\n\t// The collapsed target range could happen for example while deleting inside an inline filler\n\t// (it's mapped to collapsed position before an inline filler).\n\tif ( targetRanges.length != 1 || targetRanges[ 0 ].isCollapsed ) {\n\t\treturn false;\n\t}\n\n\tconst walker = targetRanges[ 0 ].getWalker( {\n\t\tdirection: 'backward',\n\t\tsingleCharacters: true,\n\t\tignoreElementEnd: true\n\t} );\n\n\tlet count = 0;\n\n\tfor ( const { nextPosition } of walker ) {\n\t\t// There is some element in the range so count it as a single character.\n\t\tif ( !nextPosition.parent.is( '$text' ) ) {\n\t\t\tcount++;\n\t\t} else {\n\t\t\tconst data = nextPosition.parent.data;\n\t\t\tconst offset = nextPosition.offset;\n\n\t\t\t// Count combined symbols and emoji sequences as a single character.\n\t\t\tif (\n\t\t\t\tisInsideSurrogatePair( data, offset ) ||\n\t\t\t\tisInsideCombinedSymbol( data, offset ) ||\n\t\t\t\tisInsideEmojiSequence( data, offset )\n\t\t\t) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tcount++;\n\t\t}\n\n\t\tif ( count > 1 ) {\n\t\t\treturn true;\n\t\t}\n\t}\n\n\treturn false;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/delete\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport DeleteCommand from './deletecommand';\nimport DeleteObserver, { type ViewDocumentDeleteEvent } from './deleteobserver';\n\n/**\n * The delete and backspace feature. Handles keys such as <kbd>Delete</kbd> and <kbd>Backspace</kbd>, other\n * keystrokes and user actions that result in deleting content in the editor.\n */\nexport default class Delete extends Plugin {\n\t/**\n\t * Whether pressing backspace should trigger undo action\n\t */\n\tprivate _undoOnBackspace!: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Delete' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\t\tconst modelDocument = editor.model.document;\n\n\t\tview.addObserver( DeleteObserver );\n\n\t\tthis._undoOnBackspace = false;\n\n\t\tconst deleteForwardCommand = new DeleteCommand( editor, 'forward' );\n\n\t\t// Register `deleteForward` command and add `forwardDelete` command as an alias for backward compatibility.\n\t\teditor.commands.add( 'deleteForward', deleteForwardCommand );\n\t\teditor.commands.add( 'forwardDelete', deleteForwardCommand );\n\n\t\teditor.commands.add( 'delete', new DeleteCommand( editor, 'backward' ) );\n\n\t\tthis.listenTo<ViewDocumentDeleteEvent>( viewDocument, 'delete', ( evt, data ) => {\n\t\t\t// When not in composition, we handle the action, so prevent the default one.\n\t\t\t// When in composition, it's the browser who modify the DOM (renderer is disabled).\n\t\t\tif ( !viewDocument.isComposing ) {\n\t\t\t\tdata.preventDefault();\n\t\t\t}\n\n\t\t\tconst { direction, sequence, selectionToRemove, unit } = data;\n\t\t\tconst commandName = direction === 'forward' ? 'deleteForward' : 'delete';\n\t\t\tconst commandData: Parameters<DeleteCommand[ 'execute' ]>[ 0 ] = { sequence };\n\n\t\t\tif ( unit == 'selection' ) {\n\t\t\t\tconst modelRanges = Array.from( selectionToRemove!.getRanges() ).map( viewRange => {\n\t\t\t\t\treturn editor.editing.mapper.toModelRange( viewRange );\n\t\t\t\t} );\n\n\t\t\t\tcommandData.selection = editor.model.createSelection( modelRanges );\n\t\t\t} else {\n\t\t\t\tcommandData.unit = unit;\n\t\t\t}\n\n\t\t\teditor.execute( commandName, commandData );\n\n\t\t\tview.scrollToTheSelection();\n\t\t}, { priority: 'low' } );\n\n\t\tif ( this.editor.plugins.has( 'UndoEditing' ) ) {\n\t\t\tthis.listenTo<ViewDocumentDeleteEvent>( viewDocument, 'delete', ( evt, data ) => {\n\t\t\t\tif ( this._undoOnBackspace && data.direction == 'backward' && data.sequence == 1 && data.unit == 'codePoint' ) {\n\t\t\t\t\tthis._undoOnBackspace = false;\n\n\t\t\t\t\teditor.execute( 'undo' );\n\n\t\t\t\t\tdata.preventDefault();\n\t\t\t\t\tevt.stop();\n\t\t\t\t}\n\t\t\t}, { context: '$capture' } );\n\n\t\t\tthis.listenTo( modelDocument, 'change', () => {\n\t\t\t\tthis._undoOnBackspace = false;\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * If the next user action after calling this method is pressing backspace, it would undo the last change.\n\t *\n\t * Requires {@link module:undo/undoediting~UndoEditing} plugin. If not loaded, does nothing.\n\t */\n\tpublic requestUndoOnBackspace(): void {\n\t\tif ( this.editor.plugins.has( 'UndoEditing' ) ) {\n\t\t\tthis._undoOnBackspace = true;\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/typing\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport Input from './input';\nimport Delete from './delete';\n\n/**\n * The typing feature. It handles typing.\n *\n * This is a \"glue\" plugin which loads the {@link module:typing/input~Input} and {@link module:typing/delete~Delete}\n * plugins.\n */\nexport default class Typing extends Plugin {\n\tpublic static get requires() {\n\t\treturn [ Input, Delete ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Typing' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/utils/getlasttextline\n */\n\nimport type { Model, Range } from '@ckeditor/ckeditor5-engine';\n\n/**\n * Returns the last text line from the given range.\n *\n * \"The last text line\" is understood as text (from one or more text nodes) which is limited either by a parent block\n * or by inline elements (e.g. `<softBreak>`).\n *\n * ```ts\n * const rangeToCheck = model.createRange(\n * \tmodel.createPositionAt( paragraph, 0 ),\n * \tmodel.createPositionAt( paragraph, 'end' )\n * );\n *\n * const { text, range } = getLastTextLine( rangeToCheck, model );\n * ```\n *\n * For model below, the returned `text` will be \"Foo bar baz\" and `range` will be set on whole `<paragraph>` content:\n *\n * ```xml\n * <paragraph>Foo bar baz<paragraph>\n * ```\n *\n * However, in below case, `text` will be set to \"baz\" and `range` will be set only on \"baz\".\n *\n * ```xml\n * <paragraph>Foo<softBreak></softBreak>bar<softBreak></softBreak>baz<paragraph>\n * ```\n */\nexport default function getLastTextLine( range: Range, model: Model ): LastTextLineData {\n\tlet start = range.start;\n\n\tconst text = Array.from( range.getWalker( { ignoreElementEnd: false } ) ).reduce( ( rangeText, { item } ) => {\n\t\t// Trim text to a last occurrence of an inline element and update range start.\n\t\tif ( !( item.is( '$text' ) || item.is( '$textProxy' ) ) ) {\n\t\t\tstart = model.createPositionAfter( item );\n\n\t\t\treturn '';\n\t\t}\n\n\t\treturn rangeText + item.data;\n\t}, '' );\n\n\treturn { text, range: model.createRange( start, range.end ) };\n}\n\n/**\n * The value returned by {@link module:typing/utils/getlasttextline~getLastTextLine}.\n */\nexport type LastTextLineData = {\n\n\t/**\n\t * The text from the text nodes in the last text line.\n\t */\n\ttext: string;\n\n\t/**\n\t * The range set on the text nodes in the last text line.\n\t */\n\trange: Range;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/textwatcher\n */\n\nimport { ObservableMixin, type ObservableChangeEvent } from '@ckeditor/ckeditor5-utils';\nimport getLastTextLine from './utils/getlasttextline';\n\nimport type {\n\tBatch,\n\tModel,\n\tRange,\n\tDocumentChangeEvent,\n\tDocumentSelectionChangeEvent\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * The text watcher feature.\n *\n * Fires the {@link module:typing/textwatcher~TextWatcher#event:matched:data `matched:data`},\n * {@link module:typing/textwatcher~TextWatcher#event:matched:selection `matched:selection`} and\n * {@link module:typing/textwatcher~TextWatcher#event:unmatched `unmatched`} events on typing or selection changes.\n */\nexport default class TextWatcher extends ObservableMixin() {\n\t/**\n\t * The editor's model.\n\t */\n\tpublic readonly model: Model;\n\n\t/**\n\t * The function used to match the text.\n\t *\n\t * The test callback can return 3 values:\n\t *\n\t * * `false` if there is no match,\n\t * * `true` if there is a match,\n\t * * an object if there is a match and we want to pass some additional information to the {@link #event:matched:data} event.\n\t */\n\tpublic testCallback: ( text: string ) => unknown;\n\n\t/**\n\t * Whether there is a match currently.\n\t */\n\tprivate _hasMatch: boolean;\n\n\t/**\n\t * Flag indicating whether the `TextWatcher` instance is enabled or disabled.\n\t * A disabled TextWatcher will not evaluate text.\n\t *\n\t * To disable TextWatcher:\n\t *\n\t * ```ts\n\t * const watcher = new TextWatcher( editor.model, testCallback );\n\t *\n\t * // After this a testCallback will not be called.\n\t * watcher.isEnabled = false;\n\t * ```\n\t */\n\tdeclare public isEnabled: boolean;\n\n\t/**\n\t * Creates a text watcher instance.\n\t *\n\t * @param testCallback See {@link module:typing/textwatcher~TextWatcher#testCallback}.\n\t */\n\tconstructor( model: Model, testCallback: ( text: string ) => unknown ) {\n\t\tsuper();\n\n\t\tthis.model = model;\n\t\tthis.testCallback = testCallback;\n\t\tthis._hasMatch = false;\n\n\t\tthis.set( 'isEnabled', true );\n\n\t\t// Toggle text watching on isEnabled state change.\n\t\tthis.on<ObservableChangeEvent>( 'change:isEnabled', () => {\n\t\t\tif ( this.isEnabled ) {\n\t\t\t\tthis._startListening();\n\t\t\t} else {\n\t\t\t\tthis.stopListening( model.document.selection );\n\t\t\t\tthis.stopListening( model.document );\n\t\t\t}\n\t\t} );\n\n\t\tthis._startListening();\n\t}\n\n\t/**\n\t * Flag indicating whether there is a match currently.\n\t */\n\tpublic get hasMatch(): boolean {\n\t\treturn this._hasMatch;\n\t}\n\n\t/**\n\t * Starts listening to the editor for typing and selection events.\n\t */\n\tprivate _startListening(): void {\n\t\tconst model = this.model;\n\t\tconst document = model.document;\n\n\t\tthis.listenTo<DocumentSelectionChangeEvent>( document.selection, 'change:range', ( evt, { directChange } ) => {\n\t\t\t// Indirect changes (i.e. when the user types or external changes are applied) are handled in the document's change event.\n\t\t\tif ( !directChange ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Act only on collapsed selection.\n\t\t\tif ( !document.selection.isCollapsed ) {\n\t\t\t\tif ( this.hasMatch ) {\n\t\t\t\t\tthis.fire( 'unmatched' );\n\t\t\t\t\tthis._hasMatch = false;\n\t\t\t\t}\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._evaluateTextBeforeSelection( 'selection' );\n\t\t} );\n\n\t\tthis.listenTo<DocumentChangeEvent>( document, 'change:data', ( evt, batch ) => {\n\t\t\tif ( batch.isUndo || !batch.isLocal ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._evaluateTextBeforeSelection( 'data', { batch } );\n\t\t} );\n\t}\n\n\t/**\n\t * Checks the editor content for matched text.\n\t *\n\t * @fires matched:data\n\t * @fires matched:selection\n\t * @fires unmatched\n\t *\n\t * @param suffix A suffix used for generating the event name.\n\t * @param data Data object for event.\n\t */\n\tprivate _evaluateTextBeforeSelection( suffix: 'data' | 'selection', data: { batch?: Batch } = {} ): void {\n\t\tconst model = this.model;\n\t\tconst document = model.document;\n\t\tconst selection = document.selection;\n\n\t\tconst rangeBeforeSelection = model.createRange( model.createPositionAt( selection.focus!.parent, 0 ), selection.focus! );\n\n\t\tconst { text, range } = getLastTextLine( rangeBeforeSelection, model );\n\n\t\tconst testResult = this.testCallback( text );\n\n\t\tif ( !testResult && this.hasMatch ) {\n\t\t\tthis.fire<TextWatcherUnmatchedEvent>( 'unmatched' );\n\t\t}\n\n\t\tthis._hasMatch = !!testResult;\n\n\t\tif ( testResult ) {\n\t\t\tconst eventData = Object.assign( data, { text, range } );\n\n\t\t\t// If the test callback returns an object with additional data, assign the data as well.\n\t\t\tif ( typeof testResult == 'object' ) {\n\t\t\t\tObject.assign( eventData, testResult );\n\t\t\t}\n\n\t\t\tthis.fire<TextWatcherMatchedEvent>( `matched:${ suffix }`, eventData );\n\t\t}\n\t}\n}\n\nexport type TextWatcherMatchedEvent<TCallbackResult extends Record<string, unknown> = Record<string, unknown>> = {\n\tname: 'matched' | 'matched:data' | 'matched:selection';\n\targs: [ {\n\t\ttext: string;\n\t\trange: Range;\n\t\tbatch?: Batch;\n\t} & TCallbackResult ];\n};\n\n/**\n * Fired whenever the text watcher found a match for data changes.\n *\n * @eventName ~TextWatcher#matched:data\n * @param data Event data.\n * @param data.testResult The additional data returned from the {@link module:typing/textwatcher~TextWatcher#testCallback}.\n */\nexport type TextWatcherMatchedDataEvent<TCallbackResult extends Record<string, unknown>> = {\n\tname: 'matched:data';\n\targs: [ data: TextWatcherMatchedDataEventData & TCallbackResult ];\n};\n\nexport interface TextWatcherMatchedDataEventData {\n\n\t/**\n\t * The full text before selection to which the regexp was applied.\n\t */\n\ttext: string;\n\n\t/**\n\t * The range representing the position of the `data.text`.\n\t */\n\trange: Range;\n\n\tbatch: Batch;\n}\n\n/**\n * Fired whenever the text watcher found a match for selection changes.\n *\n * @eventName ~TextWatcher#matched:selection\n * @param data Event data.\n * @param data.testResult The additional data returned from the {@link module:typing/textwatcher~TextWatcher#testCallback}.\n */\nexport type TextWatcherMatchedSelectionEvent<TCallbackResult extends Record<string, unknown>> = {\n\tname: 'matched:selection';\n\targs: [ data: TextWatcherMatchedSelectionEventData & TCallbackResult ];\n};\n\nexport interface TextWatcherMatchedSelectionEventData {\n\n\t/**\n\t * The full text before selection.\n\t */\n\ttext: string;\n\n\t/**\n\t * The range representing the position of the `data.text`.\n\t */\n\trange: Range;\n}\n\n/**\n * Fired whenever the text does not match anymore. Fired only when the text watcher found a match.\n *\n * @eventName ~TextWatcher#unmatched\n */\nexport type TextWatcherUnmatchedEvent = {\n\tname: 'unmatched';\n\targs: [];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/twostepcaretmovement\n */\n\nimport { Plugin, type Editor } from '@ckeditor/ckeditor5-core';\n\nimport { keyCodes } from '@ckeditor/ckeditor5-utils';\n\nimport type {\n\tDocumentSelection,\n\tDocumentSelectionChangeEvent,\n\tDomEventData,\n\tModel,\n\tPosition,\n\tViewDocumentArrowKeyEvent\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * This plugin enables the two-step caret (phantom) movement behavior for\n * {@link module:typing/twostepcaretmovement~TwoStepCaretMovement#registerAttribute registered attributes}\n * on arrow right (<kbd>→</kbd>) and left (<kbd>←</kbd>) key press.\n *\n * Thanks to this (phantom) caret movement the user is able to type before/after as well as at the\n * beginning/end of an attribute.\n *\n * **Note:** This plugin support righttoleft (Arabic, Hebrew, etc.) content by mirroring its behavior\n * but for the sake of simplicity examples showcase only lefttoright usecases.\n *\n * # Forward movement\n *\n * ## \"Entering\" an attribute:\n *\n * When this plugin is enabled and registered for the `a` attribute and the selection is right before it\n * (at the attribute boundary), pressing the right arrow key will not move the selection but update its\n * attributes accordingly:\n *\n * * When enabled:\n *\n * ```xml\n * foo{}<$text a=\"true\">bar</$text>\n * ```\n *\n * \t<kbd>→</kbd>\n *\n * ```xml\n * foo<$text a=\"true\">{}bar</$text>\n * ```\n *\n * * When disabled:\n *\n * ```xml\n * foo{}<$text a=\"true\">bar</$text>\n * ```\n *\n * \t<kbd>→</kbd>\n *\n * ```xml\n * foo<$text a=\"true\">b{}ar</$text>\n * ```\n *\n *\n * ## \"Leaving\" an attribute:\n *\n * * When enabled:\n *\n * ```xml\n * <$text a=\"true\">bar{}</$text>baz\n * ```\n *\n * \t<kbd>→</kbd>\n *\n * ```xml\n * <$text a=\"true\">bar</$text>{}baz\n * ```\n *\n * * When disabled:\n *\n * ```xml\n * <$text a=\"true\">bar{}</$text>baz\n * ```\n *\n * \t<kbd>→</kbd>\n *\n * ```xml\n * <$text a=\"true\">bar</$text>b{}az\n * ```\n *\n * # Backward movement\n *\n * * When enabled:\n *\n * ```xml\n * <$text a=\"true\">bar</$text>{}baz\n * ```\n *\n * \t<kbd>←</kbd>\n *\n * ```xml\n * <$text a=\"true\">bar{}</$text>baz\n * ```\n *\n * * When disabled:\n *\n * ```xml\n * <$text a=\"true\">bar</$text>{}baz\n * ```\n *\n * \t<kbd>←</kbd>\n *\n * ```xml\n * <$text a=\"true\">ba{}r</$text>b{}az\n * ```\n *\n * # Multiple attributes\n *\n * * When enabled and many attributes starts or ends at the same position:\n *\n * ```xml\n * <$text a=\"true\" b=\"true\">bar</$text>{}baz\n * ```\n *\n * \t<kbd>←</kbd>\n *\n * ```xml\n * <$text a=\"true\" b=\"true\">bar{}</$text>baz\n * ```\n *\n * * When enabled and one procedes another:\n *\n * ```xml\n * <$text a=\"true\">bar</$text><$text b=\"true\">{}bar</$text>\n * ```\n *\n * \t<kbd>←</kbd>\n *\n * ```xml\n * <$text a=\"true\">bar{}</$text><$text b=\"true\">bar</$text>\n * ```\n *\n */\nexport default class TwoStepCaretMovement extends Plugin {\n\t/**\n\t * A set of attributes to handle.\n\t */\n\tprivate attributes: Set<string>;\n\n\t/**\n\t * The current UID of the overridden gravity, as returned by\n\t * {@link module:engine/model/writer~Writer#overrideSelectionGravity}.\n\t */\n\tprivate _overrideUid: string | null;\n\n\t/**\n\t * A flag indicating that the automatic gravity restoration should not happen upon the next\n\t * gravity restoration.\n\t * {@link module:engine/model/selection~Selection#event:change:range} event.\n\t */\n\n\tprivate _isNextGravityRestorationSkipped!: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TwoStepCaretMovement' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis.attributes = new Set();\n\t\tthis._overrideUid = null;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst view = editor.editing.view;\n\t\tconst locale = editor.locale;\n\n\t\tconst modelSelection = model.document.selection;\n\n\t\t// Listen to keyboard events and handle the caret movement according to the 2-step caret logic.\n\t\tthis.listenTo<ViewDocumentArrowKeyEvent>( view.document, 'arrowKey', ( evt, data ) => {\n\t\t\t// This implementation works only for collapsed selection.\n\t\t\tif ( !modelSelection.isCollapsed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// When user tries to expand the selection or jump over the whole word or to the beginning/end then\n\t\t\t// two-steps movement is not necessary.\n\t\t\tif ( data.shiftKey || data.altKey || data.ctrlKey ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst arrowRightPressed = data.keyCode == keyCodes.arrowright;\n\t\t\tconst arrowLeftPressed = data.keyCode == keyCodes.arrowleft;\n\n\t\t\t// When neither left or right arrow has been pressed then do noting.\n\t\t\tif ( !arrowRightPressed && !arrowLeftPressed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst contentDirection = locale.contentLanguageDirection;\n\t\t\tlet isMovementHandled = false;\n\n\t\t\tif ( ( contentDirection === 'ltr' && arrowRightPressed ) || ( contentDirection === 'rtl' && arrowLeftPressed ) ) {\n\t\t\t\tisMovementHandled = this._handleForwardMovement( data );\n\t\t\t} else {\n\t\t\t\tisMovementHandled = this._handleBackwardMovement( data );\n\t\t\t}\n\n\t\t\t// Stop the keydown event if the two-step caret movement handled it. Avoid collisions\n\t\t\t// with other features which may also take over the caret movement (e.g. Widget).\n\t\t\tif ( isMovementHandled === true ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { context: '$text', priority: 'highest' } );\n\n\t\tthis._isNextGravityRestorationSkipped = false;\n\n\t\t// The automatic gravity restoration logic.\n\t\tthis.listenTo<DocumentSelectionChangeEvent>( modelSelection, 'change:range', ( evt, data ) => {\n\t\t\t// Skipping the automatic restoration is needed if the selection should change\n\t\t\t// but the gravity must remain overridden afterwards. See the #handleBackwardMovement\n\t\t\t// to learn more.\n\t\t\tif ( this._isNextGravityRestorationSkipped ) {\n\t\t\t\tthis._isNextGravityRestorationSkipped = false;\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Skip automatic restore when the gravity is not overridden — simply, there's nothing to restore\n\t\t\t// at this moment.\n\t\t\tif ( !this._isGravityOverridden ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Skip automatic restore when the change is indirect AND the selection is at the attribute boundary.\n\t\t\t// It means that e.g. if the change was external (collaboration) and the user had their\n\t\t\t// selection around the link, its gravity should remain intact in this change:range event.\n\t\t\tif ( !data.directChange && isBetweenDifferentAttributes( modelSelection.getFirstPosition()!, this.attributes ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._restoreGravity();\n\t\t} );\n\t}\n\n\t/**\n\t * Registers a given attribute for the two-step caret movement.\n\t *\n\t * @param attribute Name of the attribute to handle.\n\t */\n\tpublic registerAttribute( attribute: string ): void {\n\t\tthis.attributes.add( attribute );\n\t}\n\n\t/**\n\t * Updates the document selection and the view according to the twostep caret movement state\n\t * when moving **forwards**. Executed upon `keypress` in the {@link module:engine/view/view~View}.\n\t *\n\t * @param data Data of the key press.\n\t * @returns `true` when the handler prevented caret movement.\n\t */\n\tprivate _handleForwardMovement( data: DomEventData ): boolean {\n\t\tconst attributes = this.attributes;\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst position = selection.getFirstPosition()!;\n\n\t\t// DON'T ENGAGE 2-SCM if gravity is already overridden. It means that we just entered\n\t\t//\n\t\t// \t\t<paragraph>foo<$text attribute>{}bar</$text>baz</paragraph>\n\t\t//\n\t\t// or left the attribute\n\t\t//\n\t\t// \t\t<paragraph>foo<$text attribute>bar</$text>{}baz</paragraph>\n\t\t//\n\t\t// and the gravity will be restored automatically.\n\t\tif ( this._isGravityOverridden ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// DON'T ENGAGE 2-SCM when the selection is at the beginning of the block AND already has the\n\t\t// attribute:\n\t\t// * when the selection was initially set there using the mouse,\n\t\t// * when the editor has just started\n\t\t//\n\t\t//\t\t<paragraph><$text attribute>{}bar</$text>baz</paragraph>\n\t\t//\n\t\tif ( position.isAtStart && hasAnyAttribute( selection, attributes ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// ENGAGE 2-SCM When at least one of the observed attributes changes its value (incl. starts, ends).\n\t\t//\n\t\t//\t\t<paragraph>foo<$text attribute>bar{}</$text>baz</paragraph>\n\t\t//\t\t<paragraph>foo<$text attribute>bar{}</$text><$text otherAttribute>baz</$text></paragraph>\n\t\t//\t\t<paragraph>foo<$text attribute=1>bar{}</$text><$text attribute=2>baz</$text></paragraph>\n\t\t//\t\t<paragraph>foo{}<$text attribute>bar</$text>baz</paragraph>\n\t\t//\n\t\tif ( isBetweenDifferentAttributes( position, attributes ) ) {\n\t\t\tpreventCaretMovement( data );\n\t\t\tthis._overrideGravity();\n\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Updates the document selection and the view according to the twostep caret movement state\n\t * when moving **backwards**. Executed upon `keypress` in the {@link module:engine/view/view~View}.\n\t *\n\t * @param data Data of the key press.\n\t * @returns `true` when the handler prevented caret movement\n\t */\n\tprivate _handleBackwardMovement( data: DomEventData ): boolean {\n\t\tconst attributes = this.attributes;\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst position = selection.getFirstPosition()!;\n\n\t\t// When the gravity is already overridden (by this plugin), it means we are on the two-step position.\n\t\t// Prevent the movement, restore the gravity and update selection attributes.\n\t\t//\n\t\t//\t\t<paragraph>foo<$text attribute=1>bar</$text><$text attribute=2>{}baz</$text></paragraph>\n\t\t//\t\t<paragraph>foo<$text attribute>bar</$text><$text otherAttribute>{}baz</$text></paragraph>\n\t\t//\t\t<paragraph>foo<$text attribute>{}bar</$text>baz</paragraph>\n\t\t//\t\t<paragraph>foo<$text attribute>bar</$text>{}baz</paragraph>\n\t\t//\n\t\tif ( this._isGravityOverridden ) {\n\t\t\tpreventCaretMovement( data );\n\t\t\tthis._restoreGravity();\n\t\t\tsetSelectionAttributesFromTheNodeBefore( model, attributes, position );\n\n\t\t\treturn true;\n\t\t} else {\n\t\t\t// REMOVE SELECTION ATTRIBUTE when restoring gravity towards a non-existent content at the\n\t\t\t// beginning of the block.\n\t\t\t//\n\t\t\t// \t\t<paragraph>{}<$text attribute>bar</$text></paragraph>\n\t\t\t//\n\t\t\tif ( position.isAtStart ) {\n\t\t\t\tif ( hasAnyAttribute( selection, attributes ) ) {\n\t\t\t\t\tpreventCaretMovement( data );\n\t\t\t\t\tsetSelectionAttributesFromTheNodeBefore( model, attributes, position );\n\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\treturn false;\n\t\t\t}\n\n\t\t\t// When we are moving from natural gravity, to the position of the 2SCM, we need to override the gravity,\n\t\t\t// and make sure it won't be restored. Unless it's at the end of the block and an observed attribute.\n\t\t\t// We need to check if the caret is a one position before the attribute boundary:\n\t\t\t//\n\t\t\t//\t\t<paragraph>foo<$text attribute=1>bar</$text><$text attribute=2>b{}az</$text></paragraph>\n\t\t\t//\t\t<paragraph>foo<$text attribute>bar</$text><$text otherAttribute>b{}az</$text></paragraph>\n\t\t\t//\t\t<paragraph>foo<$text attribute>b{}ar</$text>baz</paragraph>\n\t\t\t//\t\t<paragraph>foo<$text attribute>bar</$text>b{}az</paragraph>\n\t\t\t//\n\t\t\tif ( isStepAfterAnyAttributeBoundary( position, attributes ) ) {\n\t\t\t\t// ENGAGE 2-SCM if the selection has no attribute. This may happen when the user\n\t\t\t\t// left the attribute using a FORWARD 2-SCM.\n\t\t\t\t//\n\t\t\t\t// \t\t<paragraph><$text attribute>bar</$text>{}</paragraph>\n\t\t\t\t//\n\t\t\t\tif (\n\t\t\t\t\tposition.isAtEnd &&\n\t\t\t\t\t!hasAnyAttribute( selection, attributes ) &&\n\t\t\t\t\tisBetweenDifferentAttributes( position, attributes )\n\t\t\t\t) {\n\t\t\t\t\tpreventCaretMovement( data );\n\t\t\t\t\tsetSelectionAttributesFromTheNodeBefore( model, attributes, position );\n\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\t// Skip the automatic gravity restore upon the next selection#change:range event.\n\t\t\t\t// If not skipped, it would automatically restore the gravity, which should remain\n\t\t\t\t// overridden.\n\t\t\t\tthis._isNextGravityRestorationSkipped = true;\n\t\t\t\tthis._overrideGravity();\n\n\t\t\t\t// Don't return \"true\" here because we didn't call _preventCaretMovement.\n\t\t\t\t// Returning here will destabilize the filler logic, which also listens to\n\t\t\t\t// keydown (and the event would be stopped).\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * `true` when the gravity is overridden for the plugin.\n\t */\n\tprivate get _isGravityOverridden(): boolean {\n\t\treturn !!this._overrideUid;\n\t}\n\n\t/**\n\t * Overrides the gravity using the {@link module:engine/model/writer~Writer model writer}\n\t * and stores the information about this fact in the {@link #_overrideUid}.\n\t *\n\t * A shorthand for {@link module:engine/model/writer~Writer#overrideSelectionGravity}.\n\t */\n\tprivate _overrideGravity(): void {\n\t\tthis._overrideUid = this.editor.model.change( writer => {\n\t\t\treturn writer.overrideSelectionGravity();\n\t\t} );\n\t}\n\n\t/**\n\t * Restores the gravity using the {@link module:engine/model/writer~Writer model writer}.\n\t *\n\t * A shorthand for {@link module:engine/model/writer~Writer#restoreSelectionGravity}.\n\t */\n\tprivate _restoreGravity(): void {\n\t\tthis.editor.model.change( writer => {\n\t\t\twriter.restoreSelectionGravity( this._overrideUid! );\n\t\t\tthis._overrideUid = null;\n\t\t} );\n\t}\n}\n\n/**\n * Checks whether the selection has any of given attributes.\n */\nfunction hasAnyAttribute( selection: DocumentSelection, attributes: Set<string> ): boolean {\n\tfor ( const observedAttribute of attributes ) {\n\t\tif ( selection.hasAttribute( observedAttribute ) ) {\n\t\t\treturn true;\n\t\t}\n\t}\n\n\treturn false;\n}\n\n/**\n * Applies the given attributes to the current selection using using the\n * values from the node before the current position. Uses\n * the {@link module:engine/model/writer~Writer model writer}.\n */\nfunction setSelectionAttributesFromTheNodeBefore( model: Model, attributes: Set<string>, position: Position ) {\n\tconst nodeBefore = position.nodeBefore;\n\tmodel.change( writer => {\n\t\tif ( nodeBefore ) {\n\t\t\tconst attributes: Array<[string, unknown]> = [];\n\t\t\tconst isInlineObject = model.schema.isObject( nodeBefore ) && model.schema.isInline( nodeBefore );\n\n\t\t\tfor ( const [ key, value ] of nodeBefore.getAttributes() ) {\n\t\t\t\tif (\n\t\t\t\t\tmodel.schema.checkAttribute( '$text', key ) &&\n\t\t\t\t\t( !isInlineObject || model.schema.getAttributeProperties( key ).copyFromObject !== false )\n\t\t\t\t) {\n\t\t\t\t\tattributes.push( [ key, value ] );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\twriter.setSelectionAttribute( attributes );\n\t\t} else {\n\t\t\twriter.removeSelectionAttribute( attributes );\n\t\t}\n\t} );\n}\n\n/**\n * Prevents the caret movement in the view by calling `preventDefault` on the event data.\n *\n * @alias data.preventDefault\n */\nfunction preventCaretMovement( data: DomEventData ) {\n\tdata.preventDefault();\n}\n\n/**\n * Checks whether the step before `isBetweenDifferentAttributes()`.\n */\nfunction isStepAfterAnyAttributeBoundary( position: Position, attributes: Set<string> ): boolean {\n\tconst positionBefore = position.getShiftedBy( -1 );\n\treturn isBetweenDifferentAttributes( positionBefore, attributes );\n}\n\n/**\n * Checks whether the given position is between different values of given attributes.\n */\nfunction isBetweenDifferentAttributes( position: Position, attributes: Set<string> ): boolean {\n\tconst { nodeBefore, nodeAfter } = position;\n\tfor ( const observedAttribute of attributes ) {\n\t\tconst attrBefore = nodeBefore ? nodeBefore.getAttribute( observedAttribute ) : undefined;\n\t\tconst attrAfter = nodeAfter ? nodeAfter.getAttribute( observedAttribute ) : undefined;\n\n\t\tif ( attrAfter !== attrBefore ) {\n\t\t\treturn true;\n\t\t}\n\t}\n\treturn false;\n}\n","import toString from './toString.js';\n\n/**\n * Used to match `RegExp`\n * [syntax characters](http://ecma-international.org/ecma-262/7.0/#sec-patterns).\n */\nvar reRegExpChar = /[\\\\^$.*+?()[\\]{}|]/g,\n reHasRegExpChar = RegExp(reRegExpChar.source);\n\n/**\n * Escapes the `RegExp` special characters \"^\", \"$\", \"\\\", \".\", \"*\", \"+\",\n * \"?\", \"(\", \")\", \"[\", \"]\", \"{\", \"}\", and \"|\" in `string`.\n *\n * @static\n * @memberOf _\n * @since 3.0.0\n * @category String\n * @param {string} [string=''] The string to escape.\n * @returns {string} Returns the escaped string.\n * @example\n *\n * _.escapeRegExp('[lodash](https://lodash.com/)');\n * // => '\\[lodash\\]\\(https://lodash\\.com/\\)'\n */\nfunction escapeRegExp(string) {\n string = toString(string);\n return (string && reHasRegExpChar.test(string))\n ? string.replace(reRegExpChar, '\\\\$&')\n : string;\n}\n\nexport default escapeRegExp;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/texttransformation\n */\n\nimport {\n\tPlugin,\n\ttype Editor\n} from '@ckeditor/ckeditor5-core';\n\nimport type { Position } from '@ckeditor/ckeditor5-engine';\n\nimport TextWatcher, { type TextWatcherMatchedDataEvent } from './textwatcher';\nimport type { TextTransformationConfig, TextTransformationDescription } from './typingconfig';\nimport type Delete from './delete';\n\nimport { escapeRegExp } from 'lodash-es';\n\n// All named transformations.\nconst TRANSFORMATIONS: Record<string, TextTransformationDescription> = {\n\t// Common symbols:\n\tcopyright: { from: '(c)', to: '©' },\n\tregisteredTrademark: { from: '(r)', to: '®' },\n\ttrademark: { from: '(tm)', to: '™' },\n\n\t// Mathematical:\n\toneHalf: { from: /(^|[^/a-z0-9])(1\\/2)([^/a-z0-9])$/i, to: [ null, '½', null ] },\n\toneThird: { from: /(^|[^/a-z0-9])(1\\/3)([^/a-z0-9])$/i, to: [ null, '⅓', null ] },\n\ttwoThirds: { from: /(^|[^/a-z0-9])(2\\/3)([^/a-z0-9])$/i, to: [ null, '⅔', null ] },\n\toneForth: { from: /(^|[^/a-z0-9])(1\\/4)([^/a-z0-9])$/i, to: [ null, '¼', null ] },\n\tthreeQuarters: { from: /(^|[^/a-z0-9])(3\\/4)([^/a-z0-9])$/i, to: [ null, '¾', null ] },\n\tlessThanOrEqual: { from: '<=', to: '≤' },\n\tgreaterThanOrEqual: { from: '>=', to: '≥' },\n\tnotEqual: { from: '!=', to: '≠' },\n\tarrowLeft: { from: '<-', to: '←' },\n\tarrowRight: { from: '->', to: '→' },\n\n\t// Typography:\n\thorizontalEllipsis: { from: '...', to: '…' },\n\tenDash: { from: /(^| )(--)( )$/, to: [ null, '', null ] },\n\temDash: { from: /(^| )(---)( )$/, to: [ null, '—', null ] },\n\t// Quotations:\n\t// English, US\n\tquotesPrimary: { from: buildQuotesRegExp( '\"' ), to: [ null, '“', null, '”' ] },\n\tquotesSecondary: { from: buildQuotesRegExp( '\\'' ), to: [ null, '', null, '' ] },\n\n\t// English, UK\n\tquotesPrimaryEnGb: { from: buildQuotesRegExp( '\\'' ), to: [ null, '', null, '' ] },\n\tquotesSecondaryEnGb: { from: buildQuotesRegExp( '\"' ), to: [ null, '“', null, '”' ] },\n\n\t// Polish\n\tquotesPrimaryPl: { from: buildQuotesRegExp( '\"' ), to: [ null, '„', null, '”' ] },\n\tquotesSecondaryPl: { from: buildQuotesRegExp( '\\'' ), to: [ null, '', null, '' ] }\n};\n\n// Transformation groups.\nconst TRANSFORMATION_GROUPS: Record<string, Array<string>> = {\n\tsymbols: [ 'copyright', 'registeredTrademark', 'trademark' ],\n\tmathematical: [\n\t\t'oneHalf', 'oneThird', 'twoThirds', 'oneForth', 'threeQuarters',\n\t\t'lessThanOrEqual', 'greaterThanOrEqual', 'notEqual',\n\t\t'arrowLeft', 'arrowRight'\n\t],\n\ttypography: [ 'horizontalEllipsis', 'enDash', 'emDash' ],\n\tquotes: [ 'quotesPrimary', 'quotesSecondary' ]\n};\n\n// A set of default transformations provided by the feature.\nconst DEFAULT_TRANSFORMATIONS = [\n\t'symbols',\n\t'mathematical',\n\t'typography',\n\t'quotes'\n];\n\n/**\n * The text transformation plugin.\n */\nexport default class TextTransformation extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ 'Delete', 'Input' ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TextTransformation' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\teditor.config.define( 'typing', {\n\t\t\ttransformations: {\n\t\t\t\tinclude: DEFAULT_TRANSFORMATIONS\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst model = this.editor.model;\n\t\tconst modelSelection = model.document.selection;\n\n\t\tmodelSelection.on( 'change:range', () => {\n\t\t\t// Disable plugin when selection is inside a code block.\n\t\t\tthis.isEnabled = !modelSelection.anchor!.parent.is( 'element', 'codeBlock' );\n\t\t} );\n\n\t\tthis._enableTransformationWatchers();\n\t}\n\n\t/**\n\t * Create new TextWatcher listening to the editor for typing and selection events.\n\t */\n\tprivate _enableTransformationWatchers(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst deletePlugin: Delete = editor.plugins.get( 'Delete' );\n\t\tconst normalizedTransformations = normalizeTransformations( editor.config.get( 'typing.transformations' )! );\n\n\t\tconst testCallback = ( text: string ) => {\n\t\t\tfor ( const normalizedTransformation of normalizedTransformations ) {\n\t\t\t\tconst from = normalizedTransformation.from;\n\t\t\t\tconst match = from.test( text );\n\n\t\t\t\tif ( match ) {\n\t\t\t\t\treturn { normalizedTransformation };\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\n\t\tconst watcher = new TextWatcher( editor.model, testCallback );\n\n\t\twatcher.on<TextWatcherMatchedDataEvent<{\n\t\t\tnormalizedTransformation: NormalizedTransformationConfig;\n\t\t}>>( 'matched:data', ( evt, data ) => {\n\t\t\tif ( !data.batch.isTyping ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst { from, to } = data.normalizedTransformation;\n\n\t\t\tconst matches = from.exec( data.text )!;\n\t\t\tconst replaces = to( matches.slice( 1 ) );\n\n\t\t\tconst matchedRange = data.range;\n\n\t\t\tlet changeIndex = matches.index;\n\n\t\t\tmodel.enqueueChange( writer => {\n\t\t\t\tfor ( let i = 1; i < matches.length; i++ ) {\n\t\t\t\t\tconst match = matches[ i ];\n\t\t\t\t\tconst replaceWith = replaces[ i - 1 ];\n\n\t\t\t\t\tif ( replaceWith == null ) {\n\t\t\t\t\t\tchangeIndex += match.length;\n\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\tconst replacePosition = matchedRange.start.getShiftedBy( changeIndex );\n\t\t\t\t\tconst replaceRange = model.createRange( replacePosition, replacePosition.getShiftedBy( match.length ) );\n\t\t\t\t\tconst attributes = getTextAttributesAfterPosition( replacePosition );\n\n\t\t\t\t\tmodel.insertContent( writer.createText( replaceWith, attributes ), replaceRange );\n\n\t\t\t\t\tchangeIndex += replaceWith.length;\n\t\t\t\t}\n\n\t\t\t\tmodel.enqueueChange( () => {\n\t\t\t\t\tdeletePlugin.requestUndoOnBackspace();\n\t\t\t\t} );\n\t\t\t} );\n\t\t} );\n\n\t\twatcher.bind( 'isEnabled' ).to( this );\n\t}\n}\n\n/**\n * Normalizes the configuration `from` parameter value.\n * The normalized value for the `from` parameter is a RegExp instance. If the passed `from` is already a RegExp instance,\n * it is returned unchanged.\n */\nfunction normalizeFrom( from: string | RegExp ): RegExp {\n\tif ( typeof from == 'string' ) {\n\t\treturn new RegExp( `(${ escapeRegExp( from ) })$` );\n\t}\n\n\t// `from` is already a regular expression.\n\treturn from;\n}\n\n/**\n * Normalizes the configuration `to` parameter value.\n * The normalized value for the `to` parameter is a function that takes an array and returns an array. See more in the\n * configuration description. If the passed `to` is already a function, it is returned unchanged.\n */\nfunction normalizeTo( to: TextTransformationDescription[ 'to' ] ) {\n\tif ( typeof to == 'string' ) {\n\t\treturn () => [ to ];\n\t} else if ( to instanceof Array ) {\n\t\treturn () => to;\n\t}\n\n\t// `to` is already a function.\n\treturn to;\n}\n\n/**\n * For given `position` returns attributes for the text that is after that position.\n * The text can be in the same text node as the position (`foo[]bar`) or in the next text node (`foo[]<$text bold=\"true\">bar</$text>`).\n */\nfunction getTextAttributesAfterPosition( position: Position ) {\n\tconst textNode = position.textNode ? position.textNode : position.nodeAfter;\n\n\treturn textNode!.getAttributes();\n}\n\n/**\n * Returns a RegExp pattern string that detects a sentence inside a quote.\n *\n * @param quoteCharacter The character to create a pattern for.\n */\nfunction buildQuotesRegExp( quoteCharacter: string ): RegExp {\n\treturn new RegExp( `(^|\\\\s)(${ quoteCharacter })([^${ quoteCharacter }]*)(${ quoteCharacter })$` );\n}\n\n/**\n * Reads text transformation config and returns normalized array of transformations objects.\n */\nfunction normalizeTransformations( config: TextTransformationConfig ): Array<NormalizedTransformationConfig> {\n\tconst extra = config.extra || [];\n\tconst remove = config.remove || [];\n\tconst isNotRemoved = ( transformation: TextTransformationDescription | string ) => !remove.includes( transformation );\n\n\tconst configured = config.include.concat( extra ).filter( isNotRemoved );\n\n\treturn expandGroupsAndRemoveDuplicates( configured )\n\t\t.filter( isNotRemoved ) // Filter out 'remove' transformations as they might be set in group.\n\t\t.map( transformation => (\n\t\t\ttypeof transformation == 'string' && TRANSFORMATIONS[ transformation ] ? TRANSFORMATIONS[ transformation ] : transformation )\n\t\t)\n\t\t// Filter out transformations set as string that has not been found.\n\t\t.filter( ( transformation ): transformation is TextTransformationDescription => typeof transformation === 'object' )\n\t\t.map( transformation => ( {\n\t\t\tfrom: normalizeFrom( transformation.from ),\n\t\t\tto: normalizeTo( transformation.to )\n\t\t} ) );\n}\n\n/**\n * Reads definitions and expands named groups if needed to transformation names.\n * This method also removes duplicated named transformations if any.\n */\nfunction expandGroupsAndRemoveDuplicates(\n\tdefinitions: Array<TextTransformationDescription | string>\n): Array<TextTransformationDescription | string> {\n\t// Set is using to make sure that transformation names are not duplicated.\n\tconst definedTransformations = new Set<TextTransformationDescription | string>();\n\n\tfor ( const transformationOrGroup of definitions ) {\n\t\tif ( typeof transformationOrGroup == 'string' && TRANSFORMATION_GROUPS[ transformationOrGroup ] ) {\n\t\t\tfor ( const transformation of TRANSFORMATION_GROUPS[ transformationOrGroup ] ) {\n\t\t\t\tdefinedTransformations.add( transformation );\n\t\t\t}\n\t\t} else {\n\t\t\tdefinedTransformations.add( transformationOrGroup );\n\t\t}\n\t}\n\n\treturn Array.from( definedTransformations );\n}\n\ntype NormalizedTransformationConfig = {\n\tfrom: RegExp;\n\tto: ( matches: Array<string> ) => Array<string | null>;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/utils/findattributerange\n */\n\nimport type { Position, Model, Range } from '@ckeditor/ckeditor5-engine';\n\n/**\n * Returns a model range that covers all consecutive nodes with the same `attributeName` and its `value`\n * that intersect the given `position`.\n *\n * It can be used e.g. to get the entire range on which the `linkHref` attribute needs to be changed when having a\n * selection inside a link.\n *\n * @param position The start position.\n * @param attributeName The attribute name.\n * @param value The attribute value.\n * @param model The model instance.\n * @returns The link range.\n */\nexport default function findAttributeRange(\n\tposition: Position,\n\tattributeName: string,\n\tvalue: unknown,\n\tmodel: Model\n): Range {\n\treturn model.createRange(\n\t\tfindAttributeRangeBound( position, attributeName, value, true, model ),\n\t\tfindAttributeRangeBound( position, attributeName, value, false, model )\n\t);\n}\n\n/**\n * Walks forward or backward (depends on the `lookBack` flag), node by node, as long as they have the same attribute value\n * and returns a position just before or after (depends on the `lookBack` flag) the last matched node.\n *\n * @param position The start position.\n * @param attributeName The attribute name.\n * @param value The attribute value.\n * @param lookBack Whether the walk direction is forward (`false`) or backward (`true`).\n * @returns The position just before the last matched node.\n */\nexport function findAttributeRangeBound(\n\tposition: Position,\n\tattributeName: string,\n\tvalue: unknown,\n\tlookBack: boolean,\n\tmodel: Model\n): Position {\n\t// Get node before or after position (depends on `lookBack` flag).\n\t// When position is inside text node then start searching from text node.\n\tlet node = position.textNode || ( lookBack ? position.nodeBefore : position.nodeAfter );\n\n\tlet lastNode = null;\n\n\twhile ( node && node.getAttribute( attributeName ) == value ) {\n\t\tlastNode = node;\n\t\tnode = lookBack ? node.previousSibling : node.nextSibling;\n\t}\n\n\treturn lastNode ? model.createPositionAt( lastNode, lookBack ? 'before' : 'after' ) : position;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module enter/utils\n */\n\nimport type { Schema } from '@ckeditor/ckeditor5-engine';\n\n/**\n * Returns attributes that should be preserved on the enter keystroke.\n *\n * Filtering is realized based on `copyOnEnter` attribute property. Read more about attribute properties\n * {@link module:engine/model/schema~Schema#setAttributeProperties here}.\n *\n * @param schema Model's schema.\n * @param allAttributes Attributes to filter.\n */\nexport function* getCopyOnEnterAttributes(\n\tschema: Schema,\n\tallAttributes: Iterable<[ string, unknown ]>\n): IterableIterator<[ string, unknown ]> {\n\tfor ( const attribute of allAttributes ) {\n\t\tif ( attribute && schema.getAttributeProperties( attribute[ 0 ] ).copyOnEnter ) {\n\t\t\tyield attribute;\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module enter/entercommand\n */\n\nimport { Command } from '@ckeditor/ckeditor5-core';\nimport { getCopyOnEnterAttributes } from './utils';\n\nimport type {\n\tElement,\n\tPosition,\n\tWriter\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * Enter command used by the {@link module:enter/enter~Enter Enter feature} to handle the <kbd>Enter</kbd> keystroke.\n */\nexport default class EnterCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tthis.editor.model.change( writer => {\n\t\t\tthis.enterBlock( writer );\n\t\t\tthis.fire<EnterCommandAfterExecuteEvent>( 'afterExecute', { writer } );\n\t\t} );\n\t}\n\n\t/**\n\t * Splits a block where the document selection is placed, in the way how the <kbd>Enter</kbd> key is expected to work:\n\t *\n\t * ```\n\t * <p>Foo[]bar</p> -> <p>Foo</p><p>[]bar</p>\n\t * <p>Foobar[]</p> -> <p>Foobar</p><p>[]</p>\n\t * <p>Fo[ob]ar</p> -> <p>Fo</p><p>[]ar</p>\n\t * ```\n\t *\n\t * In some cases, the split will not happen:\n\t *\n\t * ```\n\t * // The selection parent is a limit element:\n\t * <figcaption>A[bc]d</figcaption> -> <figcaption>A[]d</figcaption>\n\t *\n\t * // The selection spans over multiple elements:\n\t * <h>x[x</h><p>y]y<p> -> <h>x</h><p>[]y</p>\n\t * ```\n\t *\n\t * @param writer Writer to use when performing the enter action.\n\t * @returns Boolean indicating if the block was split.\n\t */\n\tpublic enterBlock( writer: Writer ): boolean {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst schema = model.schema;\n\t\tconst isSelectionEmpty = selection.isCollapsed;\n\t\tconst range = selection.getFirstRange()!;\n\t\tconst startElement = range.start.parent as Element;\n\t\tconst endElement = range.end.parent as Element;\n\n\t\t// Don't touch the roots and other limit elements.\n\t\tif ( schema.isLimit( startElement ) || schema.isLimit( endElement ) ) {\n\t\t\t// Delete the selected content but only if inside a single limit element.\n\t\t\t// Abort, when crossing limit elements boundary (e.g. <limit1>x[x</limit1>donttouchme<limit2>y]y</limit2>).\n\t\t\t// This is an edge case and it's hard to tell what should actually happen because such a selection\n\t\t\t// is not entirely valid.\n\t\t\tif ( !isSelectionEmpty && startElement == endElement ) {\n\t\t\t\tmodel.deleteContent( selection );\n\t\t\t}\n\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( isSelectionEmpty ) {\n\t\t\tconst attributesToCopy = getCopyOnEnterAttributes( writer.model.schema, selection.getAttributes() );\n\n\t\t\tsplitBlock( writer, range.start );\n\t\t\twriter.setSelectionAttribute( attributesToCopy );\n\n\t\t\treturn true;\n\t\t} else {\n\t\t\tconst leaveUnmerged = !( range.start.isAtStart && range.end.isAtEnd );\n\t\t\tconst isContainedWithinOneElement = ( startElement == endElement );\n\n\t\t\tmodel.deleteContent( selection, { leaveUnmerged } );\n\n\t\t\tif ( leaveUnmerged ) {\n\t\t\t\t// Partially selected elements.\n\t\t\t\t//\n\t\t\t\t// <h>x[xx]x</h>\t\t-> <h>x^x</h>\t\t\t-> <h>x</h><h>^x</h>\n\t\t\t\tif ( isContainedWithinOneElement ) {\n\t\t\t\t\tsplitBlock( writer, selection.focus! );\n\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t\t// Selection over multiple elements.\n\t\t\t\t//\n\t\t\t\t// <h>x[x</h><p>y]y<p>\t-> <h>x^</h><p>y</p>\t-> <h>x</h><p>^y</p>\n\t\t\t\telse {\n\t\t\t\t\twriter.setSelection( endElement, 0 );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t}\n}\n\n/**\n * Fired after the the {@link module:enter/entercommand~EnterCommand} is finished executing.\n *\n * @eventName ~EnterCommand#afterExecute\n */\nexport type EnterCommandAfterExecuteEvent = {\n\tname: 'afterExecute';\n\targs: [ { writer: Writer } ];\n};\n\nfunction splitBlock( writer: Writer, splitPos: Position ): void {\n\twriter.split( splitPos );\n\twriter.setSelection( splitPos.parent.nextSibling!, 0 );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module enter/enterobserver\n */\n\nimport {\n\tObserver,\n\tDomEventData,\n\tBubblingEventInfo,\n\ttype View,\n\ttype ViewDocumentInputEvent,\n\ttype BubblingEvent,\n\ttype ViewDocumentKeyDownEvent\n} from '@ckeditor/ckeditor5-engine';\n\nimport { env } from '@ckeditor/ckeditor5-utils';\n\nconst ENTER_EVENT_TYPES: Record<string, { isSoft: boolean }> = {\n\tinsertParagraph: { isSoft: false },\n\tinsertLineBreak: { isSoft: true }\n};\n\n/**\n * Enter observer introduces the {@link module:engine/view/document~Document#event:enter `Document#enter`} event.\n */\nexport default class EnterObserver extends Observer {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( view: View ) {\n\t\tsuper( view );\n\n\t\tconst doc = this.document;\n\t\tlet shiftPressed = false;\n\n\t\tdoc.on<ViewDocumentKeyDownEvent>( 'keydown', ( evt, data ) => {\n\t\t\tshiftPressed = data.shiftKey;\n\t\t} );\n\n\t\tdoc.on<ViewDocumentInputEvent>( 'beforeinput', ( evt, data ) => {\n\t\t\tif ( !this.isEnabled ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet inputType = data.inputType;\n\n\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/13321.\n\t\t\tif ( env.isSafari && shiftPressed && inputType == 'insertParagraph' ) {\n\t\t\t\tinputType = 'insertLineBreak';\n\t\t\t}\n\n\t\t\tconst domEvent = data.domEvent;\n\t\t\tconst enterEventSpec = ENTER_EVENT_TYPES[ inputType ];\n\n\t\t\tif ( !enterEventSpec ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst event = new BubblingEventInfo( doc, 'enter', data.targetRanges[ 0 ] );\n\n\t\t\tdoc.fire( event, new DomEventData( view, domEvent, {\n\t\t\t\tisSoft: enterEventSpec.isSoft\n\t\t\t} ) );\n\n\t\t\t// Stop `beforeinput` event if `enter` event was stopped.\n\t\t\t// https://github.com/ckeditor/ckeditor5/issues/753\n\t\t\tif ( event.stop.called ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic observe(): void {}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic stopObserving(): void {}\n}\n\n/**\n * Fired when the user presses the <kbd>Enter</kbd> key.\n *\n * Note: This event is fired by the {@link module:enter/enterobserver~EnterObserver observer}\n * (usually registered by the {@link module:enter/enter~Enter Enter feature} and\n * {@link module:enter/shiftenter~ShiftEnter ShiftEnter feature}).\n *\n * @eventName module:engine/view/document~Document#enter\n */\nexport type ViewDocumentEnterEvent = BubblingEvent<{\n\tname: 'enter';\n\targs: [ EnterEventData ];\n}>;\n\nexport interface EnterEventData extends DomEventData<InputEvent> {\n\n\t/**\n\t * Whether it is a soft enter (<kbd>Shift</kbd>+<kbd>Enter</kbd>) or a hard enter (<kbd>Enter</kbd>).\n\t */\n\tisSoft: boolean;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module enter/enter\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport EnterCommand from './entercommand';\nimport EnterObserver, { type ViewDocumentEnterEvent } from './enterobserver';\n\n/**\n * This plugin handles the <kbd>Enter</kbd> keystroke (hard line break) in the editor.\n *\n * See also the {@link module:enter/shiftenter~ShiftEnter} plugin.\n *\n * For more information about this feature see the {@glink api/enter package page}.\n */\nexport default class Enter extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Enter' as const;\n\t}\n\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\tview.addObserver( EnterObserver );\n\n\t\teditor.commands.add( 'enter', new EnterCommand( editor ) );\n\n\t\tthis.listenTo<ViewDocumentEnterEvent>( viewDocument, 'enter', ( evt, data ) => {\n\t\t\t// When not in composition, we handle the action, so prevent the default one.\n\t\t\t// When in composition, it's the browser who modify the DOM (renderer is disabled).\n\t\t\tif ( !viewDocument.isComposing ) {\n\t\t\t\tdata.preventDefault();\n\t\t\t}\n\n\t\t\t// The soft enter key is handled by the ShiftEnter plugin.\n\t\t\tif ( data.isSoft ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\teditor.execute( 'enter' );\n\n\t\t\tview.scrollToTheSelection();\n\t\t}, { priority: 'low' } );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module enter/shiftentercommand\n */\n\nimport { Command } from '@ckeditor/ckeditor5-core';\nimport { getCopyOnEnterAttributes } from './utils';\n\nimport type {\n\tDocumentSelection,\n\tModel,\n\tPosition,\n\tSchema,\n\tElement,\n\tWriter\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * ShiftEnter command. It is used by the {@link module:enter/shiftenter~ShiftEnter ShiftEnter feature} to handle\n * the <kbd>Shift</kbd>+<kbd>Enter</kbd> keystroke.\n */\nexport default class ShiftEnterCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\n\t\tmodel.change( writer => {\n\t\t\tsoftBreakAction( model, writer, doc.selection );\n\t\t\tthis.fire<ShiftEnterCommandAfterExecuteEvent>( 'afterExecute', { writer } );\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\n\t\tthis.isEnabled = isEnabled( model.schema, doc.selection );\n\t}\n}\n\n/**\n * Fired after the the {@link module:enter/shiftentercommand~ShiftEnterCommand} is finished executing.\n *\n * @eventName ~ShiftEnterCommand#afterExecute\n */\nexport type ShiftEnterCommandAfterExecuteEvent = {\n\tname: 'afterExecute';\n\targs: [ { writer: Writer } ];\n};\n\n/**\n * Checks whether the ShiftEnter command should be enabled in the specified selection.\n */\nfunction isEnabled( schema: Schema, selection: DocumentSelection ): boolean {\n\t// At this moment it is okay to support single range selections only.\n\t// But in the future we may need to change that.\n\tif ( selection.rangeCount > 1 ) {\n\t\treturn false;\n\t}\n\n\tconst anchorPos = selection.anchor;\n\n\t// Check whether the break element can be inserted in the current selection anchor.\n\tif ( !anchorPos || !schema.checkChild( anchorPos, 'softBreak' ) ) {\n\t\treturn false;\n\t}\n\n\tconst range = selection.getFirstRange()!;\n\tconst startElement = range.start.parent as Element;\n\tconst endElement = range.end.parent as Element;\n\n\t// Do not modify the content if selection is cross-limit elements.\n\tif ( ( isInsideLimitElement( startElement, schema ) || isInsideLimitElement( endElement, schema ) ) && startElement !== endElement ) {\n\t\treturn false;\n\t}\n\n\treturn true;\n}\n\n/**\n * Creates a break in the way that the <kbd>Shift</kbd>+<kbd>Enter</kbd> keystroke is expected to work.\n */\nfunction softBreakAction( model: Model, writer: Writer, selection: DocumentSelection ): void {\n\tconst isSelectionEmpty = selection.isCollapsed;\n\tconst range = selection.getFirstRange()!;\n\tconst startElement = range.start.parent as Element;\n\tconst endElement = range.end.parent as Element;\n\tconst isContainedWithinOneElement = ( startElement == endElement );\n\n\tif ( isSelectionEmpty ) {\n\t\tconst attributesToCopy = getCopyOnEnterAttributes( model.schema, selection.getAttributes() );\n\n\t\tinsertBreak( model, writer, range.end );\n\n\t\twriter.removeSelectionAttribute( selection.getAttributeKeys() );\n\t\twriter.setSelectionAttribute( attributesToCopy );\n\t} else {\n\t\tconst leaveUnmerged = !( range.start.isAtStart && range.end.isAtEnd );\n\n\t\tmodel.deleteContent( selection, { leaveUnmerged } );\n\n\t\t// Selection within one element:\n\t\t//\n\t\t// <h>x[xx]x</h>\t\t-> <h>x^x</h>\t\t\t-> <h>x<br>^x</h>\n\t\tif ( isContainedWithinOneElement ) {\n\t\t\tinsertBreak( model, writer, selection.focus! );\n\t\t}\n\t\t// Selection over multiple elements.\n\t\t//\n\t\t// <h>x[x</h><p>y]y<p>\t-> <h>x^</h><p>y</p>\t-> <h>x</h><p>^y</p>\n\t\t//\n\t\t// We chose not to insert a line break in this case because:\n\t\t//\n\t\t// * it's not a very common scenario,\n\t\t// * it actually surprised me when I saw the \"expected behavior\" in real life.\n\t\t//\n\t\t// It's ok if the user will need to be more specific where they want the <br> to be inserted.\n\t\telse {\n\t\t\t// Move the selection to the 2nd element (last step of the example above).\n\t\t\tif ( leaveUnmerged ) {\n\t\t\t\twriter.setSelection( endElement, 0 );\n\t\t\t}\n\t\t}\n\t}\n}\n\nfunction insertBreak( model: Model, writer: Writer, position: Position ): void {\n\tconst breakLineElement = writer.createElement( 'softBreak' );\n\n\tmodel.insertContent( breakLineElement, position );\n\twriter.setSelection( breakLineElement, 'after' );\n}\n\n/**\n * Checks whether the specified `element` is a child of the limit element.\n *\n * Checking whether the `<p>` element is inside a limit element:\n * - `<$root><p>Text.</p></$root> => false`\n * - `<$root><limitElement><p>Text</p></limitElement></$root> => true`\n */\nfunction isInsideLimitElement( element: Element, schema: Schema ): boolean {\n\t// `$root` is a limit element but in this case is an invalid element.\n\tif ( element.is( 'rootElement' ) ) {\n\t\treturn false;\n\t}\n\n\treturn schema.isLimit( element ) || isInsideLimitElement( element.parent as Element, schema );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module enter/shiftenter\n */\n\nimport ShiftEnterCommand from './shiftentercommand';\nimport EnterObserver, { type ViewDocumentEnterEvent } from './enterobserver';\nimport { Plugin } from '@ckeditor/ckeditor5-core';\n\n/**\n * This plugin handles the <kbd>Shift</kbd>+<kbd>Enter</kbd> keystroke (soft line break) in the editor.\n *\n * See also the {@link module:enter/enter~Enter} plugin.\n *\n * For more information about this feature see the {@glink api/enter package page}.\n */\nexport default class ShiftEnter extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ShiftEnter' as const;\n\t}\n\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\t\tconst conversion = editor.conversion;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\t// Configure the schema.\n\t\tschema.register( 'softBreak', {\n\t\t\tallowWhere: '$text',\n\t\t\tisInline: true\n\t\t} );\n\n\t\t// Configure converters.\n\t\tconversion.for( 'upcast' )\n\t\t\t.elementToElement( {\n\t\t\t\tmodel: 'softBreak',\n\t\t\t\tview: 'br'\n\t\t\t} );\n\n\t\tconversion.for( 'downcast' )\n\t\t\t.elementToElement( {\n\t\t\t\tmodel: 'softBreak',\n\t\t\t\tview: ( modelElement, { writer } ) => writer.createEmptyElement( 'br' )\n\t\t\t} );\n\n\t\tview.addObserver( EnterObserver );\n\n\t\teditor.commands.add( 'shiftEnter', new ShiftEnterCommand( editor ) );\n\n\t\tthis.listenTo<ViewDocumentEnterEvent>( viewDocument, 'enter', ( evt, data ) => {\n\t\t\t// When not in composition, we handle the action, so prevent the default one.\n\t\t\t// When in composition, it's the browser who modify the DOM (renderer is disabled).\n\t\t\tif ( !viewDocument.isComposing ) {\n\t\t\t\tdata.preventDefault();\n\t\t\t}\n\n\t\t\t// The hard enter key is handled by the Enter plugin.\n\t\t\tif ( !data.isSoft ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\teditor.execute( 'shiftEnter' );\n\t\t\tview.scrollToTheSelection();\n\t\t}, { priority: 'low' } );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module widget/highlightstack\n */\n\nimport { EmitterMixin, type ArrayOrItem } from '@ckeditor/ckeditor5-utils';\n\nimport type { DowncastWriter, HighlightDescriptor } from '@ckeditor/ckeditor5-engine';\n\n/**\n * Class used to handle the correct order of highlights on elements.\n *\n * When different highlights are applied to same element the correct order should be preserved:\n *\n * * highlight with highest priority should be applied,\n * * if two highlights have same priority - sort by CSS class provided in\n * {@link module:engine/conversion/downcasthelpers~HighlightDescriptor}.\n *\n * This way, highlight will be applied with the same rules it is applied on texts.\n */\nexport default class HighlightStack extends EmitterMixin() {\n\tprivate readonly _stack: Array<HighlightDescriptor> = [];\n\n\t/**\n\t * Adds highlight descriptor to the stack.\n\t *\n\t * @fires change:top\n\t */\n\tpublic add( descriptor: HighlightDescriptor, writer: DowncastWriter ): void {\n\t\tconst stack = this._stack;\n\n\t\t// Save top descriptor and insert new one. If top is changed - fire event.\n\t\tconst oldTop = stack[ 0 ];\n\t\tthis._insertDescriptor( descriptor );\n\t\tconst newTop = stack[ 0 ];\n\n\t\t// When new object is at the top and stores different information.\n\t\tif ( oldTop !== newTop && !compareDescriptors( oldTop, newTop ) ) {\n\t\t\tthis.fire<HighlightStackChangeEvent>( 'change:top', {\n\t\t\t\toldDescriptor: oldTop,\n\t\t\t\tnewDescriptor: newTop,\n\t\t\t\twriter\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Removes highlight descriptor from the stack.\n\t *\n\t * @fires change:top\n\t * @param id Id of the descriptor to remove.\n\t */\n\tpublic remove( id: string, writer: DowncastWriter ): void {\n\t\tconst stack = this._stack;\n\n\t\tconst oldTop = stack[ 0 ];\n\t\tthis._removeDescriptor( id );\n\t\tconst newTop = stack[ 0 ];\n\n\t\t// When new object is at the top and stores different information.\n\t\tif ( oldTop !== newTop && !compareDescriptors( oldTop, newTop ) ) {\n\t\t\tthis.fire<HighlightStackChangeEvent>( 'change:top', {\n\t\t\t\toldDescriptor: oldTop,\n\t\t\t\tnewDescriptor: newTop,\n\t\t\t\twriter\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Inserts a given descriptor in correct place in the stack. It also takes care about updating information\n\t * when descriptor with same id is already present.\n\t */\n\tprivate _insertDescriptor( descriptor: HighlightDescriptor ) {\n\t\tconst stack = this._stack;\n\t\tconst index = stack.findIndex( item => item.id === descriptor.id );\n\n\t\t// Inserting exact same descriptor - do nothing.\n\t\tif ( compareDescriptors( descriptor, stack[ index ] ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// If descriptor with same id but with different information is on the stack - remove it.\n\t\tif ( index > -1 ) {\n\t\t\tstack.splice( index, 1 );\n\t\t}\n\n\t\t// Find correct place to insert descriptor in the stack.\n\t\t// It has different information (for example priority) so it must be re-inserted in correct place.\n\t\tlet i = 0;\n\n\t\twhile ( stack[ i ] && shouldABeBeforeB( stack[ i ], descriptor ) ) {\n\t\t\ti++;\n\t\t}\n\n\t\tstack.splice( i, 0, descriptor );\n\t}\n\n\t/**\n\t * Removes descriptor with given id from the stack.\n\t *\n\t * @param id Descriptor's id.\n\t */\n\tprivate _removeDescriptor( id: string ) {\n\t\tconst stack = this._stack;\n\t\tconst index = stack.findIndex( item => item.id === id );\n\n\t\t// If descriptor with same id is on the list - remove it.\n\t\tif ( index > -1 ) {\n\t\t\tstack.splice( index, 1 );\n\t\t}\n\t}\n}\n\n/**\n * Compares two descriptors by checking their priority and class list.\n *\n * @returns Returns true if both descriptors are defined and have same priority and classes.\n */\nfunction compareDescriptors( a: HighlightDescriptor, b: HighlightDescriptor ) {\n\treturn a && b && a.priority == b.priority && classesToString( a.classes ) == classesToString( b.classes );\n}\n\n/**\n * Checks whenever first descriptor should be placed in the stack before second one.\n */\nfunction shouldABeBeforeB( a: HighlightDescriptor, b: HighlightDescriptor ) {\n\tif ( a.priority! > b.priority! ) {\n\t\treturn true;\n\t} else if ( a.priority! < b.priority! ) {\n\t\treturn false;\n\t}\n\n\t// When priorities are equal and names are different - use classes to compare.\n\treturn classesToString( a.classes ) > classesToString( b.classes );\n}\n\n/**\n * Converts CSS classes passed with {@link module:engine/conversion/downcasthelpers~HighlightDescriptor} to\n * sorted string.\n */\nfunction classesToString( classes: ArrayOrItem<string> ) {\n\treturn Array.isArray( classes ) ? classes.sort().join( ',' ) : classes;\n}\n\n/**\n * Fired when top element on {@link module:widget/highlightstack~HighlightStack} has been changed\n *\n * @eventName ~HighlightStack#change:top\n */\nexport type HighlightStackChangeEvent = {\n\tname: 'change' | 'change:top';\n\targs: [ HighlightStackChangeEventData ];\n};\n\n/**\n * Additional information about the change.\n */\nexport type HighlightStackChangeEventData = {\n\n\t/**\n\t * Old highlight descriptor. It will be `undefined` when first descriptor is added to the stack.\n\t */\n\toldDescriptor: HighlightDescriptor;\n\n\t/**\n\t * New highlight descriptor. It will be `undefined` when last descriptor is removed from the stack.\n\t */\n\tnewDescriptor: HighlightDescriptor;\n\n\t/**\n\t * View writer that can be used to modify element.\n\t */\n\twriter: DowncastWriter;\n};\n","export default \"<svg viewBox=\\\"0 0 16 16\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M4 0v1H1v3H0V.5A.5.5 0 0 1 .5 0H4zm8 0h3.5a.5.5 0 0 1 .5.5V4h-1V1h-3V0zM4 16H.5a.5.5 0 0 1-.5-.5V12h1v3h3v1zm8 0v-1h3v-3h1v3.5a.5.5 0 0 1-.5.5H12z\\\"/><path fill-opacity=\\\".256\\\" d=\\\"M1 1h14v14H1z\\\"/><g class=\\\"ck-icon__selected-indicator\\\"><path d=\\\"M7 0h2v1H7V0zM0 7h1v2H0V7zm15 0h1v2h-1V7zm-8 8h2v1H7v-1z\\\"/><path fill-opacity=\\\".254\\\" d=\\\"M1 1h14v14H1z\\\"/></g></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module widget/utils\n */\n\nimport {\n\tCKEditorError,\n\ttoArray,\n\ttype ObservableChangeEvent,\n\ttype GetCallback\n} from '@ckeditor/ckeditor5-utils';\n\nimport {\n\tfindOptimalInsertionRange as engineFindOptimalInsertionRange,\n\ttype AddHighlightCallback,\n\ttype HighlightDescriptor,\n\ttype RemoveHighlightCallback,\n\ttype MapperViewToModelPositionEvent,\n\ttype DocumentSelection,\n\ttype DowncastWriter,\n\ttype Model,\n\ttype Range,\n\ttype Selection,\n\ttype ViewContainerElement,\n\ttype ViewEditableElement,\n\ttype ViewElement,\n\ttype ViewTypeCheckable\n} from '@ckeditor/ckeditor5-engine';\n\nimport { IconView } from '@ckeditor/ckeditor5-ui';\n\nimport HighlightStack, { type HighlightStackChangeEvent } from './highlightstack';\nimport { getTypeAroundFakeCaretPosition } from './widgettypearound/utils';\n\nimport dragHandleIcon from '../theme/icons/drag-handle.svg';\n\n/**\n * CSS class added to each widget element.\n */\nexport const WIDGET_CLASS_NAME = 'ck-widget';\n\n/**\n * CSS class added to currently selected widget element.\n */\nexport const WIDGET_SELECTED_CLASS_NAME = 'ck-widget_selected';\n\n/**\n * Returns `true` if given {@link module:engine/view/node~Node} is an {@link module:engine/view/element~Element} and a widget.\n */\nexport function isWidget( node: ViewTypeCheckable ): boolean {\n\tif ( !node.is( 'element' ) ) {\n\t\treturn false;\n\t}\n\n\treturn !!node.getCustomProperty( 'widget' );\n}\n\n/**\n * Converts the given {@link module:engine/view/element~Element} to a widget in the following way:\n *\n * * sets the `contenteditable` attribute to `\"false\"`,\n * * adds the `ck-widget` CSS class,\n * * adds a custom {@link module:engine/view/element~Element#getFillerOffset `getFillerOffset()`} method returning `null`,\n * * adds a custom property allowing to recognize widget elements by using {@link ~isWidget `isWidget()`},\n * * implements the {@link ~setHighlightHandling view highlight on widgets}.\n *\n * This function needs to be used in conjunction with\n * {@link module:engine/conversion/downcasthelpers~DowncastHelpers downcast conversion helpers}\n * like {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToElement `elementToElement()`}.\n * Moreover, typically you will want to use `toWidget()` only for `editingDowncast`, while keeping the `dataDowncast` clean.\n *\n * For example, in order to convert a `<widget>` model element to `<div class=\"widget\">` in the view, you can define\n * such converters:\n *\n * ```ts\n * editor.conversion.for( 'editingDowncast' )\n * \t.elementToElement( {\n * \t\tmodel: 'widget',\n * \t\tview: ( modelItem, { writer } ) => {\n * \t\t\tconst div = writer.createContainerElement( 'div', { class: 'widget' } );\n *\n * \t\t\treturn toWidget( div, writer, { label: 'some widget' } );\n * \t\t}\n * \t} );\n *\n * editor.conversion.for( 'dataDowncast' )\n * \t.elementToElement( {\n * \t\tmodel: 'widget',\n * \t\tview: ( modelItem, { writer } ) => {\n * \t\t\treturn writer.createContainerElement( 'div', { class: 'widget' } );\n * \t\t}\n * \t} );\n * ```\n *\n * See the full source code of the widget (with a nested editable) schema definition and converters in\n * [this sample](https://github.com/ckeditor/ckeditor5-widget/blob/master/tests/manual/widget-with-nestededitable.js).\n *\n * @param options Additional options.\n * @param options.label Element's label provided to the {@link ~setLabel} function. It can be passed as\n * a plain string or a function returning a string. It represents the widget for assistive technologies (like screen readers).\n * @param options.hasSelectionHandle If `true`, the widget will have a selection handle added.\n * @returns Returns the same element.\n */\nexport function toWidget(\n\telement: ViewElement,\n\twriter: DowncastWriter,\n\toptions: {\n\t\tlabel?: string | ( () => string );\n\t\thasSelectionHandle?: boolean;\n\t} = {}\n): ViewElement {\n\tif ( !element.is( 'containerElement' ) ) {\n\t\t/**\n\t\t * The element passed to `toWidget()` must be a {@link module:engine/view/containerelement~ContainerElement}\n\t\t * instance.\n\t\t *\n\t\t * @error widget-to-widget-wrong-element-type\n\t\t * @param element The view element passed to `toWidget()`.\n\t\t */\n\t\tthrow new CKEditorError(\n\t\t\t'widget-to-widget-wrong-element-type',\n\t\t\tnull,\n\t\t\t{ element }\n\t\t);\n\t}\n\n\twriter.setAttribute( 'contenteditable', 'false', element );\n\n\twriter.addClass( WIDGET_CLASS_NAME, element );\n\twriter.setCustomProperty( 'widget', true, element );\n\telement.getFillerOffset = getFillerOffset;\n\n\twriter.setCustomProperty( 'widgetLabel', [], element );\n\n\tif ( options.label ) {\n\t\tsetLabel( element, options.label );\n\t}\n\n\tif ( options.hasSelectionHandle ) {\n\t\taddSelectionHandle( element, writer );\n\t}\n\n\tsetHighlightHandling( element, writer );\n\n\treturn element;\n}\n\n/**\n * Default handler for adding a highlight on a widget.\n * It adds CSS class and attributes basing on the given highlight descriptor.\n */\nfunction addHighlight( element: ViewElement, descriptor: HighlightDescriptor, writer: DowncastWriter ) {\n\tif ( descriptor.classes ) {\n\t\twriter.addClass( toArray( descriptor.classes ), element );\n\t}\n\n\tif ( descriptor.attributes ) {\n\t\tfor ( const key in descriptor.attributes ) {\n\t\t\twriter.setAttribute( key, descriptor.attributes[ key ], element );\n\t\t}\n\t}\n}\n\n/**\n * Default handler for removing a highlight from a widget.\n * It removes CSS class and attributes basing on the given highlight descriptor.\n */\nfunction removeHighlight( element: ViewElement, descriptor: HighlightDescriptor, writer: DowncastWriter ) {\n\tif ( descriptor.classes ) {\n\t\twriter.removeClass( toArray( descriptor.classes ), element );\n\t}\n\n\tif ( descriptor.attributes ) {\n\t\tfor ( const key in descriptor.attributes ) {\n\t\t\twriter.removeAttribute( key, element );\n\t\t}\n\t}\n}\n\n/**\n * Sets highlight handling methods. Uses {@link module:widget/highlightstack~HighlightStack} to\n * properly determine which highlight descriptor should be used at given time.\n */\nexport function setHighlightHandling(\n\telement: ViewElement,\n\twriter: DowncastWriter,\n\tadd: ( element: ViewElement, descriptor: HighlightDescriptor, writer: DowncastWriter ) => void = addHighlight,\n\tremove: ( element: ViewElement, descriptor: HighlightDescriptor, writer: DowncastWriter ) => void = removeHighlight\n): void {\n\tconst stack = new HighlightStack();\n\n\tstack.on<HighlightStackChangeEvent>( 'change:top', ( evt, data ) => {\n\t\tif ( data.oldDescriptor ) {\n\t\t\tremove( element, data.oldDescriptor, data.writer );\n\t\t}\n\n\t\tif ( data.newDescriptor ) {\n\t\t\tadd( element, data.newDescriptor, data.writer );\n\t\t}\n\t} );\n\n\tconst addHighlightCallback: AddHighlightCallback = ( element, descriptor, writer ) => stack.add( descriptor, writer );\n\tconst removeHighlightCallback: RemoveHighlightCallback = ( element, id, writer ) => stack.remove( id, writer );\n\n\twriter.setCustomProperty( 'addHighlight', addHighlightCallback, element );\n\twriter.setCustomProperty( 'removeHighlight', removeHighlightCallback, element );\n}\n\n/**\n * Sets label for given element.\n * It can be passed as a plain string or a function returning a string. Function will be called each time label is retrieved by\n * {@link ~getLabel `getLabel()`}.\n */\nexport function setLabel( element: ViewElement, labelOrCreator: string | ( () => string ) ): void {\n\tconst widgetLabel = element.getCustomProperty( 'widgetLabel' ) as Array<string | ( () => string )>;\n\n\twidgetLabel.push( labelOrCreator );\n}\n\n/**\n * Returns the label of the provided element.\n */\nexport function getLabel( element: ViewElement ): string {\n\tconst widgetLabel = element.getCustomProperty( 'widgetLabel' ) as Array<string | ( () => string )>;\n\n\treturn widgetLabel.reduce( ( prev: string, current: string | ( () => string ) ) => {\n\t\tif ( typeof current === 'function' ) {\n\t\t\treturn prev ? prev + '. ' + current() : current();\n\t\t} else {\n\t\t\treturn prev ? prev + '. ' + current : current;\n\t\t}\n\t}, '' );\n}\n\n/**\n * Adds functionality to the provided {@link module:engine/view/editableelement~EditableElement} to act as a widget's editable:\n *\n * * sets the `contenteditable` attribute to `true` when {@link module:engine/view/editableelement~EditableElement#isReadOnly} is `false`,\n * otherwise sets it to `false`,\n * * adds the `ck-editor__editable` and `ck-editor__nested-editable` CSS classes,\n * * adds the `ck-editor__nested-editable_focused` CSS class when the editable is focused and removes it when it is blurred.\n * * implements the {@link ~setHighlightHandling view highlight on widget's editable}.\n *\n * Similarly to {@link ~toWidget `toWidget()`} this function should be used in `editingDowncast` only and it is usually\n * used together with {@link module:engine/conversion/downcasthelpers~DowncastHelpers#elementToElement `elementToElement()`}.\n *\n * For example, in order to convert a `<nested>` model element to `<div class=\"nested\">` in the view, you can define\n * such converters:\n *\n * ```ts\n * editor.conversion.for( 'editingDowncast' )\n * \t.elementToElement( {\n * \t\tmodel: 'nested',\n * \t\tview: ( modelItem, { writer } ) => {\n * \t\t\tconst div = writer.createEditableElement( 'div', { class: 'nested' } );\n *\n * \t\t\treturn toWidgetEditable( nested, writer, { label: 'label for editable' } );\n * \t\t}\n * \t} );\n *\n * editor.conversion.for( 'dataDowncast' )\n * \t.elementToElement( {\n * \t\tmodel: 'nested',\n * \t\tview: ( modelItem, { writer } ) => {\n * \t\t\treturn writer.createContainerElement( 'div', { class: 'nested' } );\n * \t\t}\n * \t} );\n * ```\n *\n * See the full source code of the widget (with nested editable) schema definition and converters in\n * [this sample](https://github.com/ckeditor/ckeditor5-widget/blob/master/tests/manual/widget-with-nestededitable.js).\n *\n * @param options Additional options.\n * @param options.label Editable's label used by assistive technologies (e.g. screen readers).\n * @returns Returns the same element that was provided in the `editable` parameter\n */\nexport function toWidgetEditable(\n\teditable: ViewEditableElement,\n\twriter: DowncastWriter,\n\toptions: {\n\t\tlabel?: string;\n\t} = {}\n): ViewEditableElement {\n\twriter.addClass( [ 'ck-editor__editable', 'ck-editor__nested-editable' ], editable );\n\n\twriter.setAttribute( 'role', 'textbox', editable );\n\n\tif ( options.label ) {\n\t\twriter.setAttribute( 'aria-label', options.label, editable );\n\t}\n\n\t// Set initial contenteditable value.\n\twriter.setAttribute( 'contenteditable', editable.isReadOnly ? 'false' : 'true', editable );\n\n\t// Bind the contenteditable property to element#isReadOnly.\n\teditable.on<ObservableChangeEvent<boolean>>( 'change:isReadOnly', ( evt, property, is ) => {\n\t\twriter.setAttribute( 'contenteditable', is ? 'false' : 'true', editable );\n\t} );\n\n\teditable.on<ObservableChangeEvent<boolean>>( 'change:isFocused', ( evt, property, is ) => {\n\t\tif ( is ) {\n\t\t\twriter.addClass( 'ck-editor__nested-editable_focused', editable );\n\t\t} else {\n\t\t\twriter.removeClass( 'ck-editor__nested-editable_focused', editable );\n\t\t}\n\t} );\n\n\tsetHighlightHandling( editable, writer );\n\n\treturn editable;\n}\n\n/**\n * Returns a model range which is optimal (in terms of UX) for inserting a widget block.\n *\n * For instance, if a selection is in the middle of a paragraph, the collapsed range before this paragraph\n * will be returned so that it is not split. If the selection is at the end of a paragraph,\n * the collapsed range after this paragraph will be returned.\n *\n * Note: If the selection is placed in an empty block, the range in that block will be returned. If that range\n * is then passed to {@link module:engine/model/model~Model#insertContent}, the block will be fully replaced\n * by the inserted widget block.\n *\n * @param selection The selection based on which the insertion position should be calculated.\n * @param model Model instance.\n * @returns The optimal range.\n */\nexport function findOptimalInsertionRange(\n\tselection: Selection | DocumentSelection,\n\tmodel: Model\n): Range {\n\tconst selectedElement = selection.getSelectedElement();\n\n\tif ( selectedElement ) {\n\t\tconst typeAroundFakeCaretPosition = getTypeAroundFakeCaretPosition( selection );\n\n\t\t// If the WidgetTypeAround \"fake caret\" is displayed, use its position for the insertion\n\t\t// to provide the most predictable UX (https://github.com/ckeditor/ckeditor5/issues/7438).\n\t\tif ( typeAroundFakeCaretPosition ) {\n\t\t\treturn model.createRange( model.createPositionAt( selectedElement, typeAroundFakeCaretPosition ) );\n\t\t}\n\t}\n\n\treturn engineFindOptimalInsertionRange( selection, model );\n}\n\n/**\n * A util to be used in order to map view positions to correct model positions when implementing a widget\n * which renders non-empty view element for an empty model element.\n *\n * For example:\n *\n * ```\n * // Model:\n * <placeholder type=\"name\"></placeholder>\n *\n * // View:\n * <span class=\"placeholder\">name</span>\n * ```\n *\n * In such case, view positions inside `<span>` cannot be correctly mapped to the model (because the model element is empty).\n * To handle mapping positions inside `<span class=\"placeholder\">` to the model use this util as follows:\n *\n * ```ts\n * editor.editing.mapper.on(\n * \t'viewToModelPosition',\n * \tviewToModelPositionOutsideModelElement( model, viewElement => viewElement.hasClass( 'placeholder' ) )\n * );\n * ```\n *\n * The callback will try to map the view offset of selection to an expected model position.\n *\n * 1. When the position is at the end (or in the middle) of the inline widget:\n *\n * ```\n * // View:\n * <p>foo <span class=\"placeholder\">name|</span> bar</p>\n *\n * // Model:\n * <paragraph>foo <placeholder type=\"name\"></placeholder>| bar</paragraph>\n * ```\n *\n * 2. When the position is at the beginning of the inline widget:\n *\n * ```\n * // View:\n * <p>foo <span class=\"placeholder\">|name</span> bar</p>\n *\n * // Model:\n * <paragraph>foo |<placeholder type=\"name\"></placeholder> bar</paragraph>\n * ```\n *\n * @param model Model instance on which the callback operates.\n * @param viewElementMatcher Function that is passed a view element and should return `true` if the custom mapping\n * should be applied to the given view element.\n */\nexport function viewToModelPositionOutsideModelElement(\n\tmodel: Model,\n\tviewElementMatcher: ( element: ViewElement ) => boolean\n): GetCallback<MapperViewToModelPositionEvent> {\n\treturn ( evt, data ) => {\n\t\tconst { mapper, viewPosition } = data;\n\n\t\tconst viewParent = mapper.findMappedViewAncestor( viewPosition );\n\n\t\tif ( !viewElementMatcher( viewParent ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst modelParent = mapper.toModelElement( viewParent );\n\n\t\tdata.modelPosition = model.createPositionAt( modelParent!, viewPosition.isAtStart ? 'before' : 'after' );\n\t};\n}\n\n/**\n * Default filler offset function applied to all widget elements.\n */\nfunction getFillerOffset() {\n\treturn null;\n}\n\n/**\n * Adds a drag handle to the widget.\n */\nfunction addSelectionHandle( widgetElement: ViewContainerElement, writer: DowncastWriter ) {\n\tconst selectionHandle = writer.createUIElement( 'div', { class: 'ck ck-widget__selection-handle' }, function( domDocument ) {\n\t\tconst domElement = this.toDomElement( domDocument );\n\n\t\t// Use the IconView from the ui library.\n\t\tconst icon = new IconView();\n\t\ticon.set( 'content', dragHandleIcon );\n\n\t\t// Render the icon view right away to append its #element to the selectionHandle DOM element.\n\t\ticon.render();\n\n\t\tdomElement.appendChild( icon.element! );\n\n\t\treturn domElement;\n\t} );\n\n\t// Append the selection handle into the widget wrapper.\n\twriter.insert( writer.createPositionAt( widgetElement, 0 ), selectionHandle );\n\twriter.addClass( [ 'ck-widget_with-selection-handle' ], widgetElement );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module widget/widgettypearound/utils\n */\n\nimport { isWidget } from '../utils';\n\nimport type {\n\tDocumentSelection,\n\tDomConverter,\n\tElement,\n\tSchema,\n\tSelection,\n\tViewElement\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * The name of the type around model selection attribute responsible for\n * displaying a fake caret next to a selected widget.\n */\nexport const TYPE_AROUND_SELECTION_ATTRIBUTE = 'widget-type-around';\n\n/**\n * Checks if an element is a widget that qualifies to get the widget type around UI.\n */\nexport function isTypeAroundWidget( viewElement: ViewElement | undefined, modelElement: Element, schema: Schema ): boolean {\n\treturn !!viewElement && isWidget( viewElement ) && !schema.isInline( modelElement );\n}\n\n/**\n * For the passed HTML element, this helper finds the closest widget type around button ancestor.\n */\nexport function getClosestTypeAroundDomButton( domElement: HTMLElement ): HTMLElement | null {\n\treturn domElement.closest( '.ck-widget__type-around__button' );\n}\n\n/**\n * For the passed widget type around button element, this helper determines at which position\n * the paragraph would be inserted into the content if, for instance, the button was\n * clicked by the user.\n *\n * @returns The position of the button.\n */\nexport function getTypeAroundButtonPosition( domElement: HTMLElement ): 'before' | 'after' {\n\treturn domElement.classList.contains( 'ck-widget__type-around__button_before' ) ? 'before' : 'after';\n}\n\n/**\n * For the passed HTML element, this helper returns the closest view widget ancestor.\n */\nexport function getClosestWidgetViewElement( domElement: HTMLElement, domConverter: DomConverter ): ViewElement {\n\tconst widgetDomElement = domElement.closest( '.ck-widget' );\n\n\treturn domConverter.mapDomToView( widgetDomElement as any ) as ViewElement;\n}\n\n/**\n * For the passed selection instance, it returns the position of the fake caret displayed next to a widget.\n *\n * **Note**: If the fake caret is not currently displayed, `null` is returned.\n *\n * @returns The position of the fake caret or `null` when none is present.\n */\nexport function getTypeAroundFakeCaretPosition( selection: Selection | DocumentSelection ): 'before' | 'after' | null {\n\treturn selection.getAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE ) as any;\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./widgettypearound.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* global DOMParser */\n\n/**\n * @module widget/widgettypearound/widgettypearound\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\n\nimport { Template } from '@ckeditor/ckeditor5-ui';\n\nimport {\n\tEnter,\n\ttype ViewDocumentEnterEvent\n} from '@ckeditor/ckeditor5-enter';\n\nimport {\n\tDelete,\n\ttype ViewDocumentDeleteEvent,\n\ttype ViewDocumentInsertTextEvent\n} from '@ckeditor/ckeditor5-typing';\n\nimport {\n\tenv,\n\tisForwardArrowKeyCode,\n\ttype BaseEvent,\n\ttype Emitter,\n\ttype GetCallback,\n\ttype GetCallbackOptions,\n\ttype ObservableChangeEvent,\n\ttype KeystrokeInfo\n} from '@ckeditor/ckeditor5-utils';\n\nimport type {\n\tBubblingEventInfo,\n\tDocumentChangeEvent,\n\tDomEventData,\n\tDowncastInsertEvent,\n\tDowncastSelectionEvent,\n\tDowncastWriter,\n\tElement,\n\tSchema,\n\tSelectionChangeRangeEvent,\n\tViewDocumentArrowKeyEvent,\n\tViewDocumentCompositionStartEvent,\n\tViewDocumentKeyDownEvent,\n\tViewDocumentMouseDownEvent,\n\tViewElement,\n\tModelDeleteContentEvent,\n\tModelInsertContentEvent,\n\tModelInsertObjectEvent\n} from '@ckeditor/ckeditor5-engine';\n\nimport {\n\tisTypeAroundWidget,\n\tgetClosestTypeAroundDomButton,\n\tgetTypeAroundButtonPosition,\n\tgetClosestWidgetViewElement,\n\tgetTypeAroundFakeCaretPosition,\n\tTYPE_AROUND_SELECTION_ATTRIBUTE\n} from './utils';\n\nimport { isWidget } from '../utils';\nimport type Widget from '../widget';\n\nimport returnIcon from '../../theme/icons/return-arrow.svg';\n\nimport '../../theme/widgettypearound.css';\n\nconst POSSIBLE_INSERTION_POSITIONS = [ 'before', 'after' ] as const;\n\n// Do the SVG parsing once and then clone the result <svg> DOM element for each new button.\nconst RETURN_ARROW_ICON_ELEMENT = new DOMParser().parseFromString( returnIcon, 'image/svg+xml' ).firstChild!;\n\nconst PLUGIN_DISABLED_EDITING_ROOT_CLASS = 'ck-widget__type-around_disabled';\n\n/**\n * A plugin that allows users to type around widgets where normally it is impossible to place the caret due\n * to limitations of web browsers. These \"tight spots\" occur, for instance, before (or after) a widget being\n * the first (or last) child of its parent or between two block widgets.\n *\n * This plugin extends the {@link module:widget/widget~Widget `Widget`} plugin and injects the user interface\n * with two buttons into each widget instance in the editor. Each of the buttons can be clicked by the\n * user if the widget is next to the \"tight spot\". Once clicked, a paragraph is created with the selection anchored\n * in it so that users can type (or insert content, paste, etc.) straight away.\n */\nexport default class WidgetTypeAround extends Plugin {\n\t/**\n\t * A reference to the model widget element that has the fake caret active\n\t * on either side of it. It is later used to remove CSS classes associated with the fake caret\n\t * when the widget no longer needs it.\n\t */\n\tprivate _currentFakeCaretModelElement: Element | null = null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'WidgetTypeAround' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Enter, Delete ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst editingView = editor.editing.view;\n\n\t\t// Set a CSS class on the view editing root when the plugin is disabled so all the buttons\n\t\t// and lines visually disappear. All the interactions are disabled in individual plugin methods.\n\t\tthis.on<ObservableChangeEvent<boolean>>( 'change:isEnabled', ( evt, data, isEnabled ) => {\n\t\t\teditingView.change( writer => {\n\t\t\t\tfor ( const root of editingView.document.roots ) {\n\t\t\t\t\tif ( isEnabled ) {\n\t\t\t\t\t\twriter.removeClass( PLUGIN_DISABLED_EDITING_ROOT_CLASS, root );\n\t\t\t\t\t} else {\n\t\t\t\t\t\twriter.addClass( PLUGIN_DISABLED_EDITING_ROOT_CLASS, root );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tif ( !isEnabled ) {\n\t\t\t\teditor.model.change( writer => {\n\t\t\t\t\twriter.removeSelectionAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE );\n\t\t\t\t} );\n\t\t\t}\n\t\t} );\n\n\t\tthis._enableTypeAroundUIInjection();\n\t\tthis._enableInsertingParagraphsOnButtonClick();\n\t\tthis._enableInsertingParagraphsOnEnterKeypress();\n\t\tthis._enableInsertingParagraphsOnTypingKeystroke();\n\t\tthis._enableTypeAroundFakeCaretActivationUsingKeyboardArrows();\n\t\tthis._enableDeleteIntegration();\n\t\tthis._enableInsertContentIntegration();\n\t\tthis._enableInsertObjectIntegration();\n\t\tthis._enableDeleteContentIntegration();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis._currentFakeCaretModelElement = null;\n\t}\n\n\t/**\n\t * Inserts a new paragraph next to a widget element with the selection anchored in it.\n\t *\n\t * **Note**: This method is heavily user-oriented and will both focus the editing view and scroll\n\t * the viewport to the selection in the inserted paragraph.\n\t *\n\t * @param widgetModelElement The model widget element next to which a paragraph is inserted.\n\t * @param position The position where the paragraph is inserted. Either `'before'` or `'after'` the widget.\n\t */\n\tprivate _insertParagraph( widgetModelElement: Element, position: 'before' | 'after' ) {\n\t\tconst editor = this.editor;\n\t\tconst editingView = editor.editing.view;\n\n\t\tconst attributesToCopy = editor.model.schema.getAttributesWithProperty( widgetModelElement, 'copyOnReplace', true );\n\n\t\teditor.execute( 'insertParagraph', {\n\t\t\tposition: editor.model.createPositionAt( widgetModelElement, position ),\n\t\t\tattributes: attributesToCopy\n\t\t} );\n\n\t\teditingView.focus();\n\t\teditingView.scrollToTheSelection();\n\t}\n\n\t/**\n\t * A wrapper for the {@link module:utils/emittermixin~Emitter#listenTo} method that executes the callbacks only\n\t * when the plugin {@link #isEnabled is enabled}.\n\t *\n\t * @param emitter The object that fires the event.\n\t * @param event The name of the event.\n\t * @param callback The function to be called on event.\n\t * @param options Additional options.\n\t * @param options.priority The priority of this event callback. The higher the priority value the sooner\n\t * the callback will be fired. Events having the same priority are called in the order they were added.\n\t */\n\tprivate _listenToIfEnabled<TEvent extends BaseEvent>(\n\t\temitter: Emitter,\n\t\tevent: TEvent[ 'name' ],\n\t\tcallback: OmitThisParameter<GetCallback<TEvent>>,\n\t\toptions?: GetCallbackOptions<TEvent>\n\t): void {\n\t\tthis.listenTo( emitter, event, ( ...args ) => {\n\t\t\t// Do not respond if the plugin is disabled.\n\t\t\tif ( this.isEnabled ) {\n\t\t\t\tcallback( ...args );\n\t\t\t}\n\t\t}, options );\n\t}\n\n\t/**\n\t * Similar to {@link #_insertParagraph}, this method inserts a paragraph except that it\n\t * does not expect a position. Instead, it performs the insertion next to a selected widget\n\t * according to the `widget-type-around` model selection attribute value (fake caret position).\n\t *\n\t * Because this method requires the `widget-type-around` attribute to be set,\n\t * the insertion can only happen when the widget's fake caret is active (e.g. activated\n\t * using the keyboard).\n\t *\n\t * @returns Returns `true` when the paragraph was inserted (the attribute was present) and `false` otherwise.\n\t */\n\tprivate _insertParagraphAccordingToFakeCaretPosition() {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst modelSelection = model.document.selection;\n\t\tconst typeAroundFakeCaretPosition = getTypeAroundFakeCaretPosition( modelSelection );\n\n\t\tif ( !typeAroundFakeCaretPosition ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// @if CK_DEBUG_TYPING // if ( ( window as any ).logCKETyping ) {\n\t\t// @if CK_DEBUG_TYPING // \tconsole.info( '%c[WidgetTypeAround]%c Fake caret -> insert paragraph',\n\t\t// @if CK_DEBUG_TYPING // \t\t'font-weight: bold; color: green', ''\n\t\t// @if CK_DEBUG_TYPING // \t);\n\t\t// @if CK_DEBUG_TYPING // }\n\n\t\tconst selectedModelElement = modelSelection.getSelectedElement()!;\n\n\t\tthis._insertParagraph( selectedModelElement, typeAroundFakeCaretPosition );\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Creates a listener in the editing conversion pipeline that injects the widget type around\n\t * UI into every single widget instance created in the editor.\n\t *\n\t * The UI is delivered as a {@link module:engine/view/uielement~UIElement}\n\t * wrapper which renders DOM buttons that users can use to insert paragraphs.\n\t */\n\tprivate _enableTypeAroundUIInjection() {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\t\tconst t = editor.locale.t;\n\t\tconst buttonTitles = {\n\t\t\tbefore: t( 'Insert paragraph before block' ),\n\t\t\tafter: t( 'Insert paragraph after block' )\n\t\t};\n\n\t\teditor.editing.downcastDispatcher.on<DowncastInsertEvent<Element>>( 'insert', ( evt, data, conversionApi ) => {\n\t\t\tconst viewElement = conversionApi.mapper.toViewElement( data.item );\n\n\t\t\tif ( !viewElement ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Filter out non-widgets and inline widgets.\n\t\t\tif ( isTypeAroundWidget( viewElement, data.item, schema ) ) {\n\t\t\t\tinjectUIIntoWidget( conversionApi.writer, buttonTitles, viewElement! );\n\n\t\t\t\tconst widgetLabel = viewElement.getCustomProperty( 'widgetLabel' ) as Array<string | ( () => string )>;\n\n\t\t\t\twidgetLabel.push( () => {\n\t\t\t\t\treturn this.isEnabled ? t( 'Press Enter to type after or press Shift + Enter to type before the widget' ) : '';\n\t\t\t\t} );\n\t\t\t}\n\t\t}, { priority: 'low' } );\n\t}\n\n\t/**\n\t * Brings support for the fake caret that appears when either:\n\t *\n\t * * the selection moves to a widget from a position next to it using arrow keys,\n\t * * the arrow key is pressed when the widget is already selected.\n\t *\n\t * The fake caret lets the user know that they can start typing or just press\n\t * <kbd>Enter</kbd> to insert a paragraph at the position next to a widget as suggested by the fake caret.\n\t *\n\t * The fake caret disappears when the user changes the selection or the editor\n\t * gets blurred.\n\t *\n\t * The whole idea is as follows:\n\t *\n\t * 1. A user does one of the 2 scenarios described at the beginning.\n\t * 2. The \"keydown\" listener is executed and the decision is made whether to show or hide the fake caret.\n\t * 3. If it should show up, the `widget-type-around` model selection attribute is set indicating\n\t * on which side of the widget it should appear.\n\t * 4. The selection dispatcher reacts to the selection attribute and sets CSS classes responsible for the\n\t * fake caret on the view widget.\n\t * 5. If the fake caret should disappear, the selection attribute is removed and the dispatcher\n\t * does the CSS class clean-up in the view.\n\t * 6. Additionally, `change:range` and `FocusTracker#isFocused` listeners also remove the selection\n\t * attribute (the former also removes widget CSS classes).\n\t */\n\tprivate _enableTypeAroundFakeCaretActivationUsingKeyboardArrows() {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst modelSelection = model.document.selection;\n\t\tconst schema = model.schema;\n\t\tconst editingView = editor.editing.view;\n\n\t\t// This is the main listener responsible for the fake caret.\n\t\t// Note: The priority must precede the default Widget class keydown handler (\"high\").\n\t\tthis._listenToIfEnabled<ViewDocumentArrowKeyEvent>( editingView.document, 'arrowKey', ( evt, domEventData ) => {\n\t\t\tthis._handleArrowKeyPress( evt, domEventData );\n\t\t}, { context: [ isWidget, '$text' ], priority: 'high' } );\n\n\t\t// This listener makes sure the widget type around selection attribute will be gone from the model\n\t\t// selection as soon as the model range changes. This attribute only makes sense when a widget is selected\n\t\t// (and the \"fake horizontal caret\" is visible) so whenever the range changes (e.g. selection moved somewhere else),\n\t\t// let's get rid of the attribute so that the selection downcast dispatcher isn't even bothered.\n\t\tthis._listenToIfEnabled<SelectionChangeRangeEvent>( modelSelection, 'change:range', ( evt, data ) => {\n\t\t\t// Do not reset the selection attribute when the change was indirect.\n\t\t\tif ( !data.directChange ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Get rid of the widget type around attribute of the selection on every change:range.\n\t\t\t// If the range changes, it means for sure, the user is no longer in the active (\"fake horizontal caret\") mode.\n\t\t\teditor.model.change( writer => {\n\t\t\t\twriter.removeSelectionAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE );\n\t\t\t} );\n\t\t} );\n\n\t\t// Get rid of the widget type around attribute of the selection on every document change\n\t\t// that makes widget not selected any more (i.e. widget was removed).\n\t\tthis._listenToIfEnabled<DocumentChangeEvent>( model.document, 'change:data', () => {\n\t\t\tconst selectedModelElement = modelSelection.getSelectedElement();\n\n\t\t\tif ( selectedModelElement ) {\n\t\t\t\tconst selectedViewElement = editor.editing.mapper.toViewElement( selectedModelElement );\n\n\t\t\t\tif ( isTypeAroundWidget( selectedViewElement, selectedModelElement, schema ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\teditor.model.change( writer => {\n\t\t\t\twriter.removeSelectionAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE );\n\t\t\t} );\n\t\t} );\n\n\t\t// React to changes of the model selection attribute made by the arrow keys listener.\n\t\t// If the block widget is selected and the attribute changes, downcast the attribute to special\n\t\t// CSS classes associated with the active (\"fake horizontal caret\") mode of the widget.\n\t\tthis._listenToIfEnabled<DowncastSelectionEvent>( editor.editing.downcastDispatcher, 'selection', ( evt, data, conversionApi ) => {\n\t\t\tconst writer = conversionApi.writer;\n\n\t\t\tif ( this._currentFakeCaretModelElement ) {\n\t\t\t\tconst selectedViewElement = conversionApi.mapper.toViewElement( this._currentFakeCaretModelElement );\n\n\t\t\t\tif ( selectedViewElement ) {\n\t\t\t\t\t// Get rid of CSS classes associated with the active (\"fake horizontal caret\") mode from the view widget.\n\t\t\t\t\twriter.removeClass( POSSIBLE_INSERTION_POSITIONS.map( positionToWidgetCssClass ), selectedViewElement );\n\n\t\t\t\t\tthis._currentFakeCaretModelElement = null;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst selectedModelElement = data.selection.getSelectedElement();\n\n\t\t\tif ( !selectedModelElement ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst selectedViewElement = conversionApi.mapper.toViewElement( selectedModelElement );\n\n\t\t\tif ( !isTypeAroundWidget( selectedViewElement, selectedModelElement, schema ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst typeAroundFakeCaretPosition = getTypeAroundFakeCaretPosition( data.selection );\n\n\t\t\tif ( !typeAroundFakeCaretPosition ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\twriter.addClass( positionToWidgetCssClass( typeAroundFakeCaretPosition ), selectedViewElement! );\n\n\t\t\t// Remember the view widget that got the \"fake-caret\" CSS class. This class should be removed ASAP when the\n\t\t\t// selection changes\n\t\t\tthis._currentFakeCaretModelElement = selectedModelElement;\n\t\t} );\n\n\t\tthis._listenToIfEnabled<ObservableChangeEvent<boolean>>( editor.ui.focusTracker, 'change:isFocused', ( evt, name, isFocused ) => {\n\t\t\tif ( !isFocused ) {\n\t\t\t\teditor.model.change( writer => {\n\t\t\t\t\twriter.removeSelectionAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE );\n\t\t\t\t} );\n\t\t\t}\n\t\t} );\n\n\t\tfunction positionToWidgetCssClass( position: 'before' | 'after' ) {\n\t\t\treturn `ck-widget_type-around_show-fake-caret_${ position }`;\n\t\t}\n\t}\n\n\t/**\n\t * A listener executed on each \"keydown\" in the view document, a part of\n\t * {@link #_enableTypeAroundFakeCaretActivationUsingKeyboardArrows}.\n\t *\n\t * It decides whether the arrow keypress should activate the fake caret or not (also whether it should\n\t * be deactivated).\n\t *\n\t * The fake caret activation is done by setting the `widget-type-around` model selection attribute\n\t * in this listener, and stopping and preventing the event that would normally be handled by the widget\n\t * plugin that is responsible for the regular keyboard navigation near/across all widgets (that\n\t * includes inline widgets, which are ignored by the widget type around plugin).\n\t */\n\tprivate _handleArrowKeyPress( evt: BubblingEventInfo<'arrowKey'>, domEventData: DomEventData & KeystrokeInfo ) {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst modelSelection = model.document.selection;\n\t\tconst schema = model.schema;\n\t\tconst editingView = editor.editing.view;\n\n\t\tconst keyCode = domEventData.keyCode;\n\t\tconst isForward = isForwardArrowKeyCode( keyCode, editor.locale.contentLanguageDirection );\n\t\tconst selectedViewElement = editingView.document.selection.getSelectedElement()!;\n\t\tconst selectedModelElement = editor.editing.mapper.toModelElement( selectedViewElement )!;\n\t\tlet shouldStopAndPreventDefault;\n\n\t\t// Handle keyboard navigation when a type-around-compatible widget is currently selected.\n\t\tif ( isTypeAroundWidget( selectedViewElement, selectedModelElement, schema ) ) {\n\t\t\tshouldStopAndPreventDefault = this._handleArrowKeyPressOnSelectedWidget( isForward );\n\t\t}\n\t\t// Handle keyboard arrow navigation when the selection is next to a type-around-compatible widget\n\t\t// and the widget is about to be selected.\n\t\telse if ( modelSelection.isCollapsed ) {\n\t\t\tshouldStopAndPreventDefault = this._handleArrowKeyPressWhenSelectionNextToAWidget( isForward );\n\t\t}\n\t\t// Handle collapsing a non-collapsed selection that is wider than on a single widget.\n\t\telse if ( !domEventData.shiftKey ) {\n\t\t\tshouldStopAndPreventDefault = this._handleArrowKeyPressWhenNonCollapsedSelection( isForward );\n\t\t}\n\n\t\tif ( shouldStopAndPreventDefault ) {\n\t\t\tdomEventData.preventDefault();\n\t\t\tevt.stop();\n\t\t}\n\t}\n\n\t/**\n\t * Handles the keyboard navigation on \"keydown\" when a widget is currently selected and activates or deactivates\n\t * the fake caret for that widget, depending on the current value of the `widget-type-around` model\n\t * selection attribute and the direction of the pressed arrow key.\n\t *\n\t * @param isForward `true` when the pressed arrow key was responsible for the forward model selection movement\n\t * as in {@link module:utils/keyboard~isForwardArrowKeyCode}.\n\t * @returns Returns `true` when the keypress was handled and no other keydown listener of the editor should\n\t * process the event any further. Returns `false` otherwise.\n\t */\n\tprivate _handleArrowKeyPressOnSelectedWidget( isForward: boolean ) {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst modelSelection = model.document.selection;\n\t\tconst typeAroundFakeCaretPosition = getTypeAroundFakeCaretPosition( modelSelection );\n\n\t\treturn model.change( writer => {\n\t\t\t// If the fake caret is displayed...\n\t\t\tif ( typeAroundFakeCaretPosition ) {\n\t\t\t\tconst isLeavingWidget = typeAroundFakeCaretPosition === ( isForward ? 'after' : 'before' );\n\n\t\t\t\t// If the keyboard arrow works against the value of the selection attribute...\n\t\t\t\t// then remove the selection attribute but prevent default DOM actions\n\t\t\t\t// and do not let the Widget plugin listener move the selection. This brings\n\t\t\t\t// the widget back to the state, for instance, like if was selected using the mouse.\n\t\t\t\t//\n\t\t\t\t// **Note**: If leaving the widget when the fake caret is active, then the default\n\t\t\t\t// Widget handler will change the selection and, in turn, this will automatically discard\n\t\t\t\t// the selection attribute.\n\t\t\t\tif ( !isLeavingWidget ) {\n\t\t\t\t\twriter.removeSelectionAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE );\n\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t}\n\t\t\t// If the fake caret wasn't displayed, let's set it now according to the direction of the arrow\n\t\t\t// key press. This also means we cannot let the Widget plugin listener move the selection.\n\t\t\telse {\n\t\t\t\twriter.setSelectionAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE, isForward ? 'after' : 'before' );\n\n\t\t\t\treturn true;\n\t\t\t}\n\n\t\t\treturn false;\n\t\t} );\n\t}\n\n\t/**\n\t * Handles the keyboard navigation on \"keydown\" when **no** widget is selected but the selection is **directly** next\n\t * to one and upon the fake caret should become active for this widget upon arrow keypress\n\t * (AKA entering/selecting the widget).\n\t *\n\t * **Note**: This code mirrors the implementation from the widget plugin but also adds the selection attribute.\n\t * Unfortunately, there is no safe way to let the widget plugin do the selection part first and then just set the\n\t * selection attribute here in the widget type around plugin. This is why this code must duplicate some from the widget plugin.\n\t *\n\t * @param isForward `true` when the pressed arrow key was responsible for the forward model selection movement\n\t * as in {@link module:utils/keyboard~isForwardArrowKeyCode}.\n\t * @returns Returns `true` when the keypress was handled and no other keydown listener of the editor should\n\t * process the event any further. Returns `false` otherwise.\n\t */\n\tprivate _handleArrowKeyPressWhenSelectionNextToAWidget( isForward: boolean ) {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst schema = model.schema;\n\t\tconst widgetPlugin: Widget = editor.plugins.get( 'Widget' );\n\n\t\t// This is the widget the selection is about to be set on.\n\t\tconst modelElementNextToSelection = widgetPlugin._getObjectElementNextToSelection( isForward )!;\n\t\tconst viewElementNextToSelection = editor.editing.mapper.toViewElement( modelElementNextToSelection );\n\n\t\tif ( isTypeAroundWidget( viewElementNextToSelection, modelElementNextToSelection, schema ) ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\twidgetPlugin._setSelectionOverElement( modelElementNextToSelection );\n\t\t\t\twriter.setSelectionAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE, isForward ? 'before' : 'after' );\n\t\t\t} );\n\n\t\t\t// The change() block above does the same job as the Widget plugin. The event can\n\t\t\t// be safely canceled.\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Handles the keyboard navigation on \"keydown\" when a widget is currently selected (together with some other content)\n\t * and the widget is the first or last element in the selection. It activates or deactivates the fake caret for that widget.\n\t *\n\t * @param isForward `true` when the pressed arrow key was responsible for the forward model selection movement\n\t * as in {@link module:utils/keyboard~isForwardArrowKeyCode}.\n\t * @returns Returns `true` when the keypress was handled and no other keydown listener of the editor should\n\t * process the event any further. Returns `false` otherwise.\n\t */\n\tprivate _handleArrowKeyPressWhenNonCollapsedSelection( isForward: boolean ) {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst schema = model.schema;\n\t\tconst mapper = editor.editing.mapper;\n\t\tconst modelSelection = model.document.selection;\n\n\t\tconst selectedModelNode = isForward ?\n\t\t\tmodelSelection.getLastPosition()!.nodeBefore :\n\t\t\tmodelSelection.getFirstPosition()!.nodeAfter;\n\n\t\tconst selectedViewNode = mapper.toViewElement( selectedModelNode as any );\n\n\t\t// There is a widget at the collapse position so collapse the selection to the fake caret on it.\n\t\tif ( isTypeAroundWidget( selectedViewNode, selectedModelNode as any, schema ) ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\twriter.setSelection( selectedModelNode!, 'on' );\n\t\t\t\twriter.setSelectionAttribute( TYPE_AROUND_SELECTION_ATTRIBUTE, isForward ? 'after' : 'before' );\n\t\t\t} );\n\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Registers a `mousedown` listener for the view document which intercepts events\n\t * coming from the widget type around UI, which happens when a user clicks one of the buttons\n\t * that insert a paragraph next to a widget.\n\t */\n\tprivate _enableInsertingParagraphsOnButtonClick() {\n\t\tconst editor = this.editor;\n\t\tconst editingView = editor.editing.view;\n\n\t\tthis._listenToIfEnabled<ViewDocumentMouseDownEvent>( editingView.document, 'mousedown', ( evt, domEventData ) => {\n\t\t\tconst button = getClosestTypeAroundDomButton( domEventData.domTarget );\n\n\t\t\tif ( !button ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst buttonPosition = getTypeAroundButtonPosition( button );\n\t\t\tconst widgetViewElement = getClosestWidgetViewElement( button, editingView.domConverter );\n\t\t\tconst widgetModelElement = editor.editing.mapper.toModelElement( widgetViewElement );\n\n\t\t\tthis._insertParagraph( widgetModelElement!, buttonPosition );\n\n\t\t\tdomEventData.preventDefault();\n\t\t\tevt.stop();\n\t\t} );\n\t}\n\n\t/**\n\t * Creates the <kbd>Enter</kbd> key listener on the view document that allows the user to insert a paragraph\n\t * near the widget when either:\n\t *\n\t * * The fake caret was first activated using the arrow keys,\n\t * * The entire widget is selected in the model.\n\t *\n\t * In the first case, the new paragraph is inserted according to the `widget-type-around` selection\n\t * attribute (see {@link #_handleArrowKeyPress}).\n\t *\n\t * In the second case, the new paragraph is inserted based on whether a soft (<kbd>Shift</kbd>+<kbd>Enter</kbd>) keystroke\n\t * was pressed or not.\n\t */\n\tprivate _enableInsertingParagraphsOnEnterKeypress() {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\t\tconst editingView = editor.editing.view;\n\n\t\tthis._listenToIfEnabled<ViewDocumentEnterEvent>( editingView.document, 'enter', ( evt, domEventData ) => {\n\t\t\t// This event could be triggered from inside the widget but we are interested\n\t\t\t// only when the widget is selected itself.\n\t\t\tif ( evt.eventPhase != 'atTarget' ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst selectedModelElement = selection.getSelectedElement()!;\n\t\t\tconst selectedViewElement = editor.editing.mapper.toViewElement( selectedModelElement );\n\n\t\t\tconst schema = editor.model.schema;\n\t\t\tlet wasHandled;\n\n\t\t\t// First check if the widget is selected and there's a type around selection attribute associated\n\t\t\t// with the fake caret that would tell where to insert a new paragraph.\n\t\t\tif ( this._insertParagraphAccordingToFakeCaretPosition() ) {\n\t\t\t\twasHandled = true;\n\t\t\t}\n\t\t\t// Then, if there is no selection attribute associated with the fake caret, check if the widget\n\t\t\t// simply is selected and create a new paragraph according to the keystroke (Shift+)Enter.\n\t\t\telse if ( isTypeAroundWidget( selectedViewElement, selectedModelElement, schema ) ) {\n\t\t\t\tthis._insertParagraph( selectedModelElement, domEventData.isSoft ? 'before' : 'after' );\n\n\t\t\t\twasHandled = true;\n\t\t\t}\n\n\t\t\tif ( wasHandled ) {\n\t\t\t\tdomEventData.preventDefault();\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { context: isWidget } );\n\t}\n\n\t/**\n\t * Similar to the {@link #_enableInsertingParagraphsOnEnterKeypress}, it allows the user\n\t * to insert a paragraph next to a widget when the fake caret was activated using arrow\n\t * keys but it responds to typing instead of <kbd>Enter</kbd>.\n\t *\n\t * Listener enabled by this method will insert a new paragraph according to the `widget-type-around`\n\t * model selection attribute as the user simply starts typing, which creates the impression that the fake caret\n\t * behaves like a real one rendered by the browser (AKA your text appears where the caret was).\n\t *\n\t * **Note**: At the moment this listener creates 2 undo steps: one for the `insertParagraph` command\n\t * and another one for actual typing. It is not a disaster but this may need to be fixed\n\t * sooner or later.\n\t */\n\tprivate _enableInsertingParagraphsOnTypingKeystroke() {\n\t\tconst editor = this.editor;\n\t\tconst viewDocument = editor.editing.view.document;\n\n\t\t// Note: The priority must precede the default Input plugin insertText handler.\n\t\tthis._listenToIfEnabled<ViewDocumentInsertTextEvent>( viewDocument, 'insertText', ( evt, data ) => {\n\t\t\tif ( this._insertParagraphAccordingToFakeCaretPosition() ) {\n\t\t\t\t// The view selection in the event data contains the widget. If the new paragraph\n\t\t\t\t// was inserted, modify the view selection passed along with the insertText event\n\t\t\t\t// so the default event handler in the Input plugin starts typing inside the paragraph.\n\t\t\t\t// Otherwise, the typing would be over the widget.\n\t\t\t\tdata.selection = viewDocument.selection;\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\n\t\tif ( env.isAndroid ) {\n\t\t\t// On Android with English keyboard, the composition starts just by putting caret\n\t\t\t// at the word end or by selecting a table column. This is not a real composition started.\n\t\t\t// Trigger delete content on first composition key pressed.\n\t\t\tthis._listenToIfEnabled<ViewDocumentKeyDownEvent>( viewDocument, 'keydown', ( evt, data ) => {\n\t\t\t\tif ( data.keyCode == 229 ) {\n\t\t\t\t\tthis._insertParagraphAccordingToFakeCaretPosition();\n\t\t\t\t}\n\t\t\t} );\n\t\t} else {\n\t\t\t// Note: The priority must precede the default Input plugin compositionstart handler (to call it before delete content).\n\t\t\tthis._listenToIfEnabled<ViewDocumentCompositionStartEvent>( viewDocument, 'compositionstart', () => {\n\t\t\t\tthis._insertParagraphAccordingToFakeCaretPosition();\n\t\t\t}, { priority: 'high' } );\n\t\t}\n\t}\n\n\t/**\n\t * It creates a \"delete\" event listener on the view document to handle cases when the <kbd>Delete</kbd> or <kbd>Backspace</kbd>\n\t * is pressed and the fake caret is currently active.\n\t *\n\t * The fake caret should create an illusion of a real browser caret so that when it appears before or after\n\t * a widget, pressing <kbd>Delete</kbd> or <kbd>Backspace</kbd> should remove a widget or delete the content\n\t * before or after a widget (depending on the content surrounding the widget).\n\t */\n\tprivate _enableDeleteIntegration() {\n\t\tconst editor = this.editor;\n\t\tconst editingView = editor.editing.view;\n\t\tconst model = editor.model;\n\t\tconst schema = model.schema;\n\n\t\tthis._listenToIfEnabled<ViewDocumentDeleteEvent>( editingView.document, 'delete', ( evt, domEventData ) => {\n\t\t\t// This event could be triggered from inside the widget but we are interested\n\t\t\t// only when the widget is selected itself.\n\t\t\tif ( evt.eventPhase != 'atTarget' ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst typeAroundFakeCaretPosition = getTypeAroundFakeCaretPosition( model.document.selection );\n\n\t\t\t// This listener handles only these cases when the fake caret is active.\n\t\t\tif ( !typeAroundFakeCaretPosition ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst direction = domEventData.direction;\n\t\t\tconst selectedModelWidget = model.document.selection.getSelectedElement();\n\n\t\t\tconst isFakeCaretBefore = typeAroundFakeCaretPosition === 'before';\n\t\t\tconst isDeleteForward = direction == 'forward';\n\t\t\tconst shouldDeleteEntireWidget = isFakeCaretBefore === isDeleteForward;\n\n\t\t\tif ( shouldDeleteEntireWidget ) {\n\t\t\t\teditor.execute( 'delete', {\n\t\t\t\t\tselection: model.createSelection( selectedModelWidget!, 'on' )\n\t\t\t\t} );\n\t\t\t} else {\n\t\t\t\tconst range = schema.getNearestSelectionRange(\n\t\t\t\t\tmodel.createPositionAt( selectedModelWidget!, typeAroundFakeCaretPosition ),\n\t\t\t\t\tdirection\n\t\t\t\t);\n\n\t\t\t\t// If there is somewhere to move selection to, then there will be something to delete.\n\t\t\t\tif ( range ) {\n\t\t\t\t\t// If the range is NOT collapsed, then we know that the range contains an object (see getNearestSelectionRange() docs).\n\t\t\t\t\tif ( !range.isCollapsed ) {\n\t\t\t\t\t\tmodel.change( writer => {\n\t\t\t\t\t\t\twriter.setSelection( range );\n\t\t\t\t\t\t\teditor.execute( isDeleteForward ? 'deleteForward' : 'delete' );\n\t\t\t\t\t\t} );\n\t\t\t\t\t} else {\n\t\t\t\t\t\tconst probe = model.createSelection( range.start );\n\t\t\t\t\t\tmodel.modifySelection( probe, { direction } );\n\n\t\t\t\t\t\t// If the range is collapsed, let's see if a non-collapsed range exists that can could be deleted.\n\t\t\t\t\t\t// If such range exists, use the editor command because it it safe for collaboration (it merges where it can).\n\t\t\t\t\t\tif ( !probe.focus!.isEqual( range.start ) ) {\n\t\t\t\t\t\t\tmodel.change( writer => {\n\t\t\t\t\t\t\t\twriter.setSelection( range );\n\t\t\t\t\t\t\t\teditor.execute( isDeleteForward ? 'deleteForward' : 'delete' );\n\t\t\t\t\t\t\t} );\n\t\t\t\t\t\t}\n\t\t\t\t\t\t// If there is no non-collapsed range to be deleted then we are sure that there is an empty element\n\t\t\t\t\t\t// next to a widget that should be removed. \"delete\" and \"deleteForward\" commands cannot get rid of it\n\t\t\t\t\t\t// so calling Model#deleteContent here manually.\n\t\t\t\t\t\telse {\n\t\t\t\t\t\t\tconst deepestEmptyRangeAncestor = getDeepestEmptyElementAncestor( schema, range.start.parent as Element );\n\n\t\t\t\t\t\t\tmodel.deleteContent( model.createSelection( deepestEmptyRangeAncestor, 'on' ), {\n\t\t\t\t\t\t\t\tdoNotAutoparagraph: true\n\t\t\t\t\t\t\t} );\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// If some content was deleted, don't let the handler from the Widget plugin kick in.\n\t\t\t// If nothing was deleted, then the default handler will have nothing to do anyway.\n\t\t\tdomEventData.preventDefault();\n\t\t\tevt.stop();\n\t\t}, { context: isWidget } );\n\t}\n\n\t/**\n\t * Attaches the {@link module:engine/model/model~Model#event:insertContent} event listener that, for instance, allows the user to paste\n\t * content near a widget when the fake caret is first activated using the arrow keys.\n\t *\n\t * The content is inserted according to the `widget-type-around` selection attribute (see {@link #_handleArrowKeyPress}).\n\t */\n\tprivate _enableInsertContentIntegration() {\n\t\tconst editor = this.editor;\n\t\tconst model = this.editor.model;\n\t\tconst documentSelection = model.document.selection;\n\n\t\tthis._listenToIfEnabled<ModelInsertContentEvent>( editor.model, 'insertContent', ( evt, [ content, selectable ] ) => {\n\t\t\tif ( selectable && !( selectable as any ).is( 'documentSelection' ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst typeAroundFakeCaretPosition = getTypeAroundFakeCaretPosition( documentSelection );\n\n\t\t\tif ( !typeAroundFakeCaretPosition ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tevt.stop();\n\n\t\t\treturn model.change( writer => {\n\t\t\t\tconst selectedElement = documentSelection.getSelectedElement();\n\t\t\t\tconst position = model.createPositionAt( selectedElement!, typeAroundFakeCaretPosition );\n\t\t\t\tconst selection = writer.createSelection( position );\n\n\t\t\t\tconst result = model.insertContent( content, selection );\n\n\t\t\t\twriter.setSelection( selection );\n\n\t\t\t\treturn result;\n\t\t\t} );\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * Attaches the {@link module:engine/model/model~Model#event:insertObject} event listener that modifies the\n\t * `options.findOptimalPosition`parameter to position of fake caret in relation to selected element\n\t * to reflect user's intent of desired insertion position.\n\t *\n\t * The object is inserted according to the `widget-type-around` selection attribute (see {@link #_handleArrowKeyPress}).\n\t */\n\tprivate _enableInsertObjectIntegration() {\n\t\tconst editor = this.editor;\n\t\tconst model = this.editor.model;\n\t\tconst documentSelection = model.document.selection;\n\n\t\tthis._listenToIfEnabled<ModelInsertObjectEvent>( editor.model, 'insertObject', ( evt, args ) => {\n\t\t\tconst [ , selectable, options = {} ] = args;\n\n\t\t\tif ( selectable && !( selectable as any ).is( 'documentSelection' ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst typeAroundFakeCaretPosition = getTypeAroundFakeCaretPosition( documentSelection );\n\n\t\t\tif ( !typeAroundFakeCaretPosition ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\toptions.findOptimalPosition = typeAroundFakeCaretPosition;\n\t\t\targs[ 3 ] = options;\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * Attaches the {@link module:engine/model/model~Model#event:deleteContent} event listener to block the event when the fake\n\t * caret is active.\n\t *\n\t * This is required for cases that trigger {@link module:engine/model/model~Model#deleteContent `model.deleteContent()`}\n\t * before calling {@link module:engine/model/model~Model#insertContent `model.insertContent()`} like, for instance,\n\t * plain text pasting.\n\t */\n\tprivate _enableDeleteContentIntegration() {\n\t\tconst editor = this.editor;\n\t\tconst model = this.editor.model;\n\t\tconst documentSelection = model.document.selection;\n\n\t\tthis._listenToIfEnabled<ModelDeleteContentEvent>( editor.model, 'deleteContent', ( evt, [ selection ] ) => {\n\t\t\tif ( selection && !selection.is( 'documentSelection' ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst typeAroundFakeCaretPosition = getTypeAroundFakeCaretPosition( documentSelection );\n\n\t\t\t// Disable removing the selection content while pasting plain text.\n\t\t\tif ( typeAroundFakeCaretPosition ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\t}\n}\n\n/**\n * Injects the type around UI into a view widget instance.\n */\nfunction injectUIIntoWidget(\n\tviewWriter: DowncastWriter,\n\tbuttonTitles: { before: string; after: string },\n\twidgetViewElement: ViewElement ) {\n\tconst typeAroundWrapper = viewWriter.createUIElement( 'div', {\n\t\tclass: 'ck ck-reset_all ck-widget__type-around'\n\t}, function( domDocument ) {\n\t\tconst wrapperDomElement = this.toDomElement( domDocument );\n\n\t\tinjectButtons( wrapperDomElement, buttonTitles );\n\t\tinjectFakeCaret( wrapperDomElement );\n\n\t\treturn wrapperDomElement;\n\t} );\n\n\t// Inject the type around wrapper into the widget's wrapper.\n\tviewWriter.insert( viewWriter.createPositionAt( widgetViewElement, 'end' ), typeAroundWrapper );\n}\n\n/**\n * FYI: Not using the IconView class because each instance would need to be destroyed to avoid memory leaks\n * and it's pretty hard to figure out when a view (widget) is gone for good so it's cheaper to use raw\n * <svg> here.\n */\nfunction injectButtons( wrapperDomElement: HTMLElement, buttonTitles: { before: string; after: string } ) {\n\tfor ( const position of POSSIBLE_INSERTION_POSITIONS ) {\n\t\tconst buttonTemplate = new Template( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-widget__type-around__button',\n\t\t\t\t\t`ck-widget__type-around__button_${ position }`\n\t\t\t\t],\n\t\t\t\ttitle: buttonTitles[ position ],\n\t\t\t\t'aria-hidden': 'true'\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\twrapperDomElement.ownerDocument.importNode( RETURN_ARROW_ICON_ELEMENT, true )\n\t\t\t]\n\t\t} );\n\n\t\twrapperDomElement.appendChild( buttonTemplate.render() );\n\t}\n}\n\nfunction injectFakeCaret( wrapperDomElement: HTMLElement ) {\n\tconst caretTemplate = new Template( {\n\t\ttag: 'div',\n\t\tattributes: {\n\t\t\tclass: [\n\t\t\t\t'ck',\n\t\t\t\t'ck-widget__type-around__fake-caret'\n\t\t\t]\n\t\t}\n\t} );\n\n\twrapperDomElement.appendChild( caretTemplate.render() );\n}\n\n/**\n * Returns the ancestor of an element closest to the root which is empty. For instance,\n * for `<baz>`:\n *\n * ```\n * <foo>abc<bar><baz></baz></bar></foo>\n * ```\n *\n * it returns `<bar>`.\n */\nfunction getDeepestEmptyElementAncestor( schema: Schema, element: Element ) {\n\tlet deepestEmptyAncestor = element;\n\n\tfor ( const ancestor of element.getAncestors( { parentFirst: true } ) ) {\n\t\tif ( ( ancestor as any ).childCount > 1 || schema.isLimit( ancestor ) ) {\n\t\t\tbreak;\n\t\t}\n\n\t\tdeepestEmptyAncestor = ancestor as Element;\n\t}\n\n\treturn deepestEmptyAncestor;\n}\n","export default \"<svg viewBox=\\\"0 0 10 8\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M9.055.263v3.972h-6.77M1 4.216l2-2.038m-2 2 2 2.038\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module widget/verticalnavigation\n */\n\nimport {\n\tkeyCodes,\n\tRect,\n\ttype GetCallback\n} from '@ckeditor/ckeditor5-utils';\n\nimport type {\n\tDocumentSelection,\n\tEditingController,\n\tModel,\n\tPosition,\n\tRange,\n\tSchema,\n\tSelection,\n\tViewDocumentArrowKeyEvent\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * Returns 'keydown' handler for up/down arrow keys that modifies the caret movement if it's in a text line next to an object.\n *\n * @param editing The editing controller.\n */\nexport default function verticalNavigationHandler(\n\tediting: EditingController\n): GetCallback<ViewDocumentArrowKeyEvent> {\n\tconst model = editing.model;\n\n\treturn ( evt, data ) => {\n\t\tconst arrowUpPressed = data.keyCode == keyCodes.arrowup;\n\t\tconst arrowDownPressed = data.keyCode == keyCodes.arrowdown;\n\t\tconst expandSelection = data.shiftKey;\n\t\tconst selection = model.document.selection;\n\n\t\tif ( !arrowUpPressed && !arrowDownPressed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst isForward = arrowDownPressed;\n\n\t\t// Navigation is in the opposite direction than the selection direction so this is shrinking of the selection.\n\t\t// Selection for sure will not approach any object.\n\t\tif ( expandSelection && selectionWillShrink( selection, isForward ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Find a range between selection and closest limit element.\n\t\tconst range = findTextRangeFromSelection( editing, selection, isForward );\n\n\t\t// There is no selection position inside the limit element.\n\t\tif ( !range ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// If already at the edge of a limit element.\n\t\tif ( range.isCollapsed ) {\n\t\t\t// A collapsed selection at limit edge - nothing more to do.\n\t\t\tif ( selection.isCollapsed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// A non collapsed selection is at the limit edge while expanding the selection - let others do their stuff.\n\t\t\telse if ( expandSelection ) {\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\n\t\t// If the range is a single line (there is no word wrapping) then move the selection to the position closest to the limit element.\n\t\t//\n\t\t// We can't move the selection directly to the isObject element (eg. table cell) because of dual position at the end/beginning\n\t\t// of wrapped line (it's at the same time at the end of one line and at the start of the next line).\n\t\tif ( range.isCollapsed || isSingleLineRange( editing, range, isForward ) ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\tconst newPosition = isForward ? range.end : range.start;\n\n\t\t\t\tif ( expandSelection ) {\n\t\t\t\t\tconst newSelection = model.createSelection( selection.anchor );\n\t\t\t\t\tnewSelection.setFocus( newPosition );\n\n\t\t\t\t\twriter.setSelection( newSelection );\n\t\t\t\t} else {\n\t\t\t\t\twriter.setSelection( newPosition );\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tevt.stop();\n\t\t\tdata.preventDefault();\n\t\t\tdata.stopPropagation();\n\t\t}\n\t};\n}\n\n/**\n * Finds the range between selection and closest limit element (in the direction of navigation).\n * The position next to limit element is adjusted to the closest allowed `$text` position.\n *\n * Returns `null` if, according to the schema, the resulting range cannot contain a `$text` element.\n *\n * @param editing The editing controller.\n * @param selection The current selection.\n * @param isForward The expected navigation direction.\n */\nfunction findTextRangeFromSelection( editing: EditingController, selection: Selection | DocumentSelection, isForward: boolean ) {\n\tconst model = editing.model;\n\n\tif ( isForward ) {\n\t\tconst startPosition = selection.isCollapsed ? selection.focus! : selection.getLastPosition()!;\n\t\tconst endPosition = getNearestNonInlineLimit( model, startPosition, 'forward' );\n\n\t\t// There is no limit element, browser should handle this.\n\t\tif ( !endPosition ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst range = model.createRange( startPosition, endPosition );\n\t\tconst lastRangePosition = getNearestTextPosition( model.schema, range, 'backward' );\n\n\t\tif ( lastRangePosition ) {\n\t\t\treturn model.createRange( startPosition, lastRangePosition );\n\t\t}\n\n\t\treturn null;\n\t} else {\n\t\tconst endPosition = selection.isCollapsed ? selection.focus! : selection.getFirstPosition()!;\n\t\tconst startPosition = getNearestNonInlineLimit( model, endPosition, 'backward' );\n\n\t\t// There is no limit element, browser should handle this.\n\t\tif ( !startPosition ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst range = model.createRange( startPosition, endPosition );\n\t\tconst firstRangePosition = getNearestTextPosition( model.schema, range, 'forward' );\n\n\t\tif ( firstRangePosition ) {\n\t\t\treturn model.createRange( firstRangePosition, endPosition );\n\t\t}\n\n\t\treturn null;\n\t}\n}\n\n/**\n * Finds the limit element position that is closest to startPosition.\n *\n * @param direction Search direction.\n */\nfunction getNearestNonInlineLimit( model: Model, startPosition: Position, direction: 'forward' | 'backward' ) {\n\tconst schema = model.schema;\n\tconst range = model.createRangeIn( startPosition.root );\n\n\tconst walkerValueType = direction == 'forward' ? 'elementStart' : 'elementEnd';\n\n\tfor ( const { previousPosition, item, type } of range.getWalker( { startPosition, direction } ) ) {\n\t\tif ( schema.isLimit( item ) && !schema.isInline( item ) ) {\n\t\t\treturn previousPosition;\n\t\t}\n\n\t\t// Stop looking for isLimit element if the next element is a block element (it is for sure not single line).\n\t\tif ( type == walkerValueType && schema.isBlock( item ) ) {\n\t\t\treturn null;\n\t\t}\n\t}\n\n\treturn null;\n}\n\n/**\n * Basing on the provided range, finds the first or last (depending on `direction`) position inside the range\n * that can contain `$text` (according to schema).\n *\n * @param schema The schema.\n * @param range The range to find the position in.\n * @param direction Search direction.\n * @returns The nearest selection position.\n *\n */\nfunction getNearestTextPosition( schema: Schema, range: Range, direction: 'forward' | 'backward' ) {\n\tconst position = direction == 'backward' ? range.end : range.start;\n\n\tif ( schema.checkChild( position, '$text' ) ) {\n\t\treturn position;\n\t}\n\n\tfor ( const { nextPosition } of range.getWalker( { direction } ) ) {\n\t\tif ( schema.checkChild( nextPosition, '$text' ) ) {\n\t\t\treturn nextPosition;\n\t\t}\n\t}\n\n\treturn null;\n}\n\n/**\n * Checks if the DOM range corresponding to the provided model range renders as a single line by analyzing DOMRects\n * (verifying if they visually wrap content to the next line).\n *\n * @param editing The editing controller.\n * @param modelRange The current table cell content range.\n * @param isForward The expected navigation direction.\n */\nfunction isSingleLineRange( editing: EditingController, modelRange: Range, isForward: boolean ) {\n\tconst model = editing.model;\n\tconst domConverter = editing.view.domConverter;\n\n\t// Wrapped lines contain exactly the same position at the end of current line\n\t// and at the beginning of next line. That position's client rect is at the end\n\t// of current line. In case of caret at first position of the last line that 'dual'\n\t// position would be detected as it's not the last line.\n\tif ( isForward ) {\n\t\tconst probe = model.createSelection( modelRange.start );\n\n\t\tmodel.modifySelection( probe );\n\n\t\t// If the new position is at the end of the container then we can't use this position\n\t\t// because it would provide incorrect result for eg caption of image and selection\n\t\t// just before end of it. Also in this case there is no \"dual\" position.\n\t\tif ( !probe.focus!.isAtEnd && !modelRange.start.isEqual( probe.focus! ) ) {\n\t\t\tmodelRange = model.createRange( probe.focus!, modelRange.end );\n\t\t}\n\t}\n\n\tconst viewRange = editing.mapper.toViewRange( modelRange );\n\tconst domRange = domConverter.viewRangeToDom( viewRange );\n\tconst rects = Rect.getDomRangeRects( domRange );\n\n\tlet boundaryVerticalPosition;\n\n\tfor ( const rect of rects ) {\n\t\tif ( boundaryVerticalPosition === undefined ) {\n\t\t\tboundaryVerticalPosition = Math.round( rect.bottom );\n\t\t\tcontinue;\n\t\t}\n\n\t\t// Let's check if this rect is in new line.\n\t\tif ( Math.round( rect.top ) >= boundaryVerticalPosition ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tboundaryVerticalPosition = Math.max( boundaryVerticalPosition, Math.round( rect.bottom ) );\n\t}\n\n\treturn true;\n}\n\nfunction selectionWillShrink( selection: DocumentSelection, isForward: boolean ) {\n\treturn !selection.isCollapsed && selection.isBackward == isForward;\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./widget.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module widget/widget\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\n\nimport {\n\tMouseObserver,\n\ttype DomEventData,\n\ttype DowncastSelectionEvent,\n\ttype DowncastWriter,\n\ttype Element,\n\ttype Node,\n\ttype ViewDocumentArrowKeyEvent,\n\ttype ViewDocumentFragment,\n\ttype ViewDocumentMouseDownEvent,\n\ttype ViewElement\n} from '@ckeditor/ckeditor5-engine';\n\nimport { Delete, type ViewDocumentDeleteEvent } from '@ckeditor/ckeditor5-typing';\n\nimport {\n\tenv,\n\tgetLocalizedArrowKeyCodeDirection,\n\ttype EventInfo,\n\ttype KeystrokeInfo\n} from '@ckeditor/ckeditor5-utils';\n\nimport WidgetTypeAround from './widgettypearound/widgettypearound';\nimport verticalNavigationHandler from './verticalnavigation';\nimport { getLabel, isWidget, WIDGET_SELECTED_CLASS_NAME } from './utils';\n\nimport '../theme/widget.css';\n\n/**\n * The widget plugin. It enables base support for widgets.\n *\n * See {@glink api/widget package page} for more details and documentation.\n *\n * This plugin enables multiple behaviors required by widgets:\n *\n * * The model to view selection converter for the editing pipeline (it handles widget custom selection rendering).\n * If a converted selection wraps around a widget element, that selection is marked as\n * {@link module:engine/view/selection~Selection#isFake fake}. Additionally, the `ck-widget_selected` CSS class\n * is added to indicate that widget has been selected.\n * * The mouse and keyboard events handling on and around widget elements.\n */\nexport default class Widget extends Plugin {\n\t/**\n\t * Holds previously selected widgets.\n\t */\n\tprivate _previouslySelected = new Set<ViewElement>();\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Widget' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ WidgetTypeAround, Delete ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\t// Model to view selection converter.\n\t\t// Converts selection placed over widget element to fake selection.\n\t\t//\n\t\t// By default, the selection is downcasted by the engine to surround the attribute element, even though its only\n\t\t// child is an inline widget. A similar thing also happens when a collapsed marker is rendered as a UI element\n\t\t// next to an inline widget: the view selection contains both the widget and the marker.\n\t\t//\n\t\t// This prevents creating a correct fake selection when this inline widget is selected. Normalize the selection\n\t\t// in these cases based on the model:\n\t\t//\n\t\t//\t\t[<attributeElement><inlineWidget /></attributeElement>] -> <attributeElement>[<inlineWidget />]</attributeElement>\n\t\t//\t\t[<uiElement></uiElement><inlineWidget />] -> <uiElement></uiElement>[<inlineWidget />]\n\t\t//\n\t\t// Thanks to this:\n\t\t//\n\t\t// * fake selection can be set correctly,\n\t\t// * any logic depending on (View)Selection#getSelectedElement() also works OK.\n\t\t//\n\t\t// See https://github.com/ckeditor/ckeditor5/issues/9524.\n\t\tthis.editor.editing.downcastDispatcher.on<DowncastSelectionEvent>( 'selection', ( evt, data, conversionApi ) => {\n\t\t\tconst viewWriter = conversionApi.writer;\n\t\t\tconst modelSelection = data.selection;\n\n\t\t\t// The collapsed selection can't contain any widget.\n\t\t\tif ( modelSelection.isCollapsed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst selectedModelElement = modelSelection.getSelectedElement();\n\n\t\t\tif ( !selectedModelElement ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst selectedViewElement = editor.editing.mapper.toViewElement( selectedModelElement )!;\n\n\t\t\tif ( !isWidget( selectedViewElement ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( !conversionApi.consumable.consume( modelSelection, 'selection' ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tviewWriter.setSelection( viewWriter.createRangeOn( selectedViewElement ), {\n\t\t\t\tfake: true,\n\t\t\t\tlabel: getLabel( selectedViewElement )\n\t\t\t} );\n\t\t} );\n\n\t\t// Mark all widgets inside the selection with the css class.\n\t\t// This handler is registered at the 'low' priority so it's triggered after the real selection conversion.\n\t\tthis.editor.editing.downcastDispatcher.on<DowncastSelectionEvent>( 'selection', ( evt, data, conversionApi ) => {\n\t\t\t// Remove selected class from previously selected widgets.\n\t\t\tthis._clearPreviouslySelectedWidgets( conversionApi.writer );\n\n\t\t\tconst viewWriter = conversionApi.writer;\n\t\t\tconst viewSelection = viewWriter.document.selection;\n\n\t\t\tlet lastMarked = null;\n\n\t\t\tfor ( const range of viewSelection.getRanges() ) {\n\t\t\t\t// Note: There could be multiple selected widgets in a range but no fake selection.\n\t\t\t\t// All of them must be marked as selected, for instance [<widget></widget><widget></widget>]\n\t\t\t\tfor ( const value of range ) {\n\t\t\t\t\tconst node = value.item as ViewElement;\n\t\t\t\t\t// Do not mark nested widgets in selected one. See: #4594\n\t\t\t\t\tif ( isWidget( node ) && !isChild( node, lastMarked ) ) {\n\t\t\t\t\t\tviewWriter.addClass( WIDGET_SELECTED_CLASS_NAME, node );\n\t\t\t\t\t\tthis._previouslySelected.add( node );\n\t\t\t\t\t\tlastMarked = node;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}, { priority: 'low' } );\n\n\t\t// If mouse down is pressed on widget - create selection over whole widget.\n\t\tview.addObserver( MouseObserver );\n\t\tthis.listenTo<ViewDocumentMouseDownEvent>( viewDocument, 'mousedown', ( ...args ) => this._onMousedown( ...args ) );\n\n\t\t// There are two keydown listeners working on different priorities. This allows other\n\t\t// features such as WidgetTypeAround or TableKeyboard to attach their listeners in between\n\t\t// and customize the behavior even further in different content/selection scenarios.\n\t\t//\n\t\t// * The first listener handles changing the selection on arrow key press\n\t\t// if the widget is selected or if the selection is next to a widget and the widget\n\t\t// should become selected upon the arrow key press.\n\t\t//\n\t\t// * The second (late) listener makes sure the default browser action on arrow key press is\n\t\t// prevented when a widget is selected. This prevents the selection from being moved\n\t\t// from a fake selection container.\n\t\tthis.listenTo<ViewDocumentArrowKeyEvent>( viewDocument, 'arrowKey', ( ...args ) => {\n\t\t\tthis._handleSelectionChangeOnArrowKeyPress( ...args );\n\t\t}, { context: [ isWidget, '$text' ] } );\n\n\t\tthis.listenTo<ViewDocumentArrowKeyEvent>( viewDocument, 'arrowKey', ( ...args ) => {\n\t\t\tthis._preventDefaultOnArrowKeyPress( ...args );\n\t\t}, { context: '$root' } );\n\n\t\tthis.listenTo<ViewDocumentArrowKeyEvent>(\n\t\t\tviewDocument,\n\t\t\t'arrowKey',\n\t\t\tverticalNavigationHandler( this.editor.editing ),\n\t\t\t{ context: '$text' }\n\t\t);\n\n\t\t// Handle custom delete behaviour.\n\t\tthis.listenTo<ViewDocumentDeleteEvent>( viewDocument, 'delete', ( evt, data ) => {\n\t\t\tif ( this._handleDelete( data.direction == 'forward' ) ) {\n\t\t\t\tdata.preventDefault();\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { context: '$root' } );\n\t}\n\n\t/**\n\t * Handles {@link module:engine/view/document~Document#event:mousedown mousedown} events on widget elements.\n\t */\n\tprivate _onMousedown( eventInfo: EventInfo, domEventData: DomEventData<MouseEvent> ) {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\t\tlet element: ViewElement | null = domEventData.target;\n\n\t\t// Do nothing for single or double click inside nested editable.\n\t\tif ( isInsideNestedEditable( element ) ) {\n\t\t\t// But at least triple click inside nested editable causes broken selection in Safari.\n\t\t\t// For such event, we select the entire nested editable element.\n\t\t\t// See: https://github.com/ckeditor/ckeditor5/issues/1463.\n\t\t\tif ( ( env.isSafari || env.isGecko ) && domEventData.domEvent.detail >= 3 ) {\n\t\t\t\tconst mapper = editor.editing.mapper;\n\t\t\t\tconst viewElement = element.is( 'attributeElement' ) ?\n\t\t\t\t\telement.findAncestor( element => !element.is( 'attributeElement' ) )! : element;\n\t\t\t\tconst modelElement = mapper.toModelElement( viewElement )!;\n\n\t\t\t\tdomEventData.preventDefault();\n\n\t\t\t\tthis.editor.model.change( writer => {\n\t\t\t\t\twriter.setSelection( modelElement, 'in' );\n\t\t\t\t} );\n\t\t\t}\n\n\t\t\treturn;\n\t\t}\n\n\t\t// If target is not a widget element - check if one of the ancestors is.\n\t\tif ( !isWidget( element ) ) {\n\t\t\telement = element.findAncestor( isWidget );\n\n\t\t\tif ( !element ) {\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\n\t\t// On Android selection would jump to the first table cell, on other devices\n\t\t// we can't block it (and don't need to) because of drag and drop support.\n\t\tif ( env.isAndroid ) {\n\t\t\tdomEventData.preventDefault();\n\t\t}\n\n\t\t// Focus editor if is not focused already.\n\t\tif ( !viewDocument.isFocused ) {\n\t\t\tview.focus();\n\t\t}\n\n\t\t// Create model selection over widget.\n\t\tconst modelElement = editor.editing.mapper.toModelElement( element );\n\n\t\tthis._setSelectionOverElement( modelElement! );\n\t}\n\n\t/**\n\t * Handles {@link module:engine/view/document~Document#event:keydown keydown} events and changes\n\t * the model selection when:\n\t *\n\t * * arrow key is pressed when the widget is selected,\n\t * * the selection is next to a widget and the widget should become selected upon the arrow key press.\n\t *\n\t * See {@link #_preventDefaultOnArrowKeyPress}.\n\t */\n\tprivate _handleSelectionChangeOnArrowKeyPress( eventInfo: EventInfo, domEventData: DomEventData & KeystrokeInfo ) {\n\t\tconst keyCode = domEventData.keyCode;\n\n\t\tconst model = this.editor.model;\n\t\tconst schema = model.schema;\n\t\tconst modelSelection = model.document.selection;\n\t\tconst objectElement = modelSelection.getSelectedElement();\n\t\tconst direction = getLocalizedArrowKeyCodeDirection( keyCode, this.editor.locale.contentLanguageDirection );\n\t\tconst isForward = direction == 'down' || direction == 'right';\n\t\tconst isVerticalNavigation = direction == 'up' || direction == 'down';\n\n\t\t// If object element is selected.\n\t\tif ( objectElement && schema.isObject( objectElement ) ) {\n\t\t\tconst position = isForward ? modelSelection.getLastPosition() : modelSelection.getFirstPosition();\n\t\t\tconst newRange = schema.getNearestSelectionRange( position!, isForward ? 'forward' : 'backward' );\n\n\t\t\tif ( newRange ) {\n\t\t\t\tmodel.change( writer => {\n\t\t\t\t\twriter.setSelection( newRange );\n\t\t\t\t} );\n\n\t\t\t\tdomEventData.preventDefault();\n\t\t\t\teventInfo.stop();\n\t\t\t}\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Handle collapsing of the selection when there is any widget on the edge of selection.\n\t\t// This is needed because browsers have problems with collapsing such selection.\n\t\tif ( !modelSelection.isCollapsed && !domEventData.shiftKey ) {\n\t\t\tconst firstPosition = modelSelection.getFirstPosition();\n\t\t\tconst lastPosition = modelSelection.getLastPosition();\n\n\t\t\tconst firstSelectedNode = firstPosition!.nodeAfter;\n\t\t\tconst lastSelectedNode = lastPosition!.nodeBefore;\n\n\t\t\tif ( firstSelectedNode && schema.isObject( firstSelectedNode ) || lastSelectedNode && schema.isObject( lastSelectedNode ) ) {\n\t\t\t\tmodel.change( writer => {\n\t\t\t\t\twriter.setSelection( isForward ? lastPosition : firstPosition );\n\t\t\t\t} );\n\n\t\t\t\tdomEventData.preventDefault();\n\t\t\t\teventInfo.stop();\n\t\t\t}\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Return if not collapsed.\n\t\tif ( !modelSelection.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// If selection is next to object element.\n\n\t\tconst objectElementNextToSelection = this._getObjectElementNextToSelection( isForward );\n\n\t\tif ( objectElementNextToSelection && schema.isObject( objectElementNextToSelection ) ) {\n\t\t\t// Do not select an inline widget while handling up/down arrow.\n\t\t\tif ( schema.isInline( objectElementNextToSelection ) && isVerticalNavigation ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._setSelectionOverElement( objectElementNextToSelection );\n\n\t\t\tdomEventData.preventDefault();\n\t\t\teventInfo.stop();\n\t\t}\n\t}\n\n\t/**\n\t * Handles {@link module:engine/view/document~Document#event:keydown keydown} events and prevents\n\t * the default browser behavior to make sure the fake selection is not being moved from a fake selection\n\t * container.\n\t *\n\t * See {@link #_handleSelectionChangeOnArrowKeyPress}.\n\t */\n\tprivate _preventDefaultOnArrowKeyPress( eventInfo: EventInfo, domEventData: DomEventData ) {\n\t\tconst model = this.editor.model;\n\t\tconst schema = model.schema;\n\t\tconst objectElement = model.document.selection.getSelectedElement();\n\n\t\t// If object element is selected.\n\t\tif ( objectElement && schema.isObject( objectElement ) ) {\n\t\t\tdomEventData.preventDefault();\n\t\t\teventInfo.stop();\n\t\t}\n\t}\n\n\t/**\n\t * Handles delete keys: backspace and delete.\n\t *\n\t * @param isForward Set to true if delete was performed in forward direction.\n\t * @returns Returns `true` if keys were handled correctly.\n\t */\n\tprivate _handleDelete( isForward: boolean ) {\n\t\tconst modelDocument = this.editor.model.document;\n\t\tconst modelSelection = modelDocument.selection;\n\n\t\t// Do nothing when the read only mode is enabled.\n\t\tif ( !this.editor.model.canEditAt( modelSelection ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Do nothing on non-collapsed selection.\n\t\tif ( !modelSelection.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst objectElement = this._getObjectElementNextToSelection( isForward );\n\n\t\tif ( objectElement ) {\n\t\t\tthis.editor.model.change( writer => {\n\t\t\t\tlet previousNode = modelSelection.anchor!.parent;\n\n\t\t\t\t// Remove previous element if empty.\n\t\t\t\twhile ( previousNode.isEmpty ) {\n\t\t\t\t\tconst nodeToRemove = previousNode;\n\t\t\t\t\tpreviousNode = nodeToRemove.parent!;\n\n\t\t\t\t\twriter.remove( nodeToRemove as Element );\n\t\t\t\t}\n\n\t\t\t\tthis._setSelectionOverElement( objectElement );\n\t\t\t} );\n\n\t\t\treturn true;\n\t\t}\n\t}\n\n\t/**\n\t * Sets {@link module:engine/model/selection~Selection document's selection} over given element.\n\t *\n\t * @internal\n\t */\n\tpublic _setSelectionOverElement( element: Node ): void {\n\t\tthis.editor.model.change( writer => {\n\t\t\twriter.setSelection( writer.createRangeOn( element ) );\n\t\t} );\n\t}\n\n\t/**\n\t * Checks if {@link module:engine/model/element~Element element} placed next to the current\n\t * {@link module:engine/model/selection~Selection model selection} exists and is marked in\n\t * {@link module:engine/model/schema~Schema schema} as `object`.\n\t *\n\t * @internal\n\t * @param forward Direction of checking.\n\t */\n\tpublic _getObjectElementNextToSelection( forward: boolean ): Element | null {\n\t\tconst model = this.editor.model;\n\t\tconst schema = model.schema;\n\t\tconst modelSelection = model.document.selection;\n\n\t\t// Clone current selection to use it as a probe. We must leave default selection as it is so it can return\n\t\t// to its current state after undo.\n\t\tconst probe = model.createSelection( modelSelection );\n\t\tmodel.modifySelection( probe, { direction: forward ? 'forward' : 'backward' } );\n\n\t\t// The selection didn't change so there is nothing there.\n\t\tif ( probe.isEqual( modelSelection ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst objectElement = forward ? probe.focus!.nodeBefore : probe.focus!.nodeAfter;\n\n\t\tif ( !!objectElement && schema.isObject( objectElement ) ) {\n\t\t\treturn objectElement as Element;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Removes CSS class from previously selected widgets.\n\t */\n\tprivate _clearPreviouslySelectedWidgets( writer: DowncastWriter ) {\n\t\tfor ( const widget of this._previouslySelected ) {\n\t\t\twriter.removeClass( WIDGET_SELECTED_CLASS_NAME, widget );\n\t\t}\n\n\t\tthis._previouslySelected.clear();\n\t}\n}\n\n/**\n * Returns `true` when element is a nested editable or is placed inside one.\n */\nfunction isInsideNestedEditable( element: ViewElement ) {\n\tlet currentElement: ViewElement | ViewDocumentFragment | null = element;\n\n\twhile ( currentElement ) {\n\t\tif ( currentElement.is( 'editableElement' ) && !currentElement.is( 'rootElement' ) ) {\n\t\t\treturn true;\n\t\t}\n\n\t\t// Click on nested widget should select it.\n\t\tif ( isWidget( currentElement ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tcurrentElement = currentElement.parent;\n\t}\n\n\treturn false;\n}\n\n/**\n * Checks whether the specified `element` is a child of the `parent` element.\n *\n * @param element An element to check.\n * @param parent A parent for the element.\n */\nfunction isChild( element: ViewElement, parent: ViewElement | null ) {\n\tif ( !parent ) {\n\t\treturn false;\n\t}\n\n\treturn Array.from( element.getAncestors() ).includes( parent );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module widget/widgettoolbarrepository\n */\n\nimport {\n\tPlugin,\n\ttype Editor,\n\ttype ToolbarConfigItem\n} from '@ckeditor/ckeditor5-core';\n\nimport type { ViewDocumentSelection, ViewElement } from '@ckeditor/ckeditor5-engine';\n\nimport {\n\tBalloonPanelView,\n\tContextualBalloon,\n\tToolbarView,\n\ttype BalloonToolbar,\n\ttype BalloonToolbarShowEvent,\n\ttype EditorUIUpdateEvent\n} from '@ckeditor/ckeditor5-ui';\n\nimport {\n\tCKEditorError,\n\tlogWarning,\n\ttype ObservableChangeEvent,\n\ttype RectSource\n} from '@ckeditor/ckeditor5-utils';\n\nimport { isWidget } from './utils';\n\n/**\n * Widget toolbar repository plugin. A central point for registering widget toolbars. This plugin handles the whole\n * toolbar rendering process and exposes a concise API.\n *\n * To add a toolbar for your widget use the {@link ~WidgetToolbarRepository#register `WidgetToolbarRepository#register()`} method.\n *\n * The following example comes from the {@link module:image/imagetoolbar~ImageToolbar} plugin:\n *\n * ```ts\n * class ImageToolbar extends Plugin {\n * \tstatic get requires() {\n * \t\treturn [ WidgetToolbarRepository ];\n * \t}\n *\n * \tafterInit() {\n * \t\tconst editor = this.editor;\n * \t\tconst widgetToolbarRepository = editor.plugins.get( WidgetToolbarRepository );\n *\n * \t\twidgetToolbarRepository.register( 'image', {\n * \t\t\titems: editor.config.get( 'image.toolbar' ),\n * \t\t\tgetRelatedElement: getClosestSelectedImageWidget\n * \t\t} );\n * \t}\n * }\n * ```\n */\nexport default class WidgetToolbarRepository extends Plugin {\n\t/**\n\t * A map of toolbar definitions.\n\t */\n\tprivate _toolbarDefinitions = new Map<string, WidgetRepositoryToolbarDefinition>();\n\n\tprivate _balloon!: ContextualBalloon;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ContextualBalloon ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'WidgetToolbarRepository' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Disables the default balloon toolbar for all widgets.\n\t\tif ( editor.plugins.has( 'BalloonToolbar' ) ) {\n\t\t\tconst balloonToolbar: BalloonToolbar = editor.plugins.get( 'BalloonToolbar' );\n\n\t\t\tthis.listenTo<BalloonToolbarShowEvent>( balloonToolbar, 'show', evt => {\n\t\t\t\tif ( isWidgetSelected( editor.editing.view.document.selection ) ) {\n\t\t\t\t\tevt.stop();\n\t\t\t\t}\n\t\t\t}, { priority: 'high' } );\n\t\t}\n\n\t\tthis._balloon = this.editor.plugins.get( 'ContextualBalloon' );\n\n\t\tthis.on<ObservableChangeEvent>( 'change:isEnabled', () => {\n\t\t\tthis._updateToolbarsVisibility();\n\t\t} );\n\n\t\tthis.listenTo<EditorUIUpdateEvent>( editor.ui, 'update', () => {\n\t\t\tthis._updateToolbarsVisibility();\n\t\t} );\n\n\t\t// UI#update is not fired after focus is back in editor, we need to check if balloon panel should be visible.\n\t\tthis.listenTo<ObservableChangeEvent>( editor.ui.focusTracker, 'change:isFocused', () => {\n\t\t\tthis._updateToolbarsVisibility();\n\t\t}, { priority: 'low' } );\n\t}\n\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tfor ( const toolbarConfig of this._toolbarDefinitions.values() ) {\n\t\t\ttoolbarConfig.view.destroy();\n\t\t}\n\t}\n\n\t/**\n\t * Registers toolbar in the WidgetToolbarRepository. It renders it in the `ContextualBalloon` based on the value of the invoked\n\t * `getRelatedElement` function. Toolbar items are gathered from `items` array.\n\t * The balloon's CSS class is by default `ck-toolbar-container` and may be override with the `balloonClassName` option.\n\t *\n\t * Note: This method should be called in the {@link module:core/plugin~PluginInterface#afterInit `Plugin#afterInit()`}\n\t * callback (or later) to make sure that the given toolbar items were already registered by other plugins.\n\t *\n\t * @param toolbarId An id for the toolbar. Used to\n\t * @param options.ariaLabel Label used by assistive technologies to describe this toolbar element.\n\t * @param options.items Array of toolbar items.\n\t * @param options.getRelatedElement Callback which returns an element the toolbar should be attached to.\n\t * @param options.balloonClassName CSS class for the widget balloon.\n\t */\n\tpublic register(\n\t\ttoolbarId: string,\n\t\t{ ariaLabel, items, getRelatedElement, balloonClassName = 'ck-toolbar-container' }: {\n\t\t\tariaLabel?: string;\n\t\t\titems: Array<ToolbarConfigItem>;\n\t\t\tgetRelatedElement: ( selection: ViewDocumentSelection ) => ( ViewElement | null );\n\t\t\tballoonClassName?: string;\n\t\t}\n\t): void {\n\t\t// Trying to register a toolbar without any item.\n\t\tif ( !items.length ) {\n\t\t\t/**\n\t\t\t * When {@link module:widget/widgettoolbarrepository~WidgetToolbarRepository#register registering} a new widget toolbar, you\n\t\t\t * need to provide a non-empty array with the items that will be inserted into the toolbar.\n\t\t\t *\n\t\t\t * If you see this error when integrating the editor, you likely forgot to configure one of the widget toolbars.\n\t\t\t *\n\t\t\t * See for instance:\n\t\t\t *\n\t\t\t * * {@link module:table/tableconfig~TableConfig#contentToolbar `config.table.contentToolbar`}\n\t\t\t * * {@link module:image/imageconfig~ImageConfig#toolbar `config.image.toolbar`}\n\t\t\t *\n\t\t\t * @error widget-toolbar-no-items\n\t\t\t * @param toolbarId The id of the toolbar that has not been configured correctly.\n\t\t\t */\n\t\t\tlogWarning( 'widget-toolbar-no-items', { toolbarId } );\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst toolbarView = new ToolbarView( editor.locale );\n\n\t\ttoolbarView.ariaLabel = ariaLabel || t( 'Widget toolbar' );\n\n\t\tif ( this._toolbarDefinitions.has( toolbarId ) ) {\n\t\t\t/**\n\t\t\t * Toolbar with the given id was already added.\n\t\t\t *\n\t\t\t * @error widget-toolbar-duplicated\n\t\t\t * @param toolbarId Toolbar id.\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'widget-toolbar-duplicated', this, { toolbarId } );\n\t\t}\n\n\t\tconst toolbarDefinition = {\n\t\t\tview: toolbarView,\n\t\t\tgetRelatedElement,\n\t\t\tballoonClassName,\n\t\t\titemsConfig: items,\n\t\t\tinitialized: false\n\t\t};\n\n\t\t// Register the toolbar so it becomes available for Alt+F10 and Esc navigation.\n\t\teditor.ui.addToolbar( toolbarView, {\n\t\t\tisContextual: true,\n\t\t\tbeforeFocus: () => {\n\t\t\t\tconst relatedElement = getRelatedElement( editor.editing.view.document.selection );\n\n\t\t\t\tif ( relatedElement ) {\n\t\t\t\t\tthis._showToolbar( toolbarDefinition, relatedElement );\n\t\t\t\t}\n\t\t\t},\n\t\t\tafterBlur: () => {\n\t\t\t\tthis._hideToolbar( toolbarDefinition );\n\t\t\t}\n\t\t} );\n\n\t\tthis._toolbarDefinitions.set( toolbarId, toolbarDefinition );\n\t}\n\n\t/**\n\t * Iterates over stored toolbars and makes them visible or hidden.\n\t */\n\tprivate _updateToolbarsVisibility() {\n\t\tlet maxRelatedElementDepth = 0;\n\t\tlet deepestRelatedElement = null;\n\t\tlet deepestToolbarDefinition = null;\n\n\t\tfor ( const definition of this._toolbarDefinitions.values() ) {\n\t\t\tconst relatedElement = definition.getRelatedElement( this.editor.editing.view.document.selection );\n\n\t\t\tif ( !this.isEnabled || !relatedElement ) {\n\t\t\t\tif ( this._isToolbarInBalloon( definition ) ) {\n\t\t\t\t\tthis._hideToolbar( definition );\n\t\t\t\t}\n\t\t\t} else if ( !this.editor.ui.focusTracker.isFocused ) {\n\t\t\t\tif ( this._isToolbarVisible( definition ) ) {\n\t\t\t\t\tthis._hideToolbar( definition );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconst relatedElementDepth = relatedElement.getAncestors().length;\n\n\t\t\t\t// Many toolbars can express willingness to be displayed but they do not know about\n\t\t\t\t// each other. Figure out which toolbar is deepest in the view tree to decide which\n\t\t\t\t// should be displayed. For instance, if a selected image is inside a table cell, display\n\t\t\t\t// the ImageToolbar rather than the TableToolbar (#60).\n\t\t\t\tif ( relatedElementDepth > maxRelatedElementDepth ) {\n\t\t\t\t\tmaxRelatedElementDepth = relatedElementDepth;\n\t\t\t\t\tdeepestRelatedElement = relatedElement;\n\t\t\t\t\tdeepestToolbarDefinition = definition;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif ( deepestToolbarDefinition ) {\n\t\t\tthis._showToolbar( deepestToolbarDefinition, deepestRelatedElement! );\n\t\t}\n\t}\n\n\t/**\n\t * Hides the given toolbar.\n\t */\n\tprivate _hideToolbar( toolbarDefinition: WidgetRepositoryToolbarDefinition ) {\n\t\tthis._balloon.remove( toolbarDefinition.view );\n\t\tthis.stopListening( this._balloon, 'change:visibleView' );\n\t}\n\n\t/**\n\t * Shows up the toolbar if the toolbar is not visible.\n\t * Otherwise, repositions the toolbar's balloon when toolbar's view is the most top view in balloon stack.\n\t *\n\t * It might happen here that the toolbar's view is under another view. Then do nothing as the other toolbar view\n\t * should be still visible after the {@link module:ui/editorui/editorui~EditorUI#event:update}.\n\t */\n\tprivate _showToolbar( toolbarDefinition: WidgetRepositoryToolbarDefinition, relatedElement: ViewElement ) {\n\t\tif ( this._isToolbarVisible( toolbarDefinition ) ) {\n\t\t\trepositionContextualBalloon( this.editor, relatedElement );\n\t\t} else if ( !this._isToolbarInBalloon( toolbarDefinition ) ) {\n\t\t\tif ( !toolbarDefinition.initialized ) {\n\t\t\t\ttoolbarDefinition.initialized = true;\n\t\t\t\ttoolbarDefinition.view.fillFromConfig( toolbarDefinition.itemsConfig, this.editor.ui.componentFactory );\n\t\t\t}\n\n\t\t\tthis._balloon.add( {\n\t\t\t\tview: toolbarDefinition.view,\n\t\t\t\tposition: getBalloonPositionData( this.editor, relatedElement ),\n\t\t\t\tballoonClassName: toolbarDefinition.balloonClassName\n\t\t\t} );\n\n\t\t\t// Update toolbar position each time stack with toolbar view is switched to visible.\n\t\t\t// This is in a case target element has changed when toolbar was in invisible stack\n\t\t\t// e.g. target image was wrapped by a block quote.\n\t\t\t// See https://github.com/ckeditor/ckeditor5-widget/issues/92.\n\t\t\tthis.listenTo<ObservableChangeEvent>( this._balloon, 'change:visibleView', () => {\n\t\t\t\tfor ( const definition of this._toolbarDefinitions.values() ) {\n\t\t\t\t\tif ( this._isToolbarVisible( definition ) ) {\n\t\t\t\t\t\tconst relatedElement = definition.getRelatedElement( this.editor.editing.view.document.selection );\n\t\t\t\t\t\trepositionContextualBalloon( this.editor, relatedElement! );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t}\n\n\tprivate _isToolbarVisible( toolbar: WidgetRepositoryToolbarDefinition ) {\n\t\treturn this._balloon.visibleView === toolbar.view;\n\t}\n\n\tprivate _isToolbarInBalloon( toolbar: WidgetRepositoryToolbarDefinition ) {\n\t\treturn this._balloon.hasView( toolbar.view );\n\t}\n}\n\nfunction repositionContextualBalloon( editor: Editor, relatedElement: ViewElement ) {\n\tconst balloon: ContextualBalloon = editor.plugins.get( 'ContextualBalloon' );\n\tconst position = getBalloonPositionData( editor, relatedElement );\n\n\tballoon.updatePosition( position );\n}\n\nfunction getBalloonPositionData( editor: Editor, relatedElement: ViewElement ) {\n\tconst editingView = editor.editing.view;\n\tconst defaultPositions = BalloonPanelView.defaultPositions;\n\n\treturn {\n\t\ttarget: editingView.domConverter.mapViewToDom( relatedElement ) as RectSource | undefined,\n\t\tpositions: [\n\t\t\tdefaultPositions.northArrowSouth,\n\t\t\tdefaultPositions.northArrowSouthWest,\n\t\t\tdefaultPositions.northArrowSouthEast,\n\t\t\tdefaultPositions.southArrowNorth,\n\t\t\tdefaultPositions.southArrowNorthWest,\n\t\t\tdefaultPositions.southArrowNorthEast,\n\t\t\tdefaultPositions.viewportStickyNorth\n\t\t]\n\t};\n}\n\nfunction isWidgetSelected( selection: ViewDocumentSelection ) {\n\tconst viewElement = selection.getSelectedElement();\n\n\treturn !!( viewElement && isWidget( viewElement ) );\n}\n\n/**\n * The toolbar definition object used by the toolbar repository to manage toolbars.\n * It contains information necessary to display the toolbar in the\n * {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon contextual balloon} and\n * update it during its life (display) cycle.\n */\ninterface WidgetRepositoryToolbarDefinition {\n\n\t/**\n\t * The UI view of the toolbar.\n\t */\n\tview: ToolbarView;\n\n\t/**\n\t * A function that returns an engine {@link module:engine/view/view~View}\n\t * element the toolbar is to be attached to. For instance, an image widget or a table widget (or `null` when\n\t * there is no such element). The function accepts an instance of {@link module:engine/view/selection~Selection}.\n\t */\n\tgetRelatedElement: ( selection: ViewDocumentSelection ) => ViewElement | null | undefined;\n\n\t/**\n\t * CSS class for the widget balloon when a toolbar is displayed.\n\t */\n\tballoonClassName: string;\n\n\titemsConfig: Array<ToolbarConfigItem>;\n\n\tinitialized: boolean;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module widget/widgetresize/resizerstate\n */\n\nimport { ObservableMixin, Rect } from '@ckeditor/ckeditor5-utils';\n\nimport type { ResizerOptions } from '../widgetresize';\n\n/**\n * Stores the internal state of a single resizable object.\n */\nexport default class ResizeState extends ObservableMixin() {\n\t/**\n\t * The position of the handle that initiated the resizing. E.g. `\"top-left\"`, `\"bottom-right\"` etc. or `null`\n\t * if unknown.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public activeHandlePosition: string | null;\n\n\t/**\n\t * The width (percents) proposed, but not committed yet, in the current resize process.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public proposedWidthPercents: number | null;\n\n\t/**\n\t * The width (pixels) proposed, but not committed yet, in the current resize process.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public proposedWidth: number | null;\n\n\t/**\n\t * The height (pixels) proposed, but not committed yet, in the current resize process.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public proposedHeight: number | null;\n\n\t/**\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public proposedHandleHostWidth: number | null;\n\n\t/**\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public proposedHandleHostHeight: number | null;\n\n\t/**\n\t * The reference point of the resizer where the dragging started. It is used to measure the distance the user cursor\n\t * traveled, so how much the image should be enlarged.\n\t * This information is only known after the DOM was rendered, so it will be updated later.\n\t *\n\t * @internal\n\t */\n\tpublic _referenceCoordinates: { x: number; y: number } | null;\n\n\t/**\n\t * Resizer options.\n\t */\n\tprivate readonly _options: ResizerOptions;\n\n\t/**\n\t * The original width (pixels) of the resized object when the resize process was started.\n\t *\n\t * @readonly\n\t */\n\tprivate _originalWidth?: number;\n\n\t/**\n\t * The original height (pixels) of the resized object when the resize process was started.\n\t *\n\t * @readonly\n\t */\n\tprivate _originalHeight?: number;\n\n\t/**\n\t * The original width (percents) of the resized object when the resize process was started.\n\t *\n\t * @readonly\n\t */\n\tprivate _originalWidthPercents?: number;\n\n\t/**\n\t * A width to height ratio of the resized image.\n\t *\n\t * @readonly\n\t */\n\tprivate _aspectRatio?: number;\n\n\t/**\n\t * @param options Resizer options.\n\t */\n\tconstructor( options: ResizerOptions ) {\n\t\tsuper();\n\n\t\tthis.set( 'activeHandlePosition', null );\n\t\tthis.set( 'proposedWidthPercents', null );\n\t\tthis.set( 'proposedWidth', null );\n\t\tthis.set( 'proposedHeight', null );\n\t\tthis.set( 'proposedHandleHostWidth', null );\n\t\tthis.set( 'proposedHandleHostHeight', null );\n\n\t\tthis._options = options;\n\t\tthis._referenceCoordinates = null;\n\t}\n\n\t/**\n\t * The original width (pixels) of the resized object when the resize process was started.\n\t */\n\tpublic get originalWidth(): number | undefined {\n\t\treturn this._originalWidth;\n\t}\n\n\t/**\n\t * The original height (pixels) of the resized object when the resize process was started.\n\t */\n\tpublic get originalHeight(): number | undefined {\n\t\treturn this._originalHeight;\n\t}\n\n\t/**\n\t * The original width (percents) of the resized object when the resize process was started.\n\t */\n\tpublic get originalWidthPercents(): number | undefined {\n\t\treturn this._originalWidthPercents;\n\t}\n\n\t/**\n\t * A width to height ratio of the resized image.\n\t */\n\tpublic get aspectRatio(): number | undefined {\n\t\treturn this._aspectRatio;\n\t}\n\n\t/**\n\t *\n\t * @param domResizeHandle The handle used to calculate the reference point.\n\t */\n\tpublic begin( domResizeHandle: HTMLElement, domHandleHost: HTMLElement, domResizeHost: HTMLElement ): void {\n\t\tconst clientRect = new Rect( domHandleHost );\n\n\t\tthis.activeHandlePosition = getHandlePosition( domResizeHandle )!;\n\n\t\tthis._referenceCoordinates = getAbsoluteBoundaryPoint( domHandleHost, getOppositePosition( this.activeHandlePosition ) );\n\n\t\tthis._originalWidth = clientRect.width;\n\t\tthis._originalHeight = clientRect.height;\n\n\t\tthis._aspectRatio = clientRect.width / clientRect.height;\n\n\t\tconst widthStyle = domResizeHost.style.width;\n\n\t\tif ( widthStyle && widthStyle.match( /^\\d+(\\.\\d*)?%$/ ) ) {\n\t\t\tthis._originalWidthPercents = parseFloat( widthStyle );\n\t\t} else {\n\t\t\tthis._originalWidthPercents = calculateHostPercentageWidth( domResizeHost, clientRect );\n\t\t}\n\t}\n\n\tpublic update( newSize: {\n\t\twidth: number;\n\t\theight: number;\n\t\twidthPercents: number;\n\t\thandleHostWidth: number;\n\t\thandleHostHeight: number;\n\t} ): void {\n\t\tthis.proposedWidth = newSize.width;\n\t\tthis.proposedHeight = newSize.height;\n\t\tthis.proposedWidthPercents = newSize.widthPercents;\n\n\t\tthis.proposedHandleHostWidth = newSize.handleHostWidth;\n\t\tthis.proposedHandleHostHeight = newSize.handleHostHeight;\n\t}\n}\n\n/**\n * Calculates a relative width of a `domResizeHost` compared to its ancestor in percents.\n */\nfunction calculateHostPercentageWidth( domResizeHost: HTMLElement, resizeHostRect: Rect ) {\n\tconst domResizeHostParent = domResizeHost.parentElement;\n\n\t// Need to use computed style as it properly excludes parent's paddings from the returned value.\n\tlet parentWidth = parseFloat( domResizeHostParent!.ownerDocument.defaultView!.getComputedStyle( domResizeHostParent! ).width );\n\n\t// Sometimes parent width cannot be accessed. If that happens we should go up in the elements tree\n\t// and try to get width from next ancestor.\n\t// https://github.com/ckeditor/ckeditor5/issues/10776\n\tconst ancestorLevelLimit = 5;\n\tlet currentLevel = 0;\n\n\tlet checkedElement = domResizeHostParent!;\n\n\twhile ( isNaN( parentWidth ) ) {\n\t\tcheckedElement = checkedElement.parentElement!;\n\n\t\tif ( ++currentLevel > ancestorLevelLimit ) {\n\t\t\treturn 0;\n\t\t}\n\n\t\tparentWidth = parseFloat(\n\t\t\t\tdomResizeHostParent!.ownerDocument.defaultView!.getComputedStyle( checkedElement ).width\n\t\t);\n\t}\n\n\treturn resizeHostRect.width / parentWidth * 100;\n}\n\n/**\n * Returns coordinates of the top-left corner of an element, relative to the document's top-left corner.\n *\n * @param resizerPosition The position of the resize handle, e.g. `\"top-left\"`, `\"bottom-right\"`.\n */\nfunction getAbsoluteBoundaryPoint( element: HTMLElement, resizerPosition: string ) {\n\tconst elementRect = new Rect( element );\n\tconst positionParts = resizerPosition.split( '-' );\n\tconst ret = {\n\t\tx: positionParts[ 1 ] == 'right' ? elementRect.right : elementRect.left,\n\t\ty: positionParts[ 0 ] == 'bottom' ? elementRect.bottom : elementRect.top\n\t};\n\n\tret.x += element.ownerDocument.defaultView!.scrollX;\n\tret.y += element.ownerDocument.defaultView!.scrollY;\n\n\treturn ret;\n}\n\n/**\n * @param resizerPosition The expected resizer position, like `\"top-left\"`, `\"bottom-right\"`.\n * @returns A prefixed HTML class name for the resizer element.\n */\nfunction getResizerHandleClass( resizerPosition: string ) {\n\treturn `ck-widget__resizer__handle-${ resizerPosition }`;\n}\n\n/**\n * Determines the position of a given resize handle.\n *\n * @param domHandle Handle used to calculate the reference point.\n * @returns Returns a string like `\"top-left\"` or `undefined` if not matched.\n */\nfunction getHandlePosition( domHandle: HTMLElement ) {\n\tconst resizerPositions = [ 'top-left', 'top-right', 'bottom-right', 'bottom-left' ];\n\n\tfor ( const position of resizerPositions ) {\n\t\tif ( domHandle.classList.contains( getResizerHandleClass( position ) ) ) {\n\t\t\treturn position;\n\t\t}\n\t}\n}\n\n/**\n * @param position Like `\"top-left\"`.\n * @returns Inverted `position`, e.g. it returns `\"bottom-right\"` if `\"top-left\"` was given as `position`.\n */\nfunction getOppositePosition( position: string ) {\n\tconst parts = position.split( '-' );\n\tconst replacements: Record<string, string> = {\n\t\ttop: 'bottom',\n\t\tbottom: 'top',\n\t\tleft: 'right',\n\t\tright: 'left'\n\t};\n\n\treturn `${ replacements[ parts[ 0 ] ] }-${ replacements[ parts[ 1 ] ] }`;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module widget/widgetresize/resizer\n */\n\nimport { Template } from '@ckeditor/ckeditor5-ui';\nimport {\n\tRect,\n\tObservableMixin,\n\tcompareArrays,\n\ttype ObservableChangeEvent,\n\ttype DecoratedMethodEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport ResizeState from './resizerstate';\nimport SizeView from './sizeview';\n\nimport type { ResizerOptions } from '../widgetresize';\nimport type { ViewElement } from '@ckeditor/ckeditor5-engine';\n\n/**\n * Represents a resizer for a single resizable object.\n */\nexport default class Resizer extends ObservableMixin() {\n\t/**\n\t * Flag that indicates whether resizer can be used.\n\t *\n\t * @observable\n\t */\n\tdeclare public isEnabled: boolean;\n\n\t/**\n\t * Flag that indicates that resizer is currently focused.\n\t *\n\t * @observable\n\t */\n\tdeclare public isSelected: boolean;\n\n\t/**\n\t * Flag that indicates whether resizer is rendered (visible on the screen).\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public isVisible: boolean;\n\n\t/**\n\t * Stores the state of the resizable host geometry, such as the original width, the currently proposed height, etc.\n\t *\n\t * Note that a new state is created for each resize transaction.\n\t */\n\tprivate _state!: ResizeState;\n\n\t/**\n\t * A view displaying the proposed new element size during the resizing.\n\t */\n\tprivate _sizeView!: SizeView;\n\n\t/**\n\t * Options passed to the {@link #constructor}.\n\t */\n\tprivate _options: ResizerOptions;\n\n\t/**\n\t * A wrapper that is controlled by the resizer. This is usually a widget element.\n\t */\n\tprivate _viewResizerWrapper: ViewElement | null = null;\n\n\t/**\n\t * The width of the resized {@link module:widget/widgetresize~ResizerOptions#viewElement viewElement} before the resizing started.\n\t */\n\tprivate _initialViewWidth: string | undefined;\n\n\t/**\n\t * @param options Resizer options.\n\t */\n\tconstructor( options: ResizerOptions ) {\n\t\tsuper();\n\n\t\tthis._options = options;\n\n\t\tthis.set( 'isEnabled', true );\n\t\tthis.set( 'isSelected', false );\n\n\t\tthis.bind( 'isVisible' ).to( this, 'isEnabled', this, 'isSelected', ( isEnabled, isSelected ) => isEnabled && isSelected );\n\n\t\tthis.decorate( 'begin' );\n\t\tthis.decorate( 'cancel' );\n\t\tthis.decorate( 'commit' );\n\t\tthis.decorate( 'updateSize' );\n\n\t\tthis.on( 'commit', event => {\n\t\t\t// State might not be initialized yet. In this case, prevent further handling and make sure that the resizer is\n\t\t\t// cleaned up (#5195).\n\t\t\tif ( !this.state.proposedWidth && !this.state.proposedWidthPercents ) {\n\t\t\t\tthis._cleanup();\n\t\t\t\tevent.stop();\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * Stores the state of the resizable host geometry, such as the original width, the currently proposed height, etc.\n\t *\n\t * Note that a new state is created for each resize transaction.\n\t */\n\tpublic get state(): ResizeState {\n\t\treturn this._state;\n\t}\n\n\t/**\n\t * Makes resizer visible in the UI.\n\t */\n\tpublic show(): void {\n\t\tconst editingView = this._options.editor.editing.view;\n\n\t\teditingView.change( writer => {\n\t\t\twriter.removeClass( 'ck-hidden', this._viewResizerWrapper! );\n\t\t} );\n\t}\n\n\t/**\n\t * Hides resizer in the UI.\n\t */\n\tpublic hide(): void {\n\t\tconst editingView = this._options.editor.editing.view;\n\n\t\teditingView.change( writer => {\n\t\t\twriter.addClass( 'ck-hidden', this._viewResizerWrapper! );\n\t\t} );\n\t}\n\n\t/**\n\t * Attaches the resizer to the DOM.\n\t */\n\tpublic attach(): void {\n\t\t// eslint-disable-next-line @typescript-eslint/no-this-alias\n\t\tconst that = this;\n\t\tconst widgetElement = this._options.viewElement;\n\t\tconst editingView = this._options.editor.editing.view;\n\n\t\teditingView.change( writer => {\n\t\t\tconst viewResizerWrapper = writer.createUIElement( 'div', {\n\t\t\t\tclass: 'ck ck-reset_all ck-widget__resizer'\n\t\t\t}, function( domDocument ) {\n\t\t\t\tconst domElement = this.toDomElement( domDocument );\n\n\t\t\t\tthat._appendHandles( domElement );\n\t\t\t\tthat._appendSizeUI( domElement );\n\n\t\t\t\treturn domElement;\n\t\t\t} );\n\n\t\t\t// Append the resizer wrapper to the widget's wrapper.\n\t\t\twriter.insert( writer.createPositionAt( widgetElement, 'end' ), viewResizerWrapper );\n\t\t\twriter.addClass( 'ck-widget_with-resizer', widgetElement );\n\n\t\t\tthis._viewResizerWrapper = viewResizerWrapper;\n\n\t\t\tif ( !this.isVisible ) {\n\t\t\t\tthis.hide();\n\t\t\t}\n\t\t} );\n\n\t\tthis.on<ObservableChangeEvent>( 'change:isVisible', () => {\n\t\t\tif ( this.isVisible ) {\n\t\t\t\tthis.show();\n\t\t\t\tthis.redraw();\n\t\t\t} else {\n\t\t\t\tthis.hide();\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Starts the resizing process.\n\t *\n\t * Creates a new {@link #state} for the current process.\n\t *\n\t * @fires begin\n\t * @param domResizeHandle Clicked handle.\n\t */\n\tpublic begin( domResizeHandle: HTMLElement ): void {\n\t\tthis._state = new ResizeState( this._options );\n\n\t\tthis._sizeView._bindToState( this._options, this.state );\n\n\t\tthis._initialViewWidth = this._options.viewElement.getStyle( 'width' );\n\n\t\tthis.state.begin( domResizeHandle, this._getHandleHost(), this._getResizeHost() );\n\t}\n\n\t/**\n\t * Updates the proposed size based on `domEventData`.\n\t *\n\t * @fires updateSize\n\t */\n\tpublic updateSize( domEventData: MouseEvent ): void {\n\t\tconst newSize = this._proposeNewSize( domEventData );\n\t\tconst editingView = this._options.editor.editing.view;\n\n\t\teditingView.change( writer => {\n\t\t\tconst unit = this._options.unit || '%';\n\t\t\tconst newWidth = ( unit === '%' ? newSize.widthPercents : newSize.width ) + unit;\n\n\t\t\twriter.setStyle( 'width', newWidth, this._options.viewElement );\n\t\t} );\n\n\t\t// Get an actual image width, and:\n\t\t// * reflect this size to the resize wrapper\n\t\t// * apply this **real** size to the state\n\t\tconst domHandleHost = this._getHandleHost();\n\t\tconst domHandleHostRect = new Rect( domHandleHost );\n\n\t\tconst handleHostWidth = Math.round( domHandleHostRect.width );\n\t\tconst handleHostHeight = Math.round( domHandleHostRect.height );\n\n\t\t// Handle max-width limitation.\n\t\tconst domResizeHostRect = new Rect( domHandleHost );\n\n\t\tnewSize.width = Math.round( domResizeHostRect.width );\n\t\tnewSize.height = Math.round( domResizeHostRect.height );\n\n\t\tthis.redraw( domHandleHostRect );\n\n\t\tthis.state.update( {\n\t\t\t...newSize,\n\t\t\thandleHostWidth,\n\t\t\thandleHostHeight\n\t\t} );\n\t}\n\n\t/**\n\t * Applies the geometry proposed with the resizer.\n\t *\n\t * @fires commit\n\t */\n\tpublic commit(): void {\n\t\tconst unit = this._options.unit || '%';\n\t\tconst newValue = ( unit === '%' ? this.state.proposedWidthPercents : this.state.proposedWidth ) + unit;\n\n\t\t// Both cleanup and onCommit callback are very likely to make view changes. Ensure that it is made in a single step.\n\t\tthis._options.editor.editing.view.change( () => {\n\t\t\tthis._cleanup();\n\t\t\tthis._options.onCommit( newValue );\n\t\t} );\n\t}\n\n\t/**\n\t * Cancels and rejects the proposed resize dimensions, hiding the UI.\n\t *\n\t * @fires cancel\n\t */\n\tpublic cancel(): void {\n\t\tthis._cleanup();\n\t}\n\n\t/**\n\t * Destroys the resizer.\n\t */\n\tpublic destroy(): void {\n\t\tthis.cancel();\n\t}\n\n\t/**\n\t * Redraws the resizer.\n\t *\n\t * @param handleHostRect Handle host rectangle might be given to improve performance.\n\t */\n\tpublic redraw( handleHostRect?: Rect ): void {\n\t\tconst domWrapper = this._domResizerWrapper;\n\n\t\t// Refresh only if resizer exists in the DOM.\n\t\tif ( !existsInDom( domWrapper ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst widgetWrapper = domWrapper!.parentElement;\n\t\tconst handleHost = this._getHandleHost();\n\t\tconst resizerWrapper = this._viewResizerWrapper!;\n\t\tconst currentDimensions = [\n\t\t\tresizerWrapper.getStyle( 'width' ),\n\t\t\tresizerWrapper.getStyle( 'height' ),\n\t\t\tresizerWrapper.getStyle( 'left' ),\n\t\t\tresizerWrapper.getStyle( 'top' )\n\t\t];\n\t\tlet newDimensions: Array<string | undefined>;\n\n\t\tif ( widgetWrapper!.isSameNode( handleHost ) ) {\n\t\t\tconst clientRect = handleHostRect || new Rect( handleHost );\n\n\t\t\tnewDimensions = [\n\t\t\t\tclientRect.width + 'px',\n\t\t\t\tclientRect.height + 'px',\n\t\t\t\tundefined,\n\t\t\t\tundefined\n\t\t\t];\n\t\t}\n\t\t// In case a resizing host is not a widget wrapper, we need to compensate\n\t\t// for any additional offsets the resize host might have. E.g. wrapper padding\n\t\t// or simply another editable. By doing that the border and resizers are shown\n\t\t// only around the resize host.\n\t\telse {\n\t\t\tnewDimensions = [\n\t\t\t\thandleHost.offsetWidth + 'px',\n\t\t\t\thandleHost.offsetHeight + 'px',\n\t\t\t\thandleHost.offsetLeft + 'px',\n\t\t\t\thandleHost.offsetTop + 'px'\n\t\t\t];\n\t\t}\n\n\t\t// Make changes to the view only if the resizer should actually get new dimensions.\n\t\t// Otherwise, if View#change() was always called, this would cause EditorUI#update\n\t\t// loops because the WidgetResize plugin listens to EditorUI#update and updates\n\t\t// the resizer.\n\t\t// https://github.com/ckeditor/ckeditor5/issues/7633\n\t\tif ( compareArrays( currentDimensions, newDimensions ) !== 'same' ) {\n\t\t\tthis._options.editor.editing.view.change( writer => {\n\t\t\t\twriter.setStyle( {\n\t\t\t\t\twidth: newDimensions[ 0 ]!,\n\t\t\t\t\theight: newDimensions[ 1 ]!,\n\t\t\t\t\tleft: newDimensions[ 2 ]!,\n\t\t\t\t\ttop: newDimensions[ 3 ]!\n\t\t\t\t}, resizerWrapper );\n\t\t\t} );\n\t\t}\n\t}\n\n\tpublic containsHandle( domElement: HTMLElement ): boolean {\n\t\treturn this._domResizerWrapper!.contains( domElement );\n\t}\n\n\tpublic static isResizeHandle( domElement: HTMLElement ): boolean {\n\t\treturn domElement.classList.contains( 'ck-widget__resizer__handle' );\n\t}\n\n\t/**\n\t * Cleans up the context state.\n\t */\n\tprivate _cleanup(): void {\n\t\tthis._sizeView._dismiss();\n\n\t\tconst editingView = this._options.editor.editing.view;\n\n\t\teditingView.change( writer => {\n\t\t\twriter.setStyle( 'width', this._initialViewWidth!, this._options.viewElement );\n\t\t} );\n\t}\n\n\t/**\n\t * Calculates the proposed size as the resize handles are dragged.\n\t *\n\t * @param domEventData Event data that caused the size update request. It should be used to calculate the proposed size.\n\t */\n\tprivate _proposeNewSize( domEventData: MouseEvent ): { width: number; height: number; widthPercents: number } {\n\t\tconst state = this.state;\n\t\tconst currentCoordinates = extractCoordinates( domEventData );\n\t\tconst isCentered = this._options.isCentered ? this._options.isCentered( this ) : true;\n\n\t\t// Enlargement defines how much the resize host has changed in a given axis. Naturally it could be a negative number\n\t\t// meaning that it has been shrunk.\n\t\t//\n\t\t// +----------------+--+\n\t\t// | | |\n\t\t// | img | |\n\t\t// | /handle host | |\n\t\t// +----------------+ | ^\n\t\t// | | | - enlarge y\n\t\t// +-------------------+ v\n\t\t// \t\t\t\t\t<-->\n\t\t// \t\t\t\t\t enlarge x\n\t\tconst enlargement = {\n\t\t\tx: state._referenceCoordinates!.x - ( currentCoordinates.x + state.originalWidth! ),\n\t\t\ty: ( currentCoordinates.y - state.originalHeight! ) - state._referenceCoordinates!.y\n\t\t};\n\n\t\tif ( isCentered && state.activeHandlePosition!.endsWith( '-right' ) ) {\n\t\t\tenlargement.x = currentCoordinates.x - ( state._referenceCoordinates!.x + state.originalWidth! );\n\t\t}\n\n\t\t// Objects needs to be resized twice as much in horizontal axis if centered, since enlargement is counted from\n\t\t// one resized corner to your cursor. It needs to be duplicated to compensate for the other side too.\n\t\tif ( isCentered ) {\n\t\t\tenlargement.x *= 2;\n\t\t}\n\n\t\t// const resizeHost = this._getResizeHost();\n\n\t\t// The size proposed by the user. It does not consider the aspect ratio.\n\t\tlet width = Math.abs( state.originalWidth! + enlargement.x );\n\t\tlet height = Math.abs( state.originalHeight! + enlargement.y );\n\n\t\t// Dominant determination must take the ratio into account.\n\t\tconst dominant = width / state.aspectRatio! > height ? 'width' : 'height';\n\n\t\tif ( dominant == 'width' ) {\n\t\t\theight = width / state.aspectRatio!;\n\t\t} else {\n\t\t\twidth = height * state.aspectRatio!;\n\t\t}\n\n\t\treturn {\n\t\t\twidth: Math.round( width ),\n\t\t\theight: Math.round( height ),\n\t\t\twidthPercents: Math.min( Math.round( state.originalWidthPercents! / state.originalWidth! * width * 100 ) / 100, 100 )\n\t\t};\n\t}\n\n\t/**\n\t * Obtains the resize host.\n\t *\n\t * Resize host is an object that receives dimensions which are the result of resizing.\n\t */\n\tprivate _getResizeHost(): HTMLElement {\n\t\tconst widgetWrapper = this._domResizerWrapper!.parentElement;\n\n\t\treturn this._options.getResizeHost( widgetWrapper! );\n\t}\n\n\t/**\n\t * Obtains the handle host.\n\t *\n\t * Handle host is an object that the handles are aligned to.\n\t *\n\t * Handle host will not always be an entire widget itself. Take an image as an example. The image widget\n\t * contains an image and a caption. Only the image should be surrounded with handles.\n\t */\n\tprivate _getHandleHost(): HTMLElement {\n\t\tconst widgetWrapper = this._domResizerWrapper!.parentElement;\n\n\t\treturn this._options.getHandleHost( widgetWrapper! );\n\t}\n\n\t/**\n\t * DOM container of the entire resize UI.\n\t *\n\t * Note that this property will have a value only after the element bound with the resizer is rendered\n\t * (otherwise `null`).\n\t */\n\tprivate get _domResizerWrapper(): HTMLElement | null {\n\t\treturn this._options.editor.editing.view.domConverter.mapViewToDom( this._viewResizerWrapper! ) as any;\n\t}\n\n\t/**\n\t * Renders the resize handles in the DOM.\n\t *\n\t * @param domElement The resizer wrapper.\n\t */\n\tprivate _appendHandles( domElement: HTMLElement ) {\n\t\tconst resizerPositions = [ 'top-left', 'top-right', 'bottom-right', 'bottom-left' ];\n\n\t\tfor ( const currentPosition of resizerPositions ) {\n\t\t\tdomElement.appendChild( ( new Template( {\n\t\t\t\ttag: 'div',\n\t\t\t\tattributes: {\n\t\t\t\t\tclass: `ck-widget__resizer__handle ${ getResizerClass( currentPosition ) }`\n\t\t\t\t}\n\t\t\t} ).render() ) );\n\t\t}\n\t}\n\n\t/**\n\t * Sets up the {@link #_sizeView} property and adds it to the passed `domElement`.\n\t */\n\tprivate _appendSizeUI( domElement: HTMLElement ) {\n\t\tthis._sizeView = new SizeView();\n\n\t\t// Make sure icon#element is rendered before passing to appendChild().\n\t\tthis._sizeView.render();\n\n\t\tdomElement.appendChild( this._sizeView.element! );\n\t}\n}\n\n/**\n * @eventName ~Resizer#begin\n */\nexport type ResizerBeginEvent = DecoratedMethodEvent<Resizer, 'begin'>;\n\n/**\n * @eventName ~Resizer#cancel\n */\nexport type ResizerCancelEvent = DecoratedMethodEvent<Resizer, 'cancel'>;\n\n/**\n * @eventName ~Resizer#commit\n */\nexport type ResizerCommitEvent = DecoratedMethodEvent<Resizer, 'commit'>;\n\n/**\n * @eventName ~Resizer#updateSize\n */\nexport type ResizerUpdateSizeEvent = DecoratedMethodEvent<Resizer, 'updateSize'>;\n\n/**\n * @param resizerPosition Expected resizer position like `\"top-left\"`, `\"bottom-right\"`.\n * @returns A prefixed HTML class name for the resizer element\n */\nfunction getResizerClass( resizerPosition: string ) {\n\treturn `ck-widget__resizer__handle-${ resizerPosition }`;\n}\n\nfunction extractCoordinates( event: MouseEvent ) {\n\treturn {\n\t\tx: event.pageX,\n\t\ty: event.pageY\n\t};\n}\n\nfunction existsInDom( element: Node | DocumentFragment | undefined | null ) {\n\treturn element && element.ownerDocument && element.ownerDocument.contains( element );\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./widgetresize.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./clipboard.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/dragdrop\n */\n\nimport { Plugin, type Editor } from '@ckeditor/ckeditor5-core';\n\nimport {\n\tLiveRange,\n\tMouseObserver,\n\ttype DataTransfer,\n\ttype Element,\n\ttype Position,\n\ttype Range,\n\ttype ViewDocumentMouseDownEvent,\n\ttype ViewDocumentMouseUpEvent,\n\ttype ViewElement,\n\ttype ViewRange\n} from '@ckeditor/ckeditor5-engine';\n\nimport { Widget, isWidget, type WidgetToolbarRepository } from '@ckeditor/ckeditor5-widget';\n\nimport {\n\tenv,\n\tuid,\n\tdelay,\n\ttype DelayedFunc,\n\ttype ObservableChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport ClipboardPipeline, { type ClipboardContentInsertionEvent, type ViewDocumentClipboardOutputEvent } from './clipboardpipeline';\nimport ClipboardObserver, {\n\ttype ViewDocumentDragEndEvent,\n\ttype ViewDocumentDragEnterEvent,\n\ttype ViewDocumentDraggingEvent,\n\ttype ViewDocumentDragLeaveEvent,\n\ttype ViewDocumentDragStartEvent,\n\ttype ViewDocumentClipboardInputEvent\n} from './clipboardobserver';\n\nimport { throttle, type DebouncedFunc } from 'lodash-es';\n\nimport '../theme/clipboard.css';\n\n// Drag and drop events overview:\n//\n// ┌──────────────────┐\n// │ mousedown │ Sets the draggable attribute.\n// └─────────┬────────┘\n// │\n// └─────────────────────┐\n// │ │\n// │ ┌─────────V────────┐\n// │ │ mouseup │ Dragging did not start, removes the draggable attribute.\n// │ └──────────────────┘\n// │\n// ┌─────────V────────┐ Retrieves the selected model.DocumentFragment\n// │ dragstart │ and converts it to view.DocumentFragment.\n// └─────────┬────────┘\n// │\n// ┌─────────V────────┐ Processes view.DocumentFragment to text/html and text/plain\n// │ clipboardOutput │ and stores the results in data.dataTransfer.\n// └─────────┬────────┘\n// │\n// │ DOM dragover\n// ┌────────────┐\n// │ │\n// ┌─────────V────────┐ │\n// │ dragging │ │ Updates the drop target marker.\n// └─────────┬────────┘ │\n// │ │\n// ┌─────────────└────────────┘\n// │ │ │\n// │ ┌─────────V────────┐ │\n// │ │ dragleave │ │ Removes the drop target marker.\n// │ └─────────┬────────┘ │\n// │ │ │\n// ┌───│─────────────┘ │\n// │ │ │ │\n// │ │ ┌─────────V────────┐ │\n// │ │ │ dragenter │ │ Focuses the editor view.\n// │ │ └─────────┬────────┘ │\n// │ │ │ │\n// │ │ └────────────┘\n// │ │\n// │ └─────────────┐\n// │ │ │\n// │ │ ┌─────────V────────┐\n// └───┐ │ drop │ (The default handler of the clipboard pipeline).\n// │ └─────────┬────────┘\n// │ │\n// │ ┌─────────V────────┐ Resolves the final data.targetRanges.\n// │ │ clipboardInput │ Aborts if dropping on dragged content.\n// │ └─────────┬────────┘\n// │ │\n// │ ┌─────────V────────┐\n// │ │ clipboardInput │ (The default handler of the clipboard pipeline).\n// │ └─────────┬────────┘\n// │ │\n// │ ┌───────────V───────────┐\n// │ │ inputTransformation │ (The default handler of the clipboard pipeline).\n// │ └───────────┬───────────┘\n// │ │\n// │ ┌──────────V──────────┐\n// │ │ contentInsertion │ Updates the document selection to drop range.\n// │ └──────────┬──────────┘\n// │ │\n// │ ┌──────────V──────────┐\n// │ │ contentInsertion │ (The default handler of the clipboard pipeline).\n// │ └──────────┬──────────┘\n// │ │\n// │ ┌──────────V──────────┐\n// │ │ contentInsertion │ Removes the content from the original range if the insertion was successful.\n// │ └──────────┬──────────┘\n// │ │\n// └─────────────┐\n// │\n// ┌─────────V────────┐\n// │ dragend │ Removes the drop marker and cleans the state.\n// └──────────────────┘\n//\n\n/**\n * The drag and drop feature. It works on top of the {@link module:clipboard/clipboardpipeline~ClipboardPipeline}.\n *\n * Read more about the clipboard integration in the {@glink framework/deep-dive/clipboard clipboard deep-dive} guide.\n */\nexport default class DragDrop extends Plugin {\n\t/**\n\t * The live range over the original content that is being dragged.\n\t */\n\tprivate _draggedRange!: LiveRange | null;\n\n\t/**\n\t * The UID of current dragging that is used to verify if the drop started in the same editor as the drag start.\n\t *\n\t * **Note**: This is a workaround for broken 'dragend' events (they are not fired if the source text node got removed).\n\t */\n\tprivate _draggingUid!: string;\n\n\t/**\n\t * The reference to the model element that currently has a `draggable` attribute set (it is set while dragging).\n\t */\n\tprivate _draggableElement!: Element | null;\n\n\t/**\n\t * A throttled callback updating the drop marker.\n\t */\n\tprivate _updateDropMarkerThrottled!: DebouncedFunc<( targetRange: Range ) => void>;\n\n\t/**\n\t * A delayed callback removing the drop marker.\n\t */\n\tprivate _removeDropMarkerDelayed!: DelayedFunc<() => void>;\n\n\t/**\n\t * A delayed callback removing draggable attributes.\n\t */\n\tprivate _clearDraggableAttributesDelayed!: DelayedFunc<() => void>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'DragDrop' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ClipboardPipeline, Widget ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\n\t\tthis._draggedRange = null;\n\t\tthis._draggingUid = '';\n\t\tthis._draggableElement = null;\n\t\tthis._updateDropMarkerThrottled = throttle( targetRange => this._updateDropMarker( targetRange ), 40 );\n\t\tthis._removeDropMarkerDelayed = delay( () => this._removeDropMarker(), 40 );\n\t\tthis._clearDraggableAttributesDelayed = delay( () => this._clearDraggableAttributes(), 40 );\n\n\t\tif ( editor.plugins.has( 'DragDropExperimental' ) ) {\n\t\t\tthis.forceDisabled( 'DragDropExperimental' );\n\n\t\t\treturn;\n\t\t}\n\n\t\tview.addObserver( ClipboardObserver );\n\t\tview.addObserver( MouseObserver );\n\n\t\tthis._setupDragging();\n\t\tthis._setupContentInsertionIntegration();\n\t\tthis._setupClipboardInputIntegration();\n\t\tthis._setupDropMarker();\n\t\tthis._setupDraggableAttributeHandling();\n\n\t\tthis.listenTo<ObservableChangeEvent<boolean>>( editor, 'change:isReadOnly', ( evt, name, isReadOnly ) => {\n\t\t\tif ( isReadOnly ) {\n\t\t\t\tthis.forceDisabled( 'readOnlyMode' );\n\t\t\t} else {\n\t\t\t\tthis.clearForceDisabled( 'readOnlyMode' );\n\t\t\t}\n\t\t} );\n\n\t\tthis.on<ObservableChangeEvent<boolean>>( 'change:isEnabled', ( evt, name, isEnabled ) => {\n\t\t\tif ( !isEnabled ) {\n\t\t\t\tthis._finalizeDragging( false );\n\t\t\t}\n\t\t} );\n\n\t\tif ( env.isAndroid ) {\n\t\t\tthis.forceDisabled( 'noAndroidSupport' );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tif ( this._draggedRange ) {\n\t\t\tthis._draggedRange.detach();\n\t\t\tthis._draggedRange = null;\n\t\t}\n\n\t\tthis._updateDropMarkerThrottled.cancel();\n\t\tthis._removeDropMarkerDelayed.cancel();\n\t\tthis._clearDraggableAttributesDelayed.cancel();\n\n\t\treturn super.destroy();\n\t}\n\n\t/**\n\t * Drag and drop events handling.\n\t */\n\tprivate _setupDragging(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst modelDocument = model.document;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\t// The handler for the drag start; it is responsible for setting data transfer object.\n\t\tthis.listenTo<ViewDocumentDragStartEvent>( viewDocument, 'dragstart', ( evt, data ) => {\n\t\t\tconst selection = modelDocument.selection;\n\n\t\t\t// Don't drag the editable element itself.\n\t\t\tif ( data.target && data.target.is( 'editableElement' ) ) {\n\t\t\t\tdata.preventDefault();\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// TODO we could clone this node somewhere and style it to match editing view but without handles,\n\t\t\t// selection outline, WTA buttons, etc.\n\t\t\t// data.dataTransfer._native.setDragImage( data.domTarget, 0, 0 );\n\n\t\t\t// Check if this is dragstart over the widget (but not a nested editable).\n\t\t\tconst draggableWidget = data.target ? findDraggableWidget( data.target ) : null;\n\n\t\t\tif ( draggableWidget ) {\n\t\t\t\tconst modelElement = editor.editing.mapper.toModelElement( draggableWidget )!;\n\n\t\t\t\tthis._draggedRange = LiveRange.fromRange( model.createRangeOn( modelElement ) );\n\n\t\t\t\t// Disable toolbars so they won't obscure the drop area.\n\t\t\t\tif ( editor.plugins.has( 'WidgetToolbarRepository' ) ) {\n\t\t\t\t\tconst widgetToolbarRepository: WidgetToolbarRepository = editor.plugins.get( 'WidgetToolbarRepository' );\n\n\t\t\t\t\twidgetToolbarRepository.forceDisabled( 'dragDrop' );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// If this was not a widget we should check if we need to drag some text content.\n\t\t\telse if ( !viewDocument.selection.isCollapsed ) {\n\t\t\t\tconst selectedElement = viewDocument.selection.getSelectedElement();\n\n\t\t\t\tif ( !selectedElement || !isWidget( selectedElement ) ) {\n\t\t\t\t\tthis._draggedRange = LiveRange.fromRange( selection.getFirstRange()! );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( !this._draggedRange ) {\n\t\t\t\tdata.preventDefault();\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._draggingUid = uid();\n\n\t\t\tconst canEditAtDraggedRange = this.isEnabled && editor.model.canEditAt( this._draggedRange );\n\n\t\t\tdata.dataTransfer.effectAllowed = canEditAtDraggedRange ? 'copyMove' : 'copy';\n\t\t\tdata.dataTransfer.setData( 'application/ckeditor5-dragging-uid', this._draggingUid );\n\n\t\t\tconst draggedSelection = model.createSelection( this._draggedRange.toRange() );\n\t\t\tconst content = editor.data.toView( model.getSelectedContent( draggedSelection ) );\n\n\t\t\tviewDocument.fire<ViewDocumentClipboardOutputEvent>( 'clipboardOutput', {\n\t\t\t\tdataTransfer: data.dataTransfer,\n\t\t\t\tcontent,\n\t\t\t\tmethod: 'dragstart'\n\t\t\t} );\n\n\t\t\tif ( !canEditAtDraggedRange ) {\n\t\t\t\tthis._draggedRange.detach();\n\t\t\t\tthis._draggedRange = null;\n\t\t\t\tthis._draggingUid = '';\n\t\t\t}\n\t\t}, { priority: 'low' } );\n\n\t\t// The handler for finalizing drag and drop. It should always be triggered after dragging completes\n\t\t// even if it was completed in a different application.\n\t\t// Note: This is not fired if source text node got removed while downcasting a marker.\n\t\tthis.listenTo<ViewDocumentDragEndEvent>( viewDocument, 'dragend', ( evt, data ) => {\n\t\t\tthis._finalizeDragging( !data.dataTransfer.isCanceled && data.dataTransfer.dropEffect == 'move' );\n\t\t}, { priority: 'low' } );\n\n\t\t// Dragging over the editable.\n\t\tthis.listenTo<ViewDocumentDragEnterEvent>( viewDocument, 'dragenter', () => {\n\t\t\tif ( !this.isEnabled ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tview.focus();\n\t\t} );\n\n\t\t// Dragging out of the editable.\n\t\tthis.listenTo<ViewDocumentDragLeaveEvent>( viewDocument, 'dragleave', () => {\n\t\t\t// We do not know if the mouse left the editor or just some element in it, so let us wait a few milliseconds\n\t\t\t// to check if 'dragover' is not fired.\n\t\t\tthis._removeDropMarkerDelayed();\n\t\t} );\n\n\t\t// Handler for moving dragged content over the target area.\n\t\tthis.listenTo<ViewDocumentDraggingEvent>( viewDocument, 'dragging', ( evt, data ) => {\n\t\t\tif ( !this.isEnabled ) {\n\t\t\t\tdata.dataTransfer.dropEffect = 'none';\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._removeDropMarkerDelayed.cancel();\n\n\t\t\tconst targetRange = findDropTargetRange( editor, data.targetRanges, data.target );\n\n\t\t\t// Do not drop if target place is not editable.\n\t\t\tif ( !editor.model.canEditAt( targetRange ) ) {\n\t\t\t\tdata.dataTransfer.dropEffect = 'none';\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// If this is content being dragged from another editor, moving out of current editor instance\n\t\t\t// is not possible until 'dragend' event case will be fixed.\n\t\t\tif ( !this._draggedRange ) {\n\t\t\t\tdata.dataTransfer.dropEffect = 'copy';\n\t\t\t}\n\n\t\t\t// In Firefox it is already set and effect allowed remains the same as originally set.\n\t\t\tif ( !env.isGecko ) {\n\t\t\t\tif ( data.dataTransfer.effectAllowed == 'copy' ) {\n\t\t\t\t\tdata.dataTransfer.dropEffect = 'copy';\n\t\t\t\t} else if ( [ 'all', 'copyMove' ].includes( data.dataTransfer.effectAllowed ) ) {\n\t\t\t\t\tdata.dataTransfer.dropEffect = 'move';\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t/* istanbul ignore else -- @preserve */\n\t\t\tif ( targetRange ) {\n\t\t\t\tthis._updateDropMarkerThrottled( targetRange );\n\t\t\t}\n\t\t}, { priority: 'low' } );\n\t}\n\n\t/**\n\t * Integration with the `clipboardInput` event.\n\t */\n\tprivate _setupClipboardInputIntegration(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\t// Update the event target ranges and abort dropping if dropping over itself.\n\t\tthis.listenTo<ViewDocumentClipboardInputEvent>( viewDocument, 'clipboardInput', ( evt, data ) => {\n\t\t\tif ( data.method != 'drop' ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst targetRange = findDropTargetRange( editor, data.targetRanges, data.target );\n\n\t\t\t// The dragging markers must be removed after searching for the target range because sometimes\n\t\t\t// the target lands on the marker itself.\n\t\t\tthis._removeDropMarker();\n\n\t\t\t/* istanbul ignore if -- @preserve */\n\t\t\tif ( !targetRange || !editor.model.canEditAt( targetRange ) ) {\n\t\t\t\tthis._finalizeDragging( false );\n\t\t\t\tevt.stop();\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Since we cannot rely on the drag end event, we must check if the local drag range is from the current drag and drop\n\t\t\t// or it is from some previous not cleared one.\n\t\t\tif ( this._draggedRange && this._draggingUid != data.dataTransfer.getData( 'application/ckeditor5-dragging-uid' ) ) {\n\t\t\t\tthis._draggedRange.detach();\n\t\t\t\tthis._draggedRange = null;\n\t\t\t\tthis._draggingUid = '';\n\t\t\t}\n\n\t\t\t// Do not do anything if some content was dragged within the same document to the same position.\n\t\t\tconst isMove = getFinalDropEffect( data.dataTransfer ) == 'move';\n\n\t\t\tif ( isMove && this._draggedRange && this._draggedRange.containsRange( targetRange, true ) ) {\n\t\t\t\tthis._finalizeDragging( false );\n\t\t\t\tevt.stop();\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Override the target ranges with the one adjusted to the best one for a drop.\n\t\t\tdata.targetRanges = [ editor.editing.mapper.toViewRange( targetRange ) ];\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * Integration with the `contentInsertion` event of the clipboard pipeline.\n\t */\n\tprivate _setupContentInsertionIntegration(): void {\n\t\tconst clipboardPipeline = this.editor.plugins.get( ClipboardPipeline );\n\n\t\tclipboardPipeline.on<ClipboardContentInsertionEvent>( 'contentInsertion', ( evt, data ) => {\n\t\t\tif ( !this.isEnabled || data.method !== 'drop' ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Update the selection to the target range in the same change block to avoid selection post-fixing\n\t\t\t// and to be able to clone text attributes for plain text dropping.\n\t\t\tconst ranges = data.targetRanges!.map( viewRange => this.editor.editing.mapper.toModelRange( viewRange ) );\n\n\t\t\tthis.editor.model.change( writer => writer.setSelection( ranges ) );\n\t\t}, { priority: 'high' } );\n\n\t\tclipboardPipeline.on<ClipboardContentInsertionEvent>( 'contentInsertion', ( evt, data ) => {\n\t\t\tif ( !this.isEnabled || data.method !== 'drop' ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Remove dragged range content, remove markers, clean after dragging.\n\t\t\tconst isMove = getFinalDropEffect( data.dataTransfer ) == 'move';\n\n\t\t\t// Whether any content was inserted (insertion might fail if the schema is disallowing some elements\n\t\t\t// (for example an image caption allows only the content of a block but not blocks themselves.\n\t\t\t// Some integrations might not return valid range (i.e., table pasting).\n\t\t\tconst isSuccess = !data.resultRange || !data.resultRange.isCollapsed;\n\n\t\t\tthis._finalizeDragging( isSuccess && isMove );\n\t\t}, { priority: 'lowest' } );\n\t}\n\n\t/**\n\t * Adds listeners that add the `draggable` attribute to the elements while the mouse button is down so the dragging could start.\n\t */\n\tprivate _setupDraggableAttributeHandling(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\t// Add the 'draggable' attribute to the widget while pressing the selection handle.\n\t\t// This is required for widgets to be draggable. In Chrome it will enable dragging text nodes.\n\t\tthis.listenTo<ViewDocumentMouseDownEvent>( viewDocument, 'mousedown', ( evt, data ) => {\n\t\t\t// The lack of data can be caused by editor tests firing fake mouse events. This should not occur\n\t\t\t// in real-life scenarios but this greatly simplifies editor tests that would otherwise fail a lot.\n\t\t\tif ( env.isAndroid || !data ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._clearDraggableAttributesDelayed.cancel();\n\n\t\t\t// Check if this is a mousedown over the widget (but not a nested editable).\n\t\t\tlet draggableElement = findDraggableWidget( data.target );\n\n\t\t\t// Note: There is a limitation that if more than a widget is selected (a widget and some text)\n\t\t\t// and dragging starts on the widget, then only the widget is dragged.\n\n\t\t\t// If this was not a widget then we should check if we need to drag some text content.\n\t\t\t// In Chrome set a 'draggable' attribute on closest editable to allow immediate dragging of the selected text range.\n\t\t\t// In Firefox this is not needed. In Safari it makes the whole editable draggable (not just textual content).\n\t\t\t// Disabled in read-only mode because draggable=\"true\" + contenteditable=\"false\" results\n\t\t\t// in not firing selectionchange event ever, which makes the selection stuck in read-only mode.\n\t\t\tif ( env.isBlink && !draggableElement && !viewDocument.selection.isCollapsed ) {\n\t\t\t\tconst selectedElement = viewDocument.selection.getSelectedElement();\n\n\t\t\t\tif ( !selectedElement || !isWidget( selectedElement ) ) {\n\t\t\t\t\tconst editableElement = viewDocument.selection.editableElement;\n\n\t\t\t\t\tif ( editableElement && !editableElement.isReadOnly ) {\n\t\t\t\t\t\tdraggableElement = editableElement;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( draggableElement ) {\n\t\t\t\tview.change( writer => {\n\t\t\t\t\twriter.setAttribute( 'draggable', 'true', draggableElement! );\n\t\t\t\t} );\n\n\t\t\t\t// Keep the reference to the model element in case the view element gets removed while dragging.\n\t\t\t\tthis._draggableElement = editor.editing.mapper.toModelElement( draggableElement )!;\n\t\t\t}\n\t\t} );\n\n\t\t// Remove the draggable attribute in case no dragging started (only mousedown + mouseup).\n\t\tthis.listenTo<ViewDocumentMouseUpEvent>( viewDocument, 'mouseup', () => {\n\t\t\tif ( !env.isAndroid ) {\n\t\t\t\tthis._clearDraggableAttributesDelayed();\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Removes the `draggable` attribute from the element that was used for dragging.\n\t */\n\tprivate _clearDraggableAttributes(): void {\n\t\tconst editing = this.editor.editing;\n\n\t\tediting.view.change( writer => {\n\t\t\t// Remove 'draggable' attribute.\n\t\t\tif ( this._draggableElement && this._draggableElement.root.rootName != '$graveyard' ) {\n\t\t\t\twriter.removeAttribute( 'draggable', editing.mapper.toViewElement( this._draggableElement )! );\n\t\t\t}\n\n\t\t\tthis._draggableElement = null;\n\t\t} );\n\t}\n\n\t/**\n\t * Creates downcast conversion for the drop target marker.\n\t */\n\tprivate _setupDropMarker(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Drop marker conversion for hovering over widgets.\n\t\teditor.conversion.for( 'editingDowncast' ).markerToHighlight( {\n\t\t\tmodel: 'drop-target',\n\t\t\tview: {\n\t\t\t\tclasses: [ 'ck-clipboard-drop-target-range' ]\n\t\t\t}\n\t\t} );\n\n\t\t// Drop marker conversion for in text drop target.\n\t\teditor.conversion.for( 'editingDowncast' ).markerToElement( {\n\t\t\tmodel: 'drop-target',\n\t\t\tview: ( data, { writer } ) => {\n\t\t\t\tconst inText = editor.model.schema.checkChild( data.markerRange.start, '$text' );\n\n\t\t\t\tif ( !inText ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\treturn writer.createUIElement( 'span', { class: 'ck ck-clipboard-drop-target-position' }, function( domDocument ) {\n\t\t\t\t\tconst domElement = this.toDomElement( domDocument );\n\n\t\t\t\t\t// Using word joiner to make this marker as high as text and also making text not break on marker.\n\t\t\t\t\tdomElement.append( '\\u2060', domDocument.createElement( 'span' ), '\\u2060' );\n\n\t\t\t\t\treturn domElement;\n\t\t\t\t} );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Updates the drop target marker to the provided range.\n\t *\n\t * @param targetRange The range to set the marker to.\n\t */\n\tprivate _updateDropMarker( targetRange: Range ): void {\n\t\tconst editor = this.editor;\n\t\tconst markers = editor.model.markers;\n\n\t\teditor.model.change( writer => {\n\t\t\tif ( markers.has( 'drop-target' ) ) {\n\t\t\t\tif ( !markers.get( 'drop-target' )!.getRange().isEqual( targetRange ) ) {\n\t\t\t\t\twriter.updateMarker( 'drop-target', { range: targetRange } );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\twriter.addMarker( 'drop-target', {\n\t\t\t\t\trange: targetRange,\n\t\t\t\t\tusingOperation: false,\n\t\t\t\t\taffectsData: false\n\t\t\t\t} );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Removes the drop target marker.\n\t */\n\tprivate _removeDropMarker(): void {\n\t\tconst model = this.editor.model;\n\n\t\tthis._removeDropMarkerDelayed.cancel();\n\t\tthis._updateDropMarkerThrottled.cancel();\n\n\t\tif ( model.markers.has( 'drop-target' ) ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\twriter.removeMarker( 'drop-target' );\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Deletes the dragged content from its original range and clears the dragging state.\n\t *\n\t * @param moved Whether the move succeeded.\n\t */\n\tprivate _finalizeDragging( moved: boolean ): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\n\t\tthis._removeDropMarker();\n\t\tthis._clearDraggableAttributes();\n\n\t\tif ( editor.plugins.has( 'WidgetToolbarRepository' ) ) {\n\t\t\tconst widgetToolbarRepository: WidgetToolbarRepository = editor.plugins.get( 'WidgetToolbarRepository' );\n\n\t\t\twidgetToolbarRepository.clearForceDisabled( 'dragDrop' );\n\t\t}\n\n\t\tthis._draggingUid = '';\n\n\t\tif ( !this._draggedRange ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Delete moved content.\n\t\tif ( moved && this.isEnabled ) {\n\t\t\tmodel.deleteContent( model.createSelection( this._draggedRange ), { doNotAutoparagraph: true } );\n\t\t}\n\n\t\tthis._draggedRange.detach();\n\t\tthis._draggedRange = null;\n\t}\n}\n\n/**\n * Returns fixed selection range for given position and target element.\n */\nfunction findDropTargetRange( editor: Editor, targetViewRanges: Array<ViewRange> | null, targetViewElement: ViewElement ): Range | null {\n\tconst model = editor.model;\n\tconst mapper = editor.editing.mapper;\n\n\tlet range: Range | null = null;\n\n\tconst targetViewPosition = targetViewRanges ? targetViewRanges[ 0 ].start : null;\n\n\t// A UIElement is not a valid drop element, use parent (this could be a drop marker or any other UIElement).\n\tif ( targetViewElement.is( 'uiElement' ) ) {\n\t\ttargetViewElement = targetViewElement.parent as ViewElement;\n\t}\n\n\t// Quick win if the target is a widget (but not a nested editable).\n\trange = findDropTargetRangeOnWidget( editor, targetViewElement );\n\n\tif ( range ) {\n\t\treturn range;\n\t}\n\n\t// The easiest part is over, now we need to move to the model space.\n\n\t// Find target model element and position.\n\tconst targetModelElement = getClosestMappedModelElement( editor, targetViewElement );\n\tconst targetModelPosition = targetViewPosition ? mapper.toModelPosition( targetViewPosition ) : null;\n\n\t// There is no target position while hovering over an empty table cell.\n\t// In Safari, target position can be empty while hovering over a widget (e.g., a page-break).\n\t// Find the drop position inside the element.\n\tif ( !targetModelPosition ) {\n\t\treturn findDropTargetRangeInElement( editor, targetModelElement );\n\t}\n\n\t// Check if target position is between blocks and adjust drop position to the next object.\n\t// This is because while hovering over a root element next to a widget the target position can jump in crazy places.\n\trange = findDropTargetRangeBetweenBlocks( editor, targetModelPosition, targetModelElement );\n\n\tif ( range ) {\n\t\treturn range;\n\t}\n\n\t// Try fixing selection position.\n\t// In Firefox, the target position lands before widgets but in other browsers it tends to land after a widget.\n\trange = model.schema.getNearestSelectionRange( targetModelPosition, env.isGecko ? 'forward' : 'backward' );\n\n\tif ( range ) {\n\t\treturn range;\n\t}\n\n\t// There is no valid selection position inside the current limit element so find a closest object ancestor.\n\t// This happens if the model position lands directly in the <table> element itself (view target element was a `<td>`\n\t// so a nested editable, but view target position was directly in the `<figure>` element).\n\treturn findDropTargetRangeOnAncestorObject( editor, targetModelPosition.parent as Element );\n}\n\n/**\n * Returns fixed selection range for a given position and a target element if it is over the widget but not over its nested editable.\n */\nfunction findDropTargetRangeOnWidget( editor: Editor, targetViewElement: ViewElement ): Range | null {\n\tconst model = editor.model;\n\tconst mapper = editor.editing.mapper;\n\n\t// Quick win if the target is a widget.\n\tif ( isWidget( targetViewElement ) ) {\n\t\treturn model.createRangeOn( mapper.toModelElement( targetViewElement )! );\n\t}\n\n\t// Check if we are deeper over a widget (but not over a nested editable).\n\tif ( !targetViewElement.is( 'editableElement' ) ) {\n\t\t// Find a closest ancestor that is either a widget or an editable element...\n\t\tconst ancestor = targetViewElement.findAncestor( node => isWidget( node ) || node.is( 'editableElement' ) )!;\n\n\t\t// ...and if the widget was closer then it is a drop target.\n\t\tif ( isWidget( ancestor ) ) {\n\t\t\treturn model.createRangeOn( mapper.toModelElement( ancestor! )! );\n\t\t}\n\t}\n\n\treturn null;\n}\n\n/**\n * Returns fixed selection range inside a model element.\n */\nfunction findDropTargetRangeInElement( editor: Editor, targetModelElement: Element ): Range | null {\n\tconst model = editor.model;\n\tconst schema = model.schema;\n\n\tconst positionAtElementStart = model.createPositionAt( targetModelElement, 0 );\n\n\treturn schema.getNearestSelectionRange( positionAtElementStart, 'forward' );\n}\n\n/**\n * Returns fixed selection range for a given position and a target element if the drop is between blocks.\n */\nfunction findDropTargetRangeBetweenBlocks( editor: Editor, targetModelPosition: Position, targetModelElement: Element ): Range | null {\n\tconst model = editor.model;\n\n\t// Check if target is between blocks.\n\tif ( !model.schema.checkChild( targetModelElement, '$block' ) ) {\n\t\treturn null;\n\t}\n\n\t// Find position between blocks.\n\tconst positionAtElementStart = model.createPositionAt( targetModelElement, 0 );\n\n\t// Get the common part of the path (inside the target element and the target position).\n\tconst commonPath = targetModelPosition.path.slice( 0, positionAtElementStart.path.length );\n\n\t// Position between the blocks.\n\tconst betweenBlocksPosition = model.createPositionFromPath( targetModelPosition.root, commonPath );\n\tconst nodeAfter = betweenBlocksPosition.nodeAfter;\n\n\t// Adjust drop position to the next object.\n\t// This is because while hovering over a root element next to a widget the target position can jump in crazy places.\n\tif ( nodeAfter && model.schema.isObject( nodeAfter ) ) {\n\t\treturn model.createRangeOn( nodeAfter );\n\t}\n\n\treturn null;\n}\n\n/**\n * Returns a selection range on the ancestor object.\n */\nfunction findDropTargetRangeOnAncestorObject( editor: Editor, element: Element ): Range | null {\n\tconst model = editor.model;\n\tlet currentElement: Element | null = element;\n\n\twhile ( currentElement ) {\n\t\tif ( model.schema.isObject( currentElement ) ) {\n\t\t\treturn model.createRangeOn( currentElement );\n\t\t}\n\n\t\tcurrentElement = currentElement.parent as Element | null;\n\t}\n\n\t/* istanbul ignore next -- @preserve */\n\treturn null;\n}\n\n/**\n * Returns the closest model element for the specified view element.\n */\nfunction getClosestMappedModelElement( editor: Editor, element: ViewElement ): Element {\n\tconst mapper = editor.editing.mapper;\n\tconst view = editor.editing.view;\n\n\tconst targetModelElement = mapper.toModelElement( element );\n\n\tif ( targetModelElement ) {\n\t\treturn targetModelElement;\n\t}\n\n\t// Find mapped ancestor if the target is inside not mapped element (for example inline code element).\n\tconst viewPosition = view.createPositionBefore( element );\n\tconst viewElement = mapper.findMappedViewAncestor( viewPosition );\n\n\treturn mapper.toModelElement( viewElement )!;\n}\n\n/**\n * Returns the drop effect that should be a result of dragging the content.\n * This function is handling a quirk when checking the effect in the 'drop' DOM event.\n */\nfunction getFinalDropEffect( dataTransfer: DataTransfer ): DataTransfer[ 'dropEffect' ] {\n\tif ( env.isGecko ) {\n\t\treturn dataTransfer.dropEffect;\n\t}\n\n\treturn [ 'all', 'copyMove' ].includes( dataTransfer.effectAllowed ) ? 'move' : 'copy';\n}\n\n/**\n * Returns a widget element that should be dragged.\n */\nfunction findDraggableWidget( target: ViewElement ): ViewElement | null {\n\t// This is directly an editable so not a widget for sure.\n\tif ( target.is( 'editableElement' ) ) {\n\t\treturn null;\n\t}\n\n\t// TODO: Let's have a isWidgetSelectionHandleDomElement() helper in ckeditor5-widget utils.\n\tif ( target.hasClass( 'ck-widget__selection-handle' ) ) {\n\t\treturn target.findAncestor( isWidget );\n\t}\n\n\t// Direct hit on a widget.\n\tif ( isWidget( target ) ) {\n\t\treturn target;\n\t}\n\n\t// Find closest ancestor that is either a widget or an editable element...\n\tconst ancestor = target.findAncestor( node => isWidget( node ) || node.is( 'editableElement' ) )!;\n\n\t// ...and if closer was the widget then enable dragging it.\n\tif ( isWidget( ancestor ) ) {\n\t\treturn ancestor;\n\t}\n\n\treturn null;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/pasteplaintext\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\n\nimport type { DocumentFragment, Schema, ViewDocumentKeyDownEvent } from '@ckeditor/ckeditor5-engine';\n\nimport ClipboardObserver from './clipboardobserver';\nimport ClipboardPipeline, { type ClipboardContentInsertionEvent } from './clipboardpipeline';\n\n/**\n * The plugin detects the user's intention to paste plain text.\n *\n * For example, it detects the <kbd>Ctrl/Cmd</kbd> + <kbd>Shift</kbd> + <kbd>V</kbd> keystroke.\n */\nexport default class PastePlainText extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'PastePlainText' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ClipboardPipeline ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\t\tconst selection = model.document.selection;\n\n\t\tlet shiftPressed = false;\n\n\t\tview.addObserver( ClipboardObserver );\n\n\t\tthis.listenTo<ViewDocumentKeyDownEvent>( viewDocument, 'keydown', ( evt, data ) => {\n\t\t\tshiftPressed = data.shiftKey;\n\t\t} );\n\n\t\teditor.plugins.get( ClipboardPipeline ).on<ClipboardContentInsertionEvent>( 'contentInsertion', ( evt, data ) => {\n\t\t\t// Plain text can be determined based on the event flag (#7799) or auto-detection (#1006). If detected,\n\t\t\t// preserve selection attributes on pasted items.\n\t\t\tif ( !shiftPressed && !isPlainTextFragment( data.content, model.schema ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tmodel.change( writer => {\n\t\t\t\t// Formatting attributes should be preserved.\n\t\t\t\tconst textAttributes = Array.from( selection.getAttributes() )\n\t\t\t\t\t.filter( ( [ key ] ) => model.schema.getAttributeProperties( key ).isFormatting );\n\n\t\t\t\tif ( !selection.isCollapsed ) {\n\t\t\t\t\tmodel.deleteContent( selection, { doNotAutoparagraph: true } );\n\t\t\t\t}\n\n\t\t\t\t// Also preserve other attributes if they survived the content deletion (because they were not fully selected).\n\t\t\t\t// For example linkHref is not a formatting attribute but it should be preserved if pasted text was in the middle\n\t\t\t\t// of a link.\n\t\t\t\ttextAttributes.push( ...selection.getAttributes() );\n\n\t\t\t\tconst range = writer.createRangeIn( data.content );\n\n\t\t\t\tfor ( const item of range.getItems() ) {\n\t\t\t\t\tif ( item.is( '$textProxy' ) ) {\n\t\t\t\t\t\twriter.setAttributes( textAttributes, item );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\t\t} );\n\t}\n}\n\n/**\n * Returns true if specified `documentFragment` represents a plain text.\n */\nfunction isPlainTextFragment( documentFragment: DocumentFragment, schema: Schema ): boolean {\n\tif ( documentFragment.childCount > 1 ) {\n\t\treturn false;\n\t}\n\n\tconst child = documentFragment.getChild( 0 )!;\n\n\tif ( schema.isObject( child ) ) {\n\t\treturn false;\n\t}\n\n\treturn Array.from( child.getAttributeKeys() ).length == 0;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/clipboard\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\n\nimport ClipboardPipeline from './clipboardpipeline';\nimport DragDrop from './dragdrop';\nimport PastePlainText from './pasteplaintext';\n\n/**\n * The clipboard feature.\n *\n * Read more about the clipboard integration in the {@glink framework/deep-dive/clipboard clipboard deep-dive} guide.\n *\n * This is a \"glue\" plugin which loads the following plugins:\n * * {@link module:clipboard/clipboardpipeline~ClipboardPipeline}\n * * {@link module:clipboard/dragdrop~DragDrop}\n * * {@link module:clipboard/pasteplaintext~PastePlainText}\n */\nexport default class Clipboard extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Clipboard' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ClipboardPipeline, DragDrop, PastePlainText ] as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module clipboard/lineview\n */\n\n/* istanbul ignore file -- @preserve */\n\nimport { View } from '@ckeditor/ckeditor5-ui';\nimport { toUnit } from '@ckeditor/ckeditor5-utils';\n\nconst toPx = toUnit( 'px' );\n\n/**\n * The horizontal drop target line view.\n */\nexport default class LineView extends View {\n\t/**\n\t * Controls whether the line is visible.\n\t *\n\t * @observable\n\t * @default false\n\t */\n\tdeclare public isVisible: boolean;\n\n\t/**\n\t * Controls the line position x coordinate.\n\t *\n\t * @observable\n\t * @default null\n\t */\n\tdeclare public left: number | null;\n\n\t/**\n\t * Controls the line width.\n\t *\n\t * @observable\n\t * @default null\n\t */\n\tdeclare public width: number | null;\n\n\t/**\n\t * Controls the line position y coordinate.\n\t *\n\t * @observable\n\t * @default null\n\t */\n\tdeclare public top: number | null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor() {\n\t\tsuper();\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.set( {\n\t\t\tisVisible: false,\n\t\t\tleft: null,\n\t\t\ttop: null,\n\t\t\twidth: null\n\t\t} );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-clipboard-drop-target-line',\n\t\t\t\t\tbind.if( 'isVisible', 'ck-hidden', value => !value )\n\t\t\t\t],\n\t\t\t\tstyle: {\n\t\t\t\t\tleft: bind.to( 'left', left => toPx( left! ) ),\n\t\t\t\t\ttop: bind.to( 'top', top => toPx( top! ) ),\n\t\t\t\t\twidth: bind.to( 'width', width => toPx( width! ) )\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module select-all/selectallcommand\n */\n\nimport { Command, type Editor } from '@ckeditor/ckeditor5-core';\nimport type { Element, Schema } from '@ckeditor/ckeditor5-engine';\n\n/**\n * The select all command.\n *\n * It is used by the {@link module:select-all/selectallediting~SelectAllEditing select all editing feature} to handle\n * the <kbd>Ctrl/⌘</kbd>+<kbd>A</kbd> keystroke.\n *\n * Executing this command changes the {@glink framework/architecture/editing-engine#model model}\n * selection so it contains the entire content of the editable root of the editor the selection is\n * {@link module:engine/model/selection~Selection#anchor anchored} in.\n *\n * If the selection was anchored in a {@glink framework/tutorials/implementing-a-block-widget nested editable}\n * (e.g. a caption of an image), the new selection will contain its entire content. Successive executions of this command\n * will expand the selection to encompass more and more content up to the entire editable root of the editor.\n */\nexport default class SelectAllCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\t// It does not affect data so should be enabled in read-only mode.\n\t\tthis.affectsData = false;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tlet scopeElement: Element | null = model.schema.getLimitElement( selection );\n\n\t\t// If an entire scope is selected, or the selection's ancestor is not a scope yet,\n\t\t// browse through ancestors to find the enclosing parent scope.\n\t\tif ( selection.containsEntireContent( scopeElement ) || !isSelectAllScope( model.schema, scopeElement ) ) {\n\t\t\tdo {\n\t\t\t\tscopeElement = scopeElement.parent as Element | null;\n\n\t\t\t\t// Do nothing, if the entire `root` is already selected.\n\t\t\t\tif ( !scopeElement ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t} while ( !isSelectAllScope( model.schema, scopeElement ) );\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\twriter.setSelection( scopeElement!, 'in' );\n\t\t} );\n\t}\n}\n\n/**\n * Checks whether the element is a valid select-all scope. Returns true, if the element is a\n * {@link module:engine/model/schema~Schema#isLimit limit}, and can contain any text or paragraph.\n *\n * @param schema Schema to check against.\n * @param element Model element.\n */\nfunction isSelectAllScope( schema: Schema, element: Element ): boolean {\n\treturn schema.isLimit( element ) && ( schema.checkChild( element, '$text' ) || schema.checkChild( element, 'paragraph' ) );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module select-all/selectallediting\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport { getCode, parseKeystroke } from '@ckeditor/ckeditor5-utils';\nimport SelectAllCommand from './selectallcommand';\nimport type { ViewDocumentKeyDownEvent } from '@ckeditor/ckeditor5-engine';\n\nconst SELECT_ALL_KEYSTROKE = parseKeystroke( 'Ctrl+A' );\n\n/**\n * The select all editing feature.\n *\n * It registers the `'selectAll'` {@link module:select-all/selectallcommand~SelectAllCommand command}\n * and the <kbd>Ctrl/⌘</kbd>+<kbd>A</kbd> keystroke listener which executes it.\n */\nexport default class SelectAllEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'SelectAllEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\teditor.commands.add( 'selectAll', new SelectAllCommand( editor ) );\n\n\t\tthis.listenTo<ViewDocumentKeyDownEvent>( viewDocument, 'keydown', ( eventInfo, domEventData ) => {\n\t\t\tif ( getCode( domEventData ) === SELECT_ALL_KEYSTROKE ) {\n\t\t\t\teditor.execute( 'selectAll' );\n\t\t\t\tdomEventData.preventDefault();\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module select-all/selectallui\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport { ButtonView } from '@ckeditor/ckeditor5-ui';\n\nimport selectAllIcon from '../theme/icons/select-all.svg';\n\n/**\n * The select all UI feature.\n *\n * It registers the `'selectAll'` UI button in the editor's\n * {@link module:ui/componentfactory~ComponentFactory component factory}. When clicked, the button\n * executes the {@link module:select-all/selectallcommand~SelectAllCommand select all command}.\n */\nexport default class SelectAllUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'SelectAllUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\teditor.ui.componentFactory.add( 'selectAll', locale => {\n\t\t\tconst command = editor.commands.get( 'selectAll' )!;\n\t\t\tconst view = new ButtonView( locale );\n\t\t\tconst t = locale.t;\n\n\t\t\tview.set( {\n\t\t\t\tlabel: t( 'Select all' ),\n\t\t\t\ticon: selectAllIcon,\n\t\t\t\tkeystroke: 'Ctrl+A',\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tview.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\n\t\t\t// Execute the command.\n\t\t\tthis.listenTo( view, 'execute', () => {\n\t\t\t\teditor.execute( 'selectAll' );\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn view;\n\t\t} );\n\t}\n}\n","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" viewBox=\\\"0 0 20 20\\\"><path d=\\\"M.75 15.5a.75.75 0 0 1 .75.75V18l.008.09A.5.5 0 0 0 2 18.5h1.75a.75.75 0 1 1 0 1.5H1.5l-.144-.007a1.5 1.5 0 0 1-1.35-1.349L0 18.5v-2.25a.75.75 0 0 1 .75-.75zm18.5 0a.75.75 0 0 1 .75.75v2.25l-.007.144a1.5 1.5 0 0 1-1.349 1.35L18.5 20h-2.25a.75.75 0 1 1 0-1.5H18a.5.5 0 0 0 .492-.41L18.5 18v-1.75a.75.75 0 0 1 .75-.75zm-10.45 3c.11 0 .2.09.2.2v1.1a.2.2 0 0 1-.2.2H7.2a.2.2 0 0 1-.2-.2v-1.1c0-.11.09-.2.2-.2h1.6zm4 0c.11 0 .2.09.2.2v1.1a.2.2 0 0 1-.2.2h-1.6a.2.2 0 0 1-.2-.2v-1.1c0-.11.09-.2.2-.2h1.6zm.45-5.5a.75.75 0 1 1 0 1.5h-8.5a.75.75 0 1 1 0-1.5h8.5zM1.3 11c.11 0 .2.09.2.2v1.6a.2.2 0 0 1-.2.2H.2a.2.2 0 0 1-.2-.2v-1.6c0-.11.09-.2.2-.2h1.1zm18.5 0c.11 0 .2.09.2.2v1.6a.2.2 0 0 1-.2.2h-1.1a.2.2 0 0 1-.2-.2v-1.6c0-.11.09-.2.2-.2h1.1zm-4.55-2a.75.75 0 1 1 0 1.5H4.75a.75.75 0 1 1 0-1.5h10.5zM1.3 7c.11 0 .2.09.2.2v1.6a.2.2 0 0 1-.2.2H.2a.2.2 0 0 1-.2-.2V7.2c0-.11.09-.2.2-.2h1.1zm18.5 0c.11 0 .2.09.2.2v1.6a.2.2 0 0 1-.2.2h-1.1a.2.2 0 0 1-.2-.2V7.2c0-.11.09-.2.2-.2h1.1zm-4.55-2a.75.75 0 1 1 0 1.5h-2.5a.75.75 0 1 1 0-1.5h2.5zm-5 0a.75.75 0 1 1 0 1.5h-5.5a.75.75 0 0 1 0-1.5h5.5zm-6.5-5a.75.75 0 0 1 0 1.5H2a.5.5 0 0 0-.492.41L1.5 2v1.75a.75.75 0 0 1-1.5 0V1.5l.007-.144A1.5 1.5 0 0 1 1.356.006L1.5 0h2.25zM18.5 0l.144.007a1.5 1.5 0 0 1 1.35 1.349L20 1.5v2.25a.75.75 0 1 1-1.5 0V2l-.008-.09A.5.5 0 0 0 18 1.5h-1.75a.75.75 0 1 1 0-1.5h2.25zM8.8 0c.11 0 .2.09.2.2v1.1a.2.2 0 0 1-.2.2H7.2a.2.2 0 0 1-.2-.2V.2c0-.11.09-.2.2-.2h1.6zm4 0c.11 0 .2.09.2.2v1.1a.2.2 0 0 1-.2.2h-1.6a.2.2 0 0 1-.2-.2V.2c0-.11.09-.2.2-.2h1.6z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module select-all/selectall\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport SelectAllEditing from './selectallediting';\nimport SelectAllUI from './selectallui';\n\n/**\n * The select all feature.\n *\n * This is a \"glue\" plugin which loads the {@link module:select-all/selectallediting~SelectAllEditing select all editing feature}\n * and the {@link module:select-all/selectallui~SelectAllUI select all UI feature}.\n *\n * Please refer to the documentation of individual features to learn more.\n */\nexport default class SelectAll extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ SelectAllEditing, SelectAllUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'SelectAll' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module undo/basecommand\n */\n\nimport { Command, type Editor } from '@ckeditor/ckeditor5-core';\n\nimport {\n\ttransformSets,\n\ttype Batch,\n\ttype Operation,\n\ttype DataControllerSetEvent,\n\ttype Range,\n\tNoOperation\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * Base class for the undo feature commands: {@link module:undo/undocommand~UndoCommand} and {@link module:undo/redocommand~RedoCommand}.\n */\nexport default abstract class BaseCommand extends Command {\n\t/**\n\t * Stack of items stored by the command. These are pairs of:\n\t *\n\t * * {@link module:engine/model/batch~Batch batch} saved by the command,\n\t * * {@link module:engine/model/selection~Selection selection} state at the moment of saving the batch.\n\t */\n\tprotected _stack: Array<{ batch: Batch; selection: { ranges: Array<Range>; isBackward: boolean } }> = [];\n\n\t/**\n\t * Stores all batches that were created by this command.\n\t *\n\t * @internal\n\t */\n\tpublic _createdBatches = new WeakSet<Batch>();\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\t// Refresh state, so the command is inactive right after initialization.\n\t\tthis.refresh();\n\n\t\t// This command should not depend on selection change.\n\t\tthis._isEnabledBasedOnSelection = false;\n\n\t\t// Set the transparent batch for the `editor.data.set()` call if the\n\t\t// batch type is not set already.\n\t\tthis.listenTo<DataControllerSetEvent>( editor.data, 'set', ( evt, data ) => {\n\t\t\t// Create a shallow copy of the options to not change the original args.\n\t\t\t// And make sure that an object is assigned to data[ 1 ].\n\t\t\tdata[ 1 ] = { ...data[ 1 ] };\n\n\t\t\tconst options = data[ 1 ];\n\n\t\t\t// If batch type is not set, default to non-undoable batch.\n\t\t\tif ( !options.batchType ) {\n\t\t\t\toptions.batchType = { isUndoable: false };\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\n\t\t// Clear the stack for the `transparent` batches.\n\t\tthis.listenTo<DataControllerSetEvent>( editor.data, 'set', ( evt, data ) => {\n\t\t\t// We can assume that the object exists and it has a `batchType` property.\n\t\t\t// It was ensured with a higher priority listener before.\n\t\t\tconst options = data[ 1 ]!;\n\n\t\t\tif ( !options.batchType!.isUndoable ) {\n\t\t\t\tthis.clearStack();\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tthis.isEnabled = this._stack.length > 0;\n\t}\n\n\t/**\n\t * Returns all batches created by this command.\n\t */\n\tpublic get createdBatches(): WeakSet<Batch> {\n\t\treturn this._createdBatches;\n\t}\n\n\t/**\n\t * Stores a batch in the command, together with the selection state of the {@link module:engine/model/document~Document document}\n\t * created by the editor which this command is registered to.\n\t *\n\t * @param batch The batch to add.\n\t */\n\tpublic addBatch( batch: Batch ): void {\n\t\tconst docSelection = this.editor.model.document.selection;\n\n\t\tconst selection = {\n\t\t\tranges: docSelection.hasOwnRange ? Array.from( docSelection.getRanges() ) : [],\n\t\t\tisBackward: docSelection.isBackward\n\t\t};\n\n\t\tthis._stack.push( { batch, selection } );\n\t\tthis.refresh();\n\t}\n\n\t/**\n\t * Removes all items from the stack.\n\t */\n\tpublic clearStack(): void {\n\t\tthis._stack = [];\n\t\tthis.refresh();\n\t}\n\n\t/**\n\t * Restores the {@link module:engine/model/document~Document#selection document selection} state after a batch was undone.\n\t *\n\t * @param ranges Ranges to be restored.\n\t * @param isBackward A flag describing whether the restored range was selected forward or backward.\n\t * @param operations Operations which has been applied since selection has been stored.\n\t */\n\tprotected _restoreSelection(\n\t\tranges: Array<Range>,\n\t\tisBackward: boolean,\n\t\toperations: Array<Operation>\n\t): void {\n\t\tconst model = this.editor.model;\n\t\tconst document = model.document;\n\n\t\t// This will keep the transformed selection ranges.\n\t\tconst selectionRanges: Array<Range> = [];\n\n\t\t// Transform all ranges from the restored selection.\n\t\tconst transformedRangeGroups = ranges.map( range => range.getTransformedByOperations( operations ) );\n\t\tconst allRanges = transformedRangeGroups.flat();\n\n\t\tfor ( const rangeGroup of transformedRangeGroups ) {\n\t\t\t// While transforming there could appear ranges that are contained by other ranges, we shall ignore them.\n\t\t\tconst transformed = rangeGroup\n\t\t\t\t.filter( range => range.root != document.graveyard )\n\t\t\t\t.filter( range => !isRangeContainedByAnyOtherRange( range, allRanges ) );\n\n\t\t\t// All the transformed ranges ended up in graveyard.\n\t\t\tif ( !transformed.length ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// After the range got transformed, we have an array of ranges. Some of those\n\t\t\t// ranges may be \"touching\" -- they can be next to each other and could be merged.\n\t\t\tnormalizeRanges( transformed );\n\n\t\t\t// For each `range` from `ranges`, we take only one transformed range.\n\t\t\t// This is because we want to prevent situation where single-range selection\n\t\t\t// got transformed to multi-range selection.\n\t\t\tselectionRanges.push( transformed[ 0 ] );\n\t\t}\n\n\t\t// @if CK_DEBUG_ENGINE // console.log( `Restored selection by undo: ${ selectionRanges.join( ', ' ) }` );\n\n\t\t// `selectionRanges` may be empty if all ranges ended up in graveyard. If that is the case, do not restore selection.\n\t\tif ( selectionRanges.length ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\twriter.setSelection( selectionRanges, { backward: isBackward } );\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Undoes a batch by reversing that batch, transforming reversed batch and finally applying it.\n\t * This is a helper method for {@link #execute}.\n\t *\n\t * @param batchToUndo The batch to be undone.\n\t * @param undoingBatch The batch that will contain undoing changes.\n\t */\n\tprotected _undo( batchToUndo: Batch, undoingBatch: Batch ): void {\n\t\tconst model = this.editor.model;\n\t\tconst document = model.document;\n\n\t\t// All changes done by the command execution will be saved as one batch.\n\t\tthis._createdBatches.add( undoingBatch );\n\n\t\tconst operationsToUndo = batchToUndo.operations.slice().filter( operation => operation.isDocumentOperation );\n\t\toperationsToUndo.reverse();\n\n\t\t// We will process each operation from `batchToUndo`, in reverse order. If there were operations A, B and C in undone batch,\n\t\t// we need to revert them in reverse order, so first C' (reversed C), then B', then A'.\n\t\tfor ( const operationToUndo of operationsToUndo ) {\n\t\t\tconst nextBaseVersion = operationToUndo.baseVersion! + 1;\n\t\t\tconst historyOperations = Array.from( document.history.getOperations( nextBaseVersion ) );\n\n\t\t\tconst transformedSets = transformSets(\n\t\t\t\t[ operationToUndo.getReversed() ],\n\t\t\t\thistoryOperations,\n\t\t\t\t{\n\t\t\t\t\tuseRelations: true,\n\t\t\t\t\tdocument: this.editor.model.document,\n\t\t\t\t\tpadWithNoOps: false,\n\t\t\t\t\tforceWeakRemove: true\n\t\t\t\t}\n\t\t\t);\n\n\t\t\tconst reversedOperations = transformedSets.operationsA;\n\n\t\t\t// After reversed operation has been transformed by all history operations, apply it.\n\t\t\tfor ( let operation of reversedOperations ) {\n\t\t\t\t// Do not apply any operation on non-editable space.\n\t\t\t\tconst affectedSelectable = operation.affectedSelectable;\n\n\t\t\t\tif ( affectedSelectable && !model.canEditAt( affectedSelectable ) ) {\n\t\t\t\t\toperation = new NoOperation( operation.baseVersion );\n\t\t\t\t}\n\n\t\t\t\t// Before applying, add the operation to the `undoingBatch`.\n\t\t\t\tundoingBatch.addOperation( operation );\n\t\t\t\tmodel.applyOperation( operation );\n\n\t\t\t\tdocument.history.setOperationAsUndone( operationToUndo, operation );\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Normalizes list of ranges by joining intersecting or \"touching\" ranges.\n *\n * @param ranges Ranges to be normalized.\n */\nfunction normalizeRanges( ranges: Array<Range> ): void {\n\tranges.sort( ( a, b ) => a.start.isBefore( b.start ) ? -1 : 1 );\n\n\tfor ( let i = 1; i < ranges.length; i++ ) {\n\t\tconst previousRange = ranges[ i - 1 ];\n\t\tconst joinedRange = previousRange.getJoined( ranges[ i ], true );\n\n\t\tif ( joinedRange ) {\n\t\t\t// Replace the ranges on the list with the new joined range.\n\t\t\ti--;\n\t\t\tranges.splice( i, 2, joinedRange );\n\t\t}\n\t}\n}\n\nfunction isRangeContainedByAnyOtherRange( range: Range, ranges: Array<Range> ): boolean {\n\treturn ranges.some( otherRange => otherRange !== range && otherRange.containsRange( range, true ) );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module undo/undocommand\n */\n\nimport BaseCommand from './basecommand';\nimport type { Batch } from '@ckeditor/ckeditor5-engine';\n\n/**\n * The undo command stores {@link module:engine/model/batch~Batch batches} applied to the\n * {@link module:engine/model/document~Document document} and is able to undo a batch by reversing it and transforming by\n * batches from {@link module:engine/model/document~Document#history history} that happened after the reversed batch.\n *\n * The undo command also takes care of restoring the {@link module:engine/model/document~Document#selection document selection}.\n */\nexport default class UndoCommand extends BaseCommand {\n\t/**\n\t * Executes the command. This method reverts a {@link module:engine/model/batch~Batch batch} added to the command's stack, transforms\n\t * and applies the reverted version on the {@link module:engine/model/document~Document document} and removes the batch from the stack.\n\t * Then, it restores the {@link module:engine/model/document~Document#selection document selection}.\n\t *\n\t * @fires execute\n\t * @fires revert\n\t * @param batch A batch that should be undone. If not set, the last added batch will be undone.\n\t */\n\tpublic override execute( batch: Batch | null = null ): void {\n\t\t// If batch is not given, set `batchIndex` to the last index in command stack.\n\t\tconst batchIndex = batch ? this._stack.findIndex( a => a.batch == batch ) : this._stack.length - 1;\n\n\t\tconst item = this._stack.splice( batchIndex, 1 )[ 0 ];\n\t\tconst undoingBatch = this.editor.model.createBatch( { isUndo: true } );\n\n\t\t// All changes have to be done in one `enqueueChange` callback so other listeners will not\n\t\t// step between consecutive operations, or won't do changes to the document before selection is properly restored.\n\t\tthis.editor.model.enqueueChange( undoingBatch, () => {\n\t\t\tthis._undo( item.batch, undoingBatch );\n\n\t\t\tconst operations = this.editor.model.document.history.getOperations( item.batch.baseVersion! );\n\t\t\tthis._restoreSelection( item.selection.ranges, item.selection.isBackward, operations );\n\t\t} );\n\n\t\t// Firing `revert` event after the change block to make sure that it includes all changes from post-fixers\n\t\t// and make sure that the selection is \"stabilized\" (the selection range is saved after undo is executed and then\n\t\t// restored on redo, so it is important that the selection range is saved after post-fixers are done).\n\t\tthis.fire<UndoCommandRevertEvent>( 'revert', item.batch, undoingBatch );\n\n\t\tthis.refresh();\n\t}\n}\n\n/**\n * Fired when execution of the command reverts some batch.\n *\n * @eventName ~UndoCommand#revert\n */\nexport type UndoCommandRevertEvent = {\n\tname: 'revert';\n\targs: [ batch: Batch, undoingBatch: Batch ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module undo/redocommand\n */\n\nimport BaseCommand from './basecommand';\n\n/**\n * The redo command stores {@link module:engine/model/batch~Batch batches} that were used to undo a batch by\n * {@link module:undo/undocommand~UndoCommand}. It is able to redo a previously undone batch by reversing the undoing\n * batches created by `UndoCommand`. The reversed batch is transformed by all the batches from\n * {@link module:engine/model/document~Document#history history} that happened after the reversed undo batch.\n *\n * The redo command also takes care of restoring the {@link module:engine/model/document~Document#selection document selection}.\n */\nexport default class RedoCommand extends BaseCommand {\n\t/**\n\t * Executes the command. This method reverts the last {@link module:engine/model/batch~Batch batch} added to\n\t * the command's stack, applies the reverted and transformed version on the\n\t * {@link module:engine/model/document~Document document} and removes the batch from the stack.\n\t * Then, it restores the {@link module:engine/model/document~Document#selection document selection}.\n\t *\n\t * @fires execute\n\t */\n\tpublic override execute(): void {\n\t\tconst item = this._stack.pop()!;\n\t\tconst redoingBatch = this.editor.model.createBatch( { isUndo: true } );\n\n\t\t// All changes have to be done in one `enqueueChange` callback so other listeners will not step between consecutive\n\t\t// operations, or won't do changes to the document before selection is properly restored.\n\t\tthis.editor.model.enqueueChange( redoingBatch, () => {\n\t\t\tconst lastOperation = item.batch.operations[ item.batch.operations.length - 1 ];\n\t\t\tconst nextBaseVersion = lastOperation.baseVersion! + 1;\n\t\t\tconst operations = this.editor.model.document.history.getOperations( nextBaseVersion );\n\n\t\t\tthis._restoreSelection( item.selection.ranges, item.selection.isBackward, operations );\n\t\t\tthis._undo( item.batch, redoingBatch );\n\t\t} );\n\n\t\tthis.refresh();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module undo/undoediting\n */\n\nimport { Plugin, type Editor } from '@ckeditor/ckeditor5-core';\n\nimport UndoCommand, { type UndoCommandRevertEvent } from './undocommand';\nimport RedoCommand from './redocommand';\n\nimport type {\n\tBatch,\n\tModelApplyOperationEvent\n} from '@ckeditor/ckeditor5-engine';\n\n/**\n * The undo engine feature.\n *\n * It introduces the `'undo'` and `'redo'` commands to the editor.\n */\nexport default class UndoEditing extends Plugin {\n\t/**\n\t * The command that manages the undo {@link module:engine/model/batch~Batch batches} stack (history).\n\t * Created and registered during the {@link #init feature initialization}.\n\t */\n\tprivate _undoCommand!: UndoCommand;\n\n\t/**\n\t * The command that manages the redo {@link module:engine/model/batch~Batch batches} stack (history).\n\t * Created and registered during the {@link #init feature initialization}.\n\t */\n\tprivate _redoCommand!: RedoCommand;\n\n\t/**\n\t * Keeps track of which batches were registered in undo.\n\t */\n\tprivate _batchRegistry = new WeakSet<Batch>();\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'UndoEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Create commands.\n\t\tthis._undoCommand = new UndoCommand( editor );\n\t\tthis._redoCommand = new RedoCommand( editor );\n\n\t\t// Register command to the editor.\n\t\teditor.commands.add( 'undo', this._undoCommand );\n\t\teditor.commands.add( 'redo', this._redoCommand );\n\n\t\tthis.listenTo<ModelApplyOperationEvent>( editor.model, 'applyOperation', ( evt, args ) => {\n\t\t\tconst operation = args[ 0 ];\n\n\t\t\t// Do not register batch if the operation is not a document operation.\n\t\t\t// This prevents from creating empty undo steps, where all operations where non-document operations.\n\t\t\t// Non-document operations creates and alters content in detached tree fragments (for example, document fragments).\n\t\t\t// Most of time this is preparing data before it is inserted into actual tree (for example during copy & paste).\n\t\t\t// Such operations should not be reversed.\n\t\t\tif ( !operation.isDocumentOperation ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst batch = operation.batch!;\n\n\t\t\tconst isRedoBatch = this._redoCommand.createdBatches.has( batch );\n\t\t\tconst isUndoBatch = this._undoCommand.createdBatches.has( batch );\n\t\t\tconst wasProcessed = this._batchRegistry.has( batch );\n\n\t\t\t// Skip the batch if it was already processed.\n\t\t\tif ( wasProcessed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Add the batch to the registry so it will not be processed again.\n\t\t\tthis._batchRegistry.add( batch );\n\n\t\t\tif ( !batch.isUndoable ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( isRedoBatch ) {\n\t\t\t\t// If this batch comes from `redoCommand`, add it to the `undoCommand` stack.\n\t\t\t\tthis._undoCommand.addBatch( batch );\n\t\t\t} else if ( !isUndoBatch ) {\n\t\t\t\t// If the batch comes neither from `redoCommand` nor from `undoCommand` then it is a new, regular batch.\n\t\t\t\t// Add the batch to the `undoCommand` stack and clear the `redoCommand` stack.\n\t\t\t\tthis._undoCommand.addBatch( batch );\n\t\t\t\tthis._redoCommand.clearStack();\n\t\t\t}\n\t\t}, { priority: 'highest' } );\n\n\t\tthis.listenTo<UndoCommandRevertEvent>( this._undoCommand, 'revert', ( evt, undoneBatch, undoingBatch ) => {\n\t\t\tthis._redoCommand.addBatch( undoingBatch );\n\t\t} );\n\n\t\teditor.keystrokes.set( 'CTRL+Z', 'undo' );\n\t\teditor.keystrokes.set( 'CTRL+Y', 'redo' );\n\t\teditor.keystrokes.set( 'CTRL+SHIFT+Z', 'redo' );\n\t}\n}\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m5.042 9.367 2.189 1.837a.75.75 0 0 1-.965 1.149l-3.788-3.18a.747.747 0 0 1-.21-.284.75.75 0 0 1 .17-.945L6.23 4.762a.75.75 0 1 1 .964 1.15L4.863 7.866h8.917A.75.75 0 0 1 14 7.9a4 4 0 1 1-1.477 7.718l.344-1.489a2.5 2.5 0 1 0 1.094-4.73l.008-.032H5.042z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m14.958 9.367-2.189 1.837a.75.75 0 0 0 .965 1.149l3.788-3.18a.747.747 0 0 0 .21-.284.75.75 0 0 0-.17-.945L13.77 4.762a.75.75 0 1 0-.964 1.15l2.331 1.955H6.22A.75.75 0 0 0 6 7.9a4 4 0 1 0 1.477 7.718l-.344-1.489A2.5 2.5 0 1 1 6.039 9.4l-.008-.032h8.927z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module undo/undoui\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport { ButtonView } from '@ckeditor/ckeditor5-ui';\n\nimport undoIcon from '../theme/icons/undo.svg';\nimport redoIcon from '../theme/icons/redo.svg';\n\n/**\n * The undo UI feature. It introduces the `'undo'` and `'redo'` buttons to the editor.\n */\nexport default class UndoUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'UndoUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst locale = editor.locale;\n\t\tconst t = editor.t;\n\n\t\tconst localizedUndoIcon = locale.uiLanguageDirection == 'ltr' ? undoIcon : redoIcon;\n\t\tconst localizedRedoIcon = locale.uiLanguageDirection == 'ltr' ? redoIcon : undoIcon;\n\n\t\tthis._addButton( 'undo', t( 'Undo' ), 'CTRL+Z', localizedUndoIcon );\n\t\tthis._addButton( 'redo', t( 'Redo' ), 'CTRL+Y', localizedRedoIcon );\n\t}\n\n\t/**\n\t * Creates a button for the specified command.\n\t *\n\t * @param name Command name.\n\t * @param label Button label.\n\t * @param keystroke Command keystroke.\n\t * @param Icon Source of the icon.\n\t */\n\tprivate _addButton( name: 'undo' | 'redo', label: string, keystroke: string, Icon: string ) {\n\t\tconst editor = this.editor;\n\n\t\teditor.ui.componentFactory.add( name, locale => {\n\t\t\tconst command = editor.commands.get( name )!;\n\t\t\tconst view = new ButtonView( locale );\n\n\t\t\tview.set( {\n\t\t\t\tlabel,\n\t\t\t\ticon: Icon,\n\t\t\t\tkeystroke,\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tview.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\n\t\t\tthis.listenTo( view, 'execute', () => {\n\t\t\t\teditor.execute( name );\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn view;\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module undo/undo\n */\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\nimport UndoEditing from './undoediting';\nimport UndoUI from './undoui';\n\n/**\n * The undo feature.\n *\n * This is a \"glue\" plugin which loads the {@link module:undo/undoediting~UndoEditing undo editing feature}\n * and the {@link module:undo/undoui~UndoUI undo UI feature}.\n *\n * Below is an explanation of the undo mechanism working together with {@link module:engine/model/history~History History}:\n *\n * Whenever an {@link module:engine/model/operation/operation~Operation operation} is applied to the\n * {@link module:engine/model/document~Document document}, it is saved to `History` as is.\n * The {@link module:engine/model/batch~Batch batch} that owns that operation is also saved, in\n * {@link module:undo/undocommand~UndoCommand}, together with the selection that was present in the document before the\n * operation was applied. A batch is saved instead of the operation because changes are undone batch-by-batch, not operation-by-operation\n * and a batch is seen as one undo step.\n *\n * After changes happen to the document, the `History` and `UndoCommand` stack can be represented as follows:\n *\n * ```\n * History Undo stack\n * ============== ==================================\n * [operation A1] [ batch A ]\n * [operation B1] [ batch B ]\n * [operation B2] [ batch C ]\n * [operation C1]\n * [operation C2]\n * [operation B3]\n * [operation C3]\n * ```\n *\n * Where operations starting with the same letter are from same batch.\n *\n * Undoing a batch means that a set of operations which will reverse the effects of that batch needs to be generated.\n * For example, if a batch added several letters, undoing the batch should remove them. It is important to apply undoing\n * operations in the reversed order, so if a batch has operation `X`, `Y`, `Z`, reversed operations `Zr`, `Yr` and `Xr`\n * need to be applied. Otherwise reversed operation `Xr` would operate on a wrong document state, because operation `X`\n * does not know that operations `Y` and `Z` happened.\n *\n * After operations from an undone batch got {@link module:engine/model/operation/operation~Operation#getReversed reversed},\n * one needs to make sure if they are ready to be applied. In the scenario above, operation `C3` is the last operation and `C3r`\n * bases on up-to-date document state, so it can be applied to the document.\n *\n * ```\n * History Undo stack\n * ================= ==================================\n * [ operation A1 ] [ batch A ]\n * [ operation B1 ] [ batch B ]\n * [ operation B2 ] [ processing undoing batch C ]\n * [ operation C1 ]\n * [ operation C2 ]\n * [ operation B3 ]\n * [ operation C3 ]\n * [ operation C3r ]\n * ```\n *\n * Next is operation `C2`, reversed to `C2r`. `C2r` bases on `C2`, so it bases on the wrong document state. It needs to be\n * transformed by operations from history that happened after it, so it \"knows\" about them. Let us assume that `C2' = C2r * B3 * C3 * C3r`,\n * where `*` means \"transformed by\". Rest of operations from that batch are processed in the same fashion.\n *\n * ```\n * History Undo stack Redo stack\n * ================= ================================== ==================================\n * [ operation A1 ] [ batch A ] [ batch Cr ]\n * [ operation B1 ] [ batch B ]\n * [ operation B2 ]\n * [ operation C1 ]\n * [ operation C2 ]\n * [ operation B3 ]\n * [ operation C3 ]\n * [ operation C3r ]\n * [ operation C2' ]\n * [ operation C1' ]\n * ```\n *\n * Selective undo works on the same basis, however, instead of undoing the last batch in the undo stack, any batch can be undone.\n * The same algorithm applies: operations from a batch (i.e. `A1`) are reversed and then transformed by operations stored in history.\n *\n * Redo also is very similar to undo. It has its own stack that is filled with undoing (reversed batches). Operations from\n * the batch that is re-done are reversed-back, transformed in proper order and applied to the document.\n *\n * ```\n * History Undo stack Redo stack\n * ================= ================================== ==================================\n * [ operation A1 ] [ batch A ]\n * [ operation B1 ] [ batch B ]\n * [ operation B2 ] [ batch Crr ]\n * [ operation C1 ]\n * [ operation C2 ]\n * [ operation B3 ]\n * [ operation C3 ]\n * [ operation C3r ]\n * [ operation C2' ]\n * [ operation C1' ]\n * [ operation C1'r]\n * [ operation C2'r]\n * [ operation C3rr]\n * ```\n */\nexport default class Undo extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ UndoEditing, UndoUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Undo' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module upload/filereader\n */\n\n/* globals window */\n\nimport { ObservableMixin } from '@ckeditor/ckeditor5-utils';\n\ntype DomFileReader = globalThis.FileReader;\n\n/**\n * Wrapper over the native `FileReader`.\n */\nexport default class FileReader extends ObservableMixin() {\n\tpublic total!: number;\n\n\t/**\n\t * Instance of native FileReader.\n\t */\n\tprivate readonly _reader: DomFileReader;\n\n\t/**\n\t * Holds the data of an already loaded file. The file must be first loaded\n\t * by using {@link module:upload/filereader~FileReader#read `read()`}.\n\t */\n\tprivate _data?: string;\n\n\t/**\n\t * Number of bytes loaded.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public loaded: number;\n\n\t/**\n\t * Creates an instance of the FileReader.\n\t */\n\tconstructor() {\n\t\tsuper();\n\n\t\tconst reader = new window.FileReader();\n\n\t\tthis._reader = reader;\n\t\tthis._data = undefined;\n\n\t\tthis.set( 'loaded', 0 );\n\n\t\treader.onprogress = evt => {\n\t\t\tthis.loaded = evt.loaded;\n\t\t};\n\t}\n\n\t/**\n\t * Returns error that occurred during file reading.\n\t */\n\tpublic get error(): DOMException | null {\n\t\treturn this._reader.error;\n\t}\n\n\t/**\n\t * Holds the data of an already loaded file. The file must be first loaded\n\t * by using {@link module:upload/filereader~FileReader#read `read()`}.\n\t */\n\tpublic get data(): string | undefined {\n\t\treturn this._data;\n\t}\n\n\t/**\n\t * Reads the provided file.\n\t *\n\t * @param file Native File object.\n\t * @returns Returns a promise that will be resolved with file's content.\n\t * The promise will be rejected in case of an error or when the reading process is aborted.\n\t */\n\tpublic read( file: File ): Promise<string> {\n\t\tconst reader = this._reader;\n\t\tthis.total = file.size;\n\n\t\treturn new Promise( ( resolve, reject ) => {\n\t\t\treader.onload = () => {\n\t\t\t\tconst result = reader.result as string;\n\n\t\t\t\tthis._data = result;\n\n\t\t\t\tresolve( result );\n\t\t\t};\n\n\t\t\treader.onerror = () => {\n\t\t\t\treject( 'error' );\n\t\t\t};\n\n\t\t\treader.onabort = () => {\n\t\t\t\treject( 'aborted' );\n\t\t\t};\n\n\t\t\tthis._reader.readAsDataURL( file );\n\t\t} );\n\t}\n\n\t/**\n\t * Aborts file reader.\n\t */\n\tpublic abort(): void {\n\t\tthis._reader.abort();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module upload/filerepository\n */\n\nimport {\n\tPlugin,\n\tPendingActions,\n\ttype PendingAction\n} from '@ckeditor/ckeditor5-core';\n\nimport {\n\tCKEditorError,\n\tCollection,\n\tObservableMixin,\n\tlogWarning,\n\tuid,\n\ttype ObservableChangeEvent,\n\ttype CollectionChangeEvent\n} from '@ckeditor/ckeditor5-utils';\n\nimport FileReader from './filereader';\n\n/**\n * File repository plugin. A central point for managing file upload.\n *\n * To use it, first you need an upload adapter. Upload adapter's job is to handle communication with the server\n * (sending the file and handling server's response). You can use one of the existing plugins introducing upload adapters\n * (e.g. {@link module:easy-image/cloudservicesuploadadapter~CloudServicesUploadAdapter} or\n * {@link module:adapter-ckfinder/uploadadapter~CKFinderUploadAdapter}) or write your own one see\n * the {@glink framework/deep-dive/upload-adapter Custom image upload adapter deep-dive} guide.\n *\n * Then, you can use {@link module:upload/filerepository~FileRepository#createLoader `createLoader()`} and the returned\n * {@link module:upload/filerepository~FileLoader} instance to load and upload files.\n */\nexport default class FileRepository extends Plugin {\n\t/**\n\t * Collection of loaders associated with this repository.\n\t */\n\tpublic loaders = new Collection<FileLoader>();\n\n\t/**\n\t * A factory function which should be defined before using `FileRepository`.\n\t *\n\t * It should return a new instance of {@link module:upload/filerepository~UploadAdapter} that will be used to upload files.\n\t * {@link module:upload/filerepository~FileLoader} instance associated with the adapter\n\t * will be passed to that function.\n\t *\n\t * For more information and example see {@link module:upload/filerepository~UploadAdapter}.\n\t */\n\tpublic createUploadAdapter?: ( loader: FileLoader ) => UploadAdapter;\n\n\t/**\n\t * Loaders mappings used to retrieve loaders references.\n\t */\n\tprivate _loadersMap = new Map<File | Promise<File>, FileLoader>();\n\n\t/**\n\t * Reference to a pending action registered in a {@link module:core/pendingactions~PendingActions} plugin\n\t * while upload is in progress. When there is no upload then value is `null`.\n\t */\n\tprivate _pendingAction: PendingAction | null = null;\n\n\t/**\n\t * Number of bytes uploaded.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public uploaded: number;\n\n\t/**\n\t * Number of total bytes to upload.\n\t *\n\t * It might be different than the file size because of headers and additional data.\n\t * It contains `null` if value is not available yet, so it's better to use {@link #uploadedPercent} to monitor\n\t * the progress.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public uploadTotal: number | null;\n\n\t/**\n\t * Upload progress in percents.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public uploadedPercent: number;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'FileRepository' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ PendingActions ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\t// Keeps upload in a sync with pending actions.\n\t\tthis.loaders.on<CollectionChangeEvent>( 'change', () => this._updatePendingAction() );\n\n\t\tthis.set( 'uploaded', 0 );\n\t\tthis.set( 'uploadTotal', null );\n\n\t\tthis.bind( 'uploadedPercent' ).to( this, 'uploaded', this, 'uploadTotal', ( uploaded, total ) => {\n\t\t\treturn total ? ( uploaded / total * 100 ) : 0;\n\t\t} );\n\t}\n\n\t/**\n\t * Returns the loader associated with specified file or promise.\n\t *\n\t * To get loader by id use `fileRepository.loaders.get( id )`.\n\t *\n\t * @param fileOrPromise Native file or promise handle.\n\t */\n\tpublic getLoader( fileOrPromise: File | Promise<File> ): FileLoader | null {\n\t\treturn this._loadersMap.get( fileOrPromise ) || null;\n\t}\n\n\t/**\n\t * Creates a loader instance for the given file.\n\t *\n\t * Requires {@link #createUploadAdapter} factory to be defined.\n\t *\n\t * @param fileOrPromise Native File object or native Promise object which resolves to a File.\n\t */\n\tpublic createLoader( fileOrPromise: File | Promise<File> ): FileLoader | null {\n\t\tif ( !this.createUploadAdapter ) {\n\t\t\t/**\n\t\t\t * You need to enable an upload adapter in order to be able to upload files.\n\t\t\t *\n\t\t\t * This warning shows up when {@link module:upload/filerepository~FileRepository} is being used\n\t\t\t * without {@link module:upload/filerepository~FileRepository#createUploadAdapter defining an upload adapter}.\n\t\t\t *\n\t\t\t * **If you see this warning when using one of the {@glink installation/getting-started/predefined-builds\n\t\t\t * CKEditor 5 Builds}**\n\t\t\t * it means that you did not configure any of the upload adapters available by default in those builds.\n\t\t\t *\n\t\t\t * See the {@glink features/images/image-upload/image-upload comprehensive \"Image upload overview\"} to learn which upload\n\t\t\t * adapters are available in the builds and how to configure them.\n\t\t\t *\n\t\t\t * **If you see this warning when using a custom build** there is a chance that you enabled\n\t\t\t * a feature like {@link module:image/imageupload~ImageUpload},\n\t\t\t * or {@link module:image/imageupload/imageuploadui~ImageUploadUI} but you did not enable any upload adapter.\n\t\t\t * You can choose one of the existing upload adapters listed in the\n\t\t\t * {@glink features/images/image-upload/image-upload \"Image upload overview\"}.\n\t\t\t *\n\t\t\t * You can also implement your {@glink framework/deep-dive/upload-adapter own image upload adapter}.\n\t\t\t *\n\t\t\t * @error filerepository-no-upload-adapter\n\t\t\t */\n\t\t\tlogWarning( 'filerepository-no-upload-adapter' );\n\n\t\t\treturn null;\n\t\t}\n\n\t\tconst loader = new FileLoader( Promise.resolve( fileOrPromise ), this.createUploadAdapter );\n\n\t\tthis.loaders.add( loader );\n\t\tthis._loadersMap.set( fileOrPromise, loader );\n\n\t\t// Store also file => loader mapping so loader can be retrieved by file instance returned upon Promise resolution.\n\t\tif ( fileOrPromise instanceof Promise ) {\n\t\t\tloader.file\n\t\t\t\t.then( file => {\n\t\t\t\t\tthis._loadersMap.set( file!, loader );\n\t\t\t\t} )\n\t\t\t\t// Every then() must have a catch().\n\t\t\t\t// File loader state (and rejections) are handled in read() and upload().\n\t\t\t\t// Also, see the \"does not swallow the file promise rejection\" test.\n\t\t\t\t.catch( () => {} );\n\t\t}\n\n\t\tloader.on<ObservableChangeEvent>( 'change:uploaded', () => {\n\t\t\tlet aggregatedUploaded = 0;\n\n\t\t\tfor ( const loader of this.loaders ) {\n\t\t\t\taggregatedUploaded += loader.uploaded;\n\t\t\t}\n\n\t\t\tthis.uploaded = aggregatedUploaded;\n\t\t} );\n\n\t\tloader.on<ObservableChangeEvent>( 'change:uploadTotal', () => {\n\t\t\tlet aggregatedTotal = 0;\n\n\t\t\tfor ( const loader of this.loaders ) {\n\t\t\t\tif ( loader.uploadTotal ) {\n\t\t\t\t\taggregatedTotal += loader.uploadTotal;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tthis.uploadTotal = aggregatedTotal;\n\t\t} );\n\n\t\treturn loader;\n\t}\n\n\t/**\n\t * Destroys the given loader.\n\t *\n\t * @param fileOrPromiseOrLoader File or Promise associated with that loader or loader itself.\n\t */\n\tpublic destroyLoader( fileOrPromiseOrLoader: File | Promise<File> | FileLoader ): void {\n\t\tconst loader = fileOrPromiseOrLoader instanceof FileLoader ? fileOrPromiseOrLoader : this.getLoader( fileOrPromiseOrLoader )!;\n\n\t\tloader._destroy();\n\n\t\tthis.loaders.remove( loader );\n\n\t\tthis._loadersMap.forEach( ( value, key ) => {\n\t\t\tif ( value === loader ) {\n\t\t\t\tthis._loadersMap.delete( key );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Registers or deregisters pending action bound with upload progress.\n\t */\n\tprivate _updatePendingAction(): void {\n\t\tconst pendingActions = this.editor.plugins.get( PendingActions );\n\n\t\tif ( this.loaders.length ) {\n\t\t\tif ( !this._pendingAction ) {\n\t\t\t\tconst t = this.editor.t;\n\t\t\t\tconst getMessage = ( value: number ) => `${ t( 'Upload in progress' ) } ${ parseInt( value as any ) }%.`;\n\n\t\t\t\tthis._pendingAction = pendingActions.add( getMessage( this.uploadedPercent ) );\n\t\t\t\tthis._pendingAction.bind( 'message' ).to( this, 'uploadedPercent', getMessage );\n\t\t\t}\n\t\t} else {\n\t\t\tpendingActions.remove( this._pendingAction! );\n\t\t\tthis._pendingAction = null;\n\t\t}\n\t}\n}\n\n/**\n * File loader class.\n *\n * It is used to control the process of reading the file and uploading it using the specified upload adapter.\n */\nclass FileLoader extends ObservableMixin() {\n\t/**\n\t * Unique id of FileLoader instance.\n\t *\n\t * @readonly\n\t */\n\tpublic readonly id: string;\n\n\t/**\n\t * Additional wrapper over the initial file promise passed to this loader.\n\t */\n\tprivate _filePromiseWrapper: FilePromiseWrapper;\n\n\t/**\n\t * Adapter instance associated with this file loader.\n\t */\n\tprivate _adapter: UploadAdapter;\n\n\t/**\n\t * FileReader used by FileLoader.\n\t */\n\tprivate _reader: FileReader;\n\n\t/**\n\t * Current status of FileLoader. It can be one of the following:\n\t *\n\t * * 'idle',\n\t * * 'reading',\n\t * * 'uploading',\n\t * * 'aborted',\n\t * * 'error'.\n\t *\n\t * When reading status can change in a following way:\n\t *\n\t * `idle` -> `reading` -> `idle`\n\t * `idle` -> `reading -> `aborted`\n\t * `idle` -> `reading -> `error`\n\t *\n\t * When uploading status can change in a following way:\n\t *\n\t * `idle` -> `uploading` -> `idle`\n\t * `idle` -> `uploading` -> `aborted`\n\t * `idle` -> `uploading` -> `error`\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public status: 'idle' | 'reading' | 'uploading' | 'aborted' | 'error';\n\n\t/**\n\t * Number of bytes uploaded.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public uploaded: number;\n\n\t/**\n\t * Number of total bytes to upload.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public uploadTotal: number | null;\n\n\t/**\n\t * Upload progress in percents.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public uploadedPercent: number;\n\n\t/**\n\t * Response of the upload.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public uploadResponse?: UploadResponse | null;\n\n\t/**\n\t * Creates a new instance of `FileLoader`.\n\t *\n\t * @param filePromise A promise which resolves to a file instance.\n\t * @param uploadAdapterCreator The function which returns {@link module:upload/filerepository~UploadAdapter} instance.\n\t */\n\tconstructor( filePromise: Promise<File>, uploadAdapterCreator: ( loader: FileLoader ) => UploadAdapter ) {\n\t\tsuper();\n\n\t\tthis.id = uid();\n\t\tthis._filePromiseWrapper = this._createFilePromiseWrapper( filePromise );\n\t\tthis._adapter = uploadAdapterCreator( this );\n\t\tthis._reader = new FileReader();\n\n\t\tthis.set( 'status', 'idle' );\n\t\tthis.set( 'uploaded', 0 );\n\t\tthis.set( 'uploadTotal', null );\n\n\t\tthis.bind( 'uploadedPercent' ).to( this, 'uploaded', this, 'uploadTotal', ( uploaded, total ) => {\n\t\t\treturn total ? ( uploaded / total * 100 ) : 0;\n\t\t} );\n\n\t\tthis.set( 'uploadResponse', null );\n\t}\n\n\t/**\n\t * A `Promise` which resolves to a `File` instance associated with this file loader.\n\t */\n\tpublic get file(): Promise<File | null> {\n\t\tif ( !this._filePromiseWrapper ) {\n\t\t\t// Loader was destroyed, return promise which resolves to null.\n\t\t\treturn Promise.resolve( null );\n\t\t} else {\n\t\t\t// The `this._filePromiseWrapper.promise` is chained and not simply returned to handle a case when:\n\t\t\t//\n\t\t\t//\t\t* The `loader.file.then( ... )` is called by external code (returned promise is pending).\n\t\t\t//\t\t* Then `loader._destroy()` is called (call is synchronous) which destroys the `loader`.\n\t\t\t//\t\t* Promise returned by the first `loader.file.then( ... )` call is resolved.\n\t\t\t//\n\t\t\t// Returning `this._filePromiseWrapper.promise` will still resolve to a `File` instance so there\n\t\t\t// is an additional check needed in the chain to see if `loader` was destroyed in the meantime.\n\t\t\treturn this._filePromiseWrapper.promise.then( file => this._filePromiseWrapper ? file : null );\n\t\t}\n\t}\n\n\t/**\n\t * Returns the file data. To read its data, you need for first load the file\n\t * by using the {@link module:upload/filerepository~FileLoader#read `read()`} method.\n\t */\n\tpublic get data(): string | undefined {\n\t\treturn this._reader.data;\n\t}\n\n\t/**\n\t * Reads file using {@link module:upload/filereader~FileReader}.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `filerepository-read-wrong-status` when status\n\t * is different than `idle`.\n\t *\n\t * Example usage:\n\t *\n\t * ```ts\n\t * fileLoader.read()\n\t * \t.then( data => { ... } )\n\t * \t.catch( err => {\n\t * \t\tif ( err === 'aborted' ) {\n\t * \t\t\tconsole.log( 'Reading aborted.' );\n\t * \t\t} else {\n\t * \t\t\tconsole.log( 'Reading error.', err );\n\t * \t\t}\n\t * \t} );\n\t * ```\n\t *\n\t * @returns Returns promise that will be resolved with read data. Promise will be rejected if error\n\t * occurs or if read process is aborted.\n\t */\n\tpublic read(): Promise<string> {\n\t\tif ( this.status != 'idle' ) {\n\t\t\t/**\n\t\t\t * You cannot call read if the status is different than idle.\n\t\t\t *\n\t\t\t * @error filerepository-read-wrong-status\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'filerepository-read-wrong-status', this );\n\t\t}\n\n\t\tthis.status = 'reading';\n\n\t\treturn this.file\n\t\t\t.then( file => this._reader.read( file! ) )\n\t\t\t.then( data => {\n\t\t\t\t// Edge case: reader was aborted after file was read - double check for proper status.\n\t\t\t\t// It can happen when image was deleted during its upload.\n\t\t\t\tif ( this.status !== 'reading' ) {\n\t\t\t\t\tthrow this.status;\n\t\t\t\t}\n\n\t\t\t\tthis.status = 'idle';\n\n\t\t\t\treturn data;\n\t\t\t} )\n\t\t\t.catch( err => {\n\t\t\t\tif ( err === 'aborted' ) {\n\t\t\t\t\tthis.status = 'aborted';\n\t\t\t\t\tthrow 'aborted';\n\t\t\t\t}\n\n\t\t\t\tthis.status = 'error';\n\t\t\t\tthrow this._reader.error ? this._reader.error : err;\n\t\t\t} );\n\t}\n\n\t/**\n\t * Reads file using the provided {@link module:upload/filerepository~UploadAdapter}.\n\t *\n\t * Throws {@link module:utils/ckeditorerror~CKEditorError CKEditorError} `filerepository-upload-wrong-status` when status\n\t * is different than `idle`.\n\t * Example usage:\n\t *\n\t * ```ts\n\t * fileLoader.upload()\n\t * \t.then( data => { ... } )\n\t * \t.catch( e => {\n\t * \t\tif ( e === 'aborted' ) {\n\t * \t\t\tconsole.log( 'Uploading aborted.' );\n\t * \t\t} else {\n\t * \t\t\tconsole.log( 'Uploading error.', e );\n\t * \t\t}\n\t * \t} );\n\t * ```\n\t *\n\t * @returns Returns promise that will be resolved with response data. Promise will be rejected if error\n\t * occurs or if read process is aborted.\n\t */\n\tpublic upload(): Promise<UploadResponse> {\n\t\tif ( this.status != 'idle' ) {\n\t\t\t/**\n\t\t\t * You cannot call upload if the status is different than idle.\n\t\t\t *\n\t\t\t * @error filerepository-upload-wrong-status\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'filerepository-upload-wrong-status', this );\n\t\t}\n\n\t\tthis.status = 'uploading';\n\n\t\treturn this.file\n\t\t\t.then( () => this._adapter.upload() )\n\t\t\t.then( data => {\n\t\t\t\tthis.uploadResponse = data;\n\t\t\t\tthis.status = 'idle';\n\n\t\t\t\treturn data;\n\t\t\t} )\n\t\t\t.catch( err => {\n\t\t\t\tif ( this.status === 'aborted' ) {\n\t\t\t\t\tthrow 'aborted';\n\t\t\t\t}\n\n\t\t\t\tthis.status = 'error';\n\t\t\t\tthrow err;\n\t\t\t} );\n\t}\n\n\t/**\n\t * Aborts loading process.\n\t */\n\tpublic abort(): void {\n\t\tconst status = this.status;\n\t\tthis.status = 'aborted';\n\n\t\tif ( !this._filePromiseWrapper.isFulfilled ) {\n\t\t\t// Edge case: file loader is aborted before read() is called\n\t\t\t// so it might happen that no one handled the rejection of this promise.\n\t\t\t// See https://github.com/ckeditor/ckeditor5-upload/pull/100\n\t\t\tthis._filePromiseWrapper.promise.catch( () => {} );\n\n\t\t\tthis._filePromiseWrapper.rejecter( 'aborted' );\n\t\t} else if ( status == 'reading' ) {\n\t\t\tthis._reader.abort();\n\t\t} else if ( status == 'uploading' && this._adapter.abort ) {\n\t\t\tthis._adapter.abort();\n\t\t}\n\n\t\tthis._destroy();\n\t}\n\n\t/**\n\t * Performs cleanup.\n\t *\n\t * @internal\n\t */\n\tpublic _destroy(): void {\n\t\tthis._filePromiseWrapper = undefined as any;\n\t\tthis._reader = undefined as any;\n\t\tthis._adapter = undefined as any;\n\t\tthis.uploadResponse = undefined;\n\t}\n\n\t/**\n\t * Wraps a given file promise into another promise giving additional\n\t * control (resolving, rejecting, checking if fulfilled) over it.\n\t *\n\t * @param filePromise The initial file promise to be wrapped.\n\t */\n\tprivate _createFilePromiseWrapper( filePromise: Promise<File> ): FilePromiseWrapper {\n\t\tconst wrapper: Partial<FilePromiseWrapper> = {};\n\n\t\twrapper.promise = new Promise<File>( ( resolve, reject ) => {\n\t\t\twrapper.rejecter = reject;\n\t\t\twrapper.isFulfilled = false;\n\n\t\t\tfilePromise\n\t\t\t\t.then( file => {\n\t\t\t\t\twrapper.isFulfilled = true;\n\t\t\t\t\tresolve( file );\n\t\t\t\t} )\n\t\t\t\t.catch( err => {\n\t\t\t\t\twrapper.isFulfilled = true;\n\t\t\t\t\treject( err );\n\t\t\t\t} );\n\t\t} );\n\n\t\treturn wrapper as FilePromiseWrapper;\n\t}\n}\n\nexport type { FileLoader };\n\n/**\n * Upload adapter interface used by the {@link module:upload/filerepository~FileRepository file repository}\n * to handle file upload. An upload adapter is a bridge between the editor and server that handles file uploads.\n * It should contain a logic necessary to initiate an upload process and monitor its progress.\n *\n * Learn how to develop your own upload adapter for CKEditor 5 in the\n * {@glink framework/deep-dive/upload-adapter \"Custom upload adapter\"} guide.\n *\n * @interface UploadAdapter\n */\nexport interface UploadAdapter {\n\n\t/**\n\t * Executes the upload process.\n\t * This method should return a promise that will resolve when data will be uploaded to server. Promise should be\n\t * resolved with an object containing information about uploaded file:\n\t *\n\t * ```json\n\t * {\n\t * \tdefault: 'http://server/default-size.image.png'\n\t * }\n\t * ```\n\t *\n\t * Additionally, other image sizes can be provided:\n\t *\n\t * ```json\n\t * {\n\t * \tdefault: 'http://server/default-size.image.png',\n\t * \t'160': 'http://server/size-160.image.png',\n\t * \t'500': 'http://server/size-500.image.png',\n\t * \t'1000': 'http://server/size-1000.image.png',\n\t * \t'1052': 'http://server/default-size.image.png'\n\t * }\n\t * ```\n\t *\n\t * You can also pass additional properties from the server. In this case you need to wrap URLs\n\t * in the `urls` object and pass additional properties along the `urls` property.\n\t *\n\t * ```json\n\t * {\n\t * \tmyCustomProperty: 'foo',\n\t * \turls: {\n\t * \t\tdefault: 'http://server/default-size.image.png',\n\t * \t\t'160': 'http://server/size-160.image.png',\n\t * \t\t'500': 'http://server/size-500.image.png',\n\t * \t\t'1000': 'http://server/size-1000.image.png',\n\t * \t\t'1052': 'http://server/default-size.image.png'\n\t * \t}\n\t * }\n\t * ```\n\t *\n\t * NOTE: When returning multiple images, the widest returned one should equal the default one. It is essential to\n\t * correctly set `width` attribute of the image. See this discussion:\n\t * https://github.com/ckeditor/ckeditor5-easy-image/issues/4 for more information.\n\t *\n\t * Take a look at {@link module:upload/filerepository~UploadAdapter example Adapter implementation} and\n\t * {@link module:upload/filerepository~FileRepository#createUploadAdapter createUploadAdapter method}.\n\t *\n\t * @returns Promise that should be resolved when data is uploaded.\n\t */\n\tupload(): Promise<UploadResponse>;\n\n\t/**\n\t * Aborts the upload process.\n\t * After aborting it should reject promise returned from {@link #upload upload()}.\n\t *\n\t * Take a look at {@link module:upload/filerepository~UploadAdapter example Adapter implementation} and\n\t * {@link module:upload/filerepository~FileRepository#createUploadAdapter createUploadAdapter method}.\n\t */\n\tabort?(): void;\n}\n\nexport type UploadResponse = Record<string, unknown>;\n\n/**\n * Object returned by {@link module:upload/filerepository~FileLoader#_createFilePromiseWrapper} method\n * to add more control over the initial file promise passed to {@link module:upload/filerepository~FileLoader}.\n */\ntype FilePromiseWrapper = {\n\n\t/**\n\t * Wrapper promise which can be chained for further processing.\n\t */\n\tpromise: Promise<File>;\n\n\t/**\n\t * Rejects the promise when called.\n\t */\n\trejecter: ( reason?: unknown ) => void;\n\n\t/**\n\t * Whether original promise is already fulfilled.\n\t */\n\tisFulfilled: boolean;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module upload/ui/filedialogbuttonview\n */\n\nimport { ButtonView, View, type ButtonExecuteEvent } from '@ckeditor/ckeditor5-ui';\n\nimport type { Locale } from '@ckeditor/ckeditor5-utils';\n\n/**\n * The file dialog button view.\n *\n * This component provides a button that opens the native file selection dialog.\n * It can be used to implement the UI of a file upload feature.\n *\n * ```ts\n * const view = new FileDialogButtonView( locale );\n *\n * view.set( {\n * \tacceptedType: 'image/*',\n * \tallowMultipleFiles: true\n * } );\n *\n * view.buttonView.set( {\n * \tlabel: t( 'Insert image' ),\n * \ticon: imageIcon,\n * \ttooltip: true\n * } );\n *\n * view.on( 'done', ( evt, files ) => {\n * \tfor ( const file of Array.from( files ) ) {\n * \t\tconsole.log( 'Selected file', file );\n * \t}\n * } );\n * ```\n */\nexport default class FileDialogButtonView extends View {\n\t/**\n\t * The button view of the component.\n\t */\n\tpublic buttonView: ButtonView;\n\n\t/**\n\t * A hidden `<input>` view used to execute file dialog.\n\t */\n\tprivate _fileInputView: FileInputView;\n\n\t/**\n\t * Accepted file types. Can be provided in form of file extensions, media type or one of:\n\t * * `audio/*`,\n\t * * `video/*`,\n\t * * `image/*`.\n\t *\n\t * @observable\n\t */\n\tdeclare public acceptedType: string;\n\n\t/**\n\t * Indicates if multiple files can be selected. Defaults to `true`.\n\t *\n\t * @observable\n\t */\n\tdeclare public allowMultipleFiles: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.buttonView = new ButtonView( locale );\n\n\t\tthis._fileInputView = new FileInputView( locale );\n\t\tthis._fileInputView.bind( 'acceptedType' ).to( this );\n\t\tthis._fileInputView.bind( 'allowMultipleFiles' ).to( this );\n\n\t\tthis._fileInputView.delegate( 'done' ).to( this );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'span',\n\t\t\tattributes: {\n\t\t\t\tclass: 'ck-file-dialog-button'\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\tthis.buttonView,\n\t\t\t\tthis._fileInputView\n\t\t\t]\n\t\t} );\n\n\t\tthis.buttonView.on<ButtonExecuteEvent>( 'execute', () => {\n\t\t\tthis._fileInputView.open();\n\t\t} );\n\t}\n\n\t/**\n\t * Focuses the {@link #buttonView}.\n\t */\n\tpublic focus(): void {\n\t\tthis.buttonView.focus();\n\t}\n}\n\n/**\n * The hidden file input view class.\n */\nclass FileInputView extends View<HTMLInputElement> {\n\t/**\n\t * Accepted file types. Can be provided in form of file extensions, media type or one of:\n\t * * `audio/*`,\n\t * * `video/*`,\n\t * * `image/*`.\n\t *\n\t * @observable\n\t */\n\tdeclare public acceptedType?: string;\n\n\t/**\n\t * Indicates if multiple files can be selected. Defaults to `false`.\n\t *\n\t * @observable\n\t */\n\tdeclare public allowMultipleFiles: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale?: Locale ) {\n\t\tsuper( locale );\n\n\t\tthis.set( 'acceptedType', undefined );\n\t\tthis.set( 'allowMultipleFiles', false );\n\n\t\tconst bind = this.bindTemplate;\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'input',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck-hidden'\n\t\t\t\t],\n\t\t\t\ttype: 'file',\n\t\t\t\ttabindex: '-1',\n\t\t\t\taccept: bind.to( 'acceptedType' ),\n\t\t\t\tmultiple: bind.to( 'allowMultipleFiles' )\n\t\t\t},\n\n\t\t\ton: {\n\t\t\t\t// Removing from code coverage since we cannot programmatically set input element files.\n\t\t\t\tchange: bind.to( /* istanbul ignore next -- @preserve */ () => {\n\t\t\t\t\tif ( this.element && this.element.files && this.element.files.length ) {\n\t\t\t\t\t\tthis.fire<FileInputViewDoneEvent>( 'done', this.element.files );\n\t\t\t\t\t}\n\n\t\t\t\t\tthis.element!.value = '';\n\t\t\t\t} )\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Opens file dialog.\n\t */\n\tpublic open(): void {\n\t\tthis.element!.click();\n\t}\n}\n\n/**\n * Fired when file dialog is closed with file selected.\n *\n * ```ts\n * view.on( 'done', ( evt, files ) => {\n * \tfor ( const file of files ) {\n * \t\tconsole.log( 'Selected file', file );\n * \t}\n * }\n * ```\n */\nexport type FileInputViewDoneEvent = {\n\tname: 'done';\n\targs: [ files: FileList ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals window, document */\n\n/**\n * @module adapter-ckfinder/utils\n */\n\nconst TOKEN_COOKIE_NAME = 'ckCsrfToken';\nconst TOKEN_LENGTH = 40;\nconst tokenCharset = 'abcdefghijklmnopqrstuvwxyz0123456789';\n\n/**\n * Returns the CSRF token value. The value is a hash stored in `document.cookie`\n * under the `ckCsrfToken` key. The CSRF token can be used to secure the communication\n * between the web browser and the CKFinder server.\n */\nexport function getCsrfToken(): string {\n\tlet token = getCookie( TOKEN_COOKIE_NAME );\n\n\tif ( !token || token.length != TOKEN_LENGTH ) {\n\t\ttoken = generateToken( TOKEN_LENGTH );\n\t\tsetCookie( TOKEN_COOKIE_NAME, token );\n\t}\n\n\treturn token;\n}\n\n/**\n * Returns the value of the cookie with a given name or `null` if the cookie is not found.\n */\nexport function getCookie( name: string ): string | null {\n\tname = name.toLowerCase();\n\tconst parts = document.cookie.split( ';' );\n\n\tfor ( const part of parts ) {\n\t\tconst pair = part.split( '=' );\n\t\tconst key = decodeURIComponent( pair[ 0 ].trim().toLowerCase() );\n\n\t\tif ( key === name ) {\n\t\t\treturn decodeURIComponent( pair[ 1 ] );\n\t\t}\n\t}\n\n\treturn null;\n}\n\n/**\n * Sets the value of the cookie with a given name.\n */\nexport function setCookie( name: string, value: string ): void {\n\tdocument.cookie = encodeURIComponent( name ) + '=' + encodeURIComponent( value ) + ';path=/';\n}\n\n/**\n * Generates the CSRF token with the given length.\n */\nfunction generateToken( length: number ): string {\n\tlet result = '';\n\tconst randValues = new Uint8Array( length );\n\n\twindow.crypto.getRandomValues( randValues );\n\n\tfor ( let j = 0; j < randValues.length; j++ ) {\n\t\tconst character = tokenCharset.charAt( randValues[ j ] % tokenCharset.length );\n\t\tresult += Math.random() > 0.5 ? character.toUpperCase() : character;\n\t}\n\n\treturn result;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals XMLHttpRequest, FormData */\n\n/**\n * @module adapter-ckfinder/uploadadapter\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport {\n\tFileRepository,\n\ttype UploadAdapter as UploadAdapterInterface,\n\ttype FileLoader,\n\ttype UploadResponse\n} from 'ckeditor5/src/upload';\nimport type { LocaleTranslate } from 'ckeditor5/src/utils';\n\nimport { getCsrfToken } from './utils';\n\n/**\n * A plugin that enables file uploads in CKEditor 5 using the CKFinder serverside connector.\n *\n * See the {@glink features/file-management/ckfinder \"CKFinder file manager integration\"} guide to learn how to configure\n * and use this feature as well as find out more about the full integration with the file manager\n * provided by the {@link module:ckfinder/ckfinder~CKFinder} plugin.\n *\n * Check out the {@glink features/images/image-upload/image-upload comprehensive \"Image upload overview\"} guide to learn\n * about other ways to upload images into CKEditor 5.\n */\nexport default class CKFinderUploadAdapter extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ FileRepository ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CKFinderUploadAdapter' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst url = this.editor.config.get( 'ckfinder.uploadUrl' )! as string;\n\n\t\tif ( !url ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Register CKFinderAdapter\n\t\tthis.editor.plugins.get( FileRepository ).createUploadAdapter = loader => new UploadAdapter( loader, url, this.editor.t );\n\t}\n}\n\n/**\n * Upload adapter for CKFinder.\n */\nclass UploadAdapter implements UploadAdapterInterface {\n\t/**\n\t * FileLoader instance to use during the upload.\n\t */\n\tpublic loader: FileLoader;\n\n\t/**\n\t * Upload URL.\n\t */\n\tpublic url: string;\n\n\t/**\n\t * Locale translation method.\n\t */\n\tpublic t: LocaleTranslate;\n\n\tprivate xhr?: XMLHttpRequest;\n\n\t/**\n\t * Creates a new adapter instance.\n\t */\n\tconstructor( loader: FileLoader, url: string, t: LocaleTranslate ) {\n\t\tthis.loader = loader;\n\t\tthis.url = url;\n\t\tthis.t = t;\n\t}\n\n\t/**\n\t * Starts the upload process.\n\t *\n\t * @see module:upload/filerepository~UploadAdapter#upload\n\t */\n\tpublic upload() {\n\t\treturn this.loader.file.then( file => {\n\t\t\treturn new Promise<UploadResponse>( ( resolve, reject ) => {\n\t\t\t\tthis._initRequest();\n\t\t\t\tthis._initListeners( resolve, reject, file! );\n\t\t\t\tthis._sendRequest( file! );\n\t\t\t} );\n\t\t} );\n\t}\n\n\t/**\n\t * Aborts the upload process.\n\t *\n\t * @see module:upload/filerepository~UploadAdapter#abort\n\t */\n\tpublic abort() {\n\t\tif ( this.xhr ) {\n\t\t\tthis.xhr.abort();\n\t\t}\n\t}\n\n\t/**\n\t * Initializes the XMLHttpRequest object.\n\t */\n\tprivate _initRequest() {\n\t\tconst xhr = this.xhr = new XMLHttpRequest();\n\n\t\txhr.open( 'POST', this.url, true );\n\t\txhr.responseType = 'json';\n\t}\n\n\t/**\n\t * Initializes XMLHttpRequest listeners.\n\t *\n\t * @param resolve Callback function to be called when the request is successful.\n\t * @param reject Callback function to be called when the request cannot be completed.\n\t * @param file File instance to be uploaded.\n\t */\n\tprivate _initListeners(\n\t\tresolve: ( value: UploadResponse ) => void,\n\t\treject: ( reason?: unknown ) => void,\n\t\tfile: File\n\t) {\n\t\tconst xhr = this.xhr!;\n\t\tconst loader = this.loader;\n\t\tconst t = this.t;\n\t\tconst genericError = t( 'Cannot upload file:' ) + ` ${ file.name }.`;\n\n\t\txhr.addEventListener( 'error', () => reject( genericError ) );\n\t\txhr.addEventListener( 'abort', () => reject() );\n\t\txhr.addEventListener( 'load', () => {\n\t\t\tconst response = xhr.response;\n\n\t\t\tif ( !response || !response.uploaded ) {\n\t\t\t\treturn reject( response && response.error && response.error.message ? response.error.message : genericError );\n\t\t\t}\n\n\t\t\tresolve( {\n\t\t\t\tdefault: response.url\n\t\t\t} );\n\t\t} );\n\n\t\t// Upload progress when it's supported.\n\t\t/* istanbul ignore else -- @preserve */\n\t\tif ( xhr.upload ) {\n\t\t\txhr.upload.addEventListener( 'progress', evt => {\n\t\t\t\tif ( evt.lengthComputable ) {\n\t\t\t\t\tloader.uploadTotal = evt.total;\n\t\t\t\t\tloader.uploaded = evt.loaded;\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Prepares the data and sends the request.\n\t *\n\t * @param file File instance to be uploaded.\n\t */\n\tprivate _sendRequest( file: File ) {\n\t\t// Prepare form data.\n\t\tconst data = new FormData();\n\t\tdata.append( 'upload', file );\n\t\tdata.append( 'ckCsrfToken', getCsrfToken() );\n\n\t\t// Send request.\n\t\tthis.xhr!.send( data );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport type { Command, Editor } from 'ckeditor5/src/core';\n\nimport {\n\tLiveRange,\n\ttype DocumentChangeEvent,\n\ttype Item,\n\ttype Text\n} from 'ckeditor5/src/engine';\n\nimport { first } from 'ckeditor5/src/utils';\n\nimport type Autoformat from './autoformat';\n\nimport type { Delete } from 'ckeditor5/src/typing';\n\n/**\n * The block autoformatting engine. It allows to format various block patterns. For example,\n * it can be configured to turn a paragraph starting with `*` and followed by a space into a list item.\n *\n * The autoformatting operation is integrated with the undo manager,\n * so the autoformatting step can be undone if the user's intention was not to format the text.\n *\n * See the {@link module:autoformat/blockautoformatediting~blockAutoformatEditing `blockAutoformatEditing`} documentation\n * to learn how to create custom block autoformatters. You can also use\n * the {@link module:autoformat/autoformat~Autoformat} feature which enables a set of default autoformatters\n * (lists, headings, bold and italic).\n *\n * @module autoformat/blockautoformatediting\n */\n\n/**\n * Creates a listener triggered on {@link module:engine/model/document~Document#event:change:data `change:data`} event in the document.\n * Calls the callback when inserted text matches the regular expression or the command name\n * if provided instead of the callback.\n *\n * Examples of usage:\n *\n * To convert a paragraph into heading 1 when `- ` is typed, using just the command name:\n *\n * ```ts\n * blockAutoformatEditing( editor, plugin, /^\\- $/, 'heading1' );\n * ```\n *\n * To convert a paragraph into heading 1 when `- ` is typed, using just the callback:\n *\n * ```ts\n * blockAutoformatEditing( editor, plugin, /^\\- $/, ( context ) => {\n * \tconst { match } = context;\n * \tconst headingLevel = match[ 1 ].length;\n *\n * \teditor.execute( 'heading', {\n * \t\tformatId: `heading${ headingLevel }`\n * \t} );\n * } );\n * ```\n *\n * @param editor The editor instance.\n * @param plugin The autoformat plugin instance.\n * @param pattern The regular expression to execute on just inserted text. The regular expression is tested against the text\n * from the beginning until the caret position.\n * @param callbackOrCommand The callback to execute or the command to run when the text is matched.\n * In case of providing the callback, it receives the following parameter:\n * * match RegExp.exec() result of matching the pattern to inserted text.\n */\nexport default function blockAutoformatEditing(\n\teditor: Editor,\n\tplugin: Autoformat,\n\tpattern: RegExp,\n\tcallbackOrCommand: string | ( ( context: { match: RegExpExecArray } ) => unknown )\n): void {\n\tlet callback: ( context: { match: RegExpExecArray } ) => unknown;\n\tlet command: Command | null = null;\n\n\tif ( typeof callbackOrCommand == 'function' ) {\n\t\tcallback = callbackOrCommand;\n\t} else {\n\t\t// We assume that the actual command name was provided.\n\t\tcommand = editor.commands.get( callbackOrCommand )!;\n\n\t\tcallback = () => {\n\t\t\teditor.execute( callbackOrCommand );\n\t\t};\n\t}\n\n\teditor.model.document.on<DocumentChangeEvent>( 'change:data', ( evt, batch ) => {\n\t\tif ( command && !command.isEnabled || !plugin.isEnabled ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst range = first( editor.model.document.selection.getRanges() )!;\n\n\t\tif ( !range.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( batch.isUndo || !batch.isLocal ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst changes = Array.from( editor.model.document.differ.getChanges() );\n\t\tconst entry = changes[ 0 ];\n\n\t\t// Typing is represented by only a single change.\n\t\tif ( changes.length != 1 || entry.type !== 'insert' || entry.name != '$text' || entry.length != 1 ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst blockToFormat = entry.position.parent;\n\n\t\t// Block formatting should be disabled in codeBlocks (#5800).\n\t\tif ( blockToFormat.is( 'element', 'codeBlock' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Only list commands and custom callbacks can be applied inside a list.\n\t\tif ( blockToFormat.is( 'element', 'listItem' ) &&\n\t\t\ttypeof callbackOrCommand !== 'function' &&\n\t\t\t![ 'numberedList', 'bulletedList', 'todoList' ].includes( callbackOrCommand )\n\t\t) {\n\t\t\treturn;\n\t\t}\n\n\t\t// In case a command is bound, do not re-execute it over an existing block style which would result in a style removal.\n\t\t// Instead, just drop processing so that autoformat trigger text is not lost. E.g. writing \"# \" in a level 1 heading.\n\t\tif ( command && command.value === true ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst firstNode = blockToFormat.getChild( 0 ) as Text;\n\n\t\tconst firstNodeRange = editor.model.createRangeOn( firstNode );\n\n\t\t// Range is only expected to be within or at the very end of the first text node.\n\t\tif ( !firstNodeRange.containsRange( range ) && !range.end.isEqual( firstNodeRange.end ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst match = pattern.exec( firstNode.data.substr( 0, range.end.offset ) );\n\n\t\t// ...and this text node's data match the pattern.\n\t\tif ( !match ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Use enqueueChange to create new batch to separate typing batch from the auto-format changes.\n\t\teditor.model.enqueueChange( writer => {\n\t\t\t// Matched range.\n\t\t\tconst start = writer.createPositionAt( blockToFormat, 0 );\n\t\t\tconst end = writer.createPositionAt( blockToFormat, match[ 0 ].length );\n\t\t\tconst range = new LiveRange( start, end );\n\n\t\t\tconst wasChanged = callback( { match } );\n\n\t\t\t// Remove matched text.\n\t\t\tif ( wasChanged !== false ) {\n\t\t\t\twriter.remove( range );\n\n\t\t\t\tconst selectionRange = editor.model.document.selection.getFirstRange()!;\n\t\t\t\tconst blockRange = writer.createRangeIn( blockToFormat );\n\n\t\t\t\t// If the block is empty and the document selection has been moved when\n\t\t\t\t// applying formatting (e.g. is now in newly created block).\n\t\t\t\tif ( blockToFormat.isEmpty && !blockRange.isEqual( selectionRange ) && !blockRange.containsRange( selectionRange, true ) ) {\n\t\t\t\t\twriter.remove( blockToFormat as Item );\n\t\t\t\t}\n\t\t\t}\n\t\t\trange.detach();\n\n\t\t\teditor.model.enqueueChange( () => {\n\t\t\t\tconst deletePlugin: Delete = editor.plugins.get( 'Delete' );\n\n\t\t\t\tdeletePlugin.requestUndoOnBackspace();\n\t\t\t} );\n\t\t} );\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * The inline autoformatting engine. It allows to format various inline patterns. For example,\n * it can be configured to make \"foo\" bold when typed `**foo**` (the `**` markers will be removed).\n *\n * The autoformatting operation is integrated with the undo manager,\n * so the autoformatting step can be undone if the user's intention was not to format the text.\n *\n * See the {@link module:autoformat/inlineautoformatediting~inlineAutoformatEditing `inlineAutoformatEditing`} documentation\n * to learn how to create custom inline autoformatters. You can also use\n * the {@link module:autoformat/autoformat~Autoformat} feature which enables a set of default autoformatters\n * (lists, headings, bold and italic).\n *\n * @module autoformat/inlineautoformatediting\n */\n\nimport type { Editor } from 'ckeditor5/src/core';\nimport type {\n\tDocumentChangeEvent,\n\tModel,\n\tPosition,\n\tRange,\n\tWriter\n} from 'ckeditor5/src/engine';\nimport type { Delete, LastTextLineData } from 'ckeditor5/src/typing';\n\nimport type Autoformat from './autoformat';\n\nexport type TestCallback = ( text: string ) => {\n\tremove: Array<Array<number>>;\n\tformat: Array<Array<number>>;\n};\n\n/**\n * Enables autoformatting mechanism for a given {@link module:core/editor/editor~Editor}.\n *\n * It formats the matched text by applying the given model attribute or by running the provided formatting callback.\n * On every {@link module:engine/model/document~Document#event:change:data data change} in the model document\n * the autoformatting engine checks the text on the left of the selection\n * and executes the provided action if the text matches given criteria (regular expression or callback).\n *\n * @param editor The editor instance.\n * @param plugin The autoformat plugin instance.\n * @param testRegexpOrCallback The regular expression or callback to execute on text.\n * Provided regular expression *must* have three capture groups. The first and the third capture group\n * should match opening and closing delimiters. The second capture group should match the text to format.\n *\n * ```ts\n * // Matches the `**bold text**` pattern.\n * // There are three capturing groups:\n * // - The first to match the starting `**` delimiter.\n * // - The second to match the text to format.\n * // - The third to match the ending `**` delimiter.\n * inlineAutoformatEditing( editor, plugin, /(\\*\\*)([^\\*]+?)(\\*\\*)$/g, formatCallback );\n * ```\n *\n * When a function is provided instead of the regular expression, it will be executed with the text to match as a parameter.\n * The function should return proper \"ranges\" to delete and format.\n *\n * ```ts\n * {\n * \tremove: [\n * \t\t[ 0, 1 ],\t// Remove the first letter from the given text.\n * \t\t[ 5, 6 ]\t// Remove the 6th letter from the given text.\n * \t],\n * \tformat: [\n * \t\t[ 1, 5 ]\t// Format all letters from 2nd to 5th.\n * \t]\n * }\n * ```\n *\n * @param formatCallback A callback to apply actual formatting.\n * It should return `false` if changes should not be applied (e.g. if a command is disabled).\n *\n * ```ts\n * inlineAutoformatEditing( editor, plugin, /(\\*\\*)([^\\*]+?)(\\*\\*)$/g, ( writer, rangesToFormat ) => {\n * \tconst command = editor.commands.get( 'bold' );\n *\n * \tif ( !command.isEnabled ) {\n * \t\treturn false;\n * \t}\n *\n * \tconst validRanges = editor.model.schema.getValidRanges( rangesToFormat, 'bold' );\n *\n * \tfor ( let range of validRanges ) {\n * \t\twriter.setAttribute( 'bold', true, range );\n * \t}\n * } );\n * ```\n */\nexport default function inlineAutoformatEditing(\n\teditor: Editor,\n\tplugin: Autoformat,\n\ttestRegexpOrCallback: RegExp | TestCallback,\n\tformatCallback: ( writer: Writer, rangesToFormat: Array<Range> ) => boolean | undefined\n): void {\n\tlet regExp: RegExp;\n\tlet testCallback: TestCallback | undefined;\n\n\tif ( testRegexpOrCallback instanceof RegExp ) {\n\t\tregExp = testRegexpOrCallback;\n\t} else {\n\t\ttestCallback = testRegexpOrCallback;\n\t}\n\n\t// A test callback run on changed text.\n\ttestCallback = testCallback || ( text => {\n\t\tlet result: RegExpExecArray | null;\n\t\tconst remove: Array<Array<number>> = [];\n\t\tconst format: Array<Array<number>> = [];\n\n\t\twhile ( ( result = regExp.exec( text ) ) !== null ) {\n\t\t\t// There should be full match and 3 capture groups.\n\t\t\tif ( result && result.length < 4 ) {\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\tlet {\n\t\t\t\tindex,\n\t\t\t\t'1': leftDel,\n\t\t\t\t'2': content,\n\t\t\t\t'3': rightDel\n\t\t\t} = result;\n\n\t\t\t// Real matched string - there might be some non-capturing groups so we need to recalculate starting index.\n\t\t\tconst found = leftDel + content + rightDel;\n\t\t\tindex += result[ 0 ].length - found.length;\n\n\t\t\t// Start and End offsets of delimiters to remove.\n\t\t\tconst delStart = [\n\t\t\t\tindex,\n\t\t\t\tindex + leftDel.length\n\t\t\t];\n\t\t\tconst delEnd = [\n\t\t\t\tindex + leftDel.length + content.length,\n\t\t\t\tindex + leftDel.length + content.length + rightDel.length\n\t\t\t];\n\n\t\t\tremove.push( delStart );\n\t\t\tremove.push( delEnd );\n\n\t\t\tformat.push( [ index + leftDel.length, index + leftDel.length + content.length ] );\n\t\t}\n\n\t\treturn {\n\t\t\tremove,\n\t\t\tformat\n\t\t};\n\t} );\n\n\teditor.model.document.on<DocumentChangeEvent>( 'change:data', ( evt, batch ) => {\n\t\tif ( batch.isUndo || !batch.isLocal || !plugin.isEnabled ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst model = editor.model;\n\t\tconst selection = model.document.selection;\n\n\t\t// Do nothing if selection is not collapsed.\n\t\tif ( !selection.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst changes = Array.from( model.document.differ.getChanges() );\n\t\tconst entry = changes[ 0 ];\n\n\t\t// Typing is represented by only a single change.\n\t\tif ( changes.length != 1 || entry.type !== 'insert' || entry.name != '$text' || entry.length != 1 ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst focus = selection.focus;\n\t\tconst block = focus!.parent;\n\t\tconst { text, range } = getTextAfterCode( model.createRange( model.createPositionAt( block, 0 ), focus! ), model );\n\t\tconst testOutput = testCallback!( text );\n\t\tconst rangesToFormat = testOutputToRanges( range.start, testOutput.format, model );\n\t\tconst rangesToRemove = testOutputToRanges( range.start, testOutput.remove, model );\n\n\t\tif ( !( rangesToFormat.length && rangesToRemove.length ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Use enqueueChange to create new batch to separate typing batch from the auto-format changes.\n\t\tmodel.enqueueChange( writer => {\n\t\t\t// Apply format.\n\t\t\tconst hasChanged = formatCallback( writer, rangesToFormat );\n\n\t\t\t// Strict check on `false` to have backward compatibility (when callbacks were returning `undefined`).\n\t\t\tif ( hasChanged === false ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Remove delimiters - use reversed order to not mix the offsets while removing.\n\t\t\tfor ( const range of rangesToRemove.reverse() ) {\n\t\t\t\twriter.remove( range );\n\t\t\t}\n\n\t\t\tmodel.enqueueChange( () => {\n\t\t\t\tconst deletePlugin: Delete = editor.plugins.get( 'Delete' );\n\n\t\t\t\tdeletePlugin.requestUndoOnBackspace();\n\t\t\t} );\n\t\t} );\n\t} );\n}\n\n/**\n * Converts output of the test function provided to the inlineAutoformatEditing and converts it to the model ranges\n * inside provided block.\n */\nfunction testOutputToRanges( start: Position, arrays: Array<Array<number>>, model: Model ) {\n\treturn arrays\n\t\t.filter( array => ( array[ 0 ] !== undefined && array[ 1 ] !== undefined ) )\n\t\t.map( array => {\n\t\t\treturn model.createRange( start.getShiftedBy( array[ 0 ] ), start.getShiftedBy( array[ 1 ] ) );\n\t\t} );\n}\n\n/**\n * Returns the last text line after the last code element from the given range.\n * It is similar to {@link module:typing/utils/getlasttextline.getLastTextLine `getLastTextLine()`},\n * but it ignores any text before the last `code`.\n */\nfunction getTextAfterCode( range: Range, model: Model ): LastTextLineData {\n\tlet start = range.start;\n\n\tconst text = Array.from( range.getItems() ).reduce( ( rangeText, node ) => {\n\t\t// Trim text to a last occurrence of an inline element and update range start.\n\t\tif ( !( node.is( '$text' ) || node.is( '$textProxy' ) ) || node.getAttribute( 'code' ) ) {\n\t\t\tstart = model.createPositionAfter( node );\n\n\t\t\treturn '';\n\t\t}\n\n\t\treturn rangeText + node.data;\n\t}, '' );\n\n\treturn { text, range: model.createRange( start, range.end ) };\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module autoformat/autoformat\n */\nimport type { HeadingCommand } from '@ckeditor/ckeditor5-heading';\n\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\nimport type { Range, Writer } from 'ckeditor5/src/engine';\nimport { Delete } from 'ckeditor5/src/typing';\n\nimport blockAutoformatEditing from './blockautoformatediting';\nimport inlineAutoformatEditing from './inlineautoformatediting';\n\n/**\n * Enables a set of predefined autoformatting actions.\n *\n * For a detailed overview, check the {@glink features/autoformat Autoformatting} feature guide\n * and the {@glink api/autoformat package page}.\n */\nexport default class Autoformat extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Delete ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Autoformat' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\tthis._addListAutoformats();\n\t\tthis._addBasicStylesAutoformats();\n\t\tthis._addHeadingAutoformats();\n\t\tthis._addBlockQuoteAutoformats();\n\t\tthis._addCodeBlockAutoformats();\n\t\tthis._addHorizontalLineAutoformats();\n\t}\n\n\t/**\n\t * Adds autoformatting related to the {@link module:list/list~List}.\n\t *\n\t * When typed:\n\t * - `* ` or `- ` &ndash; A paragraph will be changed into a bulleted list.\n\t * - `1. ` or `1) ` &ndash; A paragraph will be changed into a numbered list (\"1\" can be any digit or a list of digits).\n\t * - `[] ` or `[ ] ` &ndash; A paragraph will be changed into a to-do list.\n\t * - `[x] ` or `[ x ] ` &ndash; A paragraph will be changed into a checked to-do list.\n\t */\n\tprivate _addListAutoformats(): void {\n\t\tconst commands = this.editor.commands;\n\n\t\tif ( commands.get( 'bulletedList' ) ) {\n\t\t\tblockAutoformatEditing( this.editor, this, /^[*-]\\s$/, 'bulletedList' );\n\t\t}\n\n\t\tif ( commands.get( 'numberedList' ) ) {\n\t\t\tblockAutoformatEditing( this.editor, this, /^1[.|)]\\s$/, 'numberedList' );\n\t\t}\n\n\t\tif ( commands.get( 'todoList' ) ) {\n\t\t\tblockAutoformatEditing( this.editor, this, /^\\[\\s?\\]\\s$/, 'todoList' );\n\t\t}\n\n\t\tif ( commands.get( 'checkTodoList' ) ) {\n\t\t\tblockAutoformatEditing( this.editor, this, /^\\[\\s?x\\s?\\]\\s$/, () => {\n\t\t\t\tthis.editor.execute( 'todoList' );\n\t\t\t\tthis.editor.execute( 'checkTodoList' );\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Adds autoformatting related to the {@link module:basic-styles/bold~Bold},\n\t * {@link module:basic-styles/italic~Italic}, {@link module:basic-styles/code~Code}\n\t * and {@link module:basic-styles/strikethrough~Strikethrough}\n\t *\n\t * When typed:\n\t * - `**foobar**` &ndash; `**` characters are removed and `foobar` is set to bold,\n\t * - `__foobar__` &ndash; `__` characters are removed and `foobar` is set to bold,\n\t * - `*foobar*` &ndash; `*` characters are removed and `foobar` is set to italic,\n\t * - `_foobar_` &ndash; `_` characters are removed and `foobar` is set to italic,\n\t * - ``` `foobar` &ndash; ``` ` ``` characters are removed and `foobar` is set to code,\n\t * - `~~foobar~~` &ndash; `~~` characters are removed and `foobar` is set to strikethrough.\n\t */\n\tprivate _addBasicStylesAutoformats(): void {\n\t\tconst commands = this.editor.commands;\n\n\t\tif ( commands.get( 'bold' ) ) {\n\t\t\tconst boldCallback = getCallbackFunctionForInlineAutoformat( this.editor, 'bold' );\n\n\t\t\tinlineAutoformatEditing( this.editor, this, /(?:^|\\s)(\\*\\*)([^*]+)(\\*\\*)$/g, boldCallback );\n\t\t\tinlineAutoformatEditing( this.editor, this, /(?:^|\\s)(__)([^_]+)(__)$/g, boldCallback );\n\t\t}\n\n\t\tif ( commands.get( 'italic' ) ) {\n\t\t\tconst italicCallback = getCallbackFunctionForInlineAutoformat( this.editor, 'italic' );\n\n\t\t\t// The italic autoformatter cannot be triggered by the bold markers, so we need to check the\n\t\t\t// text before the pattern (e.g. `(?:^|[^\\*])`).\n\t\t\tinlineAutoformatEditing( this.editor, this, /(?:^|\\s)(\\*)([^*_]+)(\\*)$/g, italicCallback );\n\t\t\tinlineAutoformatEditing( this.editor, this, /(?:^|\\s)(_)([^_]+)(_)$/g, italicCallback );\n\t\t}\n\n\t\tif ( commands.get( 'code' ) ) {\n\t\t\tconst codeCallback = getCallbackFunctionForInlineAutoformat( this.editor, 'code' );\n\n\t\t\tinlineAutoformatEditing( this.editor, this, /(`)([^`]+)(`)$/g, codeCallback );\n\t\t}\n\n\t\tif ( commands.get( 'strikethrough' ) ) {\n\t\t\tconst strikethroughCallback = getCallbackFunctionForInlineAutoformat( this.editor, 'strikethrough' );\n\n\t\t\tinlineAutoformatEditing( this.editor, this, /(~~)([^~]+)(~~)$/g, strikethroughCallback );\n\t\t}\n\t}\n\n\t/**\n\t * Adds autoformatting related to {@link module:heading/heading~Heading}.\n\t *\n\t * It is using a number at the end of the command name to associate it with the proper trigger:\n\t *\n\t * * `heading` with a `heading1` value will be executed when typing `#`,\n\t * * `heading` with a `heading2` value will be executed when typing `##`,\n\t * * ... up to `heading6` for `######`.\n\t */\n\tprivate _addHeadingAutoformats(): void {\n\t\tconst command: HeadingCommand | undefined = this.editor.commands.get( 'heading' );\n\n\t\tif ( command ) {\n\t\t\tcommand.modelElements\n\t\t\t\t.filter( name => name.match( /^heading[1-6]$/ ) )\n\t\t\t\t.forEach( modelName => {\n\t\t\t\t\tconst level = modelName[ 7 ];\n\t\t\t\t\tconst pattern = new RegExp( `^(#{${ level }})\\\\s$` );\n\n\t\t\t\t\tblockAutoformatEditing( this.editor, this, pattern, () => {\n\t\t\t\t\t\t// Should only be active if command is enabled and heading style associated with pattern is inactive.\n\t\t\t\t\t\tif ( !command.isEnabled || command.value === modelName ) {\n\t\t\t\t\t\t\treturn false;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tthis.editor.execute( 'heading', { value: modelName } );\n\t\t\t\t\t} );\n\t\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Adds autoformatting related to {@link module:block-quote/blockquote~BlockQuote}.\n\t *\n\t * When typed:\n\t * * `> ` &ndash; A paragraph will be changed to a block quote.\n\t */\n\tprivate _addBlockQuoteAutoformats(): void {\n\t\tif ( this.editor.commands.get( 'blockQuote' ) ) {\n\t\t\tblockAutoformatEditing( this.editor, this, /^>\\s$/, 'blockQuote' );\n\t\t}\n\t}\n\n\t/**\n\t * Adds autoformatting related to {@link module:code-block/codeblock~CodeBlock}.\n\t *\n\t * When typed:\n\t * - `` ``` `` &ndash; A paragraph will be changed to a code block.\n\t */\n\tprivate _addCodeBlockAutoformats(): void {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\n\t\tif ( editor.commands.get( 'codeBlock' ) ) {\n\t\t\tblockAutoformatEditing( editor, this, /^```$/, () => {\n\t\t\t\tif ( selection.getFirstPosition()!.parent.is( 'element', 'listItem' ) ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\t\t\t\tthis.editor.execute( 'codeBlock', {\n\t\t\t\t\tusePreviousLanguageChoice: true\n\t\t\t\t} );\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Adds autoformatting related to {@link module:horizontal-line/horizontalline~HorizontalLine}.\n\t *\n\t * When typed:\n\t * - `` --- `` &ndash; Will be replaced with a horizontal line.\n\t */\n\tprivate _addHorizontalLineAutoformats(): void {\n\t\tif ( this.editor.commands.get( 'horizontalLine' ) ) {\n\t\t\tblockAutoformatEditing( this.editor, this, /^---$/, 'horizontalLine' );\n\t\t}\n\t}\n}\n\n/**\n * Helper function for getting `inlineAutoformatEditing` callbacks that checks if command is enabled.\n */\nfunction getCallbackFunctionForInlineAutoformat( editor: Editor, attributeKey: string ) {\n\treturn ( writer: Writer, rangesToFormat: Array<Range> ): boolean | undefined => {\n\t\tconst command = editor.commands.get( attributeKey )!;\n\n\t\tif ( !command.isEnabled ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst validRanges = editor.model.schema.getValidRanges( rangesToFormat, attributeKey );\n\n\t\tfor ( const range of validRanges ) {\n\t\t\twriter.setAttribute( attributeKey, true, range );\n\t\t}\n\n\t\t// After applying attribute to the text, remove given attribute from the selection.\n\t\t// This way user is able to type a text without attribute used by auto formatter.\n\t\twriter.removeSelectionAttribute( attributeKey );\n\t};\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module basic-styles/attributecommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\n\n/**\n * An extension of the base {@link module:core/command~Command} class, which provides utilities for a command\n * that toggles a single attribute on a text or an element.\n *\n * `AttributeCommand` uses {@link module:engine/model/document~Document#selection}\n * to decide which nodes (if any) should be changed, and applies or removes the attribute from them.\n *\n * The command checks the {@link module:engine/model/model~Model#schema} to decide if it can be enabled\n * for the current selection and to which nodes the attribute can be applied.\n */\nexport default class AttributeCommand extends Command {\n\t/**\n\t * Flag indicating whether the command is active. The command is active when the\n\t * {@link module:engine/model/selection~Selection#hasAttribute selection has the attribute} which means that:\n\t *\n\t * * If the selection is not empty &ndash; That the attribute is set on the first node in the selection that allows this attribute.\n\t * * If the selection is empty &ndash; That the selection has the attribute itself (which means that newly typed\n\t * text will have this attribute, too).\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public value: boolean;\n\n\t/**\n\t * The attribute that will be set by the command.\n\t */\n\tpublic readonly attributeKey: string;\n\n\t/**\n\t * @param attributeKey Attribute that will be set by the command.\n\t */\n\tconstructor( editor: Editor, attributeKey: string ) {\n\t\tsuper( editor );\n\n\t\tthis.attributeKey = attributeKey;\n\t}\n\n\t/**\n\t * Updates the command's {@link #value} and {@link #isEnabled} based on the current selection.\n\t */\n\tpublic override refresh(): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\n\t\tthis.value = this._getValueFromFirstAllowedNode();\n\t\tthis.isEnabled = model.schema.checkAttributeInSelection( doc.selection, this.attributeKey );\n\t}\n\n\t/**\n\t * Executes the command &mdash; applies the attribute to the selection or removes it from the selection.\n\t *\n\t * If the command is active (`value == true`), it will remove attributes. Otherwise, it will set attributes.\n\t *\n\t * The execution result differs, depending on the {@link module:engine/model/document~Document#selection}:\n\t *\n\t * * If the selection is on a range, the command applies the attribute to all nodes in that range\n\t * (if they are allowed to have this attribute by the {@link module:engine/model/schema~Schema schema}).\n\t * * If the selection is collapsed in a non-empty node, the command applies the attribute to the\n\t * {@link module:engine/model/document~Document#selection} itself (note that typed characters copy attributes from the selection).\n\t * * If the selection is collapsed in an empty node, the command applies the attribute to the parent node of the selection (note\n\t * that the selection inherits all attributes from a node if it is in an empty node).\n\t *\n\t * @fires execute\n\t * @param options Command options.\n\t * @param options.forceValue If set, it will force the command behavior. If `true`,\n\t * the command will apply the attribute, otherwise the command will remove the attribute.\n\t * If not set, the command will look for its current value to decide what it should do.\n\t */\n\tpublic override execute( options: { forceValue?: boolean } = {} ): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\t\tconst selection = doc.selection;\n\t\tconst value = ( options.forceValue === undefined ) ? !this.value : options.forceValue;\n\n\t\tmodel.change( writer => {\n\t\t\tif ( selection.isCollapsed ) {\n\t\t\t\tif ( value ) {\n\t\t\t\t\twriter.setSelectionAttribute( this.attributeKey, true );\n\t\t\t\t} else {\n\t\t\t\t\twriter.removeSelectionAttribute( this.attributeKey );\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconst ranges = model.schema.getValidRanges( selection.getRanges(), this.attributeKey );\n\n\t\t\t\tfor ( const range of ranges ) {\n\t\t\t\t\tif ( value ) {\n\t\t\t\t\t\twriter.setAttribute( this.attributeKey, value, range );\n\t\t\t\t\t} else {\n\t\t\t\t\t\twriter.removeAttribute( this.attributeKey, range );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Checks the attribute value of the first node in the selection that allows the attribute.\n\t * For the collapsed selection returns the selection attribute.\n\t *\n\t * @returns The attribute value.\n\t */\n\tprivate _getValueFromFirstAllowedNode(): boolean {\n\t\tconst model = this.editor.model;\n\t\tconst schema = model.schema;\n\t\tconst selection = model.document.selection;\n\n\t\tif ( selection.isCollapsed ) {\n\t\t\treturn selection.hasAttribute( this.attributeKey );\n\t\t}\n\n\t\tfor ( const range of selection.getRanges() ) {\n\t\t\tfor ( const item of range.getItems() ) {\n\t\t\t\tif ( schema.checkAttribute( item, this.attributeKey ) ) {\n\t\t\t\t\treturn item.hasAttribute( this.attributeKey );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module basic-styles/bold/boldediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport AttributeCommand from '../attributecommand';\n\nconst BOLD = 'bold';\n\n/**\n * The bold editing feature.\n *\n * It registers the `'bold'` command and introduces the `bold` attribute in the model which renders to the view\n * as a `<strong>` element.\n */\nexport default class BoldEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'BoldEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\t// Allow bold attribute on text nodes.\n\t\teditor.model.schema.extend( '$text', { allowAttributes: BOLD } );\n\t\teditor.model.schema.setAttributeProperties( BOLD, {\n\t\t\tisFormatting: true,\n\t\t\tcopyOnEnter: true\n\t\t} );\n\n\t\t// Build converter from model to view for data and editing pipelines.\n\t\teditor.conversion.attributeToElement( {\n\t\t\tmodel: BOLD,\n\t\t\tview: 'strong',\n\t\t\tupcastAlso: [\n\t\t\t\t'b',\n\t\t\t\tviewElement => {\n\t\t\t\t\tconst fontWeight = viewElement.getStyle( 'font-weight' );\n\n\t\t\t\t\tif ( !fontWeight ) {\n\t\t\t\t\t\treturn null;\n\t\t\t\t\t}\n\n\t\t\t\t\t// Value of the `font-weight` attribute can be defined as a string or a number.\n\t\t\t\t\tif ( fontWeight == 'bold' || Number( fontWeight ) >= 600 ) {\n\t\t\t\t\t\treturn {\n\t\t\t\t\t\t\tname: true,\n\t\t\t\t\t\t\tstyles: [ 'font-weight' ]\n\t\t\t\t\t\t};\n\t\t\t\t\t}\n\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\n\t\t// Create bold command.\n\t\teditor.commands.add( BOLD, new AttributeCommand( editor, BOLD ) );\n\n\t\t// Set the Ctrl+B keystroke.\n\t\teditor.keystrokes.set( 'CTRL+B', BOLD );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module basic-styles/bold/boldui\n */\n\nimport { Plugin, icons } from 'ckeditor5/src/core';\nimport { ButtonView } from 'ckeditor5/src/ui';\nimport type AttributeCommand from '../attributecommand';\n\nconst BOLD = 'bold';\n\n/**\n * The bold UI feature. It introduces the Bold button.\n */\nexport default class BoldUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'BoldUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\n\t\t// Add bold button to feature components.\n\t\teditor.ui.componentFactory.add( BOLD, locale => {\n\t\t\tconst command: AttributeCommand = editor.commands.get( BOLD )!;\n\t\t\tconst view = new ButtonView( locale );\n\n\t\t\tview.set( {\n\t\t\t\tlabel: t( 'Bold' ),\n\t\t\t\ticon: icons.bold,\n\t\t\t\tkeystroke: 'CTRL+B',\n\t\t\t\ttooltip: true,\n\t\t\t\tisToggleable: true\n\t\t\t} );\n\n\t\t\tview.bind( 'isOn', 'isEnabled' ).to( command, 'value', 'isEnabled' );\n\n\t\t\t// Execute command.\n\t\t\tthis.listenTo( view, 'execute', () => {\n\t\t\t\teditor.execute( BOLD );\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn view;\n\t\t} );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./code.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module basic-styles/italic/italicediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport AttributeCommand from '../attributecommand';\n\nconst ITALIC = 'italic';\n\n/**\n * The italic editing feature.\n *\n * It registers the `'italic'` command, the <kbd>Ctrl+I</kbd> keystroke and introduces the `italic` attribute in the model\n * which renders to the view as an `<i>` element.\n */\nexport default class ItalicEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ItalicEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Allow italic attribute on text nodes.\n\t\teditor.model.schema.extend( '$text', { allowAttributes: ITALIC } );\n\t\teditor.model.schema.setAttributeProperties( ITALIC, {\n\t\t\tisFormatting: true,\n\t\t\tcopyOnEnter: true\n\t\t} );\n\n\t\teditor.conversion.attributeToElement( {\n\t\t\tmodel: ITALIC,\n\t\t\tview: 'i',\n\t\t\tupcastAlso: [\n\t\t\t\t'em',\n\t\t\t\t{\n\t\t\t\t\tstyles: {\n\t\t\t\t\t\t'font-style': 'italic'\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t]\n\t\t} );\n\n\t\t// Create italic command.\n\t\teditor.commands.add( ITALIC, new AttributeCommand( editor, ITALIC ) );\n\n\t\t// Set the Ctrl+I keystroke.\n\t\teditor.keystrokes.set( 'CTRL+I', ITALIC );\n\t}\n}\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m9.586 14.633.021.004c-.036.335.095.655.393.962.082.083.173.15.274.201h1.474a.6.6 0 1 1 0 1.2H5.304a.6.6 0 0 1 0-1.2h1.15c.474-.07.809-.182 1.005-.334.157-.122.291-.32.404-.597l2.416-9.55a1.053 1.053 0 0 0-.281-.823 1.12 1.12 0 0 0-.442-.296H8.15a.6.6 0 0 1 0-1.2h6.443a.6.6 0 1 1 0 1.2h-1.195c-.376.056-.65.155-.823.296-.215.175-.423.439-.623.79l-2.366 9.347z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module basic-styles/italic/italicui\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { ButtonView } from 'ckeditor5/src/ui';\nimport type AttributeCommand from '../attributecommand';\n\nimport italicIcon from '../../theme/icons/italic.svg';\n\nconst ITALIC = 'italic';\n\n/**\n * The italic UI feature. It introduces the Italic button.\n */\nexport default class ItalicUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ItalicUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\n\t\t// Add bold button to feature components.\n\t\teditor.ui.componentFactory.add( ITALIC, locale => {\n\t\t\tconst command: AttributeCommand = editor.commands.get( ITALIC )!;\n\t\t\tconst view = new ButtonView( locale );\n\n\t\t\tview.set( {\n\t\t\t\tlabel: t( 'Italic' ),\n\t\t\t\ticon: italicIcon,\n\t\t\t\tkeystroke: 'CTRL+I',\n\t\t\t\ttooltip: true,\n\t\t\t\tisToggleable: true\n\t\t\t} );\n\n\t\t\tview.bind( 'isOn', 'isEnabled' ).to( command, 'value', 'isEnabled' );\n\n\t\t\t// Execute command.\n\t\t\tthis.listenTo( view, 'execute', () => {\n\t\t\t\teditor.execute( ITALIC );\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn view;\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module block-quote/blockquotecommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport { first } from 'ckeditor5/src/utils';\nimport type { DocumentFragment, Element, Position, Range, Schema, Writer } from 'ckeditor5/src/engine';\n\n/**\n * The block quote command plugin.\n *\n * @extends module:core/command~Command\n */\nexport default class BlockQuoteCommand extends Command {\n\t/**\n\t * Whether the selection starts in a block quote.\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public value: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tthis.value = this._getValue();\n\t\tthis.isEnabled = this._checkEnabled();\n\t}\n\n\t/**\n\t * Executes the command. When the command {@link #value is on}, all top-most block quotes within\n\t * the selection will be removed. If it is off, all selected blocks will be wrapped with\n\t * a block quote.\n\t *\n\t * @fires execute\n\t * @param options Command options.\n\t * @param options.forceValue If set, it will force the command behavior. If `true`, the command will apply a block quote,\n\t * otherwise the command will remove the block quote. If not set, the command will act basing on its current value.\n\t */\n\tpublic override execute( options: { forceValue?: boolean } = {} ): void {\n\t\tconst model = this.editor.model;\n\t\tconst schema = model.schema;\n\t\tconst selection = model.document.selection;\n\n\t\tconst blocks = Array.from( selection.getSelectedBlocks() );\n\n\t\tconst value = ( options.forceValue === undefined ) ? !this.value : options.forceValue;\n\n\t\tmodel.change( writer => {\n\t\t\tif ( !value ) {\n\t\t\t\tthis._removeQuote( writer, blocks.filter( findQuote ) );\n\t\t\t} else {\n\t\t\t\tconst blocksToQuote = blocks.filter( block => {\n\t\t\t\t\t// Already quoted blocks needs to be considered while quoting too\n\t\t\t\t\t// in order to reuse their <bQ> elements.\n\t\t\t\t\treturn findQuote( block ) || checkCanBeQuoted( schema, block );\n\t\t\t\t} );\n\n\t\t\t\tthis._applyQuote( writer, blocksToQuote );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Checks the command's {@link #value}.\n\t */\n\tprivate _getValue(): boolean {\n\t\tconst selection = this.editor.model.document.selection;\n\n\t\tconst firstBlock = first( selection.getSelectedBlocks() );\n\n\t\t// In the current implementation, the block quote must be an immediate parent of a block element.\n\t\treturn !!( firstBlock && findQuote( firstBlock ) );\n\t}\n\n\t/**\n\t * Checks whether the command can be enabled in the current context.\n\t *\n\t * @returns Whether the command should be enabled.\n\t */\n\tprivate _checkEnabled(): boolean {\n\t\tif ( this.value ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tconst selection = this.editor.model.document.selection;\n\t\tconst schema = this.editor.model.schema;\n\n\t\tconst firstBlock = first( selection.getSelectedBlocks() );\n\n\t\tif ( !firstBlock ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn checkCanBeQuoted( schema, firstBlock );\n\t}\n\n\t/**\n\t * Removes the quote from given blocks.\n\t *\n\t * If blocks which are supposed to be \"unquoted\" are in the middle of a quote,\n\t * start it or end it, then the quote will be split (if needed) and the blocks\n\t * will be moved out of it, so other quoted blocks remained quoted.\n\t */\n\tprivate _removeQuote( writer: Writer, blocks: Array<Element> ): void {\n\t\t// Unquote all groups of block. Iterate in the reverse order to not break following ranges.\n\t\tgetRangesOfBlockGroups( writer, blocks ).reverse().forEach( groupRange => {\n\t\t\tif ( groupRange.start.isAtStart && groupRange.end.isAtEnd ) {\n\t\t\t\twriter.unwrap( groupRange.start.parent as Element );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// The group of blocks are at the beginning of an <bQ> so let's move them left (out of the <bQ>).\n\t\t\tif ( groupRange.start.isAtStart ) {\n\t\t\t\tconst positionBefore = writer.createPositionBefore( groupRange.start.parent as Element );\n\n\t\t\t\twriter.move( groupRange, positionBefore );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// The blocks are in the middle of an <bQ> so we need to split the <bQ> after the last block\n\t\t\t// so we move the items there.\n\t\t\tif ( !groupRange.end.isAtEnd ) {\n\t\t\t\twriter.split( groupRange.end );\n\t\t\t}\n\n\t\t\t// Now we are sure that groupRange.end.isAtEnd is true, so let's move the blocks right.\n\n\t\t\tconst positionAfter = writer.createPositionAfter( groupRange.end.parent as Element );\n\n\t\t\twriter.move( groupRange, positionAfter );\n\t\t} );\n\t}\n\n\t/**\n\t * Applies the quote to given blocks.\n\t */\n\tprivate _applyQuote( writer: Writer, blocks: Array<Element> ): void {\n\t\tconst quotesToMerge: Array<Element | DocumentFragment> = [];\n\n\t\t// Quote all groups of block. Iterate in the reverse order to not break following ranges.\n\t\tgetRangesOfBlockGroups( writer, blocks ).reverse().forEach( groupRange => {\n\t\t\tlet quote = findQuote( groupRange.start );\n\n\t\t\tif ( !quote ) {\n\t\t\t\tquote = writer.createElement( 'blockQuote' );\n\n\t\t\t\twriter.wrap( groupRange, quote );\n\t\t\t}\n\n\t\t\tquotesToMerge.push( quote );\n\t\t} );\n\n\t\t// Merge subsequent <bQ> elements. Reverse the order again because this time we want to go through\n\t\t// the <bQ> elements in the source order (due to how merge works it moves the right element's content\n\t\t// to the first element and removes the right one. Since we may need to merge a couple of subsequent `<bQ>` elements\n\t\t// we want to keep the reference to the first (furthest left) one.\n\t\tquotesToMerge.reverse().reduce( ( currentQuote, nextQuote ) => {\n\t\t\tif ( currentQuote.nextSibling == nextQuote ) {\n\t\t\t\twriter.merge( writer.createPositionAfter( currentQuote ) );\n\n\t\t\t\treturn currentQuote;\n\t\t\t}\n\n\t\t\treturn nextQuote;\n\t\t} );\n\t}\n}\n\nfunction findQuote( elementOrPosition: Element | Position ): Element | DocumentFragment | null {\n\treturn elementOrPosition.parent!.name == 'blockQuote' ? elementOrPosition.parent : null;\n}\n\n/**\n * Returns a minimal array of ranges containing groups of subsequent blocks.\n *\n * content: abcdefgh\n * blocks: [ a, b, d, f, g, h ]\n * output ranges: [ab]c[d]e[fgh]\n */\nfunction getRangesOfBlockGroups( writer: Writer, blocks: Array<Element> ): Array<Range> {\n\tlet startPosition;\n\tlet i = 0;\n\tconst ranges = [];\n\n\twhile ( i < blocks.length ) {\n\t\tconst block = blocks[ i ];\n\t\tconst nextBlock = blocks[ i + 1 ];\n\n\t\tif ( !startPosition ) {\n\t\t\tstartPosition = writer.createPositionBefore( block );\n\t\t}\n\n\t\tif ( !nextBlock || block.nextSibling != nextBlock ) {\n\t\t\tranges.push( writer.createRange( startPosition, writer.createPositionAfter( block ) ) );\n\t\t\tstartPosition = null;\n\t\t}\n\n\t\ti++;\n\t}\n\n\treturn ranges;\n}\n\n/**\n * Checks whether <bQ> can wrap the block.\n */\nfunction checkCanBeQuoted( schema: Schema, block: Element ): boolean {\n\t// TMP will be replaced with schema.checkWrap().\n\tconst isBQAllowed = schema.checkChild( block.parent as Element, 'blockQuote' );\n\tconst isBlockAllowedInBQ = schema.checkChild( [ '$root', 'blockQuote' ], block );\n\n\treturn isBQAllowed && isBlockAllowedInBQ;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module block-quote/blockquoteediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { Enter, type ViewDocumentEnterEvent } from 'ckeditor5/src/enter';\nimport { Delete, type ViewDocumentDeleteEvent } from 'ckeditor5/src/typing';\n\nimport BlockQuoteCommand from './blockquotecommand';\n\n/**\n * The block quote editing.\n *\n * Introduces the `'blockQuote'` command and the `'blockQuote'` model element.\n *\n * @extends module:core/plugin~Plugin\n */\nexport default class BlockQuoteEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'BlockQuoteEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Enter, Delete ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\n\t\teditor.commands.add( 'blockQuote', new BlockQuoteCommand( editor ) );\n\n\t\tschema.register( 'blockQuote', {\n\t\t\tinheritAllFrom: '$container'\n\t\t} );\n\n\t\teditor.conversion.elementToElement( { model: 'blockQuote', view: 'blockquote' } );\n\n\t\t// Postfixer which cleans incorrect model states connected with block quotes.\n\t\teditor.model.document.registerPostFixer( writer => {\n\t\t\tconst changes = editor.model.document.differ.getChanges();\n\n\t\t\tfor ( const entry of changes ) {\n\t\t\t\tif ( entry.type == 'insert' ) {\n\t\t\t\t\tconst element = entry.position.nodeAfter;\n\n\t\t\t\t\tif ( !element ) {\n\t\t\t\t\t\t// We are inside a text node.\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\tif ( element.is( 'element', 'blockQuote' ) && element.isEmpty ) {\n\t\t\t\t\t\t// Added an empty blockQuote - remove it.\n\t\t\t\t\t\twriter.remove( element );\n\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t} else if ( element.is( 'element', 'blockQuote' ) && !schema.checkChild( entry.position, element ) ) {\n\t\t\t\t\t\t// Added a blockQuote in incorrect place. Unwrap it so the content inside is not lost.\n\t\t\t\t\t\twriter.unwrap( element );\n\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t} else if ( element.is( 'element' ) ) {\n\t\t\t\t\t\t// Just added an element. Check that all children meet the scheme rules.\n\t\t\t\t\t\tconst range = writer.createRangeIn( element );\n\n\t\t\t\t\t\tfor ( const child of range.getItems() ) {\n\t\t\t\t\t\t\tif (\n\t\t\t\t\t\t\t\tchild.is( 'element', 'blockQuote' ) &&\n\t\t\t\t\t\t\t\t!schema.checkChild( writer.createPositionBefore( child ), child )\n\t\t\t\t\t\t\t) {\n\t\t\t\t\t\t\t\twriter.unwrap( child );\n\n\t\t\t\t\t\t\t\treturn true;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t} else if ( entry.type == 'remove' ) {\n\t\t\t\t\tconst parent = entry.position.parent;\n\n\t\t\t\t\tif ( parent.is( 'element', 'blockQuote' ) && parent.isEmpty ) {\n\t\t\t\t\t\t// Something got removed and now blockQuote is empty. Remove the blockQuote as well.\n\t\t\t\t\t\twriter.remove( parent );\n\n\t\t\t\t\t\treturn true;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\treturn false;\n\t\t} );\n\n\t\tconst viewDocument = this.editor.editing.view.document;\n\t\tconst selection = editor.model.document.selection;\n\t\tconst blockQuoteCommand: BlockQuoteCommand = editor.commands.get( 'blockQuote' )!;\n\n\t\t// Overwrite default Enter key behavior.\n\t\t// If Enter key is pressed with selection collapsed in empty block inside a quote, break the quote.\n\t\tthis.listenTo<ViewDocumentEnterEvent>( viewDocument, 'enter', ( evt, data ) => {\n\t\t\tif ( !selection.isCollapsed || !blockQuoteCommand.value ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst positionParent = selection.getLastPosition()!.parent;\n\n\t\t\tif ( positionParent.isEmpty ) {\n\t\t\t\teditor.execute( 'blockQuote' );\n\t\t\t\teditor.editing.view.scrollToTheSelection();\n\n\t\t\t\tdata.preventDefault();\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { context: 'blockquote' } );\n\n\t\t// Overwrite default Backspace key behavior.\n\t\t// If Backspace key is pressed with selection collapsed in first empty block inside a quote, break the quote.\n\t\tthis.listenTo<ViewDocumentDeleteEvent>( viewDocument, 'delete', ( evt, data ) => {\n\t\t\tif ( data.direction != 'backward' || !selection.isCollapsed || !blockQuoteCommand!.value ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst positionParent = selection.getLastPosition()!.parent;\n\n\t\t\tif ( positionParent.isEmpty && !positionParent.previousSibling ) {\n\t\t\t\teditor.execute( 'blockQuote' );\n\t\t\t\teditor.editing.view.scrollToTheSelection();\n\n\t\t\t\tdata.preventDefault();\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { context: 'blockquote' } );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./blockquote.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module block-quote/blockquoteui\n */\n\nimport { Plugin, icons } from 'ckeditor5/src/core';\nimport { ButtonView } from 'ckeditor5/src/ui';\nimport type BlockQuoteCommand from './blockquotecommand';\n\nimport '../theme/blockquote.css';\n\n/**\n * The block quote UI plugin.\n *\n * It introduces the `'blockQuote'` button.\n *\n * @extends module:core/plugin~Plugin\n */\nexport default class BlockQuoteUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'BlockQuoteUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\n\t\teditor.ui.componentFactory.add( 'blockQuote', locale => {\n\t\t\tconst command: BlockQuoteCommand = editor.commands.get( 'blockQuote' )!;\n\t\t\tconst buttonView = new ButtonView( locale );\n\n\t\t\tbuttonView.set( {\n\t\t\t\tlabel: t( 'Block quote' ),\n\t\t\t\ticon: icons.quote,\n\t\t\t\ttooltip: true,\n\t\t\t\tisToggleable: true\n\t\t\t} );\n\n\t\t\t// Bind button model to command.\n\t\t\tbuttonView.bind( 'isOn', 'isEnabled' ).to( command, 'value', 'isEnabled' );\n\n\t\t\t// Execute command.\n\t\t\tthis.listenTo( buttonView, 'execute', () => {\n\t\t\t\teditor.execute( 'blockQuote' );\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn buttonView;\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ckbox/ckboxui\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { ButtonView } from 'ckeditor5/src/ui';\n\nimport browseFilesIcon from '../theme/icons/browse-files.svg';\nimport type CKBoxCommand from './ckboxcommand';\n\n/**\n * The CKBoxUI plugin. It introduces the `'ckbox'` toolbar button.\n */\nexport default class CKBoxUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CKBoxUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\tconst editor = this.editor;\n\n\t\tconst command: CKBoxCommand | undefined = editor.commands.get( 'ckbox' );\n\n\t\t// Do not register the `ckbox` button if the command does not exist.\n\t\tif ( !command ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst t = editor.t;\n\t\tconst componentFactory = editor.ui.componentFactory;\n\n\t\tcomponentFactory.add( 'ckbox', locale => {\n\t\t\tconst button = new ButtonView( locale );\n\n\t\t\tbutton.set( {\n\t\t\t\tlabel: t( 'Open file manager' ),\n\t\t\t\ticon: browseFilesIcon,\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tbutton.bind( 'isOn', 'isEnabled' ).to( command, 'value', 'isEnabled' );\n\n\t\t\tbutton.on( 'execute', () => {\n\t\t\t\teditor.execute( 'ckbox' );\n\t\t\t} );\n\n\t\t\treturn button;\n\t\t} );\n\t}\n}\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M11.627 16.5zm5.873-.196zm0-7.001V8h-13v8.5h4.341c.191.54.457 1.044.785 1.5H2a1.5 1.5 0 0 1-1.5-1.5v-13A1.5 1.5 0 0 1 2 2h4.5a1.5 1.5 0 0 1 1.06.44L9.122 4H16a1.5 1.5 0 0 1 1.5 1.5v1A1.5 1.5 0 0 1 19 8v2.531a6.027 6.027 0 0 0-1.5-1.228zM16 6.5v-1H8.5l-2-2H2v13h1V8a1.5 1.5 0 0 1 1.5-1.5H16z\\\"/><path d=\\\"M14.5 19.5a5 5 0 1 1 0-10 5 5 0 0 1 0 10zM15 14v-2h-1v2h-2v1h2v2h1v-2h2v-1h-2z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* global atob */\n\n/**\n * @module ckbox/utils\n */\n\nimport type { InitializedToken } from '@ckeditor/ckeditor5-cloud-services';\nimport type { CKBoxImageUrls } from './ckboxconfig';\n\n/**\n * Converts image source set provided by the CKBox into an object containing:\n * - responsive URLs for the \"webp\" image format,\n * - one fallback URL for browsers that do not support the \"webp\" format.\n */\nexport function getImageUrls( imageUrls: CKBoxImageUrls ): {\n\timageFallbackUrl: string;\n\timageSources: Array<{\n\t\tsrcset: string;\n\t\tsizes: string;\n\t\ttype: string;\n\t}>;\n} {\n\tconst responsiveUrls: Array<string> = [];\n\tlet maxWidth = 0;\n\n\tfor ( const key in imageUrls ) {\n\t\tconst width = parseInt( key, 10 );\n\n\t\tif ( !isNaN( width ) ) {\n\t\t\tif ( width > maxWidth ) {\n\t\t\t\tmaxWidth = width;\n\t\t\t}\n\n\t\t\tresponsiveUrls.push( `${ imageUrls[ key ] } ${ key }w` );\n\t\t}\n\t}\n\n\tconst imageSources = [ {\n\t\tsrcset: responsiveUrls.join( ',' ),\n\t\tsizes: `(max-width: ${ maxWidth }px) 100vw, ${ maxWidth }px`,\n\t\ttype: 'image/webp'\n\t} ];\n\n\treturn {\n\t\timageFallbackUrl: imageUrls.default,\n\t\timageSources\n\t};\n}\n\n/**\n * Returns workspace id to use for communication with the CKBox service.\n *\n * @param defaultWorkspaceId The default workspace to use taken from editor config.\n */\nexport function getWorkspaceId( token: InitializedToken, defaultWorkspaceId?: string ): string | null {\n\tconst [ , binaryTokenPayload ] = token.value.split( '.' );\n\tconst payload = JSON.parse( atob( binaryTokenPayload ) );\n\tconst workspaces = ( payload.auth && payload.auth.ckbox && payload.auth.ckbox.workspaces ) || [ payload.aud ];\n\n\tif ( !defaultWorkspaceId ) {\n\t\treturn workspaces[ 0 ];\n\t}\n\n\tconst role = payload.auth && payload.auth.ckbox && payload.auth.ckbox.role;\n\n\tif ( role == 'superadmin' || workspaces.includes( defaultWorkspaceId ) ) {\n\t\treturn defaultWorkspaceId;\n\t}\n\n\treturn null;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* global document, window, setTimeout, URL */\n\n/**\n * @module ckbox/ckboxcommand\n */\n\nimport type { Writer } from 'ckeditor5/src/engine';\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport { createElement, toMap } from 'ckeditor5/src/utils';\n\nimport type {\n\tCKBoxAssetDefinition,\n\tCKBoxAssetImageAttributesDefinition,\n\tCKBoxAssetImageDefinition,\n\tCKBoxAssetLinkAttributesDefinition,\n\tCKBoxAssetLinkDefinition,\n\tCKBoxRawAssetDefinition\n} from './ckboxconfig';\n\nimport { getImageUrls } from './utils';\n\ndeclare global {\n\t// eslint-disable-next-line no-var\n\tvar CKBox: {\n\t\tmount( wrapper: Element, options: Record<string, unknown> ): void;\n\t};\n}\n\n// Defines the waiting time (in milliseconds) for inserting the chosen asset into the model. The chosen asset is temporarily stored in the\n// `CKBoxCommand#_chosenAssets` and it is removed from there automatically after this time. See `CKBoxCommand#_chosenAssets` for more\n// details.\nconst ASSET_INSERTION_WAIT_TIMEOUT = 1000;\n\n/**\n * The CKBox command. It is used by the {@link module:ckbox/ckboxediting~CKBoxEditing CKBox editing feature} to open the CKBox file manager.\n * The file manager allows inserting an image or a link to a file into the editor content.\n *\n * ```ts\n * editor.execute( 'ckbox' );\n * ```\n *\n * **Note:** This command uses other features to perform the following tasks:\n * - To insert images it uses the {@link module:image/image/insertimagecommand~InsertImageCommand 'insertImage'} command from the\n * {@link module:image/image~Image Image feature}.\n * - To insert links to other files it uses the {@link module:link/linkcommand~LinkCommand 'link'} command from the\n * {@link module:link/link~Link Link feature}.\n */\nexport default class CKBoxCommand extends Command {\n\tdeclare public value: boolean;\n\n\t/**\n\t * A set of all chosen assets. They are stored temporarily and they are automatically removed 1 second after being chosen.\n\t * Chosen assets have to be \"remembered\" for a while to be able to map the given asset with the element inserted into the model.\n\t * This association map is then used to set the ID on the model element.\n\t *\n\t * All chosen assets are automatically removed after the timeout, because (theoretically) it may happen that they will never be\n\t * inserted into the model, even if the {@link module:link/linkcommand~LinkCommand `'link'`} command or the\n\t * {@link module:image/image/insertimagecommand~InsertImageCommand `'insertImage'`} command is enabled. Such a case may arise when\n\t * another plugin blocks the command execution. Then, in order not to keep the chosen (but not inserted) assets forever, we delete\n\t * them automatically to prevent memory leakage. The 1 second timeout is enough to insert the asset into the model and extract the\n\t * ID from the chosen asset.\n\t *\n\t * The assets are stored only if\n\t * the {@link module:ckbox/ckboxconfig~CKBoxConfig#ignoreDataId `config.ckbox.ignoreDataId`} option is set to `false` (by default).\n\t *\n\t * @internal\n\t */\n\tpublic readonly _chosenAssets = new Set<CKBoxAssetDefinition>();\n\n\t/**\n\t * The DOM element that acts as a mounting point for the CKBox dialog.\n\t */\n\tprivate _wrapper: Element | null = null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis._initListeners();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tthis.value = this._getValue();\n\t\tthis.isEnabled = this._checkEnabled();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tthis.fire<CKBoxEvent<'open'>>( 'ckbox:open' );\n\t}\n\n\t/**\n\t * Indicates if the CKBox dialog is already opened.\n\t *\n\t * @protected\n\t * @returns {Boolean}\n\t */\n\tprivate _getValue(): boolean {\n\t\treturn this._wrapper !== null;\n\t}\n\n\t/**\n\t * Checks whether the command can be enabled in the current context.\n\t */\n\tprivate _checkEnabled() {\n\t\tconst imageCommand = this.editor.commands.get( 'insertImage' )!;\n\t\tconst linkCommand = this.editor.commands.get( 'link' )!;\n\n\t\tif ( !imageCommand.isEnabled && !linkCommand.isEnabled ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Creates the options object for the CKBox dialog.\n\t *\n\t * @returns The object with properties:\n\t * - theme The theme for CKBox dialog.\n\t * - language The language for CKBox dialog.\n\t * - tokenUrl The token endpoint URL.\n\t * - serviceOrigin The base URL of the API service.\n\t * - dialog.onClose The callback function invoked after closing the CKBox dialog.\n\t * - assets.onChoose The callback function invoked after choosing the assets.\n\t */\n\tprivate _prepareOptions() {\n\t\tconst editor = this.editor;\n\t\tconst ckboxConfig = editor.config.get( 'ckbox' )!;\n\n\t\treturn {\n\t\t\ttheme: ckboxConfig.theme,\n\t\t\tlanguage: ckboxConfig.language,\n\t\t\ttokenUrl: ckboxConfig.tokenUrl,\n\t\t\tserviceOrigin: ckboxConfig.serviceOrigin,\n\t\t\tdialog: {\n\t\t\t\tonClose: () => this.fire<CKBoxEvent<'close'>>( 'ckbox:close' )\n\t\t\t},\n\t\t\tassets: {\n\t\t\t\tonChoose: ( assets: Array<CKBoxRawAssetDefinition> ) => this.fire<CKBoxEvent<'choose'>>( 'ckbox:choose', assets )\n\t\t\t}\n\t\t};\n\t}\n\n\t/**\n\t * Initializes various event listeners for the `ckbox:*` events, because all functionality of the `ckbox` command is event-based.\n\t */\n\tprivate _initListeners() {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst shouldInsertDataId = !editor.config.get( 'ckbox.ignoreDataId' );\n\n\t\t// Refresh the command after firing the `ckbox:*` event.\n\t\tthis.on<CKBoxEvent>( 'ckbox', () => {\n\t\t\tthis.refresh();\n\t\t}, { priority: 'low' } );\n\n\t\t// Handle opening of the CKBox dialog.\n\t\tthis.on<CKBoxEvent<'open'>>( 'ckbox:open', () => {\n\t\t\tif ( !this.isEnabled || this.value ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._wrapper = createElement( document, 'div', { class: 'ck ckbox-wrapper' } );\n\t\t\tdocument.body.appendChild( this._wrapper );\n\n\t\t\twindow.CKBox.mount( this._wrapper, this._prepareOptions() );\n\t\t} );\n\n\t\t// Handle closing of the CKBox dialog.\n\t\tthis.on<CKBoxEvent<'close'>>( 'ckbox:close', () => {\n\t\t\tif ( !this.value ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis._wrapper!.remove();\n\t\t\tthis._wrapper = null;\n\t\t} );\n\n\t\t// Handle choosing the assets.\n\t\tthis.on<CKBoxEvent<'choose'>>( 'ckbox:choose', ( evt, assets ) => {\n\t\t\tif ( !this.isEnabled ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst imageCommand = editor.commands.get( 'insertImage' )!;\n\t\t\tconst linkCommand = editor.commands.get( 'link' )!;\n\n\t\t\tconst assetsToProcess = prepareAssets( {\n\t\t\t\tassets,\n\t\t\t\tisImageAllowed: imageCommand.isEnabled,\n\t\t\t\tisLinkAllowed: linkCommand.isEnabled\n\t\t\t} );\n\n\t\t\tif ( assetsToProcess.length === 0 ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// All assets are inserted in one undo step.\n\t\t\tmodel.change( writer => {\n\t\t\t\tfor ( const asset of assetsToProcess ) {\n\t\t\t\t\tconst isLastAsset = asset === assetsToProcess[ assetsToProcess.length - 1 ];\n\n\t\t\t\t\tthis._insertAsset( asset, isLastAsset, writer );\n\n\t\t\t\t\t// If asset ID must be set for the inserted model element, store the asset temporarily and remove it automatically\n\t\t\t\t\t// after the timeout.\n\t\t\t\t\tif ( shouldInsertDataId ) {\n\t\t\t\t\t\tsetTimeout( () => this._chosenAssets.delete( asset ), ASSET_INSERTION_WAIT_TIMEOUT );\n\n\t\t\t\t\t\tthis._chosenAssets.add( asset );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\t\t} );\n\n\t\t// Clean up after the editor is destroyed.\n\t\tthis.listenTo( editor, 'destroy', () => {\n\t\t\tthis.fire<CKBoxEvent<'close'>>( 'ckbox:close' );\n\t\t\tthis._chosenAssets.clear();\n\t\t} );\n\t}\n\n\t/**\n\t * Inserts the asset into the model.\n\t *\n\t * @param asset The asset to be inserted.\n\t * @param isLastAsset Indicates if the current asset is the last one from the chosen set.\n\t * @param writer An instance of the model writer.\n\t */\n\tprivate _insertAsset(\n\t\tasset: CKBoxAssetDefinition,\n\t\tisLastAsset: boolean,\n\t\twriter: Writer\n\t) {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst selection = model.document.selection;\n\n\t\t// Remove the `linkHref` attribute to not affect the asset to be inserted.\n\t\twriter.removeSelectionAttribute( 'linkHref' );\n\n\t\tif ( asset.type === 'image' ) {\n\t\t\tthis._insertImage( asset );\n\t\t} else {\n\t\t\tthis._insertLink( asset, writer );\n\t\t}\n\n\t\t// Except for the last chosen asset, move the selection to the end of the current range to avoid overwriting other, already\n\t\t// inserted assets.\n\t\tif ( !isLastAsset ) {\n\t\t\twriter.setSelection( selection.getLastPosition() );\n\t\t}\n\t}\n\n\t/**\n\t * Inserts the image by calling the `insertImage` command.\n\t *\n\t * @param asset The asset to be inserted.\n\t */\n\tprivate _insertImage( asset: CKBoxAssetImageDefinition ) {\n\t\tconst editor = this.editor;\n\t\tconst { imageFallbackUrl, imageSources, imageTextAlternative } = asset.attributes;\n\n\t\teditor.execute( 'insertImage', {\n\t\t\tsource: {\n\t\t\t\tsrc: imageFallbackUrl,\n\t\t\t\tsources: imageSources,\n\t\t\t\talt: imageTextAlternative\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Inserts the link to the asset by calling the `link` command.\n\t *\n\t * @param asset The asset to be inserted.\n\t * @param writer An instance of the model writer.\n\t */\n\tprivate _insertLink( asset: CKBoxAssetLinkDefinition, writer: Writer ) {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst { linkName, linkHref } = asset.attributes;\n\n\t\t// If the selection is collapsed, insert the asset name as the link label and select it.\n\t\tif ( selection.isCollapsed ) {\n\t\t\tconst selectionAttributes = toMap( selection.getAttributes() );\n\t\t\tconst textNode = writer.createText( linkName, selectionAttributes );\n\t\t\tconst range = model.insertContent( textNode );\n\n\t\t\twriter.setSelection( range );\n\t\t}\n\n\t\teditor.execute( 'link', linkHref );\n\t}\n}\n\n/**\n * Parses the chosen assets into the internal data format. Filters out chosen assets that are not allowed.\n */\nfunction prepareAssets(\n\t{ assets, isImageAllowed, isLinkAllowed }: {\n\t\tassets: Array<CKBoxRawAssetDefinition>;\n\t\tisImageAllowed: boolean;\n\t\tisLinkAllowed: boolean;\n\t}\n): Array<CKBoxAssetDefinition> {\n\treturn assets\n\t\t.map( asset => isImage( asset ) ?\n\t\t\t{\n\t\t\t\tid: asset.data.id,\n\t\t\t\ttype: 'image',\n\t\t\t\tattributes: prepareImageAssetAttributes( asset )\n\t\t\t} as const :\n\t\t\t{\n\t\t\t\tid: asset.data.id,\n\t\t\t\ttype: 'link',\n\t\t\t\tattributes: prepareLinkAssetAttributes( asset )\n\t\t\t} as const\n\t\t)\n\t\t.filter( asset => asset.type === 'image' ? isImageAllowed : isLinkAllowed );\n}\n\n/**\n * Parses the assets attributes into the internal data format.\n *\n * @param origin The base URL for assets inserted into the editor.\n */\nfunction prepareImageAssetAttributes( asset: CKBoxRawAssetDefinition ): CKBoxAssetImageAttributesDefinition {\n\tconst { imageFallbackUrl, imageSources } = getImageUrls( asset.data.imageUrls! );\n\n\treturn {\n\t\timageFallbackUrl,\n\t\timageSources,\n\t\timageTextAlternative: asset.data.metadata!.description || ''\n\t};\n}\n\n/**\n * Parses the assets attributes into the internal data format.\n *\n * @param origin The base URL for assets inserted into the editor.\n */\nfunction prepareLinkAssetAttributes( asset: CKBoxRawAssetDefinition ): CKBoxAssetLinkAttributesDefinition {\n\treturn {\n\t\tlinkName: asset.data.name,\n\t\tlinkHref: getAssetUrl( asset )\n\t};\n}\n\n/**\n * Checks whether the asset is an image.\n */\nfunction isImage( asset: CKBoxRawAssetDefinition ) {\n\tconst metadata = asset.data.metadata;\n\n\tif ( !metadata ) {\n\t\treturn false;\n\t}\n\n\treturn metadata.width && metadata.height;\n}\n\n/**\n * Creates the URL for the asset.\n *\n * @param origin The base URL for assets inserted into the editor.\n */\nfunction getAssetUrl( asset: CKBoxRawAssetDefinition ) {\n\tconst url = new URL( asset.data.url );\n\n\turl.searchParams.set( 'download', 'true' );\n\n\treturn url.toString();\n}\n\n/**\n * Fired when the command is executed, the dialog is closed or the assets are chosen.\n *\n * @eventName ~CKBoxCommand#ckbox\n */\ntype CKBoxEvent<Name extends '' | 'choose' | 'open' | 'close' = ''> = {\n\tname: Name extends '' ? 'ckbox' : `ckbox:${ Name }`;\n\targs: Name extends 'choose' ? [ assets: Array<CKBoxRawAssetDefinition> ] : [];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals AbortController, FormData, URL, XMLHttpRequest, window */\n\n/**\n * @module ckbox/ckboxuploadadapter\n */\n\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\nimport {\n\tFileRepository,\n\ttype FileLoader,\n\ttype UploadAdapter,\n\ttype UploadResponse\n} from 'ckeditor5/src/upload';\n\nimport type { InitializedToken } from '@ckeditor/ckeditor5-cloud-services';\nimport type { ImageUploadCompleteEvent, ImageUploadEditing } from '@ckeditor/ckeditor5-image';\n\nimport { logError } from 'ckeditor5/src/utils';\nimport CKBoxEditing from './ckboxediting';\nimport { getImageUrls, getWorkspaceId } from './utils';\n\n/**\n * A plugin that enables file uploads in CKEditor 5 using the CKBox serverside connector.\n * See the {@glink features/file-management/ckbox CKBox file manager integration} guide to learn how to configure\n * and use this feature as well as find out more about the full integration with the file manager\n * provided by the {@link module:ckbox/ckbox~CKBox} plugin.\n *\n * Check out the {@glink features/images/image-upload/image-upload Image upload overview} guide to learn about\n * other ways to upload images into CKEditor 5.\n */\nexport default class CKBoxUploadAdapter extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ 'ImageUploadEditing', 'ImageUploadProgress', FileRepository, CKBoxEditing ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CKBoxUploadAdapter' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic async afterInit(): Promise<void> {\n\t\tconst editor = this.editor;\n\n\t\tconst hasConfiguration = !!editor.config.get( 'ckbox' );\n\t\tconst isLibraryLoaded = !!window.CKBox;\n\n\t\t// Editor supports only one upload adapter. Register the CKBox upload adapter (and potentially overwrite other one) only when the\n\t\t// integrator intentionally wants to use the CKBox plugin, i.e. when the `config.ckbox` exists or the CKBox JavaScript library is\n\t\t// loaded.\n\t\tif ( !hasConfiguration && !isLibraryLoaded ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst fileRepository = editor.plugins.get( FileRepository );\n\t\tconst ckboxEditing = editor.plugins.get( CKBoxEditing );\n\n\t\tfileRepository.createUploadAdapter = loader => {\n\t\t\treturn new Adapter( loader, ckboxEditing.getToken(), editor );\n\t\t};\n\n\t\tconst shouldInsertDataId = !editor.config.get( 'ckbox.ignoreDataId' );\n\t\tconst imageUploadEditing: ImageUploadEditing = editor.plugins.get( 'ImageUploadEditing' );\n\n\t\t// Mark uploaded assets with the `ckboxImageId` attribute. Its value represents an ID in CKBox.\n\t\tif ( shouldInsertDataId ) {\n\t\t\timageUploadEditing.on<ImageUploadCompleteEvent>( 'uploadComplete', ( evt, { imageElement, data } ) => {\n\t\t\t\teditor.model.change( writer => {\n\t\t\t\t\twriter.setAttribute( 'ckboxImageId', data.ckboxImageId, imageElement );\n\t\t\t\t} );\n\t\t\t} );\n\t\t}\n\t}\n}\n\n/**\n * Upload adapter for CKBox.\n */\nclass Adapter implements UploadAdapter {\n\t/**\n\t * FileLoader instance to use during the upload.\n\t */\n\tpublic loader: FileLoader;\n\n\t/**\n\t * CKEditor Cloud Services access token.\n\t */\n\tpublic token: InitializedToken;\n\n\t/**\n\t * The editor instance.\n\t */\n\tpublic editor: Editor;\n\n\t/**\n\t * The abort controller for aborting asynchronous processes.\n\t */\n\tpublic controller: AbortController;\n\n\t/**\n\t * The base URL where all requests should be sent.\n\t */\n\tpublic serviceOrigin: string;\n\n\t/**\n\t * Creates a new adapter instance.\n\t */\n\tconstructor( loader: FileLoader, token: InitializedToken, editor: Editor ) {\n\t\tthis.loader = loader;\n\t\tthis.token = token;\n\t\tthis.editor = editor;\n\t\tthis.controller = new AbortController();\n\n\t\tthis.serviceOrigin = editor.config.get( 'ckbox.serviceOrigin' )!;\n\t}\n\n\t/**\n\t * The ID of workspace to use.\n\t */\n\tpublic getWorkspaceId(): string {\n\t\tconst t = this.editor.t;\n\t\tconst cannotAccessDefaultWorkspaceError = t( 'Cannot access default workspace.' );\n\t\tconst defaultWorkspaceId = this.editor.config.get( 'ckbox.defaultUploadWorkspaceId' );\n\t\tconst workspaceId = getWorkspaceId( this.token, defaultWorkspaceId );\n\n\t\tif ( workspaceId == null ) {\n\t\t\t/**\n\t\t\t * The user is not authorized to access workspace defined in `ckbox.defaultUploadWorkspaceId` configuration.\n\t\t\t *\n\t\t\t * @error ckbox-access-default-workspace-error\n\t\t\t */\n\t\t\tlogError( 'ckbox-access-default-workspace-error' );\n\n\t\t\tthrow cannotAccessDefaultWorkspaceError;\n\t\t}\n\n\t\treturn workspaceId;\n\t}\n\n\t/**\n\t * Resolves a promise with an array containing available categories with which the uploaded file can be associated.\n\t *\n\t * If the API returns limited results, the method will collect all items.\n\t */\n\tpublic async getAvailableCategories( offset: number = 0 ): Promise<Array<AvailableCategory>> {\n\t\tconst ITEMS_PER_REQUEST = 50;\n\t\tconst categoryUrl = new URL( 'categories', this.serviceOrigin );\n\n\t\tcategoryUrl.searchParams.set( 'limit', ITEMS_PER_REQUEST.toString() );\n\t\tcategoryUrl.searchParams.set( 'offset', offset.toString() );\n\t\tcategoryUrl.searchParams.set( 'workspaceId', this.getWorkspaceId() );\n\n\t\treturn this._sendHttpRequest( { url: categoryUrl } )\n\t\t\t.then( async data => {\n\t\t\t\tconst remainingItems = data.totalCount - ( offset + ITEMS_PER_REQUEST );\n\n\t\t\t\tif ( remainingItems > 0 ) {\n\t\t\t\t\tconst offsetItems = await this.getAvailableCategories( offset + ITEMS_PER_REQUEST );\n\n\t\t\t\t\treturn [\n\t\t\t\t\t\t...data.items,\n\t\t\t\t\t\t...offsetItems\n\t\t\t\t\t];\n\t\t\t\t}\n\n\t\t\t\treturn data.items;\n\t\t\t} )\n\t\t\t.catch( () => {\n\t\t\t\tthis.controller.signal.throwIfAborted();\n\n\t\t\t\t/**\n\t\t\t\t * Fetching a list of available categories with which an uploaded file can be associated failed.\n\t\t\t\t *\n\t\t\t\t * @error ckbox-fetch-category-http-error\n\t\t\t\t */\n\t\t\t\tlogError( 'ckbox-fetch-category-http-error' );\n\t\t\t} );\n\t}\n\n\t/**\n\t * Resolves a promise with an object containing a category with which the uploaded file is associated or an error code.\n\t */\n\tpublic async getCategoryIdForFile( file: File ): Promise<string | null> {\n\t\tconst extension = getFileExtension( file.name );\n\t\tconst allCategories = await this.getAvailableCategories();\n\n\t\t// Couldn't fetch all categories. Perhaps the authorization token is invalid.\n\t\tif ( !allCategories ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// The plugin allows defining to which category the uploaded file should be assigned.\n\t\tconst defaultCategories = this.editor.config.get( 'ckbox.defaultUploadCategories' );\n\n\t\t// If a user specifies the plugin configuration, find the first category that accepts the uploaded file.\n\t\tif ( defaultCategories ) {\n\t\t\tconst userCategory = Object.keys( defaultCategories ).find( category => {\n\t\t\t\treturn defaultCategories[ category ].find( e => e.toLowerCase() == extension );\n\t\t\t} );\n\n\t\t\t// If found, return its ID if the category exists on the server side.\n\t\t\tif ( userCategory ) {\n\t\t\t\tconst serverCategory = allCategories.find( category => category.id === userCategory || category.name === userCategory );\n\n\t\t\t\tif ( !serverCategory ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\treturn serverCategory.id;\n\t\t\t}\n\t\t}\n\n\t\t// Otherwise, find the first category that accepts the uploaded file and returns its ID.\n\t\tconst category = allCategories.find( category => category.extensions.find( e => e.toLowerCase() == extension ) );\n\n\t\tif ( !category ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn category.id;\n\t}\n\n\t/**\n\t * Starts the upload process.\n\t *\n\t * @see module:upload/filerepository~UploadAdapter#upload\n\t */\n\tpublic async upload(): Promise<UploadResponse> {\n\t\tconst t = this.editor.t;\n\t\tconst cannotFindCategoryError = t( 'Cannot determine a category for the uploaded file.' );\n\t\tconst file = ( await this.loader.file )!;\n\t\tconst category = await this.getCategoryIdForFile( file );\n\n\t\tif ( !category ) {\n\t\t\treturn Promise.reject( cannotFindCategoryError );\n\t\t}\n\n\t\tconst uploadUrl = new URL( 'assets', this.serviceOrigin );\n\t\tconst formData = new FormData();\n\n\t\tuploadUrl.searchParams.set( 'workspaceId', this.getWorkspaceId() );\n\n\t\tformData.append( 'categoryId', category );\n\t\tformData.append( 'file', file );\n\n\t\tconst requestConfig = {\n\t\t\tmethod: 'POST',\n\t\t\turl: uploadUrl,\n\t\t\tdata: formData,\n\t\t\tonUploadProgress: ( evt: ProgressEvent ) => {\n\t\t\t\t/* istanbul ignore else -- @preserve */\n\t\t\t\tif ( evt.lengthComputable ) {\n\t\t\t\t\tthis.loader.uploadTotal = evt.total;\n\t\t\t\t\tthis.loader.uploaded = evt.loaded;\n\t\t\t\t}\n\t\t\t}\n\t\t} as const;\n\n\t\treturn this._sendHttpRequest( requestConfig )\n\t\t\t.then( async data => {\n\t\t\t\tconst imageUrls = getImageUrls( data.imageUrls );\n\n\t\t\t\treturn {\n\t\t\t\t\tckboxImageId: data.id,\n\t\t\t\t\tdefault: imageUrls.imageFallbackUrl,\n\t\t\t\t\tsources: imageUrls.imageSources\n\t\t\t\t};\n\t\t\t} )\n\t\t\t.catch( () => {\n\t\t\t\tconst genericError = t( 'Cannot upload file:' ) + ` ${ file.name }.`;\n\n\t\t\t\treturn Promise.reject( genericError );\n\t\t\t} );\n\t}\n\n\t/**\n\t * Aborts the upload process.\n\t *\n\t * @see module:upload/filerepository~UploadAdapter#abort\n\t */\n\tpublic abort(): void {\n\t\tthis.controller.abort();\n\t}\n\n\t/**\n\t * Sends the HTTP request.\n\t *\n\t * @param config.url the URL where the request will be sent.\n\t * @param config.method The HTTP method.\n\t * @param config.data Additional data to send.\n\t * @param config.onUploadProgress A callback informing about the upload progress.\n\t */\n\tprivate _sendHttpRequest( { url, method = 'GET', data, onUploadProgress }: {\n\t\turl: URL;\n\t\tmethod?: 'GET' | 'POST';\n\t\tdata?: FormData | null;\n\t\tonUploadProgress?: ( evt: ProgressEvent ) => void;\n\t} ) {\n\t\tconst signal = this.controller.signal;\n\n\t\tconst xhr = new XMLHttpRequest();\n\t\txhr.open( method, url.toString(), true );\n\t\txhr.setRequestHeader( 'Authorization', this.token.value );\n\t\txhr.setRequestHeader( 'CKBox-Version', 'CKEditor 5' );\n\t\txhr.responseType = 'json';\n\n\t\t// The callback is attached to the `signal#abort` event.\n\t\tconst abortCallback = () => {\n\t\t\txhr.abort();\n\t\t};\n\n\t\treturn new Promise<any>( ( resolve, reject ) => {\n\t\t\tsignal.addEventListener( 'abort', abortCallback );\n\n\t\t\txhr.addEventListener( 'loadstart', () => {\n\t\t\t\tsignal.addEventListener( 'abort', abortCallback );\n\t\t\t} );\n\n\t\t\txhr.addEventListener( 'loadend', () => {\n\t\t\t\tsignal.removeEventListener( 'abort', abortCallback );\n\t\t\t} );\n\n\t\t\txhr.addEventListener( 'error', () => {\n\t\t\t\treject();\n\t\t\t} );\n\n\t\t\txhr.addEventListener( 'abort', () => {\n\t\t\t\treject();\n\t\t\t} );\n\n\t\t\txhr.addEventListener( 'load', async () => {\n\t\t\t\tconst response = xhr.response;\n\n\t\t\t\tif ( !response || response.statusCode >= 400 ) {\n\t\t\t\t\treturn reject( response && response.message );\n\t\t\t\t}\n\n\t\t\t\treturn resolve( response );\n\t\t\t} );\n\n\t\t\t/* istanbul ignore else -- @preserve */\n\t\t\tif ( onUploadProgress ) {\n\t\t\t\txhr.upload.addEventListener( 'progress', evt => {\n\t\t\t\t\tonUploadProgress( evt );\n\t\t\t\t} );\n\t\t\t}\n\n\t\t\t// Send the request.\n\t\t\txhr.send( data );\n\t\t} );\n\t}\n}\n\nexport interface AvailableCategory {\n\tid: string;\n\tname: string;\n\textensions: Array<string>;\n}\n\n/**\n * Returns an extension from the given value.\n */\nfunction getFileExtension( value: string ) {\n\tconst extensionRegExp = /\\.(?<ext>[^.]+)$/;\n\tconst match = value.match( extensionRegExp );\n\n\treturn match!.groups!.ext.toLowerCase();\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* globals window */\n\n/**\n * @module ckbox/ckboxediting\n */\n\nimport type { CloudServices, CloudServicesCore, InitializedToken } from '@ckeditor/ckeditor5-cloud-services';\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\nimport {\n\tRange,\n\ttype DocumentSelection,\n\ttype DowncastAttributeEvent,\n\ttype DowncastWriter,\n\ttype Element,\n\ttype Item,\n\ttype Node,\n\ttype UpcastElementEvent,\n\ttype ViewElement,\n\ttype Writer\n} from 'ckeditor5/src/engine';\nimport { CKEditorError, logError } from 'ckeditor5/src/utils';\n\nimport type { CKBoxAssetDefinition } from './ckboxconfig';\n\nimport CKBoxCommand from './ckboxcommand';\nimport CKBoxUploadAdapter from './ckboxuploadadapter';\n\n/**\n * The CKBox editing feature. It introduces the {@link module:ckbox/ckboxcommand~CKBoxCommand CKBox command} and\n * {@link module:ckbox/ckboxuploadadapter~CKBoxUploadAdapter CKBox upload adapter}.\n */\nexport default class CKBoxEditing extends Plugin {\n\t/**\n\t * CKEditor Cloud Services access token.\n\t */\n\tprivate _token!: InitializedToken;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CKBoxEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ 'CloudServices', 'LinkEditing', 'PictureEditing', CKBoxUploadAdapter ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic async init(): Promise<void> {\n\t\tconst editor = this.editor;\n\t\tconst hasConfiguration = !!editor.config.get( 'ckbox' );\n\t\tconst isLibraryLoaded = !!window.CKBox;\n\n\t\t// Proceed with plugin initialization only when the integrator intentionally wants to use it, i.e. when the `config.ckbox` exists or\n\t\t// the CKBox JavaScript library is loaded.\n\t\tif ( !hasConfiguration && !isLibraryLoaded ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._initConfig();\n\n\t\tconst cloudServicesCore: CloudServicesCore = editor.plugins.get( 'CloudServicesCore' );\n\t\tconst ckboxTokenUrl = editor.config.get( 'ckbox.tokenUrl' )!;\n\t\tconst cloudServicesTokenUrl = editor.config.get( 'cloudServices.tokenUrl' );\n\n\t\t// To avoid fetching the same token twice we need to compare the `ckbox.tokenUrl` and `cloudServices.tokenUrl` values.\n\t\t// If they are equal, it's enough to take the token generated by the `CloudServices` plugin.\n\t\tif ( ckboxTokenUrl === cloudServicesTokenUrl ) {\n\t\t\tconst cloudServices: CloudServices = editor.plugins.get( 'CloudServices' );\n\n\t\t\tthis._token = cloudServices.token!;\n\t\t}\n\t\t// Otherwise, create a new token manually.\n\t\telse {\n\t\t\tthis._token = await cloudServicesCore.createToken( ckboxTokenUrl ).init();\n\t\t}\n\n\t\t// Extending the schema, registering converters and applying fixers only make sense if the configuration option to assign\n\t\t// the assets ID with the model elements is enabled.\n\t\tif ( !editor.config.get( 'ckbox.ignoreDataId' ) ) {\n\t\t\tthis._initSchema();\n\t\t\tthis._initConversion();\n\t\t\tthis._initFixers();\n\t\t}\n\n\t\t// Registering the `ckbox` command makes sense only if the CKBox library is loaded, as the `ckbox` command opens the CKBox dialog.\n\t\tif ( isLibraryLoaded ) {\n\t\t\teditor.commands.add( 'ckbox', new CKBoxCommand( editor ) );\n\t\t}\n\t}\n\n\t/**\n\t * Returns a token used by the CKBox plugin for communication with the CKBox service.\n\t */\n\tpublic getToken(): InitializedToken {\n\t\treturn this._token;\n\t}\n\n\t/**\n\t * Initializes the `ckbox` editor configuration.\n\t */\n\tprivate _initConfig() {\n\t\tconst editor = this.editor;\n\n\t\teditor.config.define( 'ckbox', {\n\t\t\tserviceOrigin: 'https://api.ckbox.io',\n\t\t\tdefaultUploadCategories: null,\n\t\t\tignoreDataId: false,\n\t\t\tlanguage: editor.locale.uiLanguage,\n\t\t\ttheme: 'default',\n\t\t\ttokenUrl: editor.config.get( 'cloudServices.tokenUrl' )\n\t\t} );\n\n\t\tconst tokenUrl = editor.config.get( 'ckbox.tokenUrl' );\n\n\t\tif ( !tokenUrl ) {\n\t\t\t/**\n\t\t\t * The {@link module:ckbox/ckboxconfig~CKBoxConfig#tokenUrl `config.ckbox.tokenUrl`} or the\n\t\t\t * {@link module:cloud-services/cloudservicesconfig~CloudServicesConfig#tokenUrl `config.cloudServices.tokenUrl`}\n\t\t\t * configuration is required for the CKBox plugin.\n\t\t\t *\n\t\t\t * ```ts\n\t\t\t * ClassicEditor.create( document.createElement( 'div' ), {\n\t\t\t * \tckbox: {\n\t\t\t * \t\ttokenUrl: \"YOUR_TOKEN_URL\"\n\t\t\t * \t\t// ...\n\t\t\t * \t}\n\t\t\t * \t// ...\n\t\t\t * } );\n\t\t\t * ```\n\t\t\t *\n\t\t\t * @error ckbox-plugin-missing-token-url\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'ckbox-plugin-missing-token-url', this );\n\t\t}\n\n\t\tif ( !editor.plugins.has( 'ImageBlockEditing' ) && !editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\t\t/**\n\t\t\t * The CKBox feature requires one of the following plugins to be loaded to work correctly:\n\t\t\t *\n\t\t\t * * {@link module:image/imageblock~ImageBlock},\n\t\t\t * * {@link module:image/imageinline~ImageInline},\n\t\t\t * * {@link module:image/image~Image} (loads both `ImageBlock` and `ImageInline`)\n\t\t\t *\n\t\t\t * Please make sure your editor configuration is correct.\n\t\t\t *\n\t\t\t * @error ckbox-plugin-image-feature-missing\n\t\t\t * @param {module:core/editor/editor~Editor} editor\n\t\t\t */\n\t\t\tlogError( 'ckbox-plugin-image-feature-missing', editor );\n\t\t}\n\t}\n\n\t/**\n\t * Extends the schema to allow the `ckboxImageId` and `ckboxLinkId` attributes for links and images.\n\t */\n\tprivate _initSchema() {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\n\t\tschema.extend( '$text', { allowAttributes: 'ckboxLinkId' } );\n\n\t\tif ( schema.isRegistered( 'imageBlock' ) ) {\n\t\t\tschema.extend( 'imageBlock', { allowAttributes: [ 'ckboxImageId', 'ckboxLinkId' ] } );\n\t\t}\n\n\t\tif ( schema.isRegistered( 'imageInline' ) ) {\n\t\t\tschema.extend( 'imageInline', { allowAttributes: [ 'ckboxImageId', 'ckboxLinkId' ] } );\n\t\t}\n\n\t\tschema.addAttributeCheck( ( context, attributeName ) => {\n\t\t\tconst isLink = !!context.last.getAttribute( 'linkHref' );\n\n\t\t\tif ( !isLink && attributeName === 'ckboxLinkId' ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Configures the upcast and downcast conversions for the `ckboxImageId` and `ckboxLinkId` attributes.\n\t */\n\tprivate _initConversion() {\n\t\tconst editor = this.editor;\n\n\t\t// Convert `ckboxLinkId` => `data-ckbox-resource-id`.\n\t\teditor.conversion.for( 'downcast' ).add( dispatcher => {\n\t\t\t// Due to custom converters for linked block images, handle the `ckboxLinkId` attribute manually.\n\t\t\tdispatcher.on<DowncastAttributeEvent<Element>>( 'attribute:ckboxLinkId:imageBlock', ( evt, data, conversionApi ) => {\n\t\t\t\tconst { writer, mapper, consumable } = conversionApi;\n\n\t\t\t\tif ( !consumable.consume( data.item, evt.name ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst viewFigure = mapper.toViewElement( data.item );\n\t\t\t\tconst linkInImage = [ ...viewFigure!.getChildren() ]\n\t\t\t\t\t.find( ( child: any ) => child.name === 'a' ) as ViewElement | undefined;\n\n\t\t\t\t// No link inside an image - no conversion needed.\n\t\t\t\tif ( !linkInImage ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tif ( data.item.hasAttribute( 'ckboxLinkId' ) ) {\n\t\t\t\t\twriter.setAttribute( 'data-ckbox-resource-id', data.item.getAttribute( 'ckboxLinkId' ), linkInImage );\n\t\t\t\t} else {\n\t\t\t\t\twriter.removeAttribute( 'data-ckbox-resource-id', linkInImage );\n\t\t\t\t}\n\t\t\t}, { priority: 'low' } );\n\n\t\t\tdispatcher.on<DowncastAttributeEvent>( 'attribute:ckboxLinkId', ( evt, data, conversionApi ) => {\n\t\t\t\tconst { writer, mapper, consumable } = conversionApi;\n\n\t\t\t\tif ( !consumable.consume( data.item, evt.name ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// Remove the previous attribute value if it was applied.\n\t\t\t\tif ( data.attributeOldValue ) {\n\t\t\t\t\tconst viewElement = createLinkElement( writer, data.attributeOldValue as string );\n\n\t\t\t\t\twriter.unwrap( mapper.toViewRange( data.range ), viewElement );\n\t\t\t\t}\n\n\t\t\t\t// Add the new attribute value if specified in a model element.\n\t\t\t\tif ( data.attributeNewValue ) {\n\t\t\t\t\tconst viewElement = createLinkElement( writer, data.attributeNewValue as string );\n\n\t\t\t\t\tif ( data.item.is( 'selection' ) ) {\n\t\t\t\t\t\tconst viewSelection = writer.document.selection;\n\n\t\t\t\t\t\twriter.wrap( viewSelection.getFirstRange()!, viewElement );\n\t\t\t\t\t} else {\n\t\t\t\t\t\twriter.wrap( mapper.toViewRange( data.range ), viewElement );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}, { priority: 'low' } );\n\t\t} );\n\n\t\t// Convert `data-ckbox-resource-id` => `ckboxLinkId`.\n\t\t//\n\t\t// The helper conversion does not handle all cases, so take care of the `data-ckbox-resource-id` attribute manually for images\n\t\t// and links.\n\t\teditor.conversion.for( 'upcast' ).add( dispatcher => {\n\t\t\tdispatcher.on<UpcastElementEvent>( 'element:a', ( evt, data, conversionApi ) => {\n\t\t\t\tconst { writer, consumable } = conversionApi;\n\n\t\t\t\t// Upcast the `data-ckbox-resource-id` attribute only for valid link elements.\n\t\t\t\tif ( !data.viewItem.getAttribute( 'href' ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst consumableAttributes = { attributes: [ 'data-ckbox-resource-id' ] };\n\n\t\t\t\tif ( !consumable.consume( data.viewItem, consumableAttributes ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst attributeValue = data.viewItem.getAttribute( 'data-ckbox-resource-id' );\n\n\t\t\t\t// Missing the `data-ckbox-resource-id` attribute.\n\t\t\t\tif ( !attributeValue ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tif ( data.modelRange ) {\n\t\t\t\t\t// If the `<a>` element contains more than single children (e.g. a linked image), set the `ckboxLinkId` for each\n\t\t\t\t\t// allowed child.\n\t\t\t\t\tfor ( let item of data.modelRange.getItems() ) {\n\t\t\t\t\t\tif ( item.is( '$textProxy' ) ) {\n\t\t\t\t\t\t\titem = item.textNode;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Do not copy the `ckboxLinkId` attribute when wrapping an element in a block element, e.g. when\n\t\t\t\t\t\t// auto-paragraphing.\n\t\t\t\t\t\tif ( shouldUpcastAttributeForNode( item ) ) {\n\t\t\t\t\t\t\twriter.setAttribute( 'ckboxLinkId', attributeValue, item );\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\t// Otherwise, just set the `ckboxLinkId` for the model element.\n\t\t\t\t\tconst modelElement = data.modelCursor.nodeBefore || data.modelCursor.parent;\n\n\t\t\t\t\twriter.setAttribute( 'ckboxLinkId', attributeValue, modelElement as Element );\n\t\t\t\t}\n\t\t\t}, { priority: 'low' } );\n\t\t} );\n\n\t\t// Convert `ckboxImageId` => `data-ckbox-resource-id`.\n\t\teditor.conversion.for( 'downcast' ).attributeToAttribute( {\n\t\t\tmodel: 'ckboxImageId',\n\t\t\tview: 'data-ckbox-resource-id'\n\t\t} );\n\n\t\t// Convert `data-ckbox-resource-id` => `ckboxImageId`.\n\t\teditor.conversion.for( 'upcast' ).elementToAttribute( {\n\t\t\tmodel: {\n\t\t\t\tkey: 'ckboxImageId',\n\t\t\t\tvalue: ( viewElement: ViewElement ) => viewElement.getAttribute( 'data-ckbox-resource-id' )\n\t\t\t},\n\t\t\tview: {\n\t\t\t\tattributes: {\n\t\t\t\t\t'data-ckbox-resource-id': /[\\s\\S]+/\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Registers post-fixers that add or remove the `ckboxLinkId` and `ckboxImageId` attributes.\n\t */\n\tprivate _initFixers() {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst selection = model.document.selection;\n\n\t\t// Registers the post-fixer to sync the asset ID with the model elements.\n\t\tmodel.document.registerPostFixer( syncDataIdPostFixer( editor ) );\n\n\t\t// Registers the post-fixer to remove the `ckboxLinkId` attribute from the model selection.\n\t\tmodel.document.registerPostFixer( injectSelectionPostFixer( selection ) );\n\t}\n}\n\n/**\n * A post-fixer that synchronizes the asset ID with the model element.\n */\nfunction syncDataIdPostFixer( editor: Editor ) {\n\treturn ( writer: Writer ) => {\n\t\tlet changed = false;\n\n\t\tconst model = editor.model;\n\t\tconst ckboxCommand: CKBoxCommand = editor.commands.get( 'ckbox' )!;\n\n\t\t// The ID from chosen assets are stored in the `CKBoxCommand#_chosenAssets`. If there is no command, it makes no sense to check\n\t\t// for changes in the model.\n\t\tif ( !ckboxCommand ) {\n\t\t\treturn changed;\n\t\t}\n\n\t\tfor ( const entry of model.document.differ.getChanges() ) {\n\t\t\tif ( entry.type !== 'insert' && entry.type !== 'attribute' ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tconst range = entry.type === 'insert' ?\n\t\t\t\tnew Range( entry.position, entry.position.getShiftedBy( entry.length ) ) :\n\t\t\t\tentry.range;\n\n\t\t\tconst isLinkHrefAttributeRemoval = entry.type === 'attribute' &&\n\t\t\t\tentry.attributeKey === 'linkHref' &&\n\t\t\t\tentry.attributeNewValue === null;\n\n\t\t\tfor ( const item of range.getItems() ) {\n\t\t\t\t// If the `linkHref` attribute has been removed, sync the change with the `ckboxLinkId` attribute.\n\t\t\t\tif ( isLinkHrefAttributeRemoval && item.hasAttribute( 'ckboxLinkId' ) ) {\n\t\t\t\t\twriter.removeAttribute( 'ckboxLinkId', item );\n\n\t\t\t\t\tchanged = true;\n\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\t// Otherwise, the change concerns either a new model element or an attribute change. Try to find the assets for the modified\n\t\t\t\t// model element.\n\t\t\t\tconst assets = findAssetsForItem( item, ckboxCommand._chosenAssets );\n\n\t\t\t\tfor ( const asset of assets ) {\n\t\t\t\t\tconst attributeName = asset.type === 'image' ? 'ckboxImageId' : 'ckboxLinkId';\n\n\t\t\t\t\tif ( asset.id === item.getAttribute( attributeName ) ) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\twriter.setAttribute( attributeName, asset.id, item );\n\n\t\t\t\t\tchanged = true;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn changed;\n\t};\n}\n\n/**\n * A post-fixer that removes the `ckboxLinkId` from the selection if it does not represent a link anymore.\n */\nfunction injectSelectionPostFixer( selection: DocumentSelection ) {\n\treturn ( writer: Writer ) => {\n\t\tconst shouldRemoveLinkIdAttribute = !selection.hasAttribute( 'linkHref' ) && selection.hasAttribute( 'ckboxLinkId' );\n\n\t\tif ( shouldRemoveLinkIdAttribute ) {\n\t\t\twriter.removeSelectionAttribute( 'ckboxLinkId' );\n\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t};\n}\n\n/**\n * Tries to find the asset that is associated with the model element by comparing the attributes:\n * - the image fallback URL with the `src` attribute for images,\n * - the link URL with the `href` attribute for links.\n *\n * For any model element, zero, one or more than one asset can be found (e.g. a linked image may be associated with the link asset and the\n * image asset).\n */\nfunction findAssetsForItem( item: Item, assets: Set<CKBoxAssetDefinition> ) {\n\tconst isImageElement = item.is( 'element', 'imageInline' ) || item.is( 'element', 'imageBlock' );\n\tconst isLinkElement = item.hasAttribute( 'linkHref' );\n\n\treturn [ ...assets ].filter( asset => {\n\t\tif ( asset.type === 'image' && isImageElement ) {\n\t\t\treturn asset.attributes.imageFallbackUrl === item.getAttribute( 'src' );\n\t\t}\n\n\t\tif ( asset.type === 'link' && isLinkElement ) {\n\t\t\treturn asset.attributes.linkHref === item.getAttribute( 'linkHref' );\n\t\t}\n\t} );\n}\n\n/**\n * Creates view link element with the requested ID.\n */\nfunction createLinkElement( writer: DowncastWriter, id: string ) {\n\t// Priority equal 5 is needed to merge adjacent `<a>` elements together.\n\tconst viewElement = writer.createAttributeElement( 'a', { 'data-ckbox-resource-id': id }, { priority: 5 } );\n\n\twriter.setCustomProperty( 'link', true, viewElement );\n\n\treturn viewElement;\n}\n\n/**\n * Checks if the model element may have the `ckboxLinkId` attribute.\n */\nfunction shouldUpcastAttributeForNode( node: Node ) {\n\tif ( node.is( '$text' ) ) {\n\t\treturn true;\n\t}\n\n\tif ( node.is( 'element', 'imageInline' ) || node.is( 'element', 'imageBlock' ) ) {\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ckfinder/ckfinderui\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { ButtonView } from 'ckeditor5/src/ui';\n\nimport type CKFinderCommand from './ckfindercommand';\n\nimport browseFilesIcon from '../theme/icons/browse-files.svg';\n\n/**\n * The CKFinder UI plugin. It introduces the `'ckfinder'` toolbar button.\n */\nexport default class CKFinderUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CKFinderUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst componentFactory = editor.ui.componentFactory;\n\t\tconst t = editor.t;\n\n\t\tcomponentFactory.add( 'ckfinder', locale => {\n\t\t\tconst command: CKFinderCommand = editor.commands.get( 'ckfinder' )!;\n\n\t\t\tconst button = new ButtonView( locale );\n\n\t\t\tbutton.set( {\n\t\t\t\tlabel: t( 'Insert image or file' ),\n\t\t\t\ticon: browseFilesIcon,\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tbutton.bind( 'isEnabled' ).to( command );\n\n\t\t\tbutton.on( 'execute', () => {\n\t\t\t\teditor.execute( 'ckfinder' );\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn button;\n\t\t} );\n\t}\n}\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M11.627 16.5zm5.873-.196zm0-7.001V8h-13v8.5h4.341c.191.54.457 1.044.785 1.5H2a1.5 1.5 0 0 1-1.5-1.5v-13A1.5 1.5 0 0 1 2 2h4.5a1.5 1.5 0 0 1 1.06.44L9.122 4H16a1.5 1.5 0 0 1 1.5 1.5v1A1.5 1.5 0 0 1 19 8v2.531a6.027 6.027 0 0 0-1.5-1.228zM16 6.5v-1H8.5l-2-2H2v13h1V8a1.5 1.5 0 0 1 1.5-1.5H16z\\\"/><path d=\\\"M14.5 19.5a5 5 0 1 1 0-10 5 5 0 0 1 0 10zM15 14v-2h-1v2h-2v1h2v2h1v-2h2v-1h-2z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/* global window */\n\n/**\n * @module ckfinder/ckfindercommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport { CKEditorError } from 'ckeditor5/src/utils';\nimport type { Notification } from 'ckeditor5/src/ui';\n\n/**\n * The CKFinder command. It is used by the {@link module:ckfinder/ckfinderediting~CKFinderEditing CKFinder editing feature}\n * to open the CKFinder file manager to insert an image or a link to a file into the editor content.\n *\n * ```ts\n * editor.execute( 'ckfinder' );\n * ```\n *\n * **Note:** This command uses other features to perform tasks:\n * - To insert images the {@link module:image/image/insertimagecommand~InsertImageCommand 'insertImage'} command\n * from the {@link module:image/image~Image Image feature}.\n * - To insert links to files the {@link module:link/linkcommand~LinkCommand 'link'} command\n * from the {@link module:link/link~Link Link feature}.\n */\nexport default class CKFinderCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\t// The CKFinder command does not affect data by itself.\n\t\tthis.affectsData = false;\n\n\t\t// Remove default document listener to lower its priority.\n\t\tthis.stopListening( this.editor.model.document, 'change' );\n\n\t\t// Lower this command listener priority to be sure that refresh() will be called after link & image refresh.\n\t\tthis.listenTo( this.editor.model.document, 'change', () => this.refresh(), { priority: 'low' } );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst imageCommand = this.editor.commands.get( 'insertImage' )!;\n\t\tconst linkCommand = this.editor.commands.get( 'link' )!;\n\n\t\t// The CKFinder command is enabled when one of image or link command is enabled.\n\t\tthis.isEnabled = imageCommand.isEnabled || linkCommand.isEnabled;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tconst editor = this.editor;\n\n\t\tconst openerMethod = this.editor.config.get( 'ckfinder.openerMethod' ) || 'modal';\n\n\t\tif ( openerMethod != 'popup' && openerMethod != 'modal' ) {\n\t\t\t/**\n\t\t\t * The `ckfinder.openerMethod` must be one of: \"popup\" or \"modal\".\n\t\t\t *\n\t\t\t * @error ckfinder-unknown-openermethod\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'ckfinder-unknown-openermethod', editor );\n\t\t}\n\n\t\tconst options = this.editor.config.get( 'ckfinder.options' ) || {};\n\n\t\toptions.chooseFiles = true;\n\n\t\t// Cache the user-defined onInit method\n\t\tconst originalOnInit = options.onInit;\n\n\t\t// Pass the lang code to the CKFinder if not defined by user.\n\t\tif ( !options.language ) {\n\t\t\toptions.language = editor.locale.uiLanguage;\n\t\t}\n\n\t\t// The onInit method allows to extend CKFinder's behavior. It is used to attach event listeners to file choosing related events.\n\t\toptions.onInit = finder => {\n\t\t\t// Call original options.onInit if it was defined by user.\n\t\t\tif ( originalOnInit ) {\n\t\t\t\toriginalOnInit( finder );\n\t\t\t}\n\n\t\t\tfinder.on( 'files:choose', ( evt: any ) => {\n\t\t\t\tconst files = evt.data.files.toArray();\n\n\t\t\t\t// Insert links\n\t\t\t\tconst links = files.filter( ( file: any ) => !file.isImage() );\n\t\t\t\tconst images = files.filter( ( file: any ) => file.isImage() );\n\n\t\t\t\tfor ( const linkFile of links ) {\n\t\t\t\t\teditor.execute( 'link', linkFile.getUrl() );\n\t\t\t\t}\n\n\t\t\t\tconst imagesUrls = [];\n\n\t\t\t\tfor ( const image of images ) {\n\t\t\t\t\tconst url = image.getUrl();\n\n\t\t\t\t\timagesUrls.push( url ? url : finder.request( 'file:getProxyUrl', { file: image } ) );\n\t\t\t\t}\n\n\t\t\t\tif ( imagesUrls.length ) {\n\t\t\t\t\tinsertImages( editor, imagesUrls );\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tfinder.on( 'file:choose:resizedImage', ( evt: any ) => {\n\t\t\t\tconst resizedUrl = evt.data.resizedUrl;\n\n\t\t\t\tif ( !resizedUrl ) {\n\t\t\t\t\tconst notification: Notification = editor.plugins.get( 'Notification' );\n\t\t\t\t\tconst t = editor.locale.t;\n\n\t\t\t\t\tnotification.showWarning( t( 'Could not obtain resized image URL.' ), {\n\t\t\t\t\t\ttitle: t( 'Selecting resized image failed' ),\n\t\t\t\t\t\tnamespace: 'ckfinder'\n\t\t\t\t\t} );\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tinsertImages( editor, [ resizedUrl ] );\n\t\t\t} );\n\t\t};\n\n\t\t( window as any ).CKFinder[ openerMethod ]( options );\n\t}\n}\n\nfunction insertImages( editor: Editor, urls: Array<string> ): void {\n\tconst imageCommand = editor.commands.get( 'insertImage' )!;\n\n\t// Check if inserting an image is actually possible - it might be possible to only insert a link.\n\tif ( !imageCommand.isEnabled ) {\n\t\tconst notification: Notification = editor.plugins.get( 'Notification' );\n\t\tconst t = editor.locale.t;\n\n\t\tnotification.showWarning( t( 'Could not insert image at the current position.' ), {\n\t\t\ttitle: t( 'Inserting image failed' ),\n\t\t\tnamespace: 'ckfinder'\n\t\t} );\n\n\t\treturn;\n\t}\n\n\teditor.execute( 'insertImage', { source: urls } );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ckfinder/ckfinderediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { Notification } from 'ckeditor5/src/ui';\nimport { CKEditorError } from 'ckeditor5/src/utils';\n\nimport CKFinderCommand from './ckfindercommand';\n\n/**\n * The CKFinder editing feature. It introduces the {@link module:ckfinder/ckfindercommand~CKFinderCommand CKFinder command}.\n */\nexport default class CKFinderEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CKFinderEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Notification, 'LinkEditing' ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\tif ( !editor.plugins.has( 'ImageBlockEditing' ) && !editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\t\t/**\n\t\t\t * CKFinder requires at least one plugin providing support for images loaded in the editor. Please\n\t\t\t * make sure either:\n\t\t\t *\n\t\t\t * * {@link module:image/image~Image} (which loads both types of images),\n\t\t\t * * or {@link module:image/imageblock~ImageBlock},\n\t\t\t * * or {@link module:image/imageinline~ImageInline}.\n\t\t\t *\n\t\t\t * is loaded in your editor configuration.\n\t\t\t *\n\t\t\t * @error ckfinder-missing-image-plugin\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'ckfinder-missing-image-plugin', editor );\n\t\t}\n\n\t\teditor.commands.add( 'ckfinder', new CKFinderCommand( editor ) );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n* @module easy-image/cloudservicesuploadadapter\n*/\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { FileRepository, type FileLoader, type UploadAdapter } from 'ckeditor5/src/upload';\nimport type { CloudServicesCore, CloudServices, UploadGateway, FileUploader } from '@ckeditor/ckeditor5-cloud-services';\n\n/**\n * A plugin that enables upload to [CKEditor Cloud Services](https://ckeditor.com/ckeditor-cloud-services/).\n *\n * It is mainly used by the {@link module:easy-image/easyimage~EasyImage} feature.\n *\n * After enabling this adapter you need to configure the CKEditor Cloud Services integration through\n * {@link module:cloud-services/cloudservicesconfig~CloudServicesConfig `config.cloudServices`}.\n */\nexport default class CloudServicesUploadAdapter extends Plugin {\n\tprivate _uploadGateway?: UploadGateway;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CloudServicesUploadAdapter' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ 'CloudServices', FileRepository ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\tconst cloudServices: CloudServices = editor.plugins.get( 'CloudServices' );\n\n\t\tconst token = cloudServices.token;\n\t\tconst uploadUrl = cloudServices.uploadUrl;\n\n\t\tif ( !token ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst cloudServicesCore: CloudServicesCore = editor.plugins.get( 'CloudServicesCore' );\n\t\tthis._uploadGateway = cloudServicesCore.createUploadGateway( token, uploadUrl! );\n\n\t\teditor.plugins.get( FileRepository ).createUploadAdapter = loader => {\n\t\t\treturn new Adapter( this._uploadGateway!, loader );\n\t\t};\n\t}\n}\n\nclass Adapter implements UploadAdapter {\n\tprivate uploadGateway: UploadGateway;\n\tprivate loader: FileLoader;\n\tprivate fileUploader?: FileUploader;\n\n\tconstructor( uploadGateway: UploadGateway, loader: FileLoader ) {\n\t\tthis.uploadGateway = uploadGateway;\n\n\t\tthis.loader = loader;\n\t}\n\n\tpublic upload() {\n\t\treturn this.loader.file.then( file => {\n\t\t\tthis.fileUploader = this.uploadGateway.upload( file! );\n\n\t\t\tthis.fileUploader.on( 'progress', ( evt, data ) => {\n\t\t\t\tthis.loader.uploadTotal = data.total;\n\t\t\t\tthis.loader.uploaded = data.uploaded;\n\t\t\t} );\n\n\t\t\treturn this.fileUploader.send();\n\t\t} );\n\t}\n\n\tpublic abort() {\n\t\tthis.fileUploader!.abort();\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paragraph/paragraphcommand\n */\n\nimport { Command, type Editor } from '@ckeditor/ckeditor5-core';\nimport { first } from '@ckeditor/ckeditor5-utils';\n\nimport type { Schema, Selection, DocumentSelection, Element } from '@ckeditor/ckeditor5-engine';\n\n/**\n * The paragraph command.\n */\nexport default class ParagraphCommand extends Command {\n\tpublic constructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\t// Since this command may pass selection in execution block, it should be checked directly.\n\t\tthis._isEnabledBasedOnSelection = false;\n\t}\n\n\t/**\n\t * The value of the command. Indicates whether the selection start is placed in a paragraph.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public value: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst model = this.editor.model;\n\t\tconst document = model.document;\n\t\tconst block = first( document.selection.getSelectedBlocks() );\n\n\t\tthis.value = !!block && block.is( 'element', 'paragraph' );\n\t\tthis.isEnabled = !!block && checkCanBecomeParagraph( block, model.schema );\n\t}\n\n\t/**\n\t * Executes the command. All the blocks (see {@link module:engine/model/schema~Schema}) in the selection\n\t * will be turned to paragraphs.\n\t *\n\t * @fires execute\n\t * @param options Options for the executed command.\n\t * @param options.selection The selection that the command should be applied to. By default,\n\t * if not provided, the command is applied to the {@link module:engine/model/document~Document#selection}.\n\t */\n\tpublic override execute( options: {\n\t\tselection?: Selection | DocumentSelection;\n\t} = {} ): void {\n\t\tconst model = this.editor.model;\n\t\tconst document = model.document;\n\n\t\tconst selection = options.selection || document.selection;\n\n\t\t// Don't execute command if selection is in non-editable place.\n\t\tif ( !model.canEditAt( selection ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\tconst blocks = selection.getSelectedBlocks();\n\n\t\t\tfor ( const block of blocks ) {\n\t\t\t\tif ( !block.is( 'element', 'paragraph' ) && checkCanBecomeParagraph( block, model.schema ) ) {\n\t\t\t\t\twriter.rename( block, 'paragraph' );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n}\n\n/**\n * Checks whether the given block can be replaced by a paragraph.\n *\n * @param block A block to be tested.\n * @param schema The schema of the document.\n */\nfunction checkCanBecomeParagraph( block: Element, schema: Schema ): boolean {\n\treturn schema.checkChild( block.parent as Element, 'paragraph' ) && !schema.isObject( block );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paragraph/insertparagraphcommand\n */\n\nimport { Command, type Editor } from '@ckeditor/ckeditor5-core';\nimport type { Element, Position, Writer } from '@ckeditor/ckeditor5-engine';\n\n/**\n * The insert paragraph command. It inserts a new paragraph at a specific\n * {@link module:engine/model/position~Position document position}.\n *\n * ```ts\n * // Insert a new paragraph before an element in the document.\n * editor.execute( 'insertParagraph', {\n * position: editor.model.createPositionBefore( element )\n * } );\n * ```\n *\n * If a paragraph is disallowed in the context of the specific position, the command\n * will attempt to split position ancestors to find a place where it is possible\n * to insert a paragraph.\n *\n * **Note**: This command moves the selection to the inserted paragraph.\n */\nexport default class InsertParagraphCommand extends Command {\n\tpublic constructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\t// Since this command passes position in execution block instead of selection, it should be checked directly.\n\t\tthis._isEnabledBasedOnSelection = false;\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * @param options Options for the executed command.\n\t * @param options.position The model position at which the new paragraph will be inserted.\n\t * @param options.attributes Attributes keys and values to set on a inserted paragraph.\n\t * @fires execute\n\t */\n\tpublic override execute( options: {\n\t\tposition: Position;\n\t\tattributes: Record<string, unknown>;\n\t} ): void {\n\t\tconst model = this.editor.model;\n\t\tconst attributes = options.attributes;\n\n\t\tlet position: Position | null = options.position;\n\n\t\t// Don't execute command if position is in non-editable place.\n\t\tif ( !model.canEditAt( position ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\tposition = this._findPositionToInsertParagraph( position!, writer );\n\n\t\t\tif ( !position ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst paragraph = writer.createElement( 'paragraph' );\n\n\t\t\tif ( attributes ) {\n\t\t\t\tmodel.schema.setAllowedAttributes( paragraph, attributes, writer );\n\t\t\t}\n\n\t\t\tmodel.insertContent( paragraph, position );\n\t\t\twriter.setSelection( paragraph, 'in' );\n\t\t} );\n\t}\n\n\t/**\n\t * Returns the best position to insert a new paragraph.\n\t */\n\tprivate _findPositionToInsertParagraph( position: Position, writer: Writer ): Position | null {\n\t\tconst model = this.editor.model;\n\n\t\tif ( model.schema.checkChild( position, 'paragraph' ) ) {\n\t\t\treturn position;\n\t\t}\n\n\t\tconst allowedParent = model.schema.findAllowedParent( position, 'paragraph' );\n\n\t\t// It could be there's no ancestor limit that would allow paragraph.\n\t\t// In theory, \"paragraph\" could be disallowed even in the \"$root\".\n\t\tif ( !allowedParent ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst positionParent = position.parent as Element;\n\t\tconst isTextAllowed = model.schema.checkChild( positionParent, '$text' );\n\n\t\t// At empty $block or at the end of $block.\n\t\t// <paragraph>[]</paragraph> ---> <paragraph></paragraph><paragraph>[]</paragraph>\n\t\t// <paragraph>foo[]</paragraph> ---> <paragraph>foo</paragraph><paragraph>[]</paragraph>\n\t\tif ( positionParent.isEmpty || isTextAllowed && position.isAtEnd ) {\n\t\t\treturn model.createPositionAfter( positionParent );\n\t\t}\n\n\t\t// At the start of $block with text.\n\t\t// <paragraph>[]foo</paragraph> ---> <paragraph>[]</paragraph><paragraph>foo</paragraph>\n\t\tif ( !positionParent.isEmpty && isTextAllowed && position.isAtStart ) {\n\t\t\treturn model.createPositionBefore( positionParent );\n\t\t}\n\n\t\treturn writer.split( position, allowedParent ).position;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paragraph/paragraph\n */\n\nimport ParagraphCommand from './paragraphcommand';\nimport InsertParagraphCommand from './insertparagraphcommand';\n\nimport { Plugin } from '@ckeditor/ckeditor5-core';\n\n/**\n * The paragraph feature for the editor.\n *\n * It introduces the `<paragraph>` element in the model which renders as a `<p>` element in the DOM and data.\n *\n * It also brings two editors commands:\n *\n * * The {@link module:paragraph/paragraphcommand~ParagraphCommand `'paragraph'`} command that converts all\n * blocks in the model selection into paragraphs.\n * * The {@link module:paragraph/insertparagraphcommand~InsertParagraphCommand `'insertParagraph'`} command\n * that inserts a new paragraph at a specified location in the model.\n */\nexport default class Paragraph extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Paragraph' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\n\t\teditor.commands.add( 'paragraph', new ParagraphCommand( editor ) );\n\t\teditor.commands.add( 'insertParagraph', new InsertParagraphCommand( editor ) );\n\n\t\t// Schema.\n\t\tmodel.schema.register( 'paragraph', { inheritAllFrom: '$block' } );\n\n\t\teditor.conversion.elementToElement( { model: 'paragraph', view: 'p' } );\n\n\t\t// Conversion for paragraph-like elements which has not been converted by any plugin.\n\t\teditor.conversion.for( 'upcast' ).elementToElement( {\n\t\t\tmodel: ( viewElement, { writer } ) => {\n\t\t\t\tif ( !Paragraph.paragraphLikeElements.has( viewElement.name ) ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\t// Do not auto-paragraph empty elements.\n\t\t\t\tif ( viewElement.isEmpty ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\treturn writer.createElement( 'paragraph' );\n\t\t\t},\n\t\t\tview: /.+/,\n\t\t\tconverterPriority: 'low'\n\t\t} );\n\t}\n\n\t/**\n\t * A list of element names which should be treated by the autoparagraphing algorithms as\n\t * paragraph-like. This means that e.g. the following content:\n\t *\n\t * ```html\n\t * <h1>Foo</h1>\n\t * <table>\n\t * <tr>\n\t * <td>X</td>\n\t * <td>\n\t * <ul>\n\t * <li>Y</li>\n\t * <li>Z</li>\n\t * </ul>\n\t * </td>\n\t * </tr>\n\t * </table>\n\t * ```\n\t *\n\t * contains five paragraph-like elements: `<h1>`, two `<td>`s and two `<li>`s.\n\t * Hence, if none of the features is going to convert those elements the above content will be automatically handled\n\t * by the paragraph feature and converted to:\n\t *\n\t * ```html\n\t * <p>Foo</p>\n\t * <p>X</p>\n\t * <p>Y</p>\n\t * <p>Z</p>\n\t * ```\n\t *\n\t * Note: The `<td>` containing two `<li>` elements was ignored as the innermost paragraph-like elements\n\t * have a priority upon conversion.\n\t */\n\tpublic static paragraphLikeElements = new Set( [\n\t\t'blockquote',\n\t\t'dd',\n\t\t'div',\n\t\t'dt',\n\t\t'h1',\n\t\t'h2',\n\t\t'h3',\n\t\t'h4',\n\t\t'h5',\n\t\t'h6',\n\t\t'li',\n\t\t'p',\n\t\t'td',\n\t\t'th'\n\t] );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module heading/headingcommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport { first } from 'ckeditor5/src/utils';\n\nimport type { Element, Schema } from 'ckeditor5/src/engine';\n\n/**\n * The heading command. It is used by the {@link module:heading/heading~Heading heading feature} to apply headings.\n */\nexport default class HeadingCommand extends Command {\n\t/**\n\t * If the selection starts in a heading (which {@link #modelElements is supported by this command})\n\t * the value is set to the name of that heading model element.\n\t * It is set to `false` otherwise.\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public value: false | string;\n\n\t/**\n\t * Set of defined model's elements names that this command support.\n\t * See {@link module:heading/headingconfig~HeadingOption}.\n\t */\n\tpublic readonly modelElements: Array<string>;\n\n\t/**\n\t * Creates an instance of the command.\n\t *\n\t * @param editor Editor instance.\n\t * @param modelElements Names of the element which this command can apply in the model.\n\t */\n\tconstructor( editor: Editor, modelElements: Array<string> ) {\n\t\tsuper( editor );\n\n\t\tthis.modelElements = modelElements;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst block = first( this.editor.model.document.selection.getSelectedBlocks() );\n\n\t\tthis.value = !!block && this.modelElements.includes( block.name ) && block.name;\n\t\tthis.isEnabled = !!block && this.modelElements.some( heading => checkCanBecomeHeading( block, heading, this.editor.model.schema ) );\n\t}\n\n\t/**\n\t * Executes the command. Applies the heading to the selected blocks or, if the first selected\n\t * block is a heading already, turns selected headings (of this level only) to paragraphs.\n\t *\n\t * @param options.value Name of the element which this command will apply in the model.\n\t * @fires execute\n\t */\n\tpublic override execute( options: { value: string } ): void {\n\t\tconst model = this.editor.model;\n\t\tconst document = model.document;\n\n\t\tconst modelElement = options.value;\n\n\t\tmodel.change( writer => {\n\t\t\tconst blocks = Array.from( document.selection.getSelectedBlocks() )\n\t\t\t\t.filter( block => {\n\t\t\t\t\treturn checkCanBecomeHeading( block, modelElement, model.schema );\n\t\t\t\t} );\n\n\t\t\tfor ( const block of blocks ) {\n\t\t\t\tif ( !block.is( 'element', modelElement ) ) {\n\t\t\t\t\twriter.rename( block, modelElement );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n}\n\n/**\n * Checks whether the given block can be replaced by a specific heading.\n *\n * @param block A block to be tested.\n * @param heading Command element name in the model.\n * @param schema The schema of the document.\n */\nfunction checkCanBecomeHeading( block: Element, heading: string, schema: Schema ) {\n\treturn schema.checkChild( block.parent as Element, heading ) && !schema.isObject( block );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module heading/headingediting\n */\n\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\nimport { Paragraph } from 'ckeditor5/src/paragraph';\nimport { priorities } from 'ckeditor5/src/utils';\nimport type { EnterCommandAfterExecuteEvent } from 'ckeditor5/src/enter';\nimport type { HeadingOption } from './headingconfig';\n\nimport HeadingCommand from './headingcommand';\n\nconst defaultModelElement = 'paragraph';\n\n/**\n * The headings engine feature. It handles switching between block formats &ndash; headings and paragraph.\n * This class represents the engine part of the heading feature. See also {@link module:heading/heading~Heading}.\n * It introduces `heading1`-`headingN` commands which allow to convert paragraphs into headings.\n */\nexport default class HeadingEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'HeadingEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\teditor.config.define( 'heading', {\n\t\t\toptions: [\n\t\t\t\t{ model: 'paragraph', title: 'Paragraph', class: 'ck-heading_paragraph' },\n\t\t\t\t{ model: 'heading1', view: 'h2', title: 'Heading 1', class: 'ck-heading_heading1' },\n\t\t\t\t{ model: 'heading2', view: 'h3', title: 'Heading 2', class: 'ck-heading_heading2' },\n\t\t\t\t{ model: 'heading3', view: 'h4', title: 'Heading 3', class: 'ck-heading_heading3' }\n\t\t\t]\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Paragraph ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst options: Array<HeadingOption> = editor.config.get( 'heading.options' )!;\n\n\t\tconst modelElements = [];\n\n\t\tfor ( const option of options ) {\n\t\t\t// Skip paragraph - it is defined in required Paragraph feature.\n\t\t\tif ( option.model === 'paragraph' ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// Schema.\n\t\t\teditor.model.schema.register( option.model, {\n\t\t\t\tinheritAllFrom: '$block'\n\t\t\t} );\n\n\t\t\teditor.conversion.elementToElement( option );\n\n\t\t\tmodelElements.push( option.model );\n\t\t}\n\n\t\tthis._addDefaultH1Conversion( editor );\n\n\t\t// Register the heading command for this option.\n\t\teditor.commands.add( 'heading', new HeadingCommand( editor, modelElements ) );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\t// If the enter command is added to the editor, alter its behavior.\n\t\t// Enter at the end of a heading element should create a paragraph.\n\t\tconst editor = this.editor;\n\t\tconst enterCommand = editor.commands.get( 'enter' );\n\t\tconst options: Array<HeadingOption> = editor.config.get( 'heading.options' )!;\n\n\t\tif ( enterCommand ) {\n\t\t\tthis.listenTo<EnterCommandAfterExecuteEvent>( enterCommand, 'afterExecute', ( evt, data ) => {\n\t\t\t\tconst positionParent = editor.model.document.selection.getFirstPosition()!.parent;\n\t\t\t\tconst isHeading = options.some( option => positionParent.is( 'element', option.model ) );\n\n\t\t\t\tif ( isHeading && !positionParent.is( 'element', defaultModelElement ) && positionParent.childCount === 0 ) {\n\t\t\t\t\tdata.writer.rename( positionParent, defaultModelElement );\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Adds default conversion for `h1` -> `heading1` with a low priority.\n\t *\n\t * @param editor Editor instance on which to add the `h1` conversion.\n\t */\n\tprivate _addDefaultH1Conversion( editor: Editor ) {\n\t\teditor.conversion.for( 'upcast' ).elementToElement( {\n\t\t\tmodel: 'heading1',\n\t\t\tview: 'h1',\n\t\t\t// With a `low` priority, `paragraph` plugin autoparagraphing mechanism is executed. Make sure\n\t\t\t// this listener is called before it. If not, `h1` will be transformed into a paragraph.\n\t\t\tconverterPriority: priorities.low + 1\n\t\t} );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./heading.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module heading/headingui\n */\n\nimport { Plugin, type Command } from 'ckeditor5/src/core';\nimport {\n\tModel,\n\tcreateDropdown,\n\taddListToDropdown,\n\ttype ButtonExecuteEvent,\n\ttype ListDropdownItemDefinition\n} from 'ckeditor5/src/ui';\nimport { Collection } from 'ckeditor5/src/utils';\nimport type { ParagraphCommand } from 'ckeditor5/src/paragraph';\n\nimport { getLocalizedOptions } from './utils';\nimport type HeadingCommand from './headingcommand';\n\nimport '../theme/heading.css';\n\n/**\n * The headings UI feature. It introduces the `headings` dropdown.\n */\nexport default class HeadingUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'HeadingUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst options = getLocalizedOptions( editor );\n\t\tconst defaultTitle = t( 'Choose heading' );\n\t\tconst accessibleLabel = t( 'Heading' );\n\n\t\t// Register UI component.\n\t\teditor.ui.componentFactory.add( 'heading', locale => {\n\t\t\tconst titles: Record<string, string> = {};\n\t\t\tconst itemDefinitions: Collection<ListDropdownItemDefinition> = new Collection();\n\n\t\t\tconst headingCommand: HeadingCommand = editor.commands.get( 'heading' )!;\n\t\t\tconst paragraphCommand: ParagraphCommand = editor.commands.get( 'paragraph' )!;\n\n\t\t\tconst commands: Array<Command> = [ headingCommand ];\n\n\t\t\tfor ( const option of options ) {\n\t\t\t\tconst def: ListDropdownItemDefinition = {\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: new Model( {\n\t\t\t\t\t\tlabel: option.title,\n\t\t\t\t\t\tclass: option.class,\n\t\t\t\t\t\trole: 'menuitemradio',\n\t\t\t\t\t\twithText: true\n\t\t\t\t\t} )\n\t\t\t\t};\n\n\t\t\t\tif ( option.model === 'paragraph' ) {\n\t\t\t\t\tdef.model.bind( 'isOn' ).to( paragraphCommand, 'value' );\n\t\t\t\t\tdef.model.set( 'commandName', 'paragraph' );\n\t\t\t\t\tcommands.push( paragraphCommand );\n\t\t\t\t} else {\n\t\t\t\t\tdef.model.bind( 'isOn' ).to( headingCommand, 'value', value => value === option.model );\n\t\t\t\t\tdef.model.set( {\n\t\t\t\t\t\tcommandName: 'heading',\n\t\t\t\t\t\tcommandValue: option.model\n\t\t\t\t\t} );\n\t\t\t\t}\n\n\t\t\t\t// Add the option to the collection.\n\t\t\t\titemDefinitions.add( def );\n\n\t\t\t\ttitles[ option.model ] = option.title;\n\t\t\t}\n\n\t\t\tconst dropdownView = createDropdown( locale );\n\t\t\taddListToDropdown( dropdownView, itemDefinitions, {\n\t\t\t\tariaLabel: accessibleLabel,\n\t\t\t\trole: 'menu'\n\t\t\t} );\n\n\t\t\tdropdownView.buttonView.set( {\n\t\t\t\tariaLabel: accessibleLabel,\n\t\t\t\tariaLabelledBy: undefined,\n\t\t\t\tisOn: false,\n\t\t\t\twithText: true,\n\t\t\t\ttooltip: accessibleLabel\n\t\t\t} );\n\n\t\t\tdropdownView.extendTemplate( {\n\t\t\t\tattributes: {\n\t\t\t\t\tclass: [\n\t\t\t\t\t\t'ck-heading-dropdown'\n\t\t\t\t\t]\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tdropdownView.bind( 'isEnabled' ).toMany( commands, 'isEnabled', ( ...areEnabled ) => {\n\t\t\t\treturn areEnabled.some( isEnabled => isEnabled );\n\t\t\t} );\n\n\t\t\tdropdownView.buttonView.bind( 'label' ).to( headingCommand, 'value', paragraphCommand, 'value', ( value, para ) => {\n\t\t\t\tconst whichModel = value || para && 'paragraph';\n\n\t\t\t\tif ( typeof whichModel === 'boolean' ) {\n\t\t\t\t\treturn defaultTitle;\n\t\t\t\t}\n\n\t\t\t\t// If none of the commands is active, display default title.\n\t\t\t\tif ( !titles[ whichModel ] ) {\n\t\t\t\t\treturn defaultTitle;\n\t\t\t\t}\n\n\t\t\t\treturn titles[ whichModel ];\n\t\t\t} );\n\n\t\t\t// Execute command when an item from the dropdown is selected.\n\t\t\tthis.listenTo<ButtonExecuteEvent>( dropdownView, 'execute', evt => {\n\t\t\t\tconst { commandName, commandValue } = evt.source as any;\n\t\t\t\teditor.execute( commandName, commandValue ? { value: commandValue } : undefined );\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn dropdownView;\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module heading/utils\n */\n\nimport type { Editor } from 'ckeditor5/src/core';\n\nimport type { HeadingOption } from './headingconfig';\n\n/**\n * Returns heading options as defined in `config.heading.options` but processed to consider\n * the editor localization, i.e. to display {@link module:heading/headingconfig~HeadingOption}\n * in the correct language.\n *\n * Note: The reason behind this method is that there is no way to use {@link module:utils/locale~Locale#t}\n * when the user configuration is defined because the editor does not exist yet.\n */\nexport function getLocalizedOptions( editor: Editor ): Array<HeadingOption> {\n\tconst t = editor.t;\n\tconst localizedTitles: Record<string, string> = {\n\t\t'Paragraph': t( 'Paragraph' ),\n\t\t'Heading 1': t( 'Heading 1' ),\n\t\t'Heading 2': t( 'Heading 2' ),\n\t\t'Heading 3': t( 'Heading 3' ),\n\t\t'Heading 4': t( 'Heading 4' ),\n\t\t'Heading 5': t( 'Heading 5' ),\n\t\t'Heading 6': t( 'Heading 6' )\n\t};\n\n\treturn editor.config.get( 'heading.options' )!.map( option => {\n\t\tconst title = localizedTitles[ option.title ];\n\n\t\tif ( title && title != option.title ) {\n\t\t\toption.title = title;\n\t\t}\n\n\t\treturn option;\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/utils\n */\n\nimport type {\n\tDocumentSelection,\n\tMatcherPattern,\n\tSchema,\n\tSelection,\n\tViewContainerElement,\n\tDowncastWriter,\n\tViewElement\n} from 'ckeditor5/src/engine';\nimport type { Editor } from 'ckeditor5/src/core';\nimport { first } from 'ckeditor5/src/utils';\n\nimport type ImageUtils from '../imageutils';\n\n/**\n * Creates a view element representing the inline image.\n *\n * ```html\n * <span class=\"image-inline\"><img></img></span>\n * ```\n *\n * Note that `alt` and `src` attributes are converted separately, so they are not included.\n *\n * @internal\n */\nexport function createInlineImageViewElement( writer: DowncastWriter ): ViewContainerElement {\n\treturn writer.createContainerElement( 'span', { class: 'image-inline' },\n\t\twriter.createEmptyElement( 'img' )\n\t);\n}\n\n/**\n * Creates a view element representing the block image.\n *\n * ```html\n * <figure class=\"image\"><img></img></figure>\n * ```\n *\n * Note that `alt` and `src` attributes are converted separately, so they are not included.\n *\n * @internal\n */\nexport function createBlockImageViewElement( writer: DowncastWriter ): ViewContainerElement {\n\treturn writer.createContainerElement( 'figure', { class: 'image' }, [\n\t\twriter.createEmptyElement( 'img' ),\n\t\twriter.createSlot( 'children' )\n\t] );\n}\n\n/**\n * A function returning a `MatcherPattern` for a particular type of View images.\n *\n * @internal\n * @param matchImageType The type of created image.\n */\nexport function getImgViewElementMatcher( editor: Editor, matchImageType: 'imageBlock' | 'imageInline' ): MatcherPattern {\n\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\tconst areBothImagePluginsLoaded = editor.plugins.has( 'ImageInlineEditing' ) && editor.plugins.has( 'ImageBlockEditing' );\n\n\treturn element => {\n\t\t// Check if the matched view element is an <img>.\n\t\tif ( !imageUtils.isInlineImageView( element ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// If just one of the plugins is loaded (block or inline), it will match all kinds of images.\n\t\tif ( !areBothImagePluginsLoaded ) {\n\t\t\treturn getPositiveMatchPattern( element );\n\t\t}\n\n\t\t// The <img> can be standalone, wrapped in <figure>...</figure> (ImageBlock plugin) or\n\t\t// wrapped in <figure><a>...</a></figure> (LinkImage plugin).\n\t\tconst imageType = element.getStyle( 'display' ) == 'block' || element.findAncestor( imageUtils.isBlockImageView ) ?\n\t\t\t'imageBlock' :\n\t\t\t'imageInline';\n\n\t\tif ( imageType !== matchImageType ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn getPositiveMatchPattern( element );\n\t};\n\n\tfunction getPositiveMatchPattern( element: ViewElement ) {\n\t\tconst pattern: Record<string, unknown> = {\n\t\t\tname: true\n\t\t};\n\n\t\t// This will trigger src consumption (See https://github.com/ckeditor/ckeditor5/issues/11530).\n\t\tif ( element.hasAttribute( 'src' ) ) {\n\t\t\tpattern.attributes = [ 'src' ];\n\t\t}\n\n\t\treturn pattern;\n\t}\n}\n\n/**\n * Considering the current model selection, it returns the name of the model image element\n * (`'imageBlock'` or `'imageInline'`) that will make most sense from the UX perspective if a new\n * image was inserted (also: uploaded, dropped, pasted) at that selection.\n *\n * The assumption is that inserting images into empty blocks or on other block widgets should\n * produce block images. Inline images should be inserted in other cases, e.g. in paragraphs\n * that already contain some text.\n *\n * @internal\n */\nexport function determineImageTypeForInsertionAtSelection(\n\tschema: Schema,\n\tselection: Selection | DocumentSelection\n): 'imageBlock' | 'imageInline' {\n\tconst firstBlock = first( selection.getSelectedBlocks() );\n\n\t// Insert a block image if the selection is not in/on block elements or it's on a block widget.\n\tif ( !firstBlock || schema.isObject( firstBlock ) ) {\n\t\treturn 'imageBlock';\n\t}\n\n\t// A block image should also be inserted into an empty block element\n\t// (that is not an empty list item so the list won't get split).\n\tif ( firstBlock.isEmpty && firstBlock.name != 'listItem' ) {\n\t\treturn 'imageBlock';\n\t}\n\n\t// Otherwise insert an inline image.\n\treturn 'imageInline';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imageutils\n */\n\nimport type {\n\tElement,\n\tViewElement,\n\tViewNode,\n\tDocumentSelection,\n\tViewDocumentSelection,\n\tSelection,\n\tViewSelection,\n\tDocumentFragment,\n\tViewDocumentFragment,\n\tDowncastWriter,\n\tModel,\n\tPosition\n} from 'ckeditor5/src/engine';\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\nimport { findOptimalInsertionRange, isWidget, toWidget } from 'ckeditor5/src/widget';\nimport { determineImageTypeForInsertionAtSelection } from './image/utils';\n\n/**\n * A set of helpers related to images.\n */\nexport default class ImageUtils extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageUtils' as const;\n\t}\n\n\t/**\n\t * Checks if the provided model element is an `image` or `imageInline`.\n\t */\n\tpublic isImage( modelElement?: Element | null ): modelElement is Element & { name: 'imageInline' | 'imageBlock' } {\n\t\treturn this.isInlineImage( modelElement ) || this.isBlockImage( modelElement );\n\t}\n\n\t/**\n\t * Checks if the provided view element represents an inline image.\n\t *\n\t * Also, see {@link module:image/imageutils~ImageUtils#isImageWidget}.\n\t */\n\tpublic isInlineImageView( element?: ViewElement | null ): boolean {\n\t\treturn !!element && element.is( 'element', 'img' );\n\t}\n\n\t/**\n\t * Checks if the provided view element represents a block image.\n\t *\n\t * Also, see {@link module:image/imageutils~ImageUtils#isImageWidget}.\n\t */\n\tpublic isBlockImageView( element?: ViewElement | null ): boolean {\n\t\treturn !!element && element.is( 'element', 'figure' ) && element.hasClass( 'image' );\n\t}\n\n\t/**\n\t * Handles inserting single file. This method unifies image insertion using {@link module:widget/utils~findOptimalInsertionRange}\n\t * method.\n\t *\n\t * ```ts\n\t * const imageUtils = editor.plugins.get( 'ImageUtils' );\n\t *\n\t * imageUtils.insertImage( { src: 'path/to/image.jpg' } );\n\t * ```\n\t *\n\t * @param attributes Attributes of the inserted image.\n\t * This method filters out the attributes which are disallowed by the {@link module:engine/model/schema~Schema}.\n\t * @param selectable Place to insert the image. If not specified,\n\t * the {@link module:widget/utils~findOptimalInsertionRange} logic will be applied for the block images\n\t * and `model.document.selection` for the inline images.\n\t *\n\t * **Note**: If `selectable` is passed, this helper will not be able to set selection attributes (such as `linkHref`)\n\t * and apply them to the new image. In this case, make sure all selection attributes are passed in `attributes`.\n\t *\n\t * @param imageType Image type of inserted image. If not specified,\n\t * it will be determined automatically depending of editor config or place of the insertion.\n\t * @return The inserted model image element.\n\t */\n\tpublic insertImage(\n\t\tattributes: Record<string, unknown> = {},\n\t\tselectable: Selection | Position | null = null,\n\t\timageType: ( 'imageBlock' | 'imageInline' | null ) = null\n\t): Element | null {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst selection = model.document.selection;\n\n\t\timageType = determineImageTypeForInsertion( editor, selectable || selection, imageType );\n\n\t\t// Mix declarative attributes with selection attributes because the new image should \"inherit\"\n\t\t// the latter for best UX. For instance, inline images inserted into existing links\n\t\t// should not split them. To do that, they need to have \"linkHref\" inherited from the selection.\n\t\tattributes = {\n\t\t\t...Object.fromEntries( selection.getAttributes() ),\n\t\t\t...attributes\n\t\t};\n\n\t\tfor ( const attributeName in attributes ) {\n\t\t\tif ( !model.schema.checkAttribute( imageType, attributeName ) ) {\n\t\t\t\tdelete attributes[ attributeName ];\n\t\t\t}\n\t\t}\n\n\t\treturn model.change( writer => {\n\t\t\tconst imageElement = writer.createElement( imageType!, attributes );\n\n\t\t\tmodel.insertObject( imageElement, selectable, null, {\n\t\t\t\tsetSelection: 'on',\n\t\t\t\t// If we want to insert a block image (for whatever reason) then we don't want to split text blocks.\n\t\t\t\t// This applies only when we don't have the selectable specified (i.e., we insert multiple block images at once).\n\t\t\t\tfindOptimalPosition: !selectable && imageType != 'imageInline' ? 'auto' : undefined\n\t\t\t} );\n\n\t\t\t// Inserting an image might've failed due to schema regulations.\n\t\t\tif ( imageElement.parent ) {\n\t\t\t\treturn imageElement;\n\t\t\t}\n\n\t\t\treturn null;\n\t\t} );\n\t}\n\n\t/**\n\t * Returns an image widget editing view element if one is selected or is among the selection's ancestors.\n\t */\n\tpublic getClosestSelectedImageWidget( selection: ViewSelection | ViewDocumentSelection ): ViewElement | null {\n\t\tconst selectionPosition = selection.getFirstPosition();\n\n\t\tif ( !selectionPosition ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst viewElement = selection.getSelectedElement();\n\n\t\tif ( viewElement && this.isImageWidget( viewElement ) ) {\n\t\t\treturn viewElement;\n\t\t}\n\n\t\tlet parent: ViewNode | ViewDocumentFragment | null = selectionPosition.parent;\n\n\t\twhile ( parent ) {\n\t\t\tif ( parent.is( 'element' ) && this.isImageWidget( parent ) ) {\n\t\t\t\treturn parent;\n\t\t\t}\n\n\t\t\tparent = parent.parent;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns a image model element if one is selected or is among the selection's ancestors.\n\t */\n\tpublic getClosestSelectedImageElement( selection: Selection | DocumentSelection ): Element | null {\n\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\treturn this.isImage( selectedElement ) ? selectedElement : selection.getFirstPosition()!.findAncestor( 'imageBlock' );\n\t}\n\n\t/**\n\t * Checks if image can be inserted at current model selection.\n\t *\n\t * @internal\n\t */\n\tpublic isImageAllowed(): boolean {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\n\t\treturn isImageAllowedInParent( this.editor, selection ) && isNotInsideImage( selection );\n\t}\n\n\t/**\n\t * Converts a given {@link module:engine/view/element~Element} to an image widget:\n\t * * Adds a {@link module:engine/view/element~Element#_setCustomProperty custom property} allowing to recognize the image widget\n\t * element.\n\t * * Calls the {@link module:widget/utils~toWidget} function with the proper element's label creator.\n\t *\n\t * @param writer An instance of the view writer.\n\t * @param label The element's label. It will be concatenated with the image `alt` attribute if one is present.\n\t */\n\tpublic toImageWidget( viewElement: ViewElement, writer: DowncastWriter, label: string ): ViewElement {\n\t\twriter.setCustomProperty( 'image', true, viewElement );\n\n\t\tconst labelCreator = () => {\n\t\t\tconst imgElement = this.findViewImgElement( viewElement )!;\n\t\t\tconst altText = imgElement.getAttribute( 'alt' );\n\n\t\t\treturn altText ? `${ altText } ${ label }` : label;\n\t\t};\n\n\t\treturn toWidget( viewElement, writer, { label: labelCreator } );\n\t}\n\n\t/**\n\t * Checks if a given view element is an image widget.\n\t */\n\tprotected isImageWidget( viewElement: ViewElement ): boolean {\n\t\treturn !!viewElement.getCustomProperty( 'image' ) && isWidget( viewElement );\n\t}\n\n\t/**\n\t * Checks if the provided model element is an `image`.\n\t */\n\tpublic isBlockImage( modelElement?: Element | null ): boolean {\n\t\treturn !!modelElement && modelElement.is( 'element', 'imageBlock' );\n\t}\n\n\t/**\n\t * Checks if the provided model element is an `imageInline`.\n\t */\n\tpublic isInlineImage( modelElement?: Element | null ): boolean {\n\t\treturn !!modelElement && modelElement.is( 'element', 'imageInline' );\n\t}\n\n\t/**\n\t * Get the view `<img>` from another view element, e.g. a widget (`<figure class=\"image\">`), a link (`<a>`).\n\t *\n\t * The `<img>` can be located deep in other elements, so this helper performs a deep tree search.\n\t */\n\tpublic findViewImgElement( figureView: ViewElement ): ViewElement | undefined {\n\t\tif ( this.isInlineImageView( figureView ) ) {\n\t\t\treturn figureView;\n\t\t}\n\n\t\tconst editingView = this.editor.editing.view;\n\n\t\tfor ( const { item } of editingView.createRangeIn( figureView ) ) {\n\t\t\tif ( this.isInlineImageView( item as ViewElement ) ) {\n\t\t\t\treturn item as ViewElement;\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Checks if image is allowed by schema in optimal insertion parent.\n */\nfunction isImageAllowedInParent( editor: Editor, selection: Selection | DocumentSelection ): boolean {\n\tconst imageType = determineImageTypeForInsertion( editor, selection, null );\n\n\tif ( imageType == 'imageBlock' ) {\n\t\tconst parent = getInsertImageParent( selection, editor.model );\n\n\t\tif ( editor.model.schema.checkChild( parent as Element, 'imageBlock' ) ) {\n\t\t\treturn true;\n\t\t}\n\t} else if ( editor.model.schema.checkChild( selection.focus!, 'imageInline' ) ) {\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n\n/**\n * Checks if selection is not placed inside an image (e.g. its caption).\n */\nfunction isNotInsideImage( selection: DocumentSelection ): boolean {\n\treturn [ ...selection.focus!.getAncestors() ].every( ancestor => !ancestor.is( 'element', 'imageBlock' ) );\n}\n\n/**\n * Returns a node that will be used to insert image with `model.insertContent`.\n */\nfunction getInsertImageParent( selection: Selection | DocumentSelection, model: Model ): Element | DocumentFragment {\n\tconst insertionRange = findOptimalInsertionRange( selection, model );\n\tconst parent = insertionRange.start.parent;\n\n\tif ( parent.isEmpty && !parent.is( 'element', '$root' ) ) {\n\t\treturn parent.parent!;\n\t}\n\n\treturn parent;\n}\n\n/**\n * Determine image element type name depending on editor config or place of insertion.\n *\n * @param imageType Image element type name. Used to force return of provided element name,\n * but only if there is proper plugin enabled.\n */\nfunction determineImageTypeForInsertion(\n\teditor: Editor,\n\tselectable: Position | Selection | DocumentSelection,\n\timageType: 'imageBlock' | 'imageInline' | null\n): 'imageBlock' | 'imageInline' {\n\tconst schema = editor.model.schema;\n\tconst configImageInsertType = editor.config.get( 'image.insert.type' );\n\n\tif ( !editor.plugins.has( 'ImageBlockEditing' ) ) {\n\t\treturn 'imageInline';\n\t}\n\n\tif ( !editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\treturn 'imageBlock';\n\t}\n\n\tif ( imageType ) {\n\t\treturn imageType;\n\t}\n\n\tif ( configImageInsertType === 'inline' ) {\n\t\treturn 'imageInline';\n\t}\n\n\tif ( configImageInsertType === 'block' ) {\n\t\treturn 'imageBlock';\n\t}\n\n\t// Try to replace the selected widget (e.g. another image).\n\tif ( selectable.is( 'selection' ) ) {\n\t\treturn determineImageTypeForInsertionAtSelection( schema, selectable );\n\t}\n\n\treturn schema.checkChild( selectable, 'imageInline' ) ? 'imageInline' : 'imageBlock';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/autoimage\n */\n\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\nimport { Clipboard, type ClipboardPipeline } from 'ckeditor5/src/clipboard';\nimport { LivePosition, LiveRange } from 'ckeditor5/src/engine';\nimport { Undo } from 'ckeditor5/src/undo';\nimport { Delete } from 'ckeditor5/src/typing';\nimport { global } from 'ckeditor5/src/utils';\n\nimport ImageUtils from './imageutils';\n\n// Implements the pattern: http(s)://(www.)example.com/path/to/resource.ext?query=params&maybe=too.\nconst IMAGE_URL_REGEXP = new RegExp( String( /^(http(s)?:\\/\\/)?[\\w-]+\\.[\\w.~:/[\\]@!$&'()*+,;=%-]+/.source +\n\t/\\.(jpg|jpeg|png|gif|ico|webp|JPG|JPEG|PNG|GIF|ICO|WEBP)/.source +\n\t/(\\?[\\w.~:/[\\]@!$&'()*+,;=%-]*)?/.source +\n\t/(#[\\w.~:/[\\]@!$&'()*+,;=%-]*)?$/.source ) );\n\n/**\n * The auto-image plugin. It recognizes image links in the pasted content and embeds\n * them shortly after they are injected into the document.\n */\nexport default class AutoImage extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Clipboard, ImageUtils, Undo, Delete ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'AutoImage' as const;\n\t}\n\n\t/**\n\t * The pastetoembed `setTimeout` ID. Stored as a property to allow\n\t * cleaning of the timeout.\n\t */\n\tprivate _timeoutId: ReturnType<typeof setTimeout> | null;\n\n\t/**\n\t * The position where the `<imageBlock>` element will be inserted after the timeout,\n\t * determined each time a new content is pasted into the document.\n\t */\n\tprivate _positionToInsert: LivePosition | null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis._timeoutId = null;\n\n\t\tthis._positionToInsert = null;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst modelDocument = editor.model.document;\n\t\tconst clipboardPipeline: ClipboardPipeline = editor.plugins.get( 'ClipboardPipeline' );\n\n\t\t// We need to listen on `Clipboard#inputTransformation` because we need to save positions of selection.\n\t\t// After pasting, the content between those positions will be checked for a URL that could be transformed\n\t\t// into an image.\n\t\tthis.listenTo( clipboardPipeline, 'inputTransformation', () => {\n\t\t\tconst firstRange = modelDocument.selection.getFirstRange()!;\n\n\t\t\tconst leftLivePosition = LivePosition.fromPosition( firstRange.start );\n\t\t\tleftLivePosition.stickiness = 'toPrevious';\n\n\t\t\tconst rightLivePosition = LivePosition.fromPosition( firstRange.end );\n\t\t\trightLivePosition.stickiness = 'toNext';\n\n\t\t\tmodelDocument.once( 'change:data', () => {\n\t\t\t\tthis._embedImageBetweenPositions( leftLivePosition, rightLivePosition );\n\n\t\t\t\tleftLivePosition.detach();\n\t\t\t\trightLivePosition.detach();\n\t\t\t}, { priority: 'high' } );\n\t\t} );\n\n\t\teditor.commands.get( 'undo' )!.on( 'execute', () => {\n\t\t\tif ( this._timeoutId ) {\n\t\t\t\tglobal.window.clearTimeout( this._timeoutId );\n\t\t\t\tthis._positionToInsert!.detach();\n\n\t\t\t\tthis._timeoutId = null;\n\t\t\t\tthis._positionToInsert = null;\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * Analyzes the part of the document between provided positions in search for a URL representing an image.\n\t * When the URL is found, it is automatically converted into an image.\n\t *\n\t * @param leftPosition Left position of the selection.\n\t * @param rightPosition Right position of the selection.\n\t */\n\tprivate _embedImageBetweenPositions( leftPosition: LivePosition, rightPosition: LivePosition ): void {\n\t\tconst editor = this.editor;\n\t\t// TODO: Use a marker instead of LiveRange & LivePositions.\n\t\tconst urlRange = new LiveRange( leftPosition, rightPosition );\n\t\tconst walker = urlRange.getWalker( { ignoreElementEnd: true } );\n\t\tconst selectionAttributes = Object.fromEntries( editor.model.document.selection.getAttributes() );\n\t\tconst imageUtils: ImageUtils = this.editor.plugins.get( 'ImageUtils' );\n\n\t\tlet src = '';\n\n\t\tfor ( const node of walker ) {\n\t\t\tif ( node.item.is( '$textProxy' ) ) {\n\t\t\t\tsrc += node.item.data;\n\t\t\t}\n\t\t}\n\n\t\tsrc = src.trim();\n\n\t\t// If the URL does not match the image URL regexp, let's skip that.\n\t\tif ( !src.match( IMAGE_URL_REGEXP ) ) {\n\t\t\turlRange.detach();\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Position will not be available in the `setTimeout` function so let's clone it.\n\t\tthis._positionToInsert = LivePosition.fromPosition( leftPosition );\n\n\t\t// This action mustn't be executed if undo was called between pasting and auto-embedding.\n\t\tthis._timeoutId = setTimeout( () => {\n\t\t\t// Do nothing if image element cannot be inserted at the current position.\n\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/2763.\n\t\t\t// Condition must be checked after timeout - pasting may take place on an element, replacing it. The final position matters.\n\t\t\tconst imageCommand = editor.commands.get( 'insertImage' )!;\n\n\t\t\tif ( !imageCommand.isEnabled ) {\n\t\t\t\turlRange.detach();\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\teditor.model.change( writer => {\n\t\t\t\tthis._timeoutId = null;\n\n\t\t\t\twriter.remove( urlRange );\n\t\t\t\turlRange.detach();\n\n\t\t\t\tlet insertionPosition;\n\n\t\t\t\t// Check if the position where the element should be inserted is still valid.\n\t\t\t\t// Otherwise leave it as undefined to use the logic of insertImage().\n\t\t\t\tif ( this._positionToInsert!.root.rootName !== '$graveyard' ) {\n\t\t\t\t\tinsertionPosition = this._positionToInsert!.toPosition();\n\t\t\t\t}\n\n\t\t\t\timageUtils.insertImage( { ...selectionAttributes, src }, insertionPosition );\n\n\t\t\t\tthis._positionToInsert!.detach();\n\t\t\t\tthis._positionToInsert = null;\n\t\t\t} );\n\n\t\t\tconst deletePlugin: Delete = editor.plugins.get( 'Delete' );\n\n\t\t\tdeletePlugin.requestUndoOnBackspace();\n\t\t}, 100 );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagetextalternative/imagetextalternativecommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport type ImageUtils from '../imageutils';\n\n/**\n * The image text alternative command. It is used to change the `alt` attribute of `<imageBlock>` and `<imageInline>` model elements.\n */\nexport default class ImageTextAlternativeCommand extends Command {\n\t/**\n\t * The command value: `false` if there is no `alt` attribute, otherwise the value of the `alt` attribute.\n\t *\n\t * @readonly\n\t * @observable\n\t */\n\tdeclare public value: string | false;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst editor = this.editor;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst element = imageUtils.getClosestSelectedImageElement( this.editor.model.document.selection )!;\n\n\t\tthis.isEnabled = !!element;\n\n\t\tif ( this.isEnabled && element.hasAttribute( 'alt' ) ) {\n\t\t\tthis.value = element.getAttribute( 'alt' ) as string | false;\n\t\t} else {\n\t\t\tthis.value = false;\n\t\t}\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * @fires execute\n\t * @param options\n\t * @param options.newValue The new value of the `alt` attribute to set.\n\t */\n\tpublic override execute( options: { newValue: string } ): void {\n\t\tconst editor = this.editor;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst model = editor.model;\n\t\tconst imageElement = imageUtils.getClosestSelectedImageElement( model.document.selection );\n\n\t\tmodel.change( writer => {\n\t\t\twriter.setAttribute( 'alt', options.newValue, imageElement! );\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagetextalternative/imagetextalternativeediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport ImageTextAlternativeCommand from './imagetextalternativecommand';\nimport ImageUtils from '../imageutils';\n\n/**\n * The image text alternative editing plugin.\n *\n * Registers the `'imageTextAlternative'` command.\n */\nexport default class ImageTextAlternativeEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageTextAlternativeEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tthis.editor.commands.add( 'imageTextAlternative', new ImageTextAlternativeCommand( this.editor ) );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./textalternativeform.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../../../node_modules/css-loader/dist/cjs.js!../../../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./responsiveform.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagetextalternative/ui/textalternativeformview\n */\n\nimport {\n\tButtonView,\n\tFocusCycler,\n\tLabeledFieldView,\n\tView,\n\tViewCollection,\n\tcreateLabeledInputText,\n\tsubmitHandler,\n\ttype InputView\n} from 'ckeditor5/src/ui';\nimport { FocusTracker, KeystrokeHandler, type Locale } from 'ckeditor5/src/utils';\nimport { icons } from 'ckeditor5/src/core';\n\nimport '../../../theme/textalternativeform.css';\n\n// See: #8833.\n// eslint-disable-next-line ckeditor5-rules/ckeditor-imports\nimport '@ckeditor/ckeditor5-ui/theme/components/responsive-form/responsiveform.css';\n\n/**\n * The TextAlternativeFormView class.\n */\nexport default class TextAlternativeFormView extends View {\n\t/**\n\t * Tracks information about the DOM focus in the form.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * An instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}.\n\t */\n\tpublic readonly keystrokes: KeystrokeHandler;\n\n\t/**\n\t * An input with a label.\n\t */\n\tpublic labeledInput: LabeledFieldView<InputView>;\n\n\t/**\n\t * A button used to submit the form.\n\t */\n\tpublic saveButtonView: ButtonView;\n\n\t/**\n\t * A button used to cancel the form.\n\t */\n\tpublic cancelButtonView: ButtonView;\n\n\t/**\n\t * A collection of views which can be focused in the form.\n\t */\n\tprotected readonly _focusables: ViewCollection;\n\n\t/**\n\t * Helps cycling over {@link #_focusables} in the form.\n\t */\n\tprotected readonly _focusCycler: FocusCycler;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale: Locale ) {\n\t\tsuper( locale );\n\t\tconst t = this.locale!.t;\n\n\t\tthis.focusTracker = new FocusTracker();\n\n\t\tthis.keystrokes = new KeystrokeHandler();\n\n\t\tthis.labeledInput = this._createLabeledInputView();\n\n\t\tthis.saveButtonView = this._createButton( t( 'Save' ), icons.check, 'ck-button-save' );\n\t\tthis.saveButtonView.type = 'submit';\n\n\t\tthis.cancelButtonView = this._createButton( t( 'Cancel' ), icons.cancel, 'ck-button-cancel', 'cancel' );\n\n\t\tthis._focusables = new ViewCollection();\n\n\t\tthis._focusCycler = new FocusCycler( {\n\t\t\tfocusables: this._focusables,\n\t\t\tfocusTracker: this.focusTracker,\n\t\t\tkeystrokeHandler: this.keystrokes,\n\t\t\tactions: {\n\t\t\t\t// Navigate form fields backwards using the Shift + Tab keystroke.\n\t\t\t\tfocusPrevious: 'shift + tab',\n\n\t\t\t\t// Navigate form fields forwards using the Tab key.\n\t\t\t\tfocusNext: 'tab'\n\t\t\t}\n\t\t} );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'form',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-text-alternative-form',\n\t\t\t\t\t'ck-responsive-form'\n\t\t\t\t],\n\n\t\t\t\t// https://github.com/ckeditor/ckeditor5-image/issues/40\n\t\t\t\ttabindex: '-1'\n\t\t\t},\n\n\t\t\tchildren: [\n\t\t\t\tthis.labeledInput,\n\t\t\t\tthis.saveButtonView,\n\t\t\t\tthis.cancelButtonView\n\t\t\t]\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tthis.keystrokes.listenTo( this.element! );\n\n\t\tsubmitHandler( { view: this } );\n\n\t\t[ this.labeledInput, this.saveButtonView, this.cancelButtonView ]\n\t\t\t.forEach( v => {\n\t\t\t\t// Register the view as focusable.\n\t\t\t\tthis._focusables.add( v );\n\n\t\t\t\t// Register the view in the focus tracker.\n\t\t\t\tthis.focusTracker.add( v.element! );\n\t\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.focusTracker.destroy();\n\t\tthis.keystrokes.destroy();\n\t}\n\n\t/**\n\t * Creates the button view.\n\t *\n\t * @param label The button label\n\t * @param icon The button's icon.\n\t * @param className The additional button CSS class name.\n\t * @param eventName The event name that the ButtonView#execute event will be delegated to.\n\t * @returns The button view instance.\n\t */\n\tprivate _createButton( label: string, icon: string, className: string, eventName?: string ): ButtonView {\n\t\tconst button = new ButtonView( this.locale );\n\n\t\tbutton.set( {\n\t\t\tlabel,\n\t\t\ticon,\n\t\t\ttooltip: true\n\t\t} );\n\n\t\tbutton.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: className\n\t\t\t}\n\t\t} );\n\n\t\tif ( eventName ) {\n\t\t\tbutton.delegate( 'execute' ).to( this, eventName );\n\t\t}\n\n\t\treturn button;\n\t}\n\n\t/**\n\t * Creates an input with a label.\n\t *\n\t * @returns Labeled field view instance.\n\t */\n\tprivate _createLabeledInputView(): LabeledFieldView<InputView> {\n\t\tconst t = this.locale!.t;\n\t\tconst labeledInput = new LabeledFieldView<InputView>( this.locale, createLabeledInputText );\n\n\t\tlabeledInput.label = t( 'Text alternative' );\n\n\t\treturn labeledInput;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/ui/utils\n */\n\nimport type { PositionOptions } from 'ckeditor5/src/utils';\nimport type { Editor } from 'ckeditor5/src/core';\nimport { BalloonPanelView, type ContextualBalloon } from 'ckeditor5/src/ui';\n\nimport type ImageUtils from '../../imageutils';\n\n/**\n * A helper utility that positions the\n * {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon contextual balloon} instance\n * with respect to the image in the editor content, if one is selected.\n *\n * @param editor The editor instance.\n */\nexport function repositionContextualBalloon( editor: Editor ): void {\n\tconst balloon: ContextualBalloon = editor.plugins.get( 'ContextualBalloon' );\n\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\tif ( imageUtils.getClosestSelectedImageWidget( editor.editing.view.document.selection ) ) {\n\t\tconst position = getBalloonPositionData( editor );\n\n\t\tballoon.updatePosition( position );\n\t}\n}\n\n/**\n * Returns the positioning options that control the geometry of the\n * {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon contextual balloon} with respect\n * to the selected element in the editor content.\n *\n * @param editor The editor instance.\n */\nexport function getBalloonPositionData( editor: Editor ): Partial<PositionOptions> {\n\tconst editingView = editor.editing.view;\n\tconst defaultPositions = BalloonPanelView.defaultPositions;\n\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\treturn {\n\t\ttarget: editingView.domConverter.mapViewToDom(\n\t\t\timageUtils.getClosestSelectedImageWidget( editingView.document.selection )!\n\t\t) as HTMLElement,\n\t\tpositions: [\n\t\t\tdefaultPositions.northArrowSouth,\n\t\t\tdefaultPositions.northArrowSouthWest,\n\t\t\tdefaultPositions.northArrowSouthEast,\n\t\t\tdefaultPositions.southArrowNorth,\n\t\t\tdefaultPositions.southArrowNorthWest,\n\t\t\tdefaultPositions.southArrowNorthEast,\n\t\t\tdefaultPositions.viewportStickyNorth\n\t\t]\n\t};\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagetextalternative/imagetextalternativeui\n */\n\nimport { Plugin, icons } from 'ckeditor5/src/core';\nimport {\n\tButtonView,\n\tContextualBalloon,\n\tclickOutsideHandler,\n\tCssTransitionDisablerMixin,\n\ttype ViewWithCssTransitionDisabler\n} from 'ckeditor5/src/ui';\n\nimport TextAlternativeFormView from './ui/textalternativeformview';\nimport { repositionContextualBalloon, getBalloonPositionData } from '../image/ui/utils';\nimport type { CancelEvent, SubmitEvent } from '../imageinsert/ui/imageinsertpanelview';\nimport type ImageTextAlternativeCommand from './imagetextalternativecommand';\nimport type ImageUtils from '../imageutils';\n\n/**\n * The image text alternative UI plugin.\n *\n * The plugin uses the {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon}.\n */\nexport default class ImageTextAlternativeUI extends Plugin {\n\t/**\n\t * The contextual balloon plugin instance.\n\t */\n\tprivate _balloon?: ContextualBalloon;\n\n\t/**\n\t * A form containing a textarea and buttons, used to change the `alt` text value.\n\t */\n\tprivate _form?: TextAlternativeFormView & ViewWithCssTransitionDisabler;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ContextualBalloon ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageTextAlternativeUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tthis._createButton();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\t// Destroy created UI components as they are not automatically destroyed (see ckeditor5#1341).\n\t\tif ( this._form ) {\n\t\t\tthis._form.destroy();\n\t\t}\n\t}\n\n\t/**\n\t * Creates a button showing the balloon panel for changing the image text alternative and\n\t * registers it in the editor {@link module:ui/componentfactory~ComponentFactory ComponentFactory}.\n\t */\n\tprivate _createButton(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\n\t\teditor.ui.componentFactory.add( 'imageTextAlternative', locale => {\n\t\t\tconst command: ImageTextAlternativeCommand = editor.commands.get( 'imageTextAlternative' )!;\n\t\t\tconst view = new ButtonView( locale );\n\n\t\t\tview.set( {\n\t\t\t\tlabel: t( 'Change image text alternative' ),\n\t\t\t\ticon: icons.lowVision,\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tview.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\t\t\tview.bind( 'isOn' ).to( command, 'value', value => !!value );\n\n\t\t\tthis.listenTo( view, 'execute', () => {\n\t\t\t\tthis._showForm();\n\t\t\t} );\n\n\t\t\treturn view;\n\t\t} );\n\t}\n\n\t/**\n\t * Creates the {@link module:image/imagetextalternative/ui/textalternativeformview~TextAlternativeFormView}\n\t * form.\n\t */\n\tprivate _createForm(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\t\tthis._balloon = this.editor.plugins.get( 'ContextualBalloon' );\n\n\t\tthis._form = new ( CssTransitionDisablerMixin( TextAlternativeFormView ) )( editor.locale );\n\n\t\t// Render the form so its #element is available for clickOutsideHandler.\n\t\tthis._form.render();\n\n\t\tthis.listenTo<SubmitEvent>( this._form, 'submit', () => {\n\t\t\teditor.execute( 'imageTextAlternative', {\n\t\t\t\tnewValue: this._form!.labeledInput.fieldView.element!.value\n\t\t\t} );\n\n\t\t\tthis._hideForm( true );\n\t\t} );\n\n\t\tthis.listenTo<CancelEvent>( this._form, 'cancel', () => {\n\t\t\tthis._hideForm( true );\n\t\t} );\n\n\t\t// Close the form on Esc key press.\n\t\tthis._form.keystrokes.set( 'Esc', ( data, cancel ) => {\n\t\t\tthis._hideForm( true );\n\t\t\tcancel();\n\t\t} );\n\n\t\t// Reposition the balloon or hide the form if an image widget is no longer selected.\n\t\tthis.listenTo( editor.ui, 'update', () => {\n\t\t\tif ( !imageUtils.getClosestSelectedImageWidget( viewDocument.selection ) ) {\n\t\t\t\tthis._hideForm( true );\n\t\t\t} else if ( this._isVisible ) {\n\t\t\t\trepositionContextualBalloon( editor );\n\t\t\t}\n\t\t} );\n\n\t\t// Close on click outside of balloon panel element.\n\t\tclickOutsideHandler( {\n\t\t\temitter: this._form,\n\t\t\tactivator: () => this._isVisible,\n\t\t\tcontextElements: () => [ this._balloon!.view.element! ],\n\t\t\tcallback: () => this._hideForm()\n\t\t} );\n\t}\n\n\t/**\n\t * Shows the {@link #_form} in the {@link #_balloon}.\n\t */\n\tprivate _showForm(): void {\n\t\tif ( this._isVisible ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( !this._form ) {\n\t\t\tthis._createForm();\n\t\t}\n\n\t\tconst editor = this.editor;\n\t\tconst command: ImageTextAlternativeCommand = editor.commands.get( 'imageTextAlternative' )!;\n\t\tconst labeledInput = this._form!.labeledInput;\n\n\t\tthis._form!.disableCssTransitions();\n\n\t\tif ( !this._isInBalloon ) {\n\t\t\tthis._balloon!.add( {\n\t\t\t\tview: this._form!,\n\t\t\t\tposition: getBalloonPositionData( editor )\n\t\t\t} );\n\t\t}\n\n\t\t// Make sure that each time the panel shows up, the field remains in sync with the value of\n\t\t// the command. If the user typed in the input, then canceled the balloon (`labeledInput#value`\n\t\t// stays unaltered) and re-opened it without changing the value of the command, they would see the\n\t\t// old value instead of the actual value of the command.\n\t\t// https://github.com/ckeditor/ckeditor5-image/issues/114\n\t\tlabeledInput.fieldView.value = labeledInput.fieldView.element!.value = command.value || '';\n\n\t\tthis._form!.labeledInput.fieldView.select();\n\n\t\tthis._form!.enableCssTransitions();\n\t}\n\n\t/**\n\t * Removes the {@link #_form} from the {@link #_balloon}.\n\t *\n\t * @param focusEditable Controls whether the editing view is focused afterwards.\n\t */\n\tprivate _hideForm( focusEditable: boolean = false ): void {\n\t\tif ( !this._isInBalloon ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Blur the input element before removing it from DOM to prevent issues in some browsers.\n\t\t// See https://github.com/ckeditor/ckeditor5/issues/1501.\n\t\tif ( this._form!.focusTracker.isFocused ) {\n\t\t\tthis._form!.saveButtonView.focus();\n\t\t}\n\n\t\tthis._balloon!.remove( this._form! );\n\n\t\tif ( focusEditable ) {\n\t\t\tthis.editor.editing.view.focus();\n\t\t}\n\t}\n\n\t/**\n\t * Returns `true` when the {@link #_form} is the visible view in the {@link #_balloon}.\n\t */\n\tprivate get _isVisible(): boolean {\n\t\treturn !!this._balloon && this._balloon.visibleView === this._form;\n\t}\n\n\t/**\n\t * Returns `true` when the {@link #_form} is in the {@link #_balloon}.\n\t */\n\tprivate get _isInBalloon(): boolean {\n\t\treturn !!this._balloon && this._balloon.hasView( this._form! );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagetextalternative\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport ImageTextAlternativeEditing from './imagetextalternative/imagetextalternativeediting';\nimport ImageTextAlternativeUI from './imagetextalternative/imagetextalternativeui';\n\n/**\n * The image text alternative plugin.\n *\n * For a detailed overview, check the {@glink features/images/images-styles image styles} documentation.\n *\n * This is a \"glue\" plugin which loads the\n * {@link module:image/imagetextalternative/imagetextalternativeediting~ImageTextAlternativeEditing}\n * and {@link module:image/imagetextalternative/imagetextalternativeui~ImageTextAlternativeUI} plugins.\n */\nexport default class ImageTextAlternative extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageTextAlternativeEditing, ImageTextAlternativeUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageTextAlternative' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/converters\n */\n\nimport type {\n\tDowncastDispatcher,\n\tElement,\n\tUpcastDispatcher,\n\tUpcastElementEvent,\n\tViewElement,\n\tViewElementAttributes,\n\tDowncastAttributeEvent\n} from 'ckeditor5/src/engine';\nimport { first, type GetCallback } from 'ckeditor5/src/utils';\nimport type ImageUtils from '../imageutils';\n\ntype SrcsetAttributeType = null | { data: unknown; width: unknown };\n\n/**\n * Returns a function that converts the image view representation:\n *\n * ```html\n * <figure class=\"image\"><img src=\"...\" alt=\"...\"></img></figure>\n * ```\n *\n * to the model representation:\n *\n * ```html\n * <imageBlock src=\"...\" alt=\"...\"></imageBlock>\n * ```\n *\n * The entire content of the `<figure>` element except the first `<img>` is being converted as children\n * of the `<imageBlock>` model element.\n *\n * @internal\n */\nexport function upcastImageFigure( imageUtils: ImageUtils ): ( dispatcher: UpcastDispatcher ) => void {\n\tconst converter: GetCallback<UpcastElementEvent> = ( evt, data, conversionApi ) => {\n\t\t// Do not convert if this is not an \"image figure\".\n\t\tif ( !conversionApi.consumable.test( data.viewItem, { name: true, classes: 'image' } ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Find an image element inside the figure element.\n\t\tconst viewImage = imageUtils.findViewImgElement( data.viewItem );\n\n\t\t// Do not convert if image element is absent or was already converted.\n\t\tif ( !viewImage || !conversionApi.consumable.test( viewImage, { name: true } ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Consume the figure to prevent other converters from processing it again.\n\t\tconversionApi.consumable.consume( data.viewItem, { name: true, classes: 'image' } );\n\n\t\t// Convert view image to model image.\n\t\tconst conversionResult = conversionApi.convertItem( viewImage, data.modelCursor );\n\n\t\t// Get image element from conversion result.\n\t\tconst modelImage = first( conversionResult.modelRange!.getItems() ) as Element;\n\n\t\t// When image wasn't successfully converted then finish conversion.\n\t\tif ( !modelImage ) {\n\t\t\t// Revert consumed figure so other features can convert it.\n\t\t\tconversionApi.consumable.revert( data.viewItem, { name: true, classes: 'image' } );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Convert rest of the figure element's children as an image children.\n\t\tconversionApi.convertChildren( data.viewItem, modelImage );\n\n\t\tconversionApi.updateConversionResult( modelImage, data );\n\t};\n\n\treturn dispatcher => {\n\t\tdispatcher.on<UpcastElementEvent>( 'element:figure', converter );\n\t};\n}\n\n/**\n * Returns a function that converts the image view representation:\n *\n * ```html\n * <picture><source ... /><source ... />...<img ... /></picture>\n * ```\n *\n * to the model representation as the `sources` attribute:\n *\n * ```html\n * <image[Block|Inline] ... sources=\"...\"></image[Block|Inline]>\n * ```\n *\n * @internal\n */\nexport function upcastPicture( imageUtils: ImageUtils ): ( dispatcher: UpcastDispatcher ) => void {\n\tconst sourceAttributeNames = [ 'srcset', 'media', 'type', 'sizes' ];\n\n\tconst converter: GetCallback<UpcastElementEvent> = ( evt, data, conversionApi ) => {\n\t\tconst pictureViewElement = data.viewItem;\n\n\t\t// Do not convert <picture> if already consumed.\n\t\tif ( !conversionApi.consumable.test( pictureViewElement, { name: true } ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst sources = new Map<ViewElement, Record<string, string | undefined>>();\n\n\t\t// Collect all <source /> elements attribute values.\n\t\tfor ( const childSourceElement of pictureViewElement.getChildren() ) {\n\t\t\tif ( childSourceElement.is( 'element', 'source' ) ) {\n\t\t\t\tconst attributes: Record<string, string | undefined> = {};\n\n\t\t\t\tfor ( const name of sourceAttributeNames ) {\n\t\t\t\t\tif ( childSourceElement.hasAttribute( name ) ) {\n\t\t\t\t\t\t// Don't collect <source /> attribute if already consumed somewhere else.\n\t\t\t\t\t\tif ( conversionApi.consumable.test( childSourceElement, { attributes: name } ) ) {\n\t\t\t\t\t\t\tattributes[ name ] = childSourceElement.getAttribute( name );\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tif ( Object.keys( attributes ).length ) {\n\t\t\t\t\tsources.set( childSourceElement, attributes );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tconst imgViewElement = imageUtils.findViewImgElement( pictureViewElement );\n\n\t\t// Don't convert when a picture has no <img/> inside (it is broken).\n\t\tif ( !imgViewElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\tlet modelImage = data.modelCursor.parent;\n\n\t\t// - In case of an inline image (cursor parent in a <paragraph>), the <img/> must be converted right away\n\t\t// because no converter handled it yet and otherwise there would be no model element to set the sources attribute on.\n\t\t// - In case of a block image, the <figure class=\"image\"> converter (in ImageBlockEditing) converts the\n\t\t// <img/> right away on its own and the modelCursor is already inside an imageBlock and there's nothing special\n\t\t// to do here.\n\t\tif ( !modelImage.is( 'element', 'imageBlock' ) ) {\n\t\t\tconst conversionResult = conversionApi.convertItem( imgViewElement, data.modelCursor );\n\n\t\t\t// Set image range as conversion result.\n\t\t\tdata.modelRange = conversionResult.modelRange;\n\n\t\t\t// Continue conversion where image conversion ends.\n\t\t\tdata.modelCursor = conversionResult.modelCursor;\n\n\t\t\tmodelImage = first( conversionResult.modelRange!.getItems() ) as Element;\n\t\t}\n\n\t\tconversionApi.consumable.consume( pictureViewElement, { name: true } );\n\n\t\t// Consume only these <source/> attributes that were actually collected and will be passed on\n\t\t// to the image model element.\n\t\tfor ( const [ sourceElement, attributes ] of sources ) {\n\t\t\tconversionApi.consumable.consume( sourceElement, { attributes: Object.keys( attributes ) } );\n\t\t}\n\n\t\tif ( sources.size ) {\n\t\t\tconversionApi.writer.setAttribute( 'sources', Array.from( sources.values() ), modelImage );\n\t\t}\n\n\t\t// Convert rest of the <picture> children as an image children. Other converters may want to consume them.\n\t\tconversionApi.convertChildren( pictureViewElement, modelImage );\n\t};\n\n\treturn dispatcher => {\n\t\tdispatcher.on<UpcastElementEvent>( 'element:picture', converter );\n\t};\n}\n\n/**\n * Converter used to convert the `srcset` model image attribute to the `srcset`, `sizes` and `width` attributes in the view.\n *\n * @internal\n * @param imageType The type of the image.\n */\nexport function downcastSrcsetAttribute(\n\timageUtils: ImageUtils,\n\timageType: 'imageBlock' | 'imageInline'\n): ( dispatcher: DowncastDispatcher ) => void {\n\tconst converter: GetCallback<DowncastAttributeEvent<Element>> = ( evt, data, conversionApi\t) => {\n\t\tif ( !conversionApi.consumable.consume( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst writer = conversionApi.writer;\n\t\tconst element = conversionApi.mapper.toViewElement( data.item )!;\n\t\tconst img = imageUtils.findViewImgElement( element )!;\n\n\t\tif ( data.attributeNewValue === null ) {\n\t\t\tconst srcset = data.attributeOldValue as SrcsetAttributeType;\n\n\t\t\tif ( srcset && srcset.data ) {\n\t\t\t\twriter.removeAttribute( 'srcset', img );\n\t\t\t\twriter.removeAttribute( 'sizes', img );\n\n\t\t\t\tif ( srcset.width ) {\n\t\t\t\t\twriter.removeAttribute( 'width', img );\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\tconst srcset = data.attributeNewValue as SrcsetAttributeType;\n\n\t\t\tif ( srcset && srcset.data ) {\n\t\t\t\twriter.setAttribute( 'srcset', srcset.data, img );\n\t\t\t\t// Always outputting `100vw`. See https://github.com/ckeditor/ckeditor5-image/issues/2.\n\t\t\t\twriter.setAttribute( 'sizes', '100vw', img );\n\n\t\t\t\tif ( srcset.width ) {\n\t\t\t\t\twriter.setAttribute( 'width', srcset.width, img );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t};\n\n\treturn dispatcher => {\n\t\tdispatcher.on<DowncastAttributeEvent<Element>>( `attribute:srcset:${ imageType }`, converter );\n\t};\n}\n\n/**\n * Converts the `source` model attribute to the `<picture><source /><source />...<img /></picture>`\n * view structure.\n *\n * @internal\n */\nexport function downcastSourcesAttribute( imageUtils: ImageUtils ): ( dispatcher: DowncastDispatcher ) => void {\n\tconst converter: GetCallback<DowncastAttributeEvent<Element>> = ( evt, data, conversionApi\t) => {\n\t\tif ( !conversionApi.consumable.consume( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst element = conversionApi.mapper.toViewElement( data.item )!;\n\t\tconst imgElement = imageUtils.findViewImgElement( element )!;\n\t\tconst attributeNewValue = data.attributeNewValue as null | Array<ViewElementAttributes>;\n\n\t\tif ( attributeNewValue && attributeNewValue.length ) {\n\t\t\t// Make sure <picture> does not break attribute elements, for instance <a> in linked images.\n\t\t\tconst pictureElement = viewWriter.createContainerElement( 'picture', null,\n\t\t\t\tattributeNewValue.map( sourceAttributes => {\n\t\t\t\t\treturn viewWriter.createEmptyElement( 'source', sourceAttributes );\n\t\t\t\t} )\n\t\t\t);\n\n\t\t\t// Collect all wrapping attribute elements.\n\t\t\tconst attributeElements = [];\n\t\t\tlet viewElement = imgElement.parent;\n\n\t\t\twhile ( viewElement && viewElement.is( 'attributeElement' ) ) {\n\t\t\t\tconst parentElement = viewElement.parent;\n\n\t\t\t\tviewWriter.unwrap( viewWriter.createRangeOn( imgElement ), viewElement );\n\n\t\t\t\tattributeElements.unshift( viewElement );\n\t\t\t\tviewElement = parentElement;\n\t\t\t}\n\n\t\t\t// Insert the picture and move img into it.\n\t\t\tviewWriter.insert( viewWriter.createPositionBefore( imgElement ), pictureElement );\n\t\t\tviewWriter.move( viewWriter.createRangeOn( imgElement ), viewWriter.createPositionAt( pictureElement, 'end' ) );\n\n\t\t\t// Apply collected attribute elements over the new picture element.\n\t\t\tfor ( const attributeElement of attributeElements ) {\n\t\t\t\tviewWriter.wrap( viewWriter.createRangeOn( pictureElement ), attributeElement );\n\t\t\t}\n\t\t}\n\t\t// Both setting \"sources\" to an empty array and removing the attribute should unwrap the <img />.\n\t\t// Unwrap once if the latter followed the former, though.\n\t\telse if ( imgElement.parent!.is( 'element', 'picture' ) ) {\n\t\t\tconst pictureElement = imgElement.parent;\n\n\t\t\tviewWriter.move( viewWriter.createRangeOn( imgElement ), viewWriter.createPositionBefore( pictureElement ) );\n\t\t\tviewWriter.remove( pictureElement );\n\t\t}\n\t};\n\n\treturn dispatcher => {\n\t\tdispatcher.on<DowncastAttributeEvent<Element>>( 'attribute:sources:imageBlock', converter );\n\t\tdispatcher.on<DowncastAttributeEvent<Element>>( 'attribute:sources:imageInline', converter );\n\t};\n}\n\n/**\n * Converter used to convert a given image attribute from the model to the view.\n *\n * @internal\n * @param imageType The type of the image.\n * @param attributeKey The name of the attribute to convert.\n */\nexport function downcastImageAttribute(\n\timageUtils: ImageUtils,\n\timageType: 'imageBlock' | 'imageInline',\n\tattributeKey: string\n): ( dispatcher: DowncastDispatcher ) => void {\n\tconst converter: GetCallback<DowncastAttributeEvent<Element>> = ( evt, data, conversionApi ) => {\n\t\tif ( !conversionApi.consumable.consume( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst element = conversionApi.mapper.toViewElement( data.item )!;\n\t\tconst img = imageUtils.findViewImgElement( element )!;\n\n\t\tviewWriter.setAttribute( data.attributeKey, data.attributeNewValue || '', img );\n\t};\n\n\treturn dispatcher => {\n\t\tdispatcher.on<DowncastAttributeEvent<Element>>( `attribute:${ attributeKey }:${ imageType }`, converter );\n\t};\n}\n\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/imageloadobserver\n */\n\nimport { Observer } from 'ckeditor5/src/engine';\n\n/**\n * Observes all new images added to the {@link module:engine/view/document~Document},\n * fires {@link module:engine/view/document~Document#event:imageLoaded} and\n * {@link module:engine/view/document~Document#event:layoutChanged} event every time when the new image\n * has been loaded.\n *\n * **Note:** This event is not fired for images that has been added to the document and rendered as `complete` (already loaded).\n */\nexport default class ImageLoadObserver extends Observer {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic observe( domRoot: HTMLElement ): void {\n\t\tthis.listenTo( domRoot, 'load', ( event, domEvent ) => {\n\t\t\tconst domElement = domEvent.target as HTMLElement;\n\n\t\t\tif ( this.checkShouldIgnoreEventFromTarget( domElement ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( domElement.tagName == 'IMG' ) {\n\t\t\t\tthis._fireEvents( domEvent );\n\t\t\t}\n\t\t\t// Use capture phase for better performance (#4504).\n\t\t}, { useCapture: true } );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override stopObserving( domRoot: HTMLElement ): void {\n\t\tthis.stopListening( domRoot );\n\t}\n\n\t/**\n\t * Fires {@link module:engine/view/document~Document#event:layoutChanged} and\n\t * {@link module:engine/view/document~Document#event:imageLoaded}\n\t * if observer {@link #isEnabled is enabled}.\n\t *\n\t * @param domEvent The DOM event.\n\t */\n\tprivate _fireEvents( domEvent: Event ): void {\n\t\tif ( this.isEnabled ) {\n\t\t\tthis.document.fire( 'layoutChanged' );\n\t\t\tthis.document.fire<ImageLoadedEvent>( 'imageLoaded', domEvent );\n\t\t}\n\t}\n}\n\n/**\n * Fired when an <img/> DOM element has been loaded in the DOM root.\n *\n * Introduced by {@link module:image/image/imageloadobserver~ImageLoadObserver}.\n *\n * @see module:image/image/imageloadobserver~ImageLoadObserver\n *\n * @eventName module:engine/view/document~Document#imageLoaded\n * @param data Event data.\n */\nexport type ImageLoadedEvent = {\n\tname: 'imageLoaded';\n\targs: [ Event ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/insertimagecommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport { logWarning, toArray, type ArrayOrItem } from 'ckeditor5/src/utils';\nimport type ImageUtils from '../imageutils';\n\n/**\n * Insert image command.\n *\n * The command is registered by the {@link module:image/image/imageediting~ImageEditing} plugin as `insertImage`\n * and it is also available via aliased `imageInsert` name.\n *\n * In order to insert an image at the current selection position\n * (according to the {@link module:widget/utils~findOptimalInsertionRange} algorithm),\n * execute the command and specify the image source:\n *\n * ```ts\n * editor.execute( 'insertImage', { source: 'http://url.to.the/image' } );\n * ```\n *\n * It is also possible to insert multiple images at once:\n *\n * ```ts\n * editor.execute( 'insertImage', {\n * \tsource: [\n * \t\t'path/to/image.jpg',\n * \t\t'path/to/other-image.jpg'\n * \t]\n * } );\n * ```\n *\n * If you want to take the full control over the process, you can specify individual model attributes:\n *\n * ```ts\n * editor.execute( 'insertImage', {\n * \tsource: [\n * \t\t{ src: 'path/to/image.jpg', alt: 'First alt text' },\n * \t\t{ src: 'path/to/other-image.jpg', alt: 'Second alt text', customAttribute: 'My attribute value' }\n * \t]\n * } );\n * ```\n */\nexport default class InsertImageCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tconst configImageInsertType = editor.config.get( 'image.insert.type' );\n\n\t\tif ( !editor.plugins.has( 'ImageBlockEditing' ) ) {\n\t\t\tif ( configImageInsertType === 'block' ) {\n\t\t\t\t/**\n\t\t\t\t * The {@link module:image/imageblock~ImageBlock} plugin must be enabled to allow inserting block images. See\n\t\t\t\t * {@link module:image/imageconfig~ImageInsertConfig#type} to learn more.\n\t\t\t\t *\n\t\t\t\t * @error image-block-plugin-required\n\t\t\t\t */\n\t\t\t\tlogWarning( 'image-block-plugin-required' );\n\t\t\t}\n\t\t}\n\n\t\tif ( !editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\t\tif ( configImageInsertType === 'inline' ) {\n\t\t\t\t/**\n\t\t\t\t * The {@link module:image/imageinline~ImageInline} plugin must be enabled to allow inserting inline images. See\n\t\t\t\t * {@link module:image/imageconfig~ImageInsertConfig#type} to learn more.\n\t\t\t\t *\n\t\t\t\t * @error image-inline-plugin-required\n\t\t\t\t */\n\t\t\t\tlogWarning( 'image-inline-plugin-required' );\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst imageUtils: ImageUtils = this.editor.plugins.get( 'ImageUtils' );\n\n\t\tthis.isEnabled = imageUtils.isImageAllowed();\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * @fires execute\n\t * @param options Options for the executed command.\n\t * @param options.source The image source or an array of image sources to insert.\n\t * See the documentation of the command to learn more about accepted formats.\n\t */\n\tpublic override execute( options: { source: ArrayOrItem<string | Record<string, unknown>> } ): void {\n\t\tconst sourceDefinitions = toArray<string | Record<string, unknown>>( options.source );\n\t\tconst selection = this.editor.model.document.selection;\n\t\tconst imageUtils: ImageUtils = this.editor.plugins.get( 'ImageUtils' );\n\n\t\t// In case of multiple images, each image (starting from the 2nd) will be inserted at a position that\n\t\t// follows the previous one. That will move the selection and, to stay on the safe side and make sure\n\t\t// all images inherit the same selection attributes, they are collected beforehand.\n\t\t//\n\t\t// Applying these attributes ensures, for instance, that inserting an (inline) image into a link does\n\t\t// not split that link but preserves its continuity.\n\t\t//\n\t\t// Note: Selection attributes that do not make sense for images will be filtered out by insertImage() anyway.\n\t\tconst selectionAttributes = Object.fromEntries( selection.getAttributes() );\n\n\t\tsourceDefinitions.forEach( ( sourceDefinition, index ) => {\n\t\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\t\tif ( typeof sourceDefinition === 'string' ) {\n\t\t\t\tsourceDefinition = { src: sourceDefinition };\n\t\t\t}\n\n\t\t\t// Inserting of an inline image replace the selected element and make a selection on the inserted image.\n\t\t\t// Therefore inserting multiple inline images requires creating position after each element.\n\t\t\tif ( index && selectedElement && imageUtils.isImage( selectedElement ) ) {\n\t\t\t\tconst position = this.editor.model.createPositionAfter( selectedElement );\n\n\t\t\t\timageUtils.insertImage( { ...sourceDefinition, ...selectionAttributes }, position );\n\t\t\t} else {\n\t\t\t\timageUtils.insertImage( { ...sourceDefinition, ...selectionAttributes } );\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport type ImageUtils from '../imageutils';\n\n/**\n * @module image/image/replaceimagesourcecommand\n */\n\n/**\n * Replace image source command.\n *\n * Changes image source to the one provided. Can be executed as follows:\n *\n * ```ts\n * editor.execute( 'replaceImageSource', { source: 'http://url.to.the/image' } );\n * ```\n */\nexport default class ReplaceImageSourceCommand extends Command {\n\tdeclare public value: string | null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst editor = this.editor;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst element = this.editor.model.document.selection.getSelectedElement()!;\n\n\t\tthis.isEnabled = imageUtils.isImage( element );\n\t\tthis.value = this.isEnabled ? element.getAttribute( 'src' ) as string : null;\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * @fires execute\n\t * @param options Options for the executed command.\n\t * @param options.source The image source to replace.\n\t */\n\tpublic override execute( options: { source: string } ): void {\n\t\tconst image = this.editor.model.document.selection.getSelectedElement()!;\n\t\tthis.editor.model.change( writer => {\n\t\t\twriter.setAttribute( 'src', options.source, image );\n\t\t\twriter.removeAttribute( 'srcset', image );\n\t\t\twriter.removeAttribute( 'sizes', image );\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/imageediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport type { ViewElement } from 'ckeditor5/src/engine';\nimport ImageLoadObserver from './imageloadobserver';\nimport InsertImageCommand from './insertimagecommand';\nimport ReplaceImageSourceCommand from './replaceimagesourcecommand';\nimport ImageUtils from '../imageutils';\n\n/**\n * The image engine plugin. This module loads common code shared between\n * {@link module:image/image/imageinlineediting~ImageInlineEditing} and\n * {@link module:image/image/imageblockediting~ImageBlockEditing} plugins.\n *\n * This plugin registers the {@link module:image/image/insertimagecommand~InsertImageCommand 'insertImage'} command.\n */\nexport default class ImageEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst conversion = editor.conversion;\n\n\t\t// See https://github.com/ckeditor/ckeditor5-image/issues/142.\n\t\teditor.editing.view.addObserver( ImageLoadObserver );\n\n\t\tconversion.for( 'upcast' )\n\t\t\t.attributeToAttribute( {\n\t\t\t\tview: {\n\t\t\t\t\tname: 'img',\n\t\t\t\t\tkey: 'alt'\n\t\t\t\t},\n\t\t\t\tmodel: 'alt'\n\t\t\t} )\n\t\t\t.attributeToAttribute( {\n\t\t\t\tview: {\n\t\t\t\t\tname: 'img',\n\t\t\t\t\tkey: 'srcset'\n\t\t\t\t},\n\t\t\t\tmodel: {\n\t\t\t\t\tkey: 'srcset',\n\t\t\t\t\tvalue: ( viewImage: ViewElement ) => {\n\t\t\t\t\t\tconst value: Record<string, string> = {\n\t\t\t\t\t\t\tdata: viewImage.getAttribute( 'srcset' )!\n\t\t\t\t\t\t};\n\n\t\t\t\t\t\tif ( viewImage.hasAttribute( 'width' ) ) {\n\t\t\t\t\t\t\tvalue.width = viewImage.getAttribute( 'width' )!;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\treturn value;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\n\t\tconst insertImageCommand = new InsertImageCommand( editor );\n\t\tconst replaceImageSourceCommand = new ReplaceImageSourceCommand( editor );\n\n\t\teditor.commands.add( 'insertImage', insertImageCommand );\n\t\teditor.commands.add( 'replaceImageSource', replaceImageSourceCommand );\n\n\t\t// `imageInsert` is an alias for backward compatibility.\n\t\teditor.commands.add( 'imageInsert', insertImageCommand );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/imagetypecommand\n */\n\nimport type { Element } from 'ckeditor5/src/engine';\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport type ImageUtils from '../imageutils';\n\n/**\n * The image type command. It changes the type of a selected image, depending on the configuration.\n */\nexport default class ImageTypeCommand extends Command {\n\t/**\n\t * Model element name the command converts to.\n\t */\n\tprivate readonly _modelElementName: 'imageBlock' | 'imageInline';\n\n\t/**\n\t * @inheritDoc\n\t *\n\t * @param modelElementName Model element name the command converts to.\n\t */\n\tconstructor( editor: Editor, modelElementName: 'imageBlock' | 'imageInline' ) {\n\t\tsuper( editor );\n\n\t\tthis._modelElementName = modelElementName;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst editor = this.editor;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst element = imageUtils.getClosestSelectedImageElement( this.editor.model.document.selection );\n\n\t\tif ( this._modelElementName === 'imageBlock' ) {\n\t\t\tthis.isEnabled = imageUtils.isInlineImage( element );\n\t\t} else {\n\t\t\tthis.isEnabled = imageUtils.isBlockImage( element );\n\t\t}\n\t}\n\n\t/**\n\t * Executes the command and changes the type of a selected image.\n\t *\n\t * @fires execute\n\t * @returns An object containing references to old and new model image elements\n\t * (for before and after the change) so external integrations can hook into the decorated\n\t * `execute` event and handle this change. `null` if the type change failed.\n\t */\n\tpublic override execute(): { oldElement: Element; newElement: Element } | null {\n\t\tconst editor = this.editor;\n\t\tconst model = this.editor.model;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst oldElement = imageUtils.getClosestSelectedImageElement( model.document.selection )!;\n\t\tconst attributes = Object.fromEntries( oldElement.getAttributes() );\n\n\t\t// Don't change image type if \"src\" is missing (a broken image), unless there's \"uploadId\" set.\n\t\t// This state may happen during image upload (before it finishes) and it should be possible to change type\n\t\t// of the image in the meantime.\n\t\tif ( !attributes.src && !attributes.uploadId ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn model.change( writer => {\n\t\t\t// Get all markers that contain the old image element.\n\t\t\tconst markers = Array.from( model.markers )\n\t\t\t\t.filter( marker => marker.getRange().containsItem( oldElement ) );\n\n\t\t\tconst newElement = imageUtils.insertImage( attributes, model.createSelection( oldElement, 'on' ), this._modelElementName );\n\n\t\t\tif ( !newElement ) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\tconst newElementRange = writer.createRangeOn( newElement );\n\n\t\t\t// Expand the previously intersecting markers' ranges to include the new image element.\n\t\t\tfor ( const marker of markers ) {\n\t\t\t\tconst markerRange = marker.getRange();\n\n\t\t\t\t// Join the survived part of the old marker range with the new element range\n\t\t\t\t// (loosely because there could be some new paragraph or the existing one might got split).\n\t\t\t\tconst range = markerRange.root.rootName != '$graveyard' ?\n\t\t\t\t\tmarkerRange.getJoined( newElementRange, true )! : newElementRange;\n\n\t\t\t\twriter.updateMarker( marker, { range } );\n\t\t\t}\n\n\t\t\treturn {\n\t\t\t\toldElement,\n\t\t\t\tnewElement\n\t\t\t};\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/imageblockediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { ClipboardPipeline, type ClipboardInputTransformationEvent } from 'ckeditor5/src/clipboard';\nimport { UpcastWriter, type ViewElement } from 'ckeditor5/src/engine';\n\nimport {\n\tdowncastImageAttribute,\n\tdowncastSrcsetAttribute,\n\tupcastImageFigure\n} from './converters';\n\nimport ImageEditing from './imageediting';\nimport ImageTypeCommand from './imagetypecommand';\nimport ImageUtils from '../imageutils';\nimport {\n\tgetImgViewElementMatcher,\n\tcreateBlockImageViewElement,\n\tdetermineImageTypeForInsertionAtSelection\n} from '../image/utils';\n\n/**\n * The image block plugin.\n *\n * It registers:\n *\n * * `<imageBlock>` as a block element in the document schema, and allows `alt`, `src` and `srcset` attributes.\n * * converters for editing and data pipelines.,\n * * {@link module:image/image/imagetypecommand~ImageTypeCommand `'imageTypeBlock'`} command that converts inline images into\n * block images.\n */\nexport default class ImageBlockEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageEditing, ImageUtils, ClipboardPipeline ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageBlockEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\n\t\t// Converters 'alt' and 'srcset' are added in 'ImageEditing' plugin.\n\t\tschema.register( 'imageBlock', {\n\t\t\tinheritAllFrom: '$blockObject',\n\t\t\tallowAttributes: [ 'alt', 'src', 'srcset' ]\n\t\t} );\n\n\t\tthis._setupConversion();\n\n\t\tif ( editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\t\teditor.commands.add( 'imageTypeBlock', new ImageTypeCommand( this.editor, 'imageBlock' ) );\n\n\t\t\tthis._setupClipboardIntegration();\n\t\t}\n\t}\n\n\t/**\n\t * Configures conversion pipelines to support upcasting and downcasting\n\t * block images (block image widgets) and their attributes.\n\t */\n\tprivate _setupConversion(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst conversion = editor.conversion;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\t\tconversion.for( 'dataDowncast' )\n\t\t\t.elementToStructure( {\n\t\t\t\tmodel: 'imageBlock',\n\t\t\t\tview: ( modelElement, { writer } ) => createBlockImageViewElement( writer )\n\t\t\t} );\n\n\t\tconversion.for( 'editingDowncast' )\n\t\t\t.elementToStructure( {\n\t\t\t\tmodel: 'imageBlock',\n\t\t\t\tview: ( modelElement, { writer } ) => imageUtils.toImageWidget(\n\t\t\t\t\tcreateBlockImageViewElement( writer ), writer, t( 'image widget' )\n\t\t\t\t)\n\t\t\t} );\n\n\t\tconversion.for( 'downcast' )\n\t\t\t.add( downcastImageAttribute( imageUtils, 'imageBlock', 'src' ) )\n\t\t\t.add( downcastImageAttribute( imageUtils, 'imageBlock', 'alt' ) )\n\t\t\t.add( downcastSrcsetAttribute( imageUtils, 'imageBlock' ) );\n\n\t\t// More image related upcasts are in 'ImageEditing' plugin.\n\t\tconversion.for( 'upcast' )\n\t\t\t.elementToElement( {\n\t\t\t\tview: getImgViewElementMatcher( editor, 'imageBlock' ),\n\t\t\t\tmodel: ( viewImage, { writer } ) => writer.createElement(\n\t\t\t\t\t'imageBlock',\n\t\t\t\t\tviewImage.hasAttribute( 'src' ) ? { src: viewImage.getAttribute( 'src' ) } : undefined\n\t\t\t\t)\n\t\t\t} )\n\t\t\t.add( upcastImageFigure( imageUtils ) );\n\t}\n\n\t/**\n\t * Integrates the plugin with the clipboard pipeline.\n\t *\n\t * Idea is that the feature should recognize the user's intent when an **inline** image is\n\t * pasted or dropped. If such an image is pasted/dropped:\n\t *\n\t * * into an empty block (e.g. an empty paragraph),\n\t * * on another object (e.g. some block widget).\n\t *\n\t * it gets converted into a block image on the fly. We assume this is the user's intent\n\t * if they decided to put their image there.\n\t *\n\t * See the `ImageInlineEditing` for the similar integration that works in the opposite direction.\n\t */\n\tprivate _setupClipboardIntegration(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst editingView = editor.editing.view;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst clipboardPipeline: ClipboardPipeline = editor.plugins.get( 'ClipboardPipeline' );\n\n\t\tthis.listenTo<ClipboardInputTransformationEvent>(\n\t\t\tclipboardPipeline,\n\t\t\t'inputTransformation',\n\t\t\t( evt, data ) => {\n\t\t\t\tconst docFragmentChildren = Array.from( data.content.getChildren() as IterableIterator<ViewElement> );\n\t\t\t\tlet modelRange;\n\n\t\t\t\t// Make sure only <img> elements are dropped or pasted. Otherwise, if there some other HTML\n\t\t\t\t// mixed up, this should be handled as a regular paste.\n\t\t\t\tif ( !docFragmentChildren.every( imageUtils.isInlineImageView ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// When drag and dropping, data.targetRanges specifies where to drop because\n\t\t\t\t// this is usually a different place than the current model selection (the user\n\t\t\t\t// uses a drop marker to specify the drop location).\n\t\t\t\tif ( data.targetRanges ) {\n\t\t\t\t\tmodelRange = editor.editing.mapper.toModelRange( data.targetRanges[ 0 ] );\n\t\t\t\t}\n\t\t\t\t// Pasting, however, always occurs at the current model selection.\n\t\t\t\telse {\n\t\t\t\t\tmodelRange = model.document.selection.getFirstRange();\n\t\t\t\t}\n\n\t\t\t\tconst selection = model.createSelection( modelRange );\n\n\t\t\t\t// Convert inline images into block images only when the currently selected block is empty\n\t\t\t\t// (e.g. an empty paragraph) or some object is selected (to replace it).\n\t\t\t\tif ( determineImageTypeForInsertionAtSelection( model.schema, selection ) === 'imageBlock' ) {\n\t\t\t\t\tconst writer = new UpcastWriter( editingView.document );\n\n\t\t\t\t\t// Wrap <img ... /> -> <figure class=\"image\"><img .../></figure>\n\t\t\t\t\tconst blockViewImages = docFragmentChildren.map(\n\t\t\t\t\t\tinlineViewImage => writer.createElement( 'figure', { class: 'image' }, inlineViewImage )\n\t\t\t\t\t);\n\n\t\t\t\t\tdata.content = writer.createDocumentFragment( blockViewImages );\n\t\t\t\t}\n\t\t\t} );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./image.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imageblock\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { Widget } from 'ckeditor5/src/widget';\n\nimport ImageTextAlternative from './imagetextalternative';\nimport ImageBlockEditing from './image/imageblockediting';\n\nimport '../theme/image.css';\n\n/**\n * The image block plugin.\n *\n * This is a \"glue\" plugin which loads the following plugins:\n *\n * * {@link module:image/image/imageblockediting~ImageBlockEditing},\n * * {@link module:image/imagetextalternative~ImageTextAlternative}.\n *\n * Usually, it is used in conjunction with other plugins from this package. See the {@glink api/image package page}\n * for more information.\n */\nexport default class ImageBlock extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageBlockEditing, Widget, ImageTextAlternative ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageBlock' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image/imageinlineediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { ClipboardPipeline, type ClipboardInputTransformationEvent } from 'ckeditor5/src/clipboard';\nimport { UpcastWriter, type ViewElement } from 'ckeditor5/src/engine';\n\nimport {\n\tdowncastImageAttribute,\n\tdowncastSrcsetAttribute\n} from './converters';\n\nimport ImageEditing from './imageediting';\nimport ImageTypeCommand from './imagetypecommand';\nimport ImageUtils from '../imageutils';\nimport {\n\tgetImgViewElementMatcher,\n\tcreateInlineImageViewElement,\n\tdetermineImageTypeForInsertionAtSelection\n} from '../image/utils';\n\n/**\n * The image inline plugin.\n *\n * It registers:\n *\n * * `<imageInline>` as an inline element in the document schema, and allows `alt`, `src` and `srcset` attributes.\n * * converters for editing and data pipelines.\n * * {@link module:image/image/imagetypecommand~ImageTypeCommand `'imageTypeInline'`} command that converts block images into\n * inline images.\n */\nexport default class ImageInlineEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageEditing, ImageUtils, ClipboardPipeline ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageInlineEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\n\t\t// Converters 'alt' and 'srcset' are added in 'ImageEditing' plugin.\n\t\tschema.register( 'imageInline', {\n\t\t\tinheritAllFrom: '$inlineObject',\n\t\t\tallowAttributes: [ 'alt', 'src', 'srcset' ]\n\t\t} );\n\n\t\t// Disallow inline images in captions (for now). This is the best spot to do that because\n\t\t// independent packages can introduce captions (ImageCaption, TableCaption, etc.) so better this\n\t\t// be future-proof.\n\t\tschema.addChildCheck( ( context, childDefinition ) => {\n\t\t\tif ( context.endsWith( 'caption' ) && childDefinition.name === 'imageInline' ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t} );\n\n\t\tthis._setupConversion();\n\n\t\tif ( editor.plugins.has( 'ImageBlockEditing' ) ) {\n\t\t\teditor.commands.add( 'imageTypeInline', new ImageTypeCommand( this.editor, 'imageInline' ) );\n\n\t\t\tthis._setupClipboardIntegration();\n\t\t}\n\t}\n\n\t/**\n\t * Configures conversion pipelines to support upcasting and downcasting\n\t * inline images (inline image widgets) and their attributes.\n\t */\n\tprivate _setupConversion(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst conversion = editor.conversion;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\t\tconversion.for( 'dataDowncast' )\n\t\t\t.elementToElement( {\n\t\t\t\tmodel: 'imageInline',\n\t\t\t\tview: ( modelElement, { writer } ) => writer.createEmptyElement( 'img' )\n\t\t\t} );\n\n\t\tconversion.for( 'editingDowncast' )\n\t\t\t.elementToStructure( {\n\t\t\t\tmodel: 'imageInline',\n\t\t\t\tview: ( modelElement, { writer } ) => imageUtils.toImageWidget(\n\t\t\t\t\tcreateInlineImageViewElement( writer ), writer, t( 'image widget' )\n\t\t\t\t)\n\t\t\t} );\n\n\t\tconversion.for( 'downcast' )\n\t\t\t.add( downcastImageAttribute( imageUtils, 'imageInline', 'src' ) )\n\t\t\t.add( downcastImageAttribute( imageUtils, 'imageInline', 'alt' ) )\n\t\t\t.add( downcastSrcsetAttribute( imageUtils, 'imageInline' ) );\n\n\t\t// More image related upcasts are in 'ImageEditing' plugin.\n\t\tconversion.for( 'upcast' )\n\t\t\t.elementToElement( {\n\t\t\t\tview: getImgViewElementMatcher( editor, 'imageInline' ),\n\t\t\t\tmodel: ( viewImage, { writer } ) => writer.createElement(\n\t\t\t\t\t'imageInline',\n\t\t\t\t\tviewImage.hasAttribute( 'src' ) ? { src: viewImage.getAttribute( 'src' ) } : undefined\n\t\t\t\t)\n\t\t\t} );\n\t}\n\n\t/**\n\t * Integrates the plugin with the clipboard pipeline.\n\t *\n\t * Idea is that the feature should recognize the user's intent when an **block** image is\n\t * pasted or dropped. If such an image is pasted/dropped into a non-empty block\n\t * (e.g. a paragraph with some text) it gets converted into an inline image on the fly.\n\t *\n\t * We assume this is the user's intent if they decided to put their image there.\n\t *\n\t * **Note**: If a block image has a caption, it will not be converted to an inline image\n\t * to avoid the confusion. Captions are added on purpose and they should never be lost\n\t * in the clipboard pipeline.\n\t *\n\t * See the `ImageBlockEditing` for the similar integration that works in the opposite direction.\n\t */\n\tprivate _setupClipboardIntegration(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst editingView = editor.editing.view;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst clipboardPipeline: ClipboardPipeline = editor.plugins.get( 'ClipboardPipeline' );\n\n\t\tthis.listenTo<ClipboardInputTransformationEvent>(\n\t\t\tclipboardPipeline,\n\t\t\t'inputTransformation',\n\t\t\t( evt, data ) => {\n\t\t\t\tconst docFragmentChildren = Array.from( data.content.getChildren() as IterableIterator<ViewElement> );\n\t\t\t\tlet modelRange;\n\n\t\t\t\t// Make sure only <figure class=\"image\"></figure> elements are dropped or pasted. Otherwise, if there some other HTML\n\t\t\t\t// mixed up, this should be handled as a regular paste.\n\t\t\t\tif ( !docFragmentChildren.every( imageUtils.isBlockImageView ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// When drag and dropping, data.targetRanges specifies where to drop because\n\t\t\t\t// this is usually a different place than the current model selection (the user\n\t\t\t\t// uses a drop marker to specify the drop location).\n\t\t\t\tif ( data.targetRanges ) {\n\t\t\t\t\tmodelRange = editor.editing.mapper.toModelRange( data.targetRanges[ 0 ] );\n\t\t\t\t}\n\t\t\t\t// Pasting, however, always occurs at the current model selection.\n\t\t\t\telse {\n\t\t\t\t\tmodelRange = model.document.selection.getFirstRange();\n\t\t\t\t}\n\n\t\t\t\tconst selection = model.createSelection( modelRange );\n\n\t\t\t\t// Convert block images into inline images only when pasting or dropping into non-empty blocks\n\t\t\t\t// and when the block is not an object (e.g. pasting to replace another widget).\n\t\t\t\tif ( determineImageTypeForInsertionAtSelection( model.schema, selection ) === 'imageInline' ) {\n\t\t\t\t\tconst writer = new UpcastWriter( editingView.document );\n\n\t\t\t\t\t// Unwrap <figure class=\"image\"><img .../></figure> -> <img ... />\n\t\t\t\t\t// but <figure class=\"image\"><img .../><figcaption>...</figcaption></figure> -> stays the same\n\t\t\t\t\tconst inlineViewImages = docFragmentChildren.map( blockViewImage => {\n\t\t\t\t\t// If there's just one child, it can be either <img /> or <a><img></a>.\n\t\t\t\t\t// If there are other children than <img>, this means that the block image\n\t\t\t\t\t// has a caption or some other features and this kind of image should be\n\t\t\t\t\t// pasted/dropped without modifications.\n\t\t\t\t\t\tif ( blockViewImage.childCount === 1 ) {\n\t\t\t\t\t\t// Pass the attributes which are present only in the <figure> to the <img>\n\t\t\t\t\t\t// (e.g. the style=\"width:10%\" attribute applied by the ImageResize plugin).\n\t\t\t\t\t\t\tArray.from( blockViewImage.getAttributes() )\n\t\t\t\t\t\t\t\t.forEach( attribute => writer.setAttribute(\n\t\t\t\t\t\t\t\t\t...attribute,\n\t\t\t\t\t\t\t\t\timageUtils.findViewImgElement( blockViewImage )!\n\t\t\t\t\t\t\t\t) );\n\n\t\t\t\t\t\t\treturn blockViewImage.getChild( 0 )!;\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\treturn blockViewImage;\n\t\t\t\t\t\t}\n\t\t\t\t\t} );\n\n\t\t\t\t\tdata.content = writer.createDocumentFragment( inlineViewImages );\n\t\t\t\t}\n\t\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imageinline\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { Widget } from 'ckeditor5/src/widget';\n\nimport ImageTextAlternative from './imagetextalternative';\nimport ImageInlineEditing from './image/imageinlineediting';\n\nimport '../theme/image.css';\n\n/**\n * The image inline plugin.\n *\n * This is a \"glue\" plugin which loads the following plugins:\n *\n * * {@link module:image/image/imageinlineediting~ImageInlineEditing},\n * * {@link module:image/imagetextalternative~ImageTextAlternative}.\n *\n * Usually, it is used in conjunction with other plugins from this package. See the {@glink api/image package page}\n * for more information.\n */\nexport default class ImageInline extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageInlineEditing, Widget, ImageTextAlternative ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageInline' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagecaption/imagecaptionutils\n */\n\nimport type { DocumentSelection, Element, Selection, ViewElement, Match } from 'ckeditor5/src/engine';\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport ImageUtils from '../imageutils';\n\n/**\n * The image caption utilities plugin.\n */\nexport default class ImageCaptionUtils extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageCaptionUtils' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageUtils ] as const;\n\t}\n\n\t/**\n\t * Returns the caption model element from a given image element. Returns `null` if no caption is found.\n\t */\n\tpublic getCaptionFromImageModelElement( imageModelElement: Element ): Element | null {\n\t\tfor ( const node of imageModelElement.getChildren() ) {\n\t\t\tif ( !!node && node.is( 'element', 'caption' ) ) {\n\t\t\t\treturn node;\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns the caption model element for a model selection. Returns `null` if the selection has no caption element ancestor.\n\t */\n\tpublic getCaptionFromModelSelection( selection: Selection | DocumentSelection ): Element | null {\n\t\tconst imageUtils: ImageUtils = this.editor.plugins.get( 'ImageUtils' );\n\t\tconst captionElement = selection.getFirstPosition()!.findAncestor( 'caption' );\n\n\t\tif ( !captionElement ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( imageUtils.isBlockImage( captionElement.parent as Element ) ) {\n\t\t\treturn captionElement;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * {@link module:engine/view/matcher~Matcher} pattern. Checks if a given element is a `<figcaption>` element that is placed\n\t * inside the image `<figure>` element.\n\t * @returns Returns the object accepted by {@link module:engine/view/matcher~Matcher} or `null` if the element\n\t * cannot be matched.\n\t */\n\tpublic matchImageCaptionViewElement( element: ViewElement ): Match | null {\n\t\tconst imageUtils: ImageUtils = this.editor.plugins.get( 'ImageUtils' );\n\n\t\t// Convert only captions for images.\n\t\tif ( element.name == 'figcaption' && imageUtils.isBlockImageView( element.parent as ViewElement ) ) {\n\t\t\treturn { name: true };\n\t\t}\n\n\t\treturn null;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagecaption/toggleimagecaptioncommand\n */\n\nimport type { Element, Writer } from 'ckeditor5/src/engine';\nimport { Command } from 'ckeditor5/src/core';\n\nimport ImageBlockEditing from '../image/imageblockediting';\nimport type ImageCaptionUtils from './imagecaptionutils';\nimport type ImageUtils from '../imageutils';\nimport type ImageCaptionEditing from './imagecaptionediting';\n\n/**\n * The toggle image caption command.\n *\n * This command is registered by {@link module:image/imagecaption/imagecaptionediting~ImageCaptionEditing} as the\n * `'toggleImageCaption'` editor command.\n *\n * Executing this command:\n *\n * * either adds or removes the image caption of a selected image (depending on whether the caption is present or not),\n * * removes the image caption if the selection is anchored in one.\n *\n * ```ts\n * // Toggle the presence of the caption.\n * editor.execute( 'toggleImageCaption' );\n * ```\n *\n * **Note**: Upon executing this command, the selection will be set on the image if previously anchored in the caption element.\n *\n * **Note**: You can move the selection to the caption right away as it shows up upon executing this command by using\n * the `focusCaptionOnShow` option:\n *\n * ```ts\n * editor.execute( 'toggleImageCaption', { focusCaptionOnShow: true } );\n * ```\n */\nexport default class ToggleImageCaptionCommand extends Command {\n\tdeclare public value: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst editor = this.editor;\n\t\tconst imageCaptionUtils: ImageCaptionUtils = editor.plugins.get( 'ImageCaptionUtils' );\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\t\t// Only block images can get captions.\n\t\tif ( !editor.plugins.has( ImageBlockEditing ) ) {\n\t\t\tthis.isEnabled = false;\n\t\t\tthis.value = false;\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst selection = editor.model.document.selection;\n\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\tif ( !selectedElement ) {\n\t\t\tconst ancestorCaptionElement = imageCaptionUtils.getCaptionFromModelSelection( selection );\n\n\t\t\tthis.isEnabled = !!ancestorCaptionElement;\n\t\t\tthis.value = !!ancestorCaptionElement;\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Block images support captions by default but the command should also be enabled for inline\n\t\t// images because toggling the caption when one is selected should convert it into a block image.\n\t\tthis.isEnabled = imageUtils.isImage( selectedElement );\n\n\t\tif ( !this.isEnabled ) {\n\t\t\tthis.value = false;\n\t\t} else {\n\t\t\tthis.value = !!imageCaptionUtils.getCaptionFromImageModelElement( selectedElement );\n\t\t}\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * ```ts\n\t * editor.execute( 'toggleImageCaption' );\n\t * ```\n\t *\n\t * @param options Options for the executed command.\n\t * @param options.focusCaptionOnShow When true and the caption shows up, the selection will be moved into it straight away.\n\t * @fires execute\n\t */\n\tpublic override execute( options: { focusCaptionOnShow?: boolean } = {} ): void {\n\t\tconst { focusCaptionOnShow } = options;\n\n\t\tthis.editor.model.change( writer => {\n\t\t\tif ( this.value ) {\n\t\t\t\tthis._hideImageCaption( writer );\n\t\t\t} else {\n\t\t\t\tthis._showImageCaption( writer, focusCaptionOnShow );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Shows the caption of the `<imageBlock>` or `<imageInline>`. Also:\n\t *\n\t * * it converts `<imageInline>` to `<imageBlock>` to show the caption,\n\t * * it attempts to restore the caption content from the `ImageCaptionEditing` caption registry,\n\t * * it moves the selection to the caption right away, it the `focusCaptionOnShow` option was set.\n\t */\n\tprivate _showImageCaption( writer: Writer, focusCaptionOnShow?: boolean ): void {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst imageCaptionEditing: ImageCaptionEditing = this.editor.plugins.get( 'ImageCaptionEditing' );\n\t\tconst imageUtils: ImageUtils = this.editor.plugins.get( 'ImageUtils' );\n\n\t\tlet selectedImage = selection.getSelectedElement()!;\n\n\t\tconst savedCaption = imageCaptionEditing._getSavedCaption( selectedImage );\n\n\t\t// Convert imageInline -> image first.\n\t\tif ( imageUtils.isInlineImage( selectedImage ) ) {\n\t\t\tthis.editor.execute( 'imageTypeBlock' );\n\n\t\t\t// Executing the command created a new model element. Let's pick it again.\n\t\t\tselectedImage = selection.getSelectedElement()!;\n\t\t}\n\n\t\t// Try restoring the caption from the ImageCaptionEditing plugin storage.\n\t\tconst newCaptionElement = savedCaption || writer.createElement( 'caption' );\n\n\t\twriter.append( newCaptionElement, selectedImage );\n\n\t\tif ( focusCaptionOnShow ) {\n\t\t\twriter.setSelection( newCaptionElement, 'in' );\n\t\t}\n\t}\n\n\t/**\n\t * Hides the caption of a selected image (or an image caption the selection is anchored to).\n\t *\n\t * The content of the caption is stored in the `ImageCaptionEditing` caption registry to make this\n\t * a reversible action.\n\t */\n\tprivate _hideImageCaption( writer: Writer ): void {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\t\tconst imageCaptionEditing: ImageCaptionEditing = editor.plugins.get( 'ImageCaptionEditing' );\n\t\tconst imageCaptionUtils: ImageCaptionUtils = editor.plugins.get( 'ImageCaptionUtils' );\n\t\tlet selectedImage = selection.getSelectedElement()!;\n\t\tlet captionElement: Element;\n\n\t\tif ( selectedImage ) {\n\t\t\tcaptionElement = imageCaptionUtils.getCaptionFromImageModelElement( selectedImage )!;\n\t\t} else {\n\t\t\tcaptionElement = imageCaptionUtils.getCaptionFromModelSelection( selection )!;\n\t\t\tselectedImage = captionElement!.parent as Element;\n\t\t}\n\n\t\t// Store the caption content so it can be restored quickly if the user changes their mind even if they toggle image<->imageInline.\n\t\timageCaptionEditing._saveCaption( selectedImage, captionElement );\n\n\t\twriter.setSelection( selectedImage, 'on' );\n\t\twriter.remove( captionElement );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagecaption/imagecaptionediting\n */\n\nimport { type Editor, Plugin, type CommandExecuteEvent } from 'ckeditor5/src/core';\nimport { Element, enablePlaceholder, type DocumentChangeEvent, type DiffItemAttribute, PlaceholderableElement } from 'ckeditor5/src/engine';\nimport { toWidgetEditable } from 'ckeditor5/src/widget';\nimport type { GetCallback } from 'ckeditor5/src/utils';\n\nimport ToggleImageCaptionCommand from './toggleimagecaptioncommand';\nimport ImageUtils from '../imageutils';\nimport ImageCaptionUtils from './imagecaptionutils';\n\n/**\n * The image caption engine plugin. It is responsible for:\n *\n * * registering converters for the caption element,\n * * registering converters for the caption model attribute,\n * * registering the {@link module:image/imagecaption/toggleimagecaptioncommand~ToggleImageCaptionCommand `toggleImageCaption`} command.\n */\nexport default class ImageCaptionEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageUtils, ImageCaptionUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageCaptionEditing' as const;\n\t}\n\n\t/**\n\t * A map that keeps saved JSONified image captions and image model elements they are\n\t * associated with.\n\t *\n\t * To learn more about this system, see {@link #_saveCaption}.\n\t */\n\tprivate _savedCaptionsMap: WeakMap<Element, unknown>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis._savedCaptionsMap = new WeakMap();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\n\t\t// Schema configuration.\n\t\tif ( !schema.isRegistered( 'caption' ) ) {\n\t\t\tschema.register( 'caption', {\n\t\t\t\tallowIn: 'imageBlock',\n\t\t\t\tallowContentOf: '$block',\n\t\t\t\tisLimit: true\n\t\t\t} );\n\t\t} else {\n\t\t\tschema.extend( 'caption', {\n\t\t\t\tallowIn: 'imageBlock'\n\t\t\t} );\n\t\t}\n\n\t\teditor.commands.add( 'toggleImageCaption', new ToggleImageCaptionCommand( this.editor ) );\n\n\t\tthis._setupConversion();\n\t\tthis._setupImageTypeCommandsIntegration();\n\t\tthis._registerCaptionReconversion();\n\t}\n\n\t/**\n\t * Configures conversion pipelines to support upcasting and downcasting\n\t * image captions.\n\t */\n\tprivate _setupConversion(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst imageCaptionUtils: ImageCaptionUtils = editor.plugins.get( 'ImageCaptionUtils' );\n\t\tconst t = editor.t;\n\n\t\t// View -> model converter for the data pipeline.\n\t\teditor.conversion.for( 'upcast' ).elementToElement( {\n\t\t\tview: element => imageCaptionUtils.matchImageCaptionViewElement( element ),\n\t\t\tmodel: 'caption'\n\t\t} );\n\n\t\t// Model -> view converter for the data pipeline.\n\t\teditor.conversion.for( 'dataDowncast' ).elementToElement( {\n\t\t\tmodel: 'caption',\n\t\t\tview: ( modelElement, { writer } ) => {\n\t\t\t\tif ( !imageUtils.isBlockImage( modelElement.parent as Element ) ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\treturn writer.createContainerElement( 'figcaption' );\n\t\t\t}\n\t\t} );\n\n\t\t// Model -> view converter for the editing pipeline.\n\t\teditor.conversion.for( 'editingDowncast' ).elementToElement( {\n\t\t\tmodel: 'caption',\n\t\t\tview: ( modelElement, { writer } ) => {\n\t\t\t\tif ( !imageUtils.isBlockImage( modelElement.parent as Element ) ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\tconst figcaptionElement = writer.createEditableElement( 'figcaption' );\n\t\t\t\twriter.setCustomProperty( 'imageCaption', true, figcaptionElement );\n\n\t\t\t\tfigcaptionElement.placeholder = t( 'Enter image caption' );\n\t\t\t\tenablePlaceholder( {\n\t\t\t\t\tview,\n\t\t\t\t\telement: figcaptionElement,\n\t\t\t\t\tkeepOnFocus: true\n\t\t\t\t} );\n\n\t\t\t\tconst imageAlt = ( modelElement.parent as Element ).getAttribute( 'alt' ) as string;\n\t\t\t\tconst label = imageAlt ? t( 'Caption for image: %0', [ imageAlt ] ) : t( 'Caption for the image' );\n\n\t\t\t\treturn toWidgetEditable( figcaptionElement, writer, { label } );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Integrates with {@link module:image/image/imagetypecommand~ImageTypeCommand image type commands}\n\t * to make sure the caption is preserved when the type of an image changes so it can be restored\n\t * in the future if the user decides they want their caption back.\n\t */\n\tprivate _setupImageTypeCommandsIntegration(): void {\n\t\tconst editor = this.editor;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst imageCaptionUtils: ImageCaptionUtils = editor.plugins.get( 'ImageCaptionUtils' );\n\t\tconst imageTypeInlineCommand = editor.commands.get( 'imageTypeInline' );\n\t\tconst imageTypeBlockCommand = editor.commands.get( 'imageTypeBlock' );\n\n\t\tconst handleImageTypeChange: GetCallback<CommandExecuteEvent> = evt => {\n\t\t\t// The image type command execution can be unsuccessful.\n\t\t\tif ( !evt.return ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst { oldElement, newElement } = evt.return as { oldElement: Element; newElement: Element };\n\n\t\t\t/* istanbul ignore if: paranoid check -- @preserve */\n\t\t\tif ( !oldElement ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( imageUtils.isBlockImage( oldElement ) ) {\n\t\t\t\tconst oldCaptionElement = imageCaptionUtils.getCaptionFromImageModelElement( oldElement );\n\n\t\t\t\t// If the old element was a captioned block image (the caption was visible),\n\t\t\t\t// simply save it so it can be restored.\n\t\t\t\tif ( oldCaptionElement ) {\n\t\t\t\t\tthis._saveCaption( newElement, oldCaptionElement );\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst savedOldElementCaption = this._getSavedCaption( oldElement );\n\n\t\t\t// If either:\n\t\t\t//\n\t\t\t// * the block image didn't have a visible caption,\n\t\t\t// * the block image caption was hidden (and already saved),\n\t\t\t// * the inline image was passed\n\t\t\t//\n\t\t\t// just try to \"pass\" the saved caption from the old image to the new image\n\t\t\t// so it can be retrieved in the future if the user wants it back.\n\t\t\tif ( savedOldElementCaption ) {\n\t\t\t\t// Note: Since we're writing to a WeakMap, we don't bother with removing the\n\t\t\t\t// [ oldElement, savedOldElementCaption ] pair from it.\n\t\t\t\tthis._saveCaption( newElement, savedOldElementCaption );\n\t\t\t}\n\t\t};\n\n\t\t// Presence of the commands depends on the Image(Inline|Block)Editing plugins loaded in the editor.\n\t\tif ( imageTypeInlineCommand ) {\n\t\t\tthis.listenTo<CommandExecuteEvent>( imageTypeInlineCommand, 'execute', handleImageTypeChange, { priority: 'low' } );\n\t\t}\n\n\t\tif ( imageTypeBlockCommand ) {\n\t\t\tthis.listenTo<CommandExecuteEvent>( imageTypeBlockCommand, 'execute', handleImageTypeChange, { priority: 'low' } );\n\t\t}\n\t}\n\n\t/**\n\t * Returns the saved {@link module:engine/model/element~Element#toJSON JSONified} caption\n\t * of an image model element.\n\t *\n\t * See {@link #_saveCaption}.\n\t *\n\t * @internal\n\t * @param imageModelElement The model element the caption should be returned for.\n\t * @returns The model caption element or `null` if there is none.\n\t */\n\tpublic _getSavedCaption( imageModelElement: Element ): Element | null {\n\t\tconst jsonObject = this._savedCaptionsMap.get( imageModelElement );\n\n\t\treturn jsonObject ? Element.fromJSON( jsonObject ) : null;\n\t}\n\n\t/**\n\t * Saves a {@link module:engine/model/element~Element#toJSON JSONified} caption for\n\t * an image element to allow restoring it in the future.\n\t *\n\t * A caption is saved every time it gets hidden and/or the type of an image changes. The\n\t * user should be able to restore it on demand.\n\t *\n\t * **Note**: The caption cannot be stored in the image model element attribute because,\n\t * for instance, when the model state propagates to collaborators, the attribute would get\n\t * lost (mainly because it does not convert to anything when the caption is hidden) and\n\t * the states of collaborators' models would de-synchronize causing numerous issues.\n\t *\n\t * See {@link #_getSavedCaption}.\n\t *\n\t * @internal\n\t * @param imageModelElement The model element the caption is saved for.\n\t * @param caption The caption model element to be saved.\n\t */\n\tpublic _saveCaption( imageModelElement: Element, caption: Element ): void {\n\t\tthis._savedCaptionsMap.set( imageModelElement, caption.toJSON() );\n\t}\n\n\t/**\n\t * Reconverts image caption when image alt attribute changes.\n\t * The change of alt attribute is reflected in caption's aria-label attribute.\n\t */\n\tprivate _registerCaptionReconversion(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst imageCaptionUtils: ImageCaptionUtils = editor.plugins.get( 'ImageCaptionUtils' );\n\n\t\tmodel.document.on<DocumentChangeEvent>( 'change:data', () => {\n\t\t\tconst changes = model.document.differ.getChanges();\n\n\t\t\tfor ( const change of changes as Array<DiffItemAttribute> ) {\n\t\t\t\tif ( change.attributeKey !== 'alt' ) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\tconst image = change.range.start.nodeAfter as Element;\n\n\t\t\t\tif ( imageUtils.isBlockImage( image ) ) {\n\t\t\t\t\tconst caption = imageCaptionUtils.getCaptionFromImageModelElement( image );\n\n\t\t\t\t\tif ( !caption ) {\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\n\t\t\t\t\teditor.editing.reconvertItem( caption );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagecaption/imagecaptionui\n */\n\nimport { Plugin, icons } from 'ckeditor5/src/core';\nimport { ButtonView } from 'ckeditor5/src/ui';\nimport ImageCaptionUtils from './imagecaptionutils';\nimport type ToggleImageCaptionCommand from './toggleimagecaptioncommand';\n\n/**\n * The image caption UI plugin. It introduces the `'toggleImageCaption'` UI button.\n */\nexport default class ImageCaptionUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageCaptionUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageCaptionUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst editingView = editor.editing.view;\n\t\tconst imageCaptionUtils: ImageCaptionUtils = editor.plugins.get( 'ImageCaptionUtils' );\n\t\tconst t = editor.t;\n\n\t\teditor.ui.componentFactory.add( 'toggleImageCaption', locale => {\n\t\t\tconst command: ToggleImageCaptionCommand = editor.commands.get( 'toggleImageCaption' )!;\n\t\t\tconst view = new ButtonView( locale );\n\n\t\t\tview.set( {\n\t\t\t\ticon: icons.caption,\n\t\t\t\ttooltip: true,\n\t\t\t\tisToggleable: true\n\t\t\t} );\n\n\t\t\tview.bind( 'isOn', 'isEnabled' ).to( command, 'value', 'isEnabled' );\n\t\t\tview.bind( 'label' ).to( command, 'value', value => value ? t( 'Toggle caption off' ) : t( 'Toggle caption on' ) );\n\n\t\t\tthis.listenTo( view, 'execute', () => {\n\t\t\t\teditor.execute( 'toggleImageCaption', { focusCaptionOnShow: true } );\n\n\t\t\t\t// Scroll to the selection and highlight the caption if the caption showed up.\n\t\t\t\tconst modelCaptionElement = imageCaptionUtils.getCaptionFromModelSelection( editor.model.document.selection );\n\n\t\t\t\tif ( modelCaptionElement ) {\n\t\t\t\t\tconst figcaptionElement = editor.editing.mapper.toViewElement( modelCaptionElement );\n\n\t\t\t\t\teditingView.scrollToTheSelection();\n\n\t\t\t\t\teditingView.change( writer => {\n\t\t\t\t\t\twriter.addClass( 'image__caption_highlighted', figcaptionElement! );\n\t\t\t\t\t} );\n\t\t\t\t}\n\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn view;\n\t\t} );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./imagecaption.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imageupload/utils\n */\n\n/* global fetch, File */\n\nimport type { ViewElement } from 'ckeditor5/src/engine';\nimport { global } from 'ckeditor5/src/utils';\nimport type ImageUtils from '../imageutils';\n\n/**\n * Creates a regular expression used to test for image files.\n *\n * ```ts\n * const imageType = createImageTypeRegExp( [ 'png', 'jpeg', 'svg+xml', 'vnd.microsoft.icon' ] );\n *\n * console.log( 'is supported image', imageType.test( file.type ) );\n * ```\n */\nexport function createImageTypeRegExp( types: Array<string> ): RegExp {\n\t// Sanitize the MIME type name which may include: \"+\", \"-\" or \".\".\n\tconst regExpSafeNames = types.map( type => type.replace( '+', '\\\\+' ) );\n\n\treturn new RegExp( `^image\\\\/(${ regExpSafeNames.join( '|' ) })$` );\n}\n\n/**\n * Creates a promise that fetches the image local source (Base64 or blob) and resolves with a `File` object.\n *\n * @param image Image whose source to fetch.\n * @returns A promise which resolves when an image source is fetched and converted to a `File` instance.\n * It resolves with a `File` object. If there were any errors during file processing, the promise will be rejected.\n */\nexport function fetchLocalImage( image: ViewElement ): Promise<File> {\n\treturn new Promise( ( resolve, reject ) => {\n\t\tconst imageSrc = image.getAttribute( 'src' )!;\n\n\t\t// Fetch works asynchronously and so does not block browser UI when processing data.\n\t\tfetch( imageSrc )\n\t\t\t.then( resource => resource.blob() )\n\t\t\t.then( blob => {\n\t\t\t\tconst mimeType = getImageMimeType( blob, imageSrc );\n\t\t\t\tconst ext = mimeType.replace( 'image/', '' );\n\t\t\t\tconst filename = `image.${ ext }`;\n\t\t\t\tconst file = new File( [ blob ], filename, { type: mimeType } );\n\n\t\t\t\tresolve( file );\n\t\t\t} )\n\t\t\t.catch( err => {\n\t\t\t\t// Fetch fails only, if it can't make a request due to a network failure or if anything prevented the request\n\t\t\t\t// from completing, i.e. the Content Security Policy rules. It is not possible to detect the exact cause of failure,\n\t\t\t\t// so we are just trying the fallback solution, if general TypeError is thrown.\n\t\t\t\treturn err && err.name === 'TypeError' ?\n\t\t\t\t\tconvertLocalImageOnCanvas( imageSrc ).then( resolve ).catch( reject ) :\n\t\t\t\t\treject( err );\n\t\t\t} );\n\t} );\n}\n\n/**\n * Checks whether a given node is an image element with a local source (Base64 or blob).\n *\n * @param node The node to check.\n */\nexport function isLocalImage( imageUtils: ImageUtils, node: ViewElement ): boolean {\n\tif ( !imageUtils.isInlineImageView( node ) || !node.getAttribute( 'src' ) ) {\n\t\treturn false;\n\t}\n\n\treturn !!node.getAttribute( 'src' )!.match( /^data:image\\/\\w+;base64,/g ) ||\n\t\t!!node.getAttribute( 'src' )!.match( /^blob:/g );\n}\n\n/**\n * Extracts an image type based on its blob representation or its source.\n * @param blob Image blob representation.\n * @param src Image `src` attribute value.\n */\nfunction getImageMimeType( blob: Blob, src: string ): string {\n\tif ( blob.type ) {\n\t\treturn blob.type;\n\t} else if ( src.match( /data:(image\\/\\w+);base64/ ) ) {\n\t\treturn src.match( /data:(image\\/\\w+);base64/ )![ 1 ].toLowerCase();\n\t} else {\n\t\t// Fallback to 'jpeg' as common extension.\n\t\treturn 'image/jpeg';\n\t}\n}\n\n/**\n * Creates a promise that converts the image local source (Base64 or blob) to a blob using canvas and resolves\n * with a `File` object.\n * @param imageSrc Image `src` attribute value.\n * @returns A promise which resolves when an image source is converted to a `File` instance.\n * It resolves with a `File` object. If there were any errors during file processing, the promise will be rejected.\n */\nfunction convertLocalImageOnCanvas( imageSrc: string ): Promise<File> {\n\treturn getBlobFromCanvas( imageSrc ).then( blob => {\n\t\tconst mimeType = getImageMimeType( blob, imageSrc );\n\t\tconst ext = mimeType.replace( 'image/', '' );\n\t\tconst filename = `image.${ ext }`;\n\n\t\treturn new File( [ blob ], filename, { type: mimeType } );\n\t} );\n}\n\n/**\n * Creates a promise that resolves with a `Blob` object converted from the image source (Base64 or blob).\n * @param imageSrc Image `src` attribute value.\n */\nfunction getBlobFromCanvas( imageSrc: string ): Promise<Blob> {\n\treturn new Promise( ( resolve, reject ) => {\n\t\tconst image = global.document.createElement( 'img' );\n\n\t\timage.addEventListener( 'load', () => {\n\t\t\tconst canvas = global.document.createElement( 'canvas' );\n\n\t\t\tcanvas.width = image.width;\n\t\t\tcanvas.height = image.height;\n\n\t\t\tconst ctx = canvas.getContext( '2d' )!;\n\n\t\t\tctx.drawImage( image, 0, 0 );\n\n\t\t\tcanvas.toBlob( blob => blob ? resolve( blob ) : reject() );\n\t\t} );\n\n\t\timage.addEventListener( 'error', () => reject() );\n\n\t\timage.src = imageSrc;\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imageupload/imageuploadui\n */\n\nimport type { Locale } from 'ckeditor5/src/utils';\nimport { Plugin, icons } from 'ckeditor5/src/core';\nimport { FileDialogButtonView } from 'ckeditor5/src/upload';\nimport { createImageTypeRegExp } from './utils';\nimport type UploadImageCommand from './uploadimagecommand';\n\n/**\n * The image upload button plugin.\n *\n * For a detailed overview, check the {@glink features/images/image-upload/image-upload Image upload feature} documentation.\n *\n * Adds the `'uploadImage'` button to the {@link module:ui/componentfactory~ComponentFactory UI component factory}\n * and also the `imageUpload` button as an alias for backward compatibility.\n */\nexport default class ImageUploadUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageUploadUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst componentCreator = ( locale: Locale ) => {\n\t\t\tconst view = new FileDialogButtonView( locale );\n\t\t\tconst command: UploadImageCommand = editor.commands.get( 'uploadImage' )!;\n\t\t\tconst imageTypes = editor.config.get( 'image.upload.types' )!;\n\t\t\tconst imageTypesRegExp = createImageTypeRegExp( imageTypes );\n\n\t\t\tview.set( {\n\t\t\t\tacceptedType: imageTypes.map( type => `image/${ type }` ).join( ',' ),\n\t\t\t\tallowMultipleFiles: true\n\t\t\t} );\n\n\t\t\tview.buttonView.set( {\n\t\t\t\tlabel: t( 'Insert image' ),\n\t\t\t\ticon: icons.image,\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tview.buttonView.bind( 'isEnabled' ).to( command );\n\n\t\t\tview.on( 'done', ( evt, files: FileList ) => {\n\t\t\t\tconst imagesToUpload = Array.from( files ).filter( file => imageTypesRegExp.test( file.type ) );\n\n\t\t\t\tif ( imagesToUpload.length ) {\n\t\t\t\t\teditor.execute( 'uploadImage', { file: imagesToUpload } );\n\n\t\t\t\t\teditor.editing.view.focus();\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\treturn view;\n\t\t};\n\n\t\t// Setup `uploadImage` button and add `imageUpload` button as an alias for backward compatibility.\n\t\teditor.ui.componentFactory.add( 'uploadImage', componentCreator );\n\t\teditor.ui.componentFactory.add( 'imageUpload', componentCreator );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./imageuploadprogress.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./imageuploadicon.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./imageuploadloader.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imageupload/imageuploadprogress\n */\n\n/* globals setTimeout */\n\nimport { type Editor, Plugin } from 'ckeditor5/src/core';\nimport { FileRepository, type FileLoader } from 'ckeditor5/src/upload';\n\nimport '../../theme/imageuploadprogress.css';\nimport '../../theme/imageuploadicon.css';\nimport '../../theme/imageuploadloader.css';\nimport type { GetCallback } from 'ckeditor5/src/utils';\nimport type {\n\tDowncastWriter,\n\tView,\n\tViewElement,\n\tViewContainerElement,\n\tViewUIElement,\n\tDowncastAttributeEvent,\n\tElement\n} from 'ckeditor5/src/engine';\nimport type ImageUtils from '../imageutils';\n\n/**\n * The image upload progress plugin.\n * It shows a placeholder when the image is read from the disk and a progress bar while the image is uploading.\n */\nexport default class ImageUploadProgress extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageUploadProgress' as const;\n\t}\n\n\t/**\n\t * The image placeholder that is displayed before real image data can be accessed.\n\t *\n\t * For the record, this image is a 1x1 px GIF with an aspect ratio set by CSS.\n\t */\n\tprivate placeholder: string;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis.placeholder = '';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Upload status change - update image's view according to that status.\n\t\tif ( editor.plugins.has( 'ImageBlockEditing' ) ) {\n\t\t\teditor.editing.downcastDispatcher.on<DowncastAttributeEvent>(\n\t\t\t\t'attribute:uploadStatus:imageBlock',\n\t\t\t\tthis.uploadStatusChange\n\t\t\t);\n\t\t}\n\n\t\tif ( editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\t\teditor.editing.downcastDispatcher.on<DowncastAttributeEvent>(\n\t\t\t\t'attribute:uploadStatus:imageInline',\n\t\t\t\tthis.uploadStatusChange\n\t\t\t);\n\t\t}\n\t}\n\n\t/**\n\t * This method is called each time the image `uploadStatus` attribute is changed.\n\t *\n\t * @param evt An object containing information about the fired event.\n\t * @param data Additional information about the change.\n\t */\n\tprivate uploadStatusChange: GetCallback<DowncastAttributeEvent> = ( evt, data, conversionApi ) => {\n\t\tconst editor = this.editor;\n\t\tconst modelImage = data.item as Element;\n\t\tconst uploadId = modelImage.getAttribute( 'uploadId' ) as string | number;\n\n\t\tif ( !conversionApi.consumable.consume( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst fileRepository = editor.plugins.get( FileRepository );\n\t\tconst status = uploadId ? data.attributeNewValue : null;\n\t\tconst placeholder = this.placeholder;\n\t\tconst viewFigure = editor.editing.mapper.toViewElement( modelImage )! as ViewContainerElement;\n\t\tconst viewWriter = conversionApi.writer;\n\n\t\tif ( status == 'reading' ) {\n\t\t\t// Start \"appearing\" effect and show placeholder with infinite progress bar on the top\n\t\t\t// while image is read from disk.\n\t\t\t_startAppearEffect( viewFigure, viewWriter );\n\t\t\t_showPlaceholder( imageUtils, placeholder, viewFigure, viewWriter );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Show progress bar on the top of the image when image is uploading.\n\t\tif ( status == 'uploading' ) {\n\t\t\tconst loader = fileRepository.loaders.get( uploadId );\n\n\t\t\t// Start appear effect if needed - see https://github.com/ckeditor/ckeditor5-image/issues/191.\n\t\t\t_startAppearEffect( viewFigure, viewWriter );\n\n\t\t\tif ( !loader ) {\n\t\t\t\t// There is no loader associated with uploadId - this means that image came from external changes.\n\t\t\t\t// In such cases we still want to show the placeholder until image is fully uploaded.\n\t\t\t\t// Show placeholder if needed - see https://github.com/ckeditor/ckeditor5-image/issues/191.\n\t\t\t\t_showPlaceholder( imageUtils, placeholder, viewFigure, viewWriter );\n\t\t\t} else {\n\t\t\t\t// Hide placeholder and initialize progress bar showing upload progress.\n\t\t\t\t_hidePlaceholder( viewFigure, viewWriter );\n\t\t\t\t_showProgressBar( viewFigure, viewWriter, loader, editor.editing.view );\n\t\t\t\t_displayLocalImage( imageUtils, viewFigure!, viewWriter, loader );\n\t\t\t}\n\n\t\t\treturn;\n\t\t}\n\n\t\tif ( status == 'complete' && fileRepository.loaders.get( uploadId ) ) {\n\t\t\t_showCompleteIcon( viewFigure, viewWriter, editor.editing.view );\n\t\t}\n\n\t\t// Clean up.\n\t\t_hideProgressBar( viewFigure, viewWriter );\n\t\t_hidePlaceholder( viewFigure, viewWriter );\n\t\t_stopAppearEffect( viewFigure, viewWriter );\n\t};\n}\n\n/**\n * Adds ck-appear class to the image figure if one is not already applied.\n */\nfunction _startAppearEffect( viewFigure: ViewContainerElement, writer: DowncastWriter ) {\n\tif ( !viewFigure.hasClass( 'ck-appear' ) ) {\n\t\twriter.addClass( 'ck-appear', viewFigure );\n\t}\n}\n\n/**\n * Removes ck-appear class to the image figure if one is not already removed.\n */\nfunction _stopAppearEffect( viewFigure: ViewContainerElement, writer: DowncastWriter ) {\n\twriter.removeClass( 'ck-appear', viewFigure );\n}\n\n/**\n * Shows placeholder together with infinite progress bar on given image figure.\n */\nfunction _showPlaceholder( imageUtils: ImageUtils, placeholder: string, viewFigure: ViewContainerElement, writer: DowncastWriter ) {\n\tif ( !viewFigure.hasClass( 'ck-image-upload-placeholder' ) ) {\n\t\twriter.addClass( 'ck-image-upload-placeholder', viewFigure );\n\t}\n\n\tconst viewImg = imageUtils.findViewImgElement( viewFigure )!;\n\n\tif ( viewImg.getAttribute( 'src' ) !== placeholder ) {\n\t\twriter.setAttribute( 'src', placeholder, viewImg );\n\t}\n\n\tif ( !_getUIElement( viewFigure, 'placeholder' ) ) {\n\t\twriter.insert( writer.createPositionAfter( viewImg ), _createPlaceholder( writer ) );\n\t}\n}\n\n/**\n * Removes placeholder together with infinite progress bar on given image figure.\n */\nfunction _hidePlaceholder( viewFigure: ViewContainerElement, writer: DowncastWriter ) {\n\tif ( viewFigure.hasClass( 'ck-image-upload-placeholder' ) ) {\n\t\twriter.removeClass( 'ck-image-upload-placeholder', viewFigure );\n\t}\n\n\t_removeUIElement( viewFigure, writer, 'placeholder' );\n}\n\n/**\n * Shows progress bar displaying upload progress.\n * Attaches it to the file loader to update when upload percentace is changed.\n */\nfunction _showProgressBar( viewFigure: ViewContainerElement, writer: DowncastWriter, loader: FileLoader, view: View ) {\n\tconst progressBar = _createProgressBar( writer );\n\twriter.insert( writer.createPositionAt( viewFigure, 'end' ), progressBar );\n\n\t// Update progress bar width when uploadedPercent is changed.\n\tloader.on( 'change:uploadedPercent', ( evt, name, value ) => {\n\t\tview.change( writer => {\n\t\t\twriter.setStyle( 'width', value + '%', progressBar );\n\t\t} );\n\t} );\n}\n\n/**\n * Hides upload progress bar.\n */\nfunction _hideProgressBar( viewFigure: ViewContainerElement, writer: DowncastWriter ) {\n\t_removeUIElement( viewFigure, writer, 'progressBar' );\n}\n\n/**\n * Shows complete icon and hides after a certain amount of time.\n */\nfunction _showCompleteIcon( viewFigure: ViewContainerElement, writer: DowncastWriter, view: View ) {\n\tconst completeIcon = writer.createUIElement( 'div', { class: 'ck-image-upload-complete-icon' } );\n\n\twriter.insert( writer.createPositionAt( viewFigure, 'end' ), completeIcon );\n\n\tsetTimeout( () => {\n\t\tview.change( writer => writer.remove( writer.createRangeOn( completeIcon ) ) );\n\t}, 3000 );\n}\n\n/**\n * Create progress bar element using {@link module:engine/view/uielement~UIElement}.\n */\nfunction _createProgressBar( writer: DowncastWriter ): ViewUIElement {\n\tconst progressBar = writer.createUIElement( 'div', { class: 'ck-progress-bar' } );\n\n\twriter.setCustomProperty( 'progressBar', true, progressBar );\n\n\treturn progressBar;\n}\n\n/**\n * Create placeholder element using {@link module:engine/view/uielement~UIElement}.\n */\nfunction _createPlaceholder( writer: DowncastWriter ): ViewUIElement {\n\tconst placeholder = writer.createUIElement( 'div', { class: 'ck-upload-placeholder-loader' } );\n\n\twriter.setCustomProperty( 'placeholder', true, placeholder );\n\n\treturn placeholder;\n}\n\n/**\n * Returns {@link module:engine/view/uielement~UIElement} of given unique property from image figure element.\n * Returns `undefined` if element is not found.\n */\nfunction _getUIElement( imageFigure: ViewElement, uniqueProperty: string ): ViewUIElement | undefined {\n\tfor ( const child of imageFigure.getChildren() ) {\n\t\tif ( ( child as ViewElement ).getCustomProperty( uniqueProperty ) ) {\n\t\t\treturn child as ViewUIElement;\n\t\t}\n\t}\n}\n\n/**\n * Removes {@link module:engine/view/uielement~UIElement} of given unique property from image figure element.\n */\nfunction _removeUIElement( viewFigure: ViewContainerElement, writer: DowncastWriter, uniqueProperty: string ) {\n\tconst element = _getUIElement( viewFigure, uniqueProperty );\n\n\tif ( element ) {\n\t\twriter.remove( writer.createRangeOn( element ) );\n\t}\n}\n\n/**\n * Displays local data from file loader.\n */\nfunction _displayLocalImage( imageUtils: ImageUtils, viewFigure: ViewElement, writer: DowncastWriter, loader: FileLoader ) {\n\tif ( loader.data ) {\n\t\tconst viewImg = imageUtils.findViewImgElement( viewFigure )!;\n\n\t\twriter.setAttribute( 'src', loader.data, viewImg );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport { FileRepository } from 'ckeditor5/src/upload';\nimport { Command } from 'ckeditor5/src/core';\nimport { toArray, type ArrayOrItem } from 'ckeditor5/src/utils';\nimport type { Position } from 'ckeditor5/src/engine';\n\nimport type ImageUtils from '../imageutils';\n\n/**\n * @module image/imageupload/uploadimagecommand\n */\n\n/**\n * The upload image command.\n *\n * The command is registered by the {@link module:image/imageupload/imageuploadediting~ImageUploadEditing} plugin as `uploadImage`\n * and it is also available via aliased `imageUpload` name.\n *\n * In order to upload an image at the current selection position\n * (according to the {@link module:widget/utils~findOptimalInsertionRange} algorithm),\n * execute the command and pass the native image file instance:\n *\n * ```ts\n * this.listenTo( editor.editing.view.document, 'clipboardInput', ( evt, data ) => {\n * \t// Assuming that only images were pasted:\n * \tconst images = Array.from( data.dataTransfer.files );\n *\n * \t// Upload the first image:\n * \teditor.execute( 'uploadImage', { file: images[ 0 ] } );\n * } );\n * ```\n *\n * It is also possible to insert multiple images at once:\n *\n * ```ts\n * editor.execute( 'uploadImage', {\n * \tfile: [\n * \t\tfile1,\n * \t\tfile2\n * \t]\n * } );\n * ```\n */\nexport default class UploadImageCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst editor = this.editor;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst selectedElement = editor.model.document.selection.getSelectedElement()!;\n\n\t\t// TODO: This needs refactoring.\n\t\tthis.isEnabled = imageUtils.isImageAllowed() || imageUtils.isImage( selectedElement );\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * @fires execute\n\t * @param options Options for the executed command.\n\t * @param options.file The image file or an array of image files to upload.\n\t */\n\tpublic override execute( options: { file: ArrayOrItem<File> } ): void {\n\t\tconst files = toArray( options.file );\n\t\tconst selection = this.editor.model.document.selection;\n\t\tconst imageUtils: ImageUtils = this.editor.plugins.get( 'ImageUtils' );\n\n\t\t// In case of multiple files, each file (starting from the 2nd) will be inserted at a position that\n\t\t// follows the previous one. That will move the selection and, to stay on the safe side and make sure\n\t\t// all images inherit the same selection attributes, they are collected beforehand.\n\t\t//\n\t\t// Applying these attributes ensures, for instance, that inserting an (inline) image into a link does\n\t\t// not split that link but preserves its continuity.\n\t\t//\n\t\t// Note: Selection attributes that do not make sense for images will be filtered out by insertImage() anyway.\n\t\tconst selectionAttributes = Object.fromEntries( selection.getAttributes() );\n\n\t\tfiles.forEach( ( file, index ) => {\n\t\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\t\t// Inserting of an inline image replace the selected element and make a selection on the inserted image.\n\t\t\t// Therefore inserting multiple inline images requires creating position after each element.\n\t\t\tif ( index && selectedElement && imageUtils.isImage( selectedElement ) ) {\n\t\t\t\tconst position = this.editor.model.createPositionAfter( selectedElement );\n\n\t\t\t\tthis._uploadImage( file, selectionAttributes, position );\n\t\t\t} else {\n\t\t\t\tthis._uploadImage( file, selectionAttributes );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Handles uploading single file.\n\t */\n\tprivate _uploadImage( file: File, attributes: object, position?: Position ): void {\n\t\tconst editor = this.editor;\n\t\tconst fileRepository = editor.plugins.get( FileRepository );\n\t\tconst loader = fileRepository.createLoader( file );\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\t\t// Do not throw when upload adapter is not set. FileRepository will log an error anyway.\n\t\tif ( !loader ) {\n\t\t\treturn;\n\t\t}\n\n\t\timageUtils.insertImage( { ...attributes, uploadId: loader.id }, position );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imageupload/imageuploadediting\n */\n\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\n\nimport { UpcastWriter, type Element, type Item, type Writer, type DataTransfer, type ViewElement } from 'ckeditor5/src/engine';\n\nimport { Notification } from 'ckeditor5/src/ui';\nimport { ClipboardPipeline, type ViewDocumentClipboardInputEvent } from 'ckeditor5/src/clipboard';\nimport { FileRepository, type UploadResponse, type FileLoader } from 'ckeditor5/src/upload';\nimport { env } from 'ckeditor5/src/utils';\n\nimport ImageUtils from '../imageutils';\nimport UploadImageCommand from './uploadimagecommand';\nimport { fetchLocalImage, isLocalImage } from '../../src/imageupload/utils';\nimport { createImageTypeRegExp } from './utils';\n\n/**\n * The editing part of the image upload feature. It registers the `'uploadImage'` command\n * and the `imageUpload` command as an aliased name.\n *\n * When an image is uploaded, it fires the {@link ~ImageUploadEditing#event:uploadComplete `uploadComplete`} event\n * that allows adding custom attributes to the {@link module:engine/model/element~Element image element}.\n */\nexport default class ImageUploadEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ FileRepository, Notification, ClipboardPipeline, ImageUtils ] as const;\n\t}\n\n\tpublic static get pluginName() {\n\t\treturn 'ImageUploadEditing' as const;\n\t}\n\n\t/**\n\t * An internal mapping of {@link module:upload/filerepository~FileLoader#id file loader UIDs} and\n\t * model elements during the upload.\n\t *\n\t * Model element of the uploaded image can change, for instance, when {@link module:image/image/imagetypecommand~ImageTypeCommand}\n\t * is executed as a result of adding caption or changing image style. As a result, the upload logic must keep track of the model\n\t * element (reference) and resolve the upload for the correct model element (instead of the one that landed in the `$graveyard`\n\t * after image type changed).\n\t */\n\tprivate readonly _uploadImageElements: Map<string, Element>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\teditor.config.define( 'image', {\n\t\t\tupload: {\n\t\t\t\ttypes: [ 'jpeg', 'png', 'gif', 'bmp', 'webp', 'tiff' ]\n\t\t\t}\n\t\t} );\n\n\t\tthis._uploadImageElements = new Map();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst doc = editor.model.document;\n\t\tconst conversion = editor.conversion;\n\t\tconst fileRepository = editor.plugins.get( FileRepository );\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst clipboardPipeline: ClipboardPipeline = editor.plugins.get( 'ClipboardPipeline' );\n\t\tconst imageTypes = createImageTypeRegExp( editor.config.get( 'image.upload.types' )! );\n\t\tconst uploadImageCommand = new UploadImageCommand( editor );\n\n\t\t// Register `uploadImage` command and add `imageUpload` command as an alias for backward compatibility.\n\t\teditor.commands.add( 'uploadImage', uploadImageCommand );\n\t\teditor.commands.add( 'imageUpload', uploadImageCommand );\n\n\t\t// Register upcast converter for uploadId.\n\t\tconversion.for( 'upcast' )\n\t\t\t.attributeToAttribute( {\n\t\t\t\tview: {\n\t\t\t\t\tname: 'img',\n\t\t\t\t\tkey: 'uploadId'\n\t\t\t\t},\n\t\t\t\tmodel: 'uploadId'\n\t\t\t} );\n\n\t\t// Handle pasted images.\n\t\t// For every image file, a new file loader is created and a placeholder image is\n\t\t// inserted into the content. Then, those images are uploaded once they appear in the model\n\t\t// (see Document#change listener below).\n\t\tthis.listenTo<ViewDocumentClipboardInputEvent>( editor.editing.view.document, 'clipboardInput', ( evt, data ) => {\n\t\t\t// Skip if non empty HTML data is included.\n\t\t\t// https://github.com/ckeditor/ckeditor5-upload/issues/68\n\t\t\tif ( isHtmlIncluded( data.dataTransfer ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst images = Array.from( data.dataTransfer.files ).filter( file => {\n\t\t\t\t// See https://github.com/ckeditor/ckeditor5-image/pull/254.\n\t\t\t\tif ( !file ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\treturn imageTypes.test( file.type );\n\t\t\t} );\n\n\t\t\tif ( !images.length ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tevt.stop();\n\n\t\t\teditor.model.change( writer => {\n\t\t\t\t// Set selection to paste target.\n\t\t\t\tif ( data.targetRanges ) {\n\t\t\t\t\twriter.setSelection( data.targetRanges.map( viewRange => editor.editing.mapper.toModelRange( viewRange ) ) );\n\t\t\t\t}\n\n\t\t\t\t// Upload images after the selection has changed in order to ensure the command's state is refreshed.\n\t\t\t\teditor.model.enqueueChange( () => {\n\t\t\t\t\teditor.execute( 'uploadImage', { file: images } );\n\t\t\t\t} );\n\t\t\t} );\n\t\t} );\n\n\t\t// Handle HTML pasted with images with base64 or blob sources.\n\t\t// For every image file, a new file loader is created and a placeholder image is\n\t\t// inserted into the content. Then, those images are uploaded once they appear in the model\n\t\t// (see Document#change listener below).\n\t\tthis.listenTo( clipboardPipeline, 'inputTransformation', ( evt, data ) => {\n\t\t\tconst fetchableImages = Array.from( editor.editing.view.createRangeIn( data.content ) )\n\t\t\t\t.map( value => value.item as ViewElement )\n\t\t\t\t.filter( viewElement =>\n\t\t\t\t\tisLocalImage( imageUtils, viewElement ) &&\n\t\t\t\t\t!viewElement.getAttribute( 'uploadProcessed' ) )\n\t\t\t\t.map( viewElement => { return { promise: fetchLocalImage( viewElement ), imageElement: viewElement }; } );\n\n\t\t\tif ( !fetchableImages.length ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst writer = new UpcastWriter( editor.editing.view.document );\n\n\t\t\tfor ( const fetchableImage of fetchableImages ) {\n\t\t\t\t// Set attribute marking that the image was processed already.\n\t\t\t\twriter.setAttribute( 'uploadProcessed', true, fetchableImage.imageElement );\n\n\t\t\t\tconst loader = fileRepository.createLoader( fetchableImage.promise );\n\n\t\t\t\tif ( loader ) {\n\t\t\t\t\twriter.setAttribute( 'src', '', fetchableImage.imageElement );\n\t\t\t\t\twriter.setAttribute( 'uploadId', loader.id, fetchableImage.imageElement );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\n\t\t// Prevents from the browser redirecting to the dropped image.\n\t\teditor.editing.view.document.on( 'dragover', ( evt, data ) => {\n\t\t\tdata.preventDefault();\n\t\t} );\n\n\t\t// Upload placeholder images that appeared in the model.\n\t\tdoc.on( 'change', () => {\n\t\t\t// Note: Reversing changes to start with insertions and only then handle removals. If it was the other way around,\n\t\t\t// loaders for **all** images that land in the $graveyard would abort while in fact only those that were **not** replaced\n\t\t\t// by other images should be aborted.\n\t\t\tconst changes = doc.differ.getChanges( { includeChangesInGraveyard: true } ).reverse();\n\t\t\tconst insertedImagesIds = new Set();\n\n\t\t\tfor ( const entry of changes ) {\n\t\t\t\tif ( entry.type == 'insert' && entry.name != '$text' ) {\n\t\t\t\t\tconst item = entry.position.nodeAfter!;\n\t\t\t\t\tconst isInsertedInGraveyard = entry.position.root.rootName == '$graveyard';\n\n\t\t\t\t\tfor ( const imageElement of getImagesFromChangeItem( editor, item ) ) {\n\t\t\t\t\t\t// Check if the image element still has upload id.\n\t\t\t\t\t\tconst uploadId = imageElement.getAttribute( 'uploadId' ) as string;\n\n\t\t\t\t\t\tif ( !uploadId ) {\n\t\t\t\t\t\t\tcontinue;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\t// Check if the image is loaded on this client.\n\t\t\t\t\t\tconst loader = fileRepository.loaders.get( uploadId );\n\n\t\t\t\t\t\tif ( !loader ) {\n\t\t\t\t\t\t\tcontinue;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif ( isInsertedInGraveyard ) {\n\t\t\t\t\t\t\t// If the image was inserted to the graveyard for good (**not** replaced by another image),\n\t\t\t\t\t\t\t// only then abort the loading process.\n\t\t\t\t\t\t\tif ( !insertedImagesIds.has( uploadId ) ) {\n\t\t\t\t\t\t\t\tloader.abort();\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t// Remember the upload id of the inserted image. If it acted as a replacement for another\n\t\t\t\t\t\t\t// image (which landed in the $graveyard), the related loader will not be aborted because\n\t\t\t\t\t\t\t// this is still the same image upload.\n\t\t\t\t\t\t\tinsertedImagesIds.add( uploadId );\n\n\t\t\t\t\t\t\t// Keep the mapping between the upload ID and the image model element so the upload\n\t\t\t\t\t\t\t// can later resolve in the context of the correct model element. The model element could\n\t\t\t\t\t\t\t// change for the same upload if one image was replaced by another (e.g. image type was changed),\n\t\t\t\t\t\t\t// so this may also replace an existing mapping.\n\t\t\t\t\t\t\tthis._uploadImageElements.set( uploadId, imageElement as Element );\n\n\t\t\t\t\t\t\tif ( loader.status == 'idle' ) {\n\t\t\t\t\t\t\t\t// If the image was inserted into content and has not been loaded yet, start loading it.\n\t\t\t\t\t\t\t\tthis._readAndUpload( loader );\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\n\t\t// Set the default handler for feeding the image element with `src` and `srcset` attributes.\n\t\tthis.on<ImageUploadCompleteEvent>( 'uploadComplete', ( evt, { imageElement, data } ) => {\n\t\t\tconst urls = data.urls ? data.urls as Record<string, unknown> : data;\n\n\t\t\tthis.editor.model.change( writer => {\n\t\t\t\twriter.setAttribute( 'src', urls.default, imageElement );\n\t\t\t\tthis._parseAndSetSrcsetAttributeOnImage( urls, imageElement, writer );\n\t\t\t} );\n\t\t}, { priority: 'low' } );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\tconst schema = this.editor.model.schema;\n\n\t\t// Setup schema to allow uploadId and uploadStatus for images.\n\t\t// Wait for ImageBlockEditing or ImageInlineEditing to register their elements first,\n\t\t// that's why doing this in afterInit() instead of init().\n\t\tif ( this.editor.plugins.has( 'ImageBlockEditing' ) ) {\n\t\t\tschema.extend( 'imageBlock', {\n\t\t\t\tallowAttributes: [ 'uploadId', 'uploadStatus' ]\n\t\t\t} );\n\t\t}\n\n\t\tif ( this.editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\t\tschema.extend( 'imageInline', {\n\t\t\t\tallowAttributes: [ 'uploadId', 'uploadStatus' ]\n\t\t\t} );\n\t\t}\n\t}\n\n\t/**\n\t * Reads and uploads an image.\n\t *\n\t * The image is read from the disk and as a Base64-encoded string it is set temporarily to\n\t * `image[src]`. When the image is successfully uploaded, the temporary data is replaced with the target\n\t * image's URL (the URL to the uploaded image on the server).\n\t */\n\tprotected _readAndUpload( loader: FileLoader ): Promise<void> {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst t = editor.locale.t;\n\t\tconst fileRepository = editor.plugins.get( FileRepository );\n\t\tconst notification = editor.plugins.get( Notification );\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst imageUploadElements = this._uploadImageElements;\n\n\t\tmodel.enqueueChange( { isUndoable: false }, writer => {\n\t\t\twriter.setAttribute( 'uploadStatus', 'reading', imageUploadElements.get( loader.id )! );\n\t\t} );\n\n\t\treturn loader.read()\n\t\t\t.then( () => {\n\t\t\t\tconst promise = loader.upload();\n\t\t\t\tconst imageElement = imageUploadElements.get( loader.id )!;\n\n\t\t\t\t// Force repaint in Safari. Without it, the image will display with a wrong size.\n\t\t\t\t// https://github.com/ckeditor/ckeditor5/issues/1975\n\t\t\t\t/* istanbul ignore next -- @preserve */\n\t\t\t\tif ( env.isSafari ) {\n\t\t\t\t\tconst viewFigure = editor.editing.mapper.toViewElement( imageElement )!;\n\t\t\t\t\tconst viewImg = imageUtils.findViewImgElement( viewFigure )!;\n\n\t\t\t\t\teditor.editing.view.once( 'render', () => {\n\t\t\t\t\t\t// Early returns just to be safe. There might be some code ran\n\t\t\t\t\t\t// in between the outer scope and this callback.\n\t\t\t\t\t\tif ( !viewImg.parent ) {\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tconst domFigure = editor.editing.view.domConverter.mapViewToDom( viewImg.parent ) as HTMLElement | undefined;\n\n\t\t\t\t\t\tif ( !domFigure ) {\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tconst originalDisplay = domFigure.style.display;\n\n\t\t\t\t\t\tdomFigure.style.display = 'none';\n\n\t\t\t\t\t\t// Make sure this line will never be removed during minification for having \"no effect\".\n\t\t\t\t\t\t( domFigure as any )._ckHack = domFigure.offsetHeight;\n\n\t\t\t\t\t\tdomFigure.style.display = originalDisplay;\n\t\t\t\t\t} );\n\t\t\t\t}\n\n\t\t\t\tmodel.enqueueChange( { isUndoable: false }, writer => {\n\t\t\t\t\twriter.setAttribute( 'uploadStatus', 'uploading', imageElement );\n\t\t\t\t} );\n\n\t\t\t\treturn promise;\n\t\t\t} )\n\t\t\t.then( data => {\n\t\t\t\tmodel.enqueueChange( { isUndoable: false }, writer => {\n\t\t\t\t\tconst imageElement = imageUploadElements.get( loader.id )!;\n\n\t\t\t\t\twriter.setAttribute( 'uploadStatus', 'complete', imageElement );\n\n\t\t\t\t\tthis.fire<ImageUploadCompleteEvent>( 'uploadComplete', { data, imageElement } );\n\t\t\t\t} );\n\n\t\t\t\tclean();\n\t\t\t} )\n\t\t\t.catch( error => {\n\t\t\t\t// If status is not 'error' nor 'aborted' - throw error because it means that something else went wrong,\n\t\t\t\t// it might be generic error and it would be real pain to find what is going on.\n\t\t\t\tif ( loader.status !== 'error' && loader.status !== 'aborted' ) {\n\t\t\t\t\tthrow error;\n\t\t\t\t}\n\n\t\t\t\t// Might be 'aborted'.\n\t\t\t\tif ( loader.status == 'error' && error ) {\n\t\t\t\t\tnotification.showWarning( error, {\n\t\t\t\t\t\ttitle: t( 'Upload failed' ),\n\t\t\t\t\t\tnamespace: 'upload'\n\t\t\t\t\t} );\n\t\t\t\t}\n\n\t\t\t\t// Permanently remove image from insertion batch.\n\t\t\t\tmodel.enqueueChange( { isUndoable: false }, writer => {\n\t\t\t\t\twriter.remove( imageUploadElements.get( loader.id )! );\n\t\t\t\t} );\n\n\t\t\t\tclean();\n\t\t\t} );\n\n\t\tfunction clean() {\n\t\t\tmodel.enqueueChange( { isUndoable: false }, writer => {\n\t\t\t\tconst imageElement = imageUploadElements.get( loader.id )!;\n\n\t\t\t\twriter.removeAttribute( 'uploadId', imageElement );\n\t\t\t\twriter.removeAttribute( 'uploadStatus', imageElement );\n\n\t\t\t\timageUploadElements.delete( loader.id );\n\t\t\t} );\n\n\t\t\tfileRepository.destroyLoader( loader );\n\t\t}\n\t}\n\n\t/**\n\t * Creates the `srcset` attribute based on a given file upload response and sets it as an attribute to a specific image element.\n\t *\n\t * @param data Data object from which `srcset` will be created.\n\t * @param image The image element on which the `srcset` attribute will be set.\n\t */\n\tprotected _parseAndSetSrcsetAttributeOnImage( data: Record<string, unknown>, image: Element, writer: Writer ): void {\n\t\t// Srcset attribute for responsive images support.\n\t\tlet maxWidth = 0;\n\n\t\tconst srcsetAttribute = Object.keys( data )\n\t\t\t// Filter out keys that are not integers.\n\t\t\t.filter( key => {\n\t\t\t\tconst width = parseInt( key, 10 );\n\n\t\t\t\tif ( !isNaN( width ) ) {\n\t\t\t\t\tmaxWidth = Math.max( maxWidth, width );\n\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t} )\n\n\t\t\t// Convert each key to srcset entry.\n\t\t\t.map( key => `${ data[ key ] } ${ key }w` )\n\n\t\t\t// Join all entries.\n\t\t\t.join( ', ' );\n\n\t\tif ( srcsetAttribute != '' ) {\n\t\t\twriter.setAttribute( 'srcset', {\n\t\t\t\tdata: srcsetAttribute,\n\t\t\t\twidth: maxWidth\n\t\t\t}, image );\n\t\t}\n\t}\n}\n\n/**\n * Returns `true` if non-empty `text/html` is included in the data transfer.\n */\nexport function isHtmlIncluded( dataTransfer: DataTransfer ): boolean {\n\treturn Array.from( dataTransfer.types ).includes( 'text/html' ) && dataTransfer.getData( 'text/html' ) !== '';\n}\n\nfunction getImagesFromChangeItem( editor: Editor, item: Item ): Array<Item> {\n\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\treturn Array.from( editor.model.createRangeOn( item ) )\n\t\t.filter( value => imageUtils.isImage( value.item as Element ) )\n\t\t.map( value => value.item );\n}\n\n/**\n * An event fired when an image is uploaded. You can hook into this event to provide\n * custom attributes to the {@link module:engine/model/element~Element image element} based on the data from\n * the server.\n *\n * ```ts\n * const imageUploadEditing = editor.plugins.get( 'ImageUploadEditing' );\n *\n * imageUploadEditing.on( 'uploadComplete', ( evt, { data, imageElement } ) => {\n * \teditor.model.change( writer => {\n * \t\twriter.setAttribute( 'someAttribute', 'foo', imageElement );\n * \t} );\n * } );\n * ```\n *\n * You can also stop the default handler that sets the `src` and `srcset` attributes\n * if you want to provide custom values for these attributes.\n *\n * ```ts\n * imageUploadEditing.on( 'uploadComplete', ( evt, { data, imageElement } ) => {\n * \tevt.stop();\n * } );\n * ```\n *\n * **Note**: This event is fired by the {@link module:image/imageupload/imageuploadediting~ImageUploadEditing} plugin.\n *\n * @eventName ~ImageUploadEditing#uploadComplete\n * @param data The `uploadComplete` event data.\n */\nexport type ImageUploadCompleteEvent = {\n\tname: 'uploadComplete';\n\targs: [ data: ImageUploadCompleteData];\n};\n\nexport type ImageUploadCompleteData = {\n\n\t/**\n\t * The data coming from the upload adapter.\n\t */\n\tdata: UploadResponse;\n\n\t/**\n\t * The model {@link module:engine/model/element~Element image element} that can be customized.\n\t */\n\timageElement: Element;\n};\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./imageinsertformrowview.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./imageinsert.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./imageresize.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagestyle/imagestylecommand\n */\n\nimport type { Element } from 'ckeditor5/src/engine';\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport type { ImageStyleOptionDefinition } from '../imageconfig';\nimport type ImageUtils from '../imageutils';\n\n/**\n * The image style command. It is used to apply {@link module:image/imageconfig~ImageStyleConfig#options image style option}\n * to a selected image.\n *\n * **Note**: Executing this command may change the image model element if the desired style requires an image of a different\n * type. See {@link module:image/imagestyle/imagestylecommand~ImageStyleCommand#execute} to learn more.\n */\nexport default class ImageStyleCommand extends Command {\n\t/**\n\t * An object containing names of default style options for the inline and block images.\n\t * If there is no default style option for the given image type in the configuration,\n\t * the name will be `false`.\n\t */\n\tprivate _defaultStyles: Record<string, string | false>;\n\n\t/**\n\t * The styles handled by this command.\n\t */\n\tprivate _styles: Map<string, ImageStyleOptionDefinition>;\n\n\t/**\n\t * Creates an instance of the image style command. When executed, the command applies one of\n\t * {@link module:image/imageconfig~ImageStyleConfig#options style options} to the currently selected image.\n\t *\n\t * @param editor The editor instance.\n\t * @param styles The style options that this command supports.\n\t */\n\tconstructor( editor: Editor, styles: Array<ImageStyleOptionDefinition> ) {\n\t\tsuper( editor );\n\n\t\tthis._defaultStyles = {\n\t\t\timageBlock: false,\n\t\t\timageInline: false\n\t\t};\n\n\t\tthis._styles = new Map( styles.map( style => {\n\t\t\tif ( style.isDefault ) {\n\t\t\t\tfor ( const modelElementName of style.modelElements ) {\n\t\t\t\t\tthis._defaultStyles[ modelElementName ] = style.name;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\treturn [ style.name, style ];\n\t\t} ) );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst editor = this.editor;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\t\tconst element = imageUtils.getClosestSelectedImageElement( this.editor.model.document.selection )!;\n\n\t\tthis.isEnabled = !!element;\n\n\t\tif ( !this.isEnabled ) {\n\t\t\tthis.value = false;\n\t\t} else if ( element.hasAttribute( 'imageStyle' ) ) {\n\t\t\tthis.value = element.getAttribute( 'imageStyle' );\n\t\t} else {\n\t\t\tthis.value = this._defaultStyles[ element.name ];\n\t\t}\n\t}\n\n\t/**\n\t * Executes the command and applies the style to the currently selected image:\n\t *\n\t * ```ts\n\t * editor.execute( 'imageStyle', { value: 'side' } );\n\t * ```\n\t *\n\t * **Note**: Executing this command may change the image model element if the desired style requires an image\n\t * of a different type. Learn more about {@link module:image/imageconfig~ImageStyleOptionDefinition#modelElements model element}\n\t * configuration for the style option.\n\t *\n\t * @param options.value The name of the style (as configured in {@link module:image/imageconfig~ImageStyleConfig#options}).\n\t * @fires execute\n\t */\n\tpublic override execute( options: { value?: string } = {} ): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\t\tmodel.change( writer => {\n\t\t\tconst requestedStyle = options.value;\n\n\t\t\tlet imageElement = imageUtils.getClosestSelectedImageElement( model.document.selection )!;\n\n\t\t\t// Change the image type if a style requires it.\n\t\t\tif ( requestedStyle && this.shouldConvertImageType( requestedStyle, imageElement ) ) {\n\t\t\t\tthis.editor.execute( imageUtils.isBlockImage( imageElement ) ? 'imageTypeInline' : 'imageTypeBlock' );\n\n\t\t\t\t// Update the imageElement to the newly created image.\n\t\t\t\timageElement = imageUtils.getClosestSelectedImageElement( model.document.selection )!;\n\t\t\t}\n\n\t\t\t// Default style means that there is no `imageStyle` attribute in the model.\n\t\t\t// https://github.com/ckeditor/ckeditor5-image/issues/147\n\t\t\tif ( !requestedStyle || this._styles.get( requestedStyle )!.isDefault ) {\n\t\t\t\twriter.removeAttribute( 'imageStyle', imageElement );\n\t\t\t} else {\n\t\t\t\twriter.setAttribute( 'imageStyle', requestedStyle, imageElement );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Returns `true` if requested style change would trigger the image type change.\n\t *\n\t * @param requestedStyle The name of the style (as configured in {@link module:image/imageconfig~ImageStyleConfig#options}).\n\t * @param imageElement The image model element.\n\t */\n\tpublic shouldConvertImageType( requestedStyle: string, imageElement: Element ): boolean {\n\t\tconst supportedTypes = this._styles.get( requestedStyle )!.modelElements;\n\n\t\treturn !supportedTypes.includes( imageElement.name );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagestyle/utils\n */\n\nimport { type Editor, icons, type PluginCollection } from 'ckeditor5/src/core';\nimport { logWarning } from 'ckeditor5/src/utils';\nimport type { ImageStyleConfig, ImageStyleDropdownDefinition, ImageStyleOptionDefinition } from '../imageconfig';\n\nconst {\n\tobjectFullWidth,\n\tobjectInline,\n\tobjectLeft,\tobjectRight, objectCenter,\n\tobjectBlockLeft, objectBlockRight\n} = icons;\n\n/**\n * Default image style options provided by the plugin that can be referred in the {@link module:image/imageconfig~ImageConfig#styles}\n * configuration.\n *\n * There are available 5 styles focused on formatting:\n *\n * * **`'alignLeft'`** aligns the inline or block image to the left and wraps it with the text using the `image-style-align-left` class,\n * * **`'alignRight'`** aligns the inline or block image to the right and wraps it with the text using the `image-style-align-right` class,\n * * **`'alignCenter'`** centers the block image using the `image-style-align-center` class,\n * * **`'alignBlockLeft'`** aligns the block image to the left using the `image-style-block-align-left` class,\n * * **`'alignBlockRight'`** aligns the block image to the right using the `image-style-block-align-right` class,\n *\n * and 3 semantic styles:\n *\n * * **`'inline'`** is an inline image without any CSS class,\n * * **`'block'`** is a block image without any CSS class,\n * * **`'side'`** is a block image styled with the `image-style-side` CSS class.\n */\nexport const DEFAULT_OPTIONS: Record<string, ImageStyleOptionDefinition> = {\n\t// This style represents an image placed in the line of text.\n\tget inline() {\n\t\treturn {\n\t\t\tname: 'inline',\n\t\t\ttitle: 'In line',\n\t\t\ticon: objectInline,\n\t\t\tmodelElements: [ 'imageInline' ],\n\t\t\tisDefault: true\n\t\t};\n\t},\n\n\t// This style represents an image aligned to the left and wrapped with text.\n\tget alignLeft() {\n\t\treturn {\n\t\t\tname: 'alignLeft',\n\t\t\ttitle: 'Left aligned image',\n\t\t\ticon: objectLeft,\n\t\t\tmodelElements: [ 'imageBlock', 'imageInline' ],\n\t\t\tclassName: 'image-style-align-left'\n\t\t};\n\t},\n\n\t// This style represents an image aligned to the left.\n\tget alignBlockLeft() {\n\t\treturn {\n\t\t\tname: 'alignBlockLeft',\n\t\t\ttitle: 'Left aligned image',\n\t\t\ticon: objectBlockLeft,\n\t\t\tmodelElements: [ 'imageBlock' ],\n\t\t\tclassName: 'image-style-block-align-left'\n\t\t};\n\t},\n\n\t// This style represents a centered image.\n\tget alignCenter() {\n\t\treturn {\n\t\t\tname: 'alignCenter',\n\t\t\ttitle: 'Centered image',\n\t\t\ticon: objectCenter,\n\t\t\tmodelElements: [ 'imageBlock' ],\n\t\t\tclassName: 'image-style-align-center'\n\t\t};\n\t},\n\n\t// This style represents an image aligned to the right and wrapped with text.\n\tget alignRight() {\n\t\treturn {\n\t\t\tname: 'alignRight',\n\t\t\ttitle: 'Right aligned image',\n\t\t\ticon: objectRight,\n\t\t\tmodelElements: [ 'imageBlock', 'imageInline' ],\n\t\t\tclassName: 'image-style-align-right'\n\t\t};\n\t},\n\n\t// This style represents an image aligned to the right.\n\tget alignBlockRight() {\n\t\treturn {\n\t\t\tname: 'alignBlockRight',\n\t\t\ttitle: 'Right aligned image',\n\t\t\ticon: objectBlockRight,\n\t\t\tmodelElements: [ 'imageBlock' ],\n\t\t\tclassName: 'image-style-block-align-right'\n\t\t};\n\t},\n\n\t// This option is equal to the situation when no style is applied.\n\tget block() {\n\t\treturn {\n\t\t\tname: 'block',\n\t\t\ttitle: 'Centered image',\n\t\t\ticon: objectCenter,\n\t\t\tmodelElements: [ 'imageBlock' ],\n\t\t\tisDefault: true\n\t\t};\n\t},\n\n\t// This represents a side image.\n\tget side() {\n\t\treturn {\n\t\t\tname: 'side',\n\t\t\ttitle: 'Side image',\n\t\t\ticon: objectRight,\n\t\t\tmodelElements: [ 'imageBlock' ],\n\t\t\tclassName: 'image-style-side'\n\t\t};\n\t}\n};\n\n/**\n * Default image style icons provided by the plugin that can be referred in the {@link module:image/imageconfig~ImageConfig#styles}\n * configuration.\n *\n * See {@link module:image/imageconfig~ImageStyleOptionDefinition#icon} to learn more.\n *\n * There are 7 default icons available: `'full'`, `'left'`, `'inlineLeft'`, `'center'`, `'right'`, `'inlineRight'`, and `'inline'`.\n */\nexport const DEFAULT_ICONS: Record<string, string> = {\n\tfull: objectFullWidth,\n\tleft: objectBlockLeft,\n\tright: objectBlockRight,\n\tcenter: objectCenter,\n\tinlineLeft: objectLeft,\n\tinlineRight: objectRight,\n\tinline: objectInline\n};\n\n/**\n * Default drop-downs provided by the plugin that can be referred in the {@link module:image/imageconfig~ImageConfig#toolbar}\n * configuration. The drop-downs are containers for the {@link module:image/imageconfig~ImageStyleConfig#options image style options}.\n *\n * If both of the `ImageEditing` plugins are loaded, there are 2 predefined drop-downs available:\n *\n * * **`'imageStyle:wrapText'`**, which contains the `alignLeft` and `alignRight` options, that is,\n * those that wraps the text around the image,\n * * **`'imageStyle:breakText'`**, which contains the `alignBlockLeft`, `alignCenter` and `alignBlockRight` options, that is,\n * those that breaks the text around the image.\n */\nexport const DEFAULT_DROPDOWN_DEFINITIONS: Array<ImageStyleDropdownDefinition> = [ {\n\tname: 'imageStyle:wrapText',\n\ttitle: 'Wrap text',\n\tdefaultItem: 'imageStyle:alignLeft',\n\titems: [ 'imageStyle:alignLeft', 'imageStyle:alignRight' ]\n}, {\n\tname: 'imageStyle:breakText',\n\ttitle: 'Break text',\n\tdefaultItem: 'imageStyle:block',\n\titems: [ 'imageStyle:alignBlockLeft', 'imageStyle:block', 'imageStyle:alignBlockRight' ]\n} ];\n\n/**\n * Returns a list of the normalized and validated image style options.\n *\n * @param config\n * @param config.isInlinePluginLoaded\n * Determines whether the {@link module:image/image/imageblockediting~ImageBlockEditing `ImageBlockEditing`} plugin has been loaded.\n * @param config.isBlockPluginLoaded\n * Determines whether the {@link module:image/image/imageinlineediting~ImageInlineEditing `ImageInlineEditing`} plugin has been loaded.\n * @param config.configuredStyles\n * The image styles configuration provided in the image styles {@link module:image/imageconfig~ImageConfig#styles configuration}\n * as a default or custom value.\n * @returns\n * * Each of options contains a complete icon markup.\n * * The image style options not supported by any of the loaded plugins are filtered out.\n */\nfunction normalizeStyles( config: {\n\tisInlinePluginLoaded: boolean;\n\tisBlockPluginLoaded: boolean;\n\tconfiguredStyles: ImageStyleConfig;\n}\n): Array<ImageStyleOptionDefinition> {\n\tconst configuredStyles = config.configuredStyles.options || [];\n\n\tconst styles = configuredStyles\n\t\t.map( arrangement => normalizeDefinition( arrangement ) )\n\t\t.filter( arrangement => isValidOption( arrangement, config ) );\n\n\treturn styles;\n}\n\n/**\n * Returns the default image styles configuration depending on the loaded image editing plugins.\n *\n * @param isInlinePluginLoaded\n * Determines whether the {@link module:image/image/imageblockediting~ImageBlockEditing `ImageBlockEditing`} plugin has been loaded.\n *\n * @param isBlockPluginLoaded\n * Determines whether the {@link module:image/image/imageinlineediting~ImageInlineEditing `ImageInlineEditing`} plugin has been loaded.\n *\n * @returns\n * It returns an object with the lists of the image style options and groups defined as strings related to the\n * {@link module:image/imagestyle/utils#DEFAULT_OPTIONS default options}\n */\nfunction getDefaultStylesConfiguration( isBlockPluginLoaded: boolean, isInlinePluginLoaded: boolean ): ImageStyleConfig {\n\tif ( isBlockPluginLoaded && isInlinePluginLoaded ) {\n\t\treturn {\n\t\t\toptions: [\n\t\t\t\t'inline', 'alignLeft', 'alignRight',\n\t\t\t\t'alignCenter', 'alignBlockLeft', 'alignBlockRight',\n\t\t\t\t'block', 'side'\n\t\t\t]\n\t\t};\n\t} else if ( isBlockPluginLoaded ) {\n\t\treturn {\n\t\t\toptions: [ 'block', 'side' ]\n\t\t};\n\t} else if ( isInlinePluginLoaded ) {\n\t\treturn {\n\t\t\toptions: [ 'inline', 'alignLeft', 'alignRight' ]\n\t\t};\n\t}\n\n\treturn {};\n}\n\n/**\n * Returns a list of the available predefined drop-downs' definitions depending on the loaded image editing plugins.\n */\nfunction getDefaultDropdownDefinitions( pluginCollection: PluginCollection<Editor> ): Array<ImageStyleDropdownDefinition> {\n\tif ( pluginCollection.has( 'ImageBlockEditing' ) && pluginCollection.has( 'ImageInlineEditing' ) ) {\n\t\treturn [ ...DEFAULT_DROPDOWN_DEFINITIONS ];\n\t} else {\n\t\treturn [];\n\t}\n}\n\n/**\n * Normalizes an image style option or group provided in the {@link module:image/imageconfig~ImageConfig#styles}\n * and returns it in a {@link module:image/imageconfig~ImageStyleOptionDefinition}/\n */\nfunction normalizeDefinition( definition: string | Partial<ImageStyleOptionDefinition> & { name: string } ): ImageStyleOptionDefinition {\n\tif ( typeof definition === 'string' ) {\n\t\t// Just the name of the style has been passed, but none of the defaults.\n\t\tif ( !DEFAULT_OPTIONS[ definition ] ) {\n\t\t\t// Normalize the style anyway to prevent errors.\n\t\t\tdefinition = { name: definition };\n\t\t}\n\t\t// Just the name of the style has been passed and it's one of the defaults, just use it.\n\t\t// Clone the style to avoid overriding defaults.\n\t\telse {\n\t\t\tdefinition = { ...DEFAULT_OPTIONS[ definition ] };\n\t\t}\n\t} else {\n\t\t// If an object style has been passed and if the name matches one of the defaults,\n\t\t// extend it with defaults the user wants to customize a default style.\n\t\t// Note: Don't override the userdefined style object, clone it instead.\n\t\tdefinition = extendStyle( DEFAULT_OPTIONS[ definition.name ], definition );\n\t}\n\n\t// If an icon is defined as a string and correspond with a name\n\t// in default icons, use the default icon provided by the plugin.\n\tif ( typeof definition.icon === 'string' ) {\n\t\tdefinition.icon = DEFAULT_ICONS[ definition.icon ] || definition.icon;\n\t}\n\n\treturn definition as ImageStyleOptionDefinition;\n}\n\n/**\n * Checks if the image style option is valid:\n * * if it has the modelElements fields defined and filled,\n * * if the defined modelElements are supported by any of the loaded image editing plugins.\n * It also displays a console warning these conditions are not met.\n *\n * @param option image style option\n */\nfunction isValidOption(\n\toption: ImageStyleOptionDefinition,\n\t{ isBlockPluginLoaded, isInlinePluginLoaded }: { isBlockPluginLoaded: boolean; isInlinePluginLoaded: boolean }\n): boolean {\n\tconst { modelElements, name } = option;\n\n\tif ( !modelElements || !modelElements.length || !name ) {\n\t\twarnInvalidStyle( { style: option } );\n\n\t\treturn false;\n\t} else {\n\t\tconst supportedElements = [ isBlockPluginLoaded ? 'imageBlock' : null, isInlinePluginLoaded ? 'imageInline' : null ];\n\n\t\t// Check if the option is supported by any of the loaded plugins.\n\t\tif ( !modelElements.some( elementName => supportedElements.includes( elementName ) ) ) {\n\t\t\t/**\n\t\t\t * In order to work correctly, each image style {@link module:image/imageconfig~ImageStyleOptionDefinition option}\n\t\t\t * requires specific model elements (also: types of images) to be supported by the editor.\n\t\t\t *\n\t\t\t * Model element names to which the image style option can be applied are defined in the\n\t\t\t * {@link module:image/imageconfig~ImageStyleOptionDefinition#modelElements} property of the style option\n\t\t\t * definition.\n\t\t\t *\n\t\t\t * Explore the warning in the console to find out precisely which option is not supported and which editor plugins\n\t\t\t * are missing. Make sure these plugins are loaded in your editor to get this image style option working.\n\t\t\t *\n\t\t\t * @error image-style-missing-dependency\n\t\t\t * @param {String} [option] The name of the unsupported option.\n\t\t\t * @param {String} [missingPlugins] The names of the plugins one of which has to be loaded for the particular option.\n\t\t\t */\n\t\t\tlogWarning( 'image-style-missing-dependency', {\n\t\t\t\tstyle: option,\n\t\t\t\tmissingPlugins: modelElements.map( name => name === 'imageBlock' ? 'ImageBlockEditing' : 'ImageInlineEditing' )\n\t\t\t} );\n\n\t\t\treturn false;\n\t\t}\n\t}\n\n\treturn true;\n}\n\n/**\n * Extends the default style with a style provided by the developer.\n * Note: Don't override the customdefined style object, clone it instead.\n */\nfunction extendStyle( source: ImageStyleOptionDefinition, style: Partial<ImageStyleOptionDefinition> ): ImageStyleOptionDefinition {\n\tconst extendedStyle: Record<string, any> = { ...style };\n\n\tfor ( const prop in source ) {\n\t\tif ( !Object.prototype.hasOwnProperty.call( style, prop ) ) {\n\t\t\textendedStyle[ prop ] = source[ prop as keyof typeof source ];\n\t\t}\n\t}\n\n\treturn extendedStyle as ImageStyleOptionDefinition;\n}\n\n/**\n * Displays a console warning with the 'image-style-configuration-definition-invalid' error.\n */\nfunction warnInvalidStyle( info: object ): void {\n\t/**\n\t * The image style definition provided in the configuration is invalid.\n\t *\n\t * Please make sure the definition implements properly one of the following:\n\t *\n\t * * {@link module:image/imageconfig~ImageStyleOptionDefinition image style option definition},\n\t * * {@link module:image/imageconfig~ImageStyleDropdownDefinition image style dropdown definition}\n\t *\n\t * @error image-style-configuration-definition-invalid\n\t * @param {String} [dropdown] The name of the invalid drop-down\n\t * @param {String} [style] The name of the invalid image style option\n\t */\n\tlogWarning( 'image-style-configuration-definition-invalid', info );\n}\n\nexport default {\n\tnormalizeStyles,\n\tgetDefaultStylesConfiguration,\n\tgetDefaultDropdownDefinitions,\n\twarnInvalidStyle,\n\tDEFAULT_OPTIONS,\n\tDEFAULT_ICONS,\n\tDEFAULT_DROPDOWN_DEFINITIONS\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\nimport type { DowncastAttributeEvent, Element, UpcastElementEvent } from 'ckeditor5/src/engine';\nimport { first, type GetCallback } from 'ckeditor5/src/utils';\nimport type { ImageStyleOptionDefinition } from '../imageconfig';\n\n/**\n * @module image/imagestyle/converters\n */\n\n/**\n * Returns a converter for the `imageStyle` attribute. It can be used for adding, changing and removing the attribute.\n *\n * @param styles An array containing available image style options.\n * @returns A model-to-view attribute converter.\n */\nexport function modelToViewStyleAttribute( styles: Array<ImageStyleOptionDefinition> ): GetCallback<DowncastAttributeEvent> {\n\treturn ( evt, data, conversionApi ) => {\n\t\tif ( !conversionApi.consumable.consume( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Check if there is class name associated with given value.\n\t\tconst newStyle = getStyleDefinitionByName( data.attributeNewValue as string, styles );\n\t\tconst oldStyle = getStyleDefinitionByName( data.attributeOldValue as string, styles );\n\n\t\tconst viewElement = conversionApi.mapper.toViewElement( data.item as Element )!;\n\t\tconst viewWriter = conversionApi.writer;\n\n\t\tif ( oldStyle ) {\n\t\t\tviewWriter.removeClass( oldStyle.className!, viewElement );\n\t\t}\n\n\t\tif ( newStyle ) {\n\t\t\tviewWriter.addClass( newStyle.className!, viewElement );\n\t\t}\n\t};\n}\n\n/**\n * Returns a view-to-model converter converting image CSS classes to a proper value in the model.\n *\n * @param styles Image style options for which the converter is created.\n * @returns A view-to-model converter.\n */\nexport function viewToModelStyleAttribute( styles: Array<ImageStyleOptionDefinition> ): GetCallback<UpcastElementEvent> {\n\t// Convert only nondefault styles.\n\tconst nonDefaultStyles: Record<string, Array<ImageStyleOptionDefinition>> = {\n\t\timageInline: styles.filter( style => !style.isDefault && style.modelElements.includes( 'imageInline' ) ),\n\t\timageBlock: styles.filter( style => !style.isDefault && style.modelElements.includes( 'imageBlock' ) )\n\t};\n\n\treturn ( evt, data, conversionApi ) => {\n\t\tif ( !data.modelRange ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewElement = data.viewItem;\n\t\tconst modelImageElement = first( data.modelRange.getItems() );\n\n\t\t// Run this converter only if an image has been found in the model.\n\t\t// In some cases it may not be found (for example if we run this on a figure with different type than image).\n\t\tif ( !modelImageElement ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// ...and the `imageStyle` attribute is allowed for that element, otherwise stop conversion early.\n\t\tif ( !conversionApi.schema.checkAttribute( modelImageElement, 'imageStyle' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// Convert styles one by one.\n\t\tfor ( const style of nonDefaultStyles[ ( modelImageElement as Element ).name ] ) {\n\t\t\t// Try to consume class corresponding with the style.\n\t\t\tif ( conversionApi.consumable.consume( viewElement, { classes: style.className } ) ) {\n\t\t\t\t// And convert this style to model attribute.\n\t\t\t\tconversionApi.writer.setAttribute( 'imageStyle', style.name, modelImageElement );\n\t\t\t}\n\t\t}\n\t};\n}\n\n/**\n * Returns the style with a given `name` from an array of styles.\n */\nfunction getStyleDefinitionByName( name: string, styles: Array<ImageStyleOptionDefinition> ): ImageStyleOptionDefinition | undefined {\n\tfor ( const style of styles ) {\n\t\tif ( style.name === name ) {\n\t\t\treturn style;\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagestyle/imagestyleediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport type { Element, UpcastElementEvent } from 'ckeditor5/src/engine';\n\nimport ImageStyleCommand from './imagestylecommand';\nimport ImageUtils from '../imageutils';\nimport utils from './utils';\nimport { viewToModelStyleAttribute, modelToViewStyleAttribute } from './converters';\nimport type { ImageStyleOptionDefinition } from '../imageconfig';\n\n/**\n * The image style engine plugin. It sets the default configuration, creates converters and registers\n * {@link module:image/imagestyle/imagestylecommand~ImageStyleCommand ImageStyleCommand}.\n */\nexport default class ImageStyleEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageStyleEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageUtils ] as const;\n\t}\n\n\t/**\n\t * It contains a list of the normalized and validated style options.\n\t *\n\t * * Each option contains a complete icon markup.\n\t * * The style options not supported by any of the loaded image editing plugins (\n\t * {@link module:image/image/imageinlineediting~ImageInlineEditing `ImageInlineEditing`} or\n\t * {@link module:image/image/imageblockediting~ImageBlockEditing `ImageBlockEditing`}) are filtered out.\n\t *\n\t * @internal\n\t * @readonly\n\t */\n\tpublic normalizedStyles?: Array<ImageStyleOptionDefinition>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst { normalizeStyles, getDefaultStylesConfiguration } = utils;\n\t\tconst editor = this.editor;\n\t\tconst isBlockPluginLoaded = editor.plugins.has( 'ImageBlockEditing' );\n\t\tconst isInlinePluginLoaded = editor.plugins.has( 'ImageInlineEditing' );\n\n\t\teditor.config.define( 'image.styles', getDefaultStylesConfiguration( isBlockPluginLoaded, isInlinePluginLoaded ) );\n\n\t\tthis.normalizedStyles = normalizeStyles( {\n\t\t\tconfiguredStyles: editor.config.get( 'image.styles' )!,\n\t\t\tisBlockPluginLoaded,\n\t\t\tisInlinePluginLoaded\n\t\t} );\n\n\t\tthis._setupConversion( isBlockPluginLoaded, isInlinePluginLoaded );\n\t\tthis._setupPostFixer();\n\n\t\t// Register imageStyle command.\n\t\teditor.commands.add( 'imageStyle', new ImageStyleCommand( editor, this.normalizedStyles ) );\n\t}\n\n\t/**\n\t * Sets the editor conversion taking the presence of\n\t * {@link module:image/image/imageinlineediting~ImageInlineEditing `ImageInlineEditing`}\n\t * and {@link module:image/image/imageblockediting~ImageBlockEditing `ImageBlockEditing`} plugins into consideration.\n\t */\n\tprivate _setupConversion( isBlockPluginLoaded: boolean, isInlinePluginLoaded: boolean ): void {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\n\t\tconst modelToViewConverter = modelToViewStyleAttribute( this.normalizedStyles! );\n\t\tconst viewToModelConverter = viewToModelStyleAttribute( this.normalizedStyles! );\n\n\t\teditor.editing.downcastDispatcher.on( 'attribute:imageStyle', modelToViewConverter );\n\t\teditor.data.downcastDispatcher.on( 'attribute:imageStyle', modelToViewConverter );\n\n\t\t// Allow imageStyle attribute in image and imageInline.\n\t\t// We could call it 'style' but https://github.com/ckeditor/ckeditor5-engine/issues/559.\n\t\tif ( isBlockPluginLoaded ) {\n\t\t\tschema.extend( 'imageBlock', { allowAttributes: 'imageStyle' } );\n\n\t\t\t// Converter for figure element from view to model.\n\t\t\teditor.data.upcastDispatcher.on<UpcastElementEvent>( 'element:figure', viewToModelConverter, { priority: 'low' } );\n\t\t}\n\n\t\tif ( isInlinePluginLoaded ) {\n\t\t\tschema.extend( 'imageInline', { allowAttributes: 'imageStyle' } );\n\n\t\t\t// Converter for the img element from view to model.\n\t\t\teditor.data.upcastDispatcher.on( 'element:img', viewToModelConverter, { priority: 'low' } );\n\t\t}\n\t}\n\n\t/**\n\t * Registers a post-fixer that will make sure that the style attribute value is correct for a specific image type (block vs inline).\n\t */\n\tprivate _setupPostFixer(): void {\n\t\tconst editor = this.editor;\n\t\tconst document = editor.model.document;\n\n\t\tconst imageUtils = editor.plugins.get( ImageUtils );\n\t\tconst stylesMap = new Map( this.normalizedStyles!.map( style => [ style.name, style ] ) );\n\n\t\t// Make sure that style attribute is valid for the image type.\n\t\tdocument.registerPostFixer( writer => {\n\t\t\tlet changed = false;\n\n\t\t\tfor ( const change of document.differ.getChanges() ) {\n\t\t\t\tif ( change.type == 'insert' || change.type == 'attribute' && change.attributeKey == 'imageStyle' ) {\n\t\t\t\t\tlet element = change.type == 'insert' ? change.position.nodeAfter! : change.range.start.nodeAfter!;\n\n\t\t\t\t\tif ( element && element.is( 'element', 'paragraph' ) && element.childCount > 0 ) {\n\t\t\t\t\t\telement = element.getChild( 0 )!;\n\t\t\t\t\t}\n\n\t\t\t\t\tif ( !imageUtils.isImage( element as Element ) ) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\tconst imageStyle = element.getAttribute( 'imageStyle' ) as string | undefined;\n\n\t\t\t\t\tif ( !imageStyle ) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\tconst imageStyleDefinition = stylesMap.get( imageStyle );\n\n\t\t\t\t\tif ( !imageStyleDefinition || !imageStyleDefinition.modelElements.includes( ( element as Element ).name ) ) {\n\t\t\t\t\t\twriter.removeAttribute( 'imageStyle', element );\n\t\t\t\t\t\tchanged = true;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\treturn changed;\n\t\t} );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./imagestyle.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagestyle/imagestyleui\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { ButtonView, createDropdown, addToolbarToDropdown, SplitButtonView } from 'ckeditor5/src/ui';\nimport { isObject, identity } from 'lodash-es';\nimport ImageStyleEditing from './imagestyleediting';\nimport utils from './utils';\nimport type { ImageStyleDropdownDefinition, ImageStyleOptionDefinition } from '../imageconfig';\nimport type ImageStyleCommand from './imagestylecommand';\n\nimport '../../theme/imagestyle.css';\n\n/**\n * The image style UI plugin.\n *\n * It registers buttons corresponding to the {@link module:image/imageconfig~ImageConfig#styles} configuration.\n * It also registers the {@link module:image/imagestyle/utils#DEFAULT_DROPDOWN_DEFINITIONS default drop-downs} and the\n * custom drop-downs defined by the developer in the {@link module:image/imageconfig~ImageConfig#toolbar} configuration.\n */\nexport default class ImageStyleUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageStyleEditing ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageStyleUI' as const;\n\t}\n\n\t/**\n\t * Returns the default localized style titles provided by the plugin.\n\t *\n\t * The following localized titles corresponding with\n\t * {@link module:image/imagestyle/utils#DEFAULT_OPTIONS} are available:\n\t *\n\t * * `'Wrap text'`,\n\t * * `'Break text'`,\n\t * * `'In line'`,\n\t * * `'Full size image'`,\n\t * * `'Side image'`,\n\t * * `'Left aligned image'`,\n\t * * `'Centered image'`,\n\t * * `'Right aligned image'`\n\t */\n\tpublic get localizedDefaultStylesTitles(): Record<string, string> {\n\t\tconst t = this.editor.t;\n\n\t\treturn {\n\t\t\t'Wrap text': t( 'Wrap text' ),\n\t\t\t'Break text': t( 'Break text' ),\n\t\t\t'In line': t( 'In line' ),\n\t\t\t'Full size image': t( 'Full size image' ),\n\t\t\t'Side image': t( 'Side image' ),\n\t\t\t'Left aligned image': t( 'Left aligned image' ),\n\t\t\t'Centered image': t( 'Centered image' ),\n\t\t\t'Right aligned image': t( 'Right aligned image' )\n\t\t};\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst plugins = this.editor.plugins;\n\t\tconst toolbarConfig = this.editor.config.get( 'image.toolbar' ) || [];\n\t\tconst imageStyleEditing: ImageStyleEditing = plugins.get( 'ImageStyleEditing' );\n\n\t\tconst definedStyles = translateStyles(\n\t\t\timageStyleEditing.normalizedStyles!,\n\t\t\tthis.localizedDefaultStylesTitles\n\t\t);\n\n\t\tfor ( const styleConfig of definedStyles ) {\n\t\t\tthis._createButton( styleConfig );\n\t\t}\n\n\t\tconst definedDropdowns = translateStyles(\n\t\t\t[\n\t\t\t\t...toolbarConfig.filter( isObject ) as Array<ImageStyleDropdownDefinition>,\n\t\t\t\t...utils.getDefaultDropdownDefinitions( plugins )\n\t\t\t],\n\t\t\tthis.localizedDefaultStylesTitles\n\t\t);\n\n\t\tfor ( const dropdownConfig of definedDropdowns ) {\n\t\t\tthis._createDropdown( dropdownConfig, definedStyles );\n\t\t}\n\t}\n\n\t/**\n\t * Creates a dropdown and stores it in the editor {@link module:ui/componentfactory~ComponentFactory}.\n\t */\n\tprivate _createDropdown( dropdownConfig: ImageStyleDropdownDefinition, definedStyles: Array<ImageStyleOptionDefinition> ): void {\n\t\tconst factory = this.editor.ui.componentFactory;\n\n\t\tfactory.add( dropdownConfig.name, locale => {\n\t\t\tlet defaultButton: ButtonView | undefined;\n\n\t\t\tconst { defaultItem, items, title } = dropdownConfig;\n\t\t\tconst buttonViews = items\n\t\t\t\t.filter( itemName => definedStyles.find( ( { name } ) => getUIComponentName( name ) === itemName ) )\n\t\t\t\t.map( buttonName => {\n\t\t\t\t\tconst button = factory.create( buttonName ) as ButtonView;\n\n\t\t\t\t\tif ( buttonName === defaultItem ) {\n\t\t\t\t\t\tdefaultButton = button;\n\t\t\t\t\t}\n\n\t\t\t\t\treturn button;\n\t\t\t\t} );\n\n\t\t\tif ( items.length !== buttonViews.length ) {\n\t\t\t\tutils.warnInvalidStyle( { dropdown: dropdownConfig } );\n\t\t\t}\n\n\t\t\tconst dropdownView = createDropdown( locale, SplitButtonView );\n\t\t\tconst splitButtonView = dropdownView.buttonView as SplitButtonView;\n\t\t\tconst splitButtonViewArrow = splitButtonView.arrowView;\n\n\t\t\taddToolbarToDropdown( dropdownView, buttonViews, { enableActiveItemFocusOnDropdownOpen: true } );\n\n\t\t\tsplitButtonView.set( {\n\t\t\t\tlabel: getDropdownButtonTitle( title, defaultButton!.label! ),\n\t\t\t\tclass: null,\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tsplitButtonViewArrow.unbind( 'label' );\n\t\t\tsplitButtonViewArrow.set( {\n\t\t\t\tlabel: title\n\t\t\t} );\n\n\t\t\tsplitButtonView.bind( 'icon' ).toMany( buttonViews, 'isOn', ( ...areOn ) => {\n\t\t\t\tconst index = areOn.findIndex( identity );\n\n\t\t\t\treturn ( index < 0 ) ? defaultButton!.icon : buttonViews[ index ].icon;\n\t\t\t} );\n\n\t\t\tsplitButtonView.bind( 'label' ).toMany( buttonViews, 'isOn', ( ...areOn ) => {\n\t\t\t\tconst index = areOn.findIndex( identity );\n\n\t\t\t\treturn getDropdownButtonTitle( title, ( index < 0 ) ? defaultButton!.label! : buttonViews[ index ].label! );\n\t\t\t} );\n\n\t\t\tsplitButtonView.bind( 'isOn' ).toMany( buttonViews, 'isOn', ( ...areOn ) => areOn.some( identity ) );\n\n\t\t\tsplitButtonView.bind( 'class' )\n\t\t\t\t.toMany( buttonViews, 'isOn', ( ...areOn ) => areOn.some( identity ) ? 'ck-splitbutton_flatten' : undefined );\n\n\t\t\tsplitButtonView.on( 'execute', () => {\n\t\t\t\tif ( !buttonViews.some( ( { isOn } ) => isOn ) ) {\n\t\t\t\t\tdefaultButton!.fire( 'execute' );\n\t\t\t\t} else {\n\t\t\t\t\tdropdownView.isOpen = !dropdownView.isOpen;\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tdropdownView.bind( 'isEnabled' )\n\t\t\t\t.toMany( buttonViews, 'isEnabled', ( ...areEnabled ) => areEnabled.some( identity ) );\n\n\t\t\t// Focus the editable after executing the command.\n\t\t\t// Overrides a default behaviour where the focus is moved to the dropdown button (#12125).\n\t\t\tthis.listenTo( dropdownView, 'execute', () => {\n\t\t\t\tthis.editor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn dropdownView;\n\t\t} );\n\t}\n\n\t/**\n\t * Creates a button and stores it in the editor {@link module:ui/componentfactory~ComponentFactory}.\n\t */\n\tprivate _createButton( buttonConfig: ImageStyleOptionDefinition ): void {\n\t\tconst buttonName = buttonConfig.name;\n\n\t\tthis.editor.ui.componentFactory.add( getUIComponentName( buttonName ), locale => {\n\t\t\tconst command: ImageStyleCommand = this.editor.commands.get( 'imageStyle' )!;\n\t\t\tconst view = new ButtonView( locale );\n\n\t\t\tview.set( {\n\t\t\t\tlabel: buttonConfig.title,\n\t\t\t\ticon: buttonConfig.icon,\n\t\t\t\ttooltip: true,\n\t\t\t\tisToggleable: true\n\t\t\t} );\n\n\t\t\tview.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\t\t\tview.bind( 'isOn' ).to( command, 'value', value => value === buttonName );\n\t\t\tview.on( 'execute', this._executeCommand.bind( this, buttonName ) );\n\n\t\t\treturn view;\n\t\t} );\n\t}\n\n\tprivate _executeCommand( name: string ): void {\n\t\tthis.editor.execute( 'imageStyle', { value: name } );\n\t\tthis.editor.editing.view.focus();\n\t}\n}\n\n/**\n * Returns the translated `title` from the passed styles array.\n */\nfunction translateStyles<T extends ImageStyleOptionDefinition | ImageStyleDropdownDefinition>(\n\tstyles: Array<T>,\n\ttitles: Record<string, string>\n): Array<T> {\n\tfor ( const style of styles ) {\n\t\t// Localize the titles of the styles, if a title corresponds with\n\t\t// a localized default provided by the plugin.\n\t\tif ( titles[ style.title! ] ) {\n\t\t\tstyle.title = titles[ style.title! ];\n\t\t}\n\t}\n\n\treturn styles;\n}\n\n/**\n * Returns the image style component name with the \"imageStyle:\" prefix.\n */\nfunction getUIComponentName( name: string ): string {\n\treturn `imageStyle:${ name }`;\n}\n\n/**\n * Returns title for the splitbutton containing the dropdown title and default action item title.\n */\nfunction getDropdownButtonTitle( dropdownTitle: string | undefined, buttonTitle: string ): string {\n\treturn ( dropdownTitle ? dropdownTitle + ': ' : '' ) + buttonTitle;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module indent/indentediting\n */\n\nimport { Plugin, MultiCommand } from 'ckeditor5/src/core';\n\n/**\n * The indent editing feature.\n *\n * This plugin registers the `'indent'` and `'outdent'` commands.\n *\n * **Note**: In order for the commands to work, at least one of the compatible features is required. Read more in the\n * {@link module:indent/indent~Indent indent feature} API documentation.\n */\nexport default class IndentEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'IndentEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\teditor.commands.add( 'indent', new MultiCommand( editor ) );\n\t\teditor.commands.add( 'outdent', new MultiCommand( editor ) );\n\t}\n}\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2 3.75c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm5 6c0 .414.336.75.75.75h9.5a.75.75 0 1 0 0-1.5h-9.5a.75.75 0 0 0-.75.75zM2.75 16.5h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 1 0 0 1.5zM1.632 6.95 5.02 9.358a.4.4 0 0 1-.013.661l-3.39 2.207A.4.4 0 0 1 1 11.892V7.275a.4.4 0 0 1 .632-.326z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2 3.75c0 .414.336.75.75.75h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 0 0-.75.75zm5 6c0 .414.336.75.75.75h9.5a.75.75 0 1 0 0-1.5h-9.5a.75.75 0 0 0-.75.75zM2.75 16.5h14.5a.75.75 0 1 0 0-1.5H2.75a.75.75 0 1 0 0 1.5zm1.618-9.55L.98 9.358a.4.4 0 0 0 .013.661l3.39 2.207A.4.4 0 0 0 5 11.892V7.275a.4.4 0 0 0-.632-.326z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module indent/indentui\n */\n\nimport { ButtonView } from 'ckeditor5/src/ui';\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport indentIcon from '../theme/icons/indent.svg';\nimport outdentIcon from '../theme/icons/outdent.svg';\n\n/**\n * The indent UI feature.\n *\n * This plugin registers the `'indent'` and `'outdent'` buttons.\n *\n * **Note**: In order for the commands to work, at least one of the compatible features is required. Read more in\n * the {@link module:indent/indent~Indent indent feature} API documentation.\n */\nexport default class IndentUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'IndentUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst locale = editor.locale;\n\t\tconst t = editor.t;\n\n\t\tconst localizedIndentIcon = locale.uiLanguageDirection == 'ltr' ? indentIcon : outdentIcon;\n\t\tconst localizedOutdentIcon = locale.uiLanguageDirection == 'ltr' ? outdentIcon : indentIcon;\n\n\t\tthis._defineButton( 'indent', t( 'Increase indent' ), localizedIndentIcon );\n\t\tthis._defineButton( 'outdent', t( 'Decrease indent' ), localizedOutdentIcon );\n\t}\n\n\t/**\n\t * Defines a UI button.\n\t */\n\tprivate _defineButton( commandName: 'indent' | 'outdent', label: string, icon: string ): void {\n\t\tconst editor = this.editor;\n\n\t\teditor.ui.componentFactory.add( commandName, locale => {\n\t\t\tconst command = editor.commands.get( commandName )!;\n\t\t\tconst view = new ButtonView( locale );\n\n\t\t\tview.set( {\n\t\t\t\tlabel,\n\t\t\t\ticon,\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tview.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\n\t\t\tthis.listenTo( view, 'execute', () => {\n\t\t\t\teditor.execute( commandName );\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\treturn view;\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/utils/automaticdecorators\n */\n\nimport { toMap, type ArrayOrItem } from 'ckeditor5/src/utils';\nimport type { DowncastAttributeEvent, DowncastDispatcher, Element, ViewElement } from 'ckeditor5/src/engine';\nimport type { NormalizedLinkDecoratorAutomaticDefinition } from '../utils';\n\n/**\n * Helper class that ties together all {@link module:link/linkconfig~LinkDecoratorAutomaticDefinition} and provides\n * the {@link module:engine/conversion/downcasthelpers~DowncastHelpers#attributeToElement downcast dispatchers} for them.\n */\nexport default class AutomaticDecorators {\n\t/**\n\t * Stores the definition of {@link module:link/linkconfig~LinkDecoratorAutomaticDefinition automatic decorators}.\n\t * This data is used as a source for a downcast dispatcher to create a proper conversion to output data.\n\t */\n\tprivate _definitions = new Set<NormalizedLinkDecoratorAutomaticDefinition>();\n\n\t/**\n\t * Gives information about the number of decorators stored in the {@link module:link/utils/automaticdecorators~AutomaticDecorators}\n\t * instance.\n\t */\n\tpublic get length(): number {\n\t\treturn this._definitions.size;\n\t}\n\n\t/**\n\t * Adds automatic decorator objects or an array with them to be used during downcasting.\n\t *\n\t * @param item A configuration object of automatic rules for decorating links. It might also be an array of such objects.\n\t */\n\tpublic add( item: ArrayOrItem<NormalizedLinkDecoratorAutomaticDefinition> ): void {\n\t\tif ( Array.isArray( item ) ) {\n\t\t\titem.forEach( item => this._definitions.add( item ) );\n\t\t} else {\n\t\t\tthis._definitions.add( item );\n\t\t}\n\t}\n\n\t/**\n\t * Provides the conversion helper used in the {@link module:engine/conversion/downcasthelpers~DowncastHelpers#add} method.\n\t *\n\t * @returns A dispatcher function used as conversion helper in {@link module:engine/conversion/downcasthelpers~DowncastHelpers#add}.\n\t */\n\tpublic getDispatcher(): ( dispatcher: DowncastDispatcher ) => void {\n\t\treturn dispatcher => {\n\t\t\tdispatcher.on<DowncastAttributeEvent>( 'attribute:linkHref', ( evt, data, conversionApi ) => {\n\t\t\t\t// There is only test as this behavior decorates links and\n\t\t\t\t// it is run before dispatcher which actually consumes this node.\n\t\t\t\t// This allows on writing own dispatcher with highest priority,\n\t\t\t\t// which blocks both native converter and this additional decoration.\n\t\t\t\tif ( !conversionApi.consumable.test( data.item, 'attribute:linkHref' ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// Automatic decorators for block links are handled e.g. in LinkImageEditing.\n\t\t\t\tif ( !( data.item.is( 'selection' ) || conversionApi.schema.isInline( data.item ) ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst viewWriter = conversionApi.writer;\n\t\t\t\tconst viewSelection = viewWriter.document.selection;\n\n\t\t\t\tfor ( const item of this._definitions ) {\n\t\t\t\t\tconst viewElement = viewWriter.createAttributeElement( 'a', item.attributes, {\n\t\t\t\t\t\tpriority: 5\n\t\t\t\t\t} );\n\n\t\t\t\t\tif ( item.classes ) {\n\t\t\t\t\t\tviewWriter.addClass( item.classes, viewElement );\n\t\t\t\t\t}\n\n\t\t\t\t\tfor ( const key in item.styles ) {\n\t\t\t\t\t\tviewWriter.setStyle( key, item.styles[ key ], viewElement );\n\t\t\t\t\t}\n\n\t\t\t\t\tviewWriter.setCustomProperty( 'link', true, viewElement );\n\n\t\t\t\t\tif ( item.callback( data.attributeNewValue as string | null ) ) {\n\t\t\t\t\t\tif ( data.item.is( 'selection' ) ) {\n\t\t\t\t\t\t\tviewWriter.wrap( viewSelection.getFirstRange()!, viewElement );\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tviewWriter.wrap( conversionApi.mapper.toViewRange( data.range ), viewElement );\n\t\t\t\t\t\t}\n\t\t\t\t\t} else {\n\t\t\t\t\t\tviewWriter.unwrap( conversionApi.mapper.toViewRange( data.range ), viewElement );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}, { priority: 'high' } );\n\t\t};\n\t}\n\n\t/**\n\t * Provides the conversion helper used in the {@link module:engine/conversion/downcasthelpers~DowncastHelpers#add} method\n\t * when linking images.\n\t *\n\t * @returns A dispatcher function used as conversion helper in {@link module:engine/conversion/downcasthelpers~DowncastHelpers#add}.\n\t */\n\tpublic getDispatcherForLinkedImage(): ( dispatcher: DowncastDispatcher ) => void {\n\t\treturn dispatcher => {\n\t\t\tdispatcher.on<DowncastAttributeEvent<Element>>( 'attribute:linkHref:imageBlock', ( evt, data, { writer, mapper } ) => {\n\t\t\t\tconst viewFigure = mapper.toViewElement( data.item )!;\n\t\t\t\tconst linkInImage = Array.from( viewFigure.getChildren() )\n\t\t\t\t\t.find( ( child ): child is ViewElement => child.is( 'element', 'a' ) )!;\n\n\t\t\t\tfor ( const item of this._definitions ) {\n\t\t\t\t\tconst attributes = toMap( item.attributes );\n\n\t\t\t\t\tif ( item.callback( data.attributeNewValue as string | null ) ) {\n\t\t\t\t\t\tfor ( const [ key, val ] of attributes ) {\n\t\t\t\t\t\t\t// Left for backward compatibility. Since v30 decorator should\n\t\t\t\t\t\t\t// accept `classes` and `styles` separately from `attributes`.\n\t\t\t\t\t\t\tif ( key === 'class' ) {\n\t\t\t\t\t\t\t\twriter.addClass( val, linkInImage );\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\twriter.setAttribute( key, val, linkInImage );\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif ( item.classes ) {\n\t\t\t\t\t\t\twriter.addClass( item.classes, linkInImage );\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tfor ( const key in item.styles ) {\n\t\t\t\t\t\t\twriter.setStyle( key, item.styles[ key ], linkInImage );\n\t\t\t\t\t\t}\n\t\t\t\t\t} else {\n\t\t\t\t\t\tfor ( const [ key, val ] of attributes ) {\n\t\t\t\t\t\t\tif ( key === 'class' ) {\n\t\t\t\t\t\t\t\twriter.removeClass( val, linkInImage );\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\twriter.removeAttribute( key, linkInImage );\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif ( item.classes ) {\n\t\t\t\t\t\t\twriter.removeClass( item.classes, linkInImage );\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tfor ( const key in item.styles ) {\n\t\t\t\t\t\t\twriter.removeStyle( key, linkInImage );\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\t\t};\n\t}\n}\n","import baseSlice from './_baseSlice.js';\n\n/**\n * Casts `array` to a slice if it's needed.\n *\n * @private\n * @param {Array} array The array to inspect.\n * @param {number} start The start position.\n * @param {number} [end=array.length] The end position.\n * @returns {Array} Returns the cast slice.\n */\nfunction castSlice(array, start, end) {\n var length = array.length;\n end = end === undefined ? length : end;\n return (!start && end >= length) ? array : baseSlice(array, start, end);\n}\n\nexport default castSlice;\n","/** Used to compose unicode character classes. */\nvar rsAstralRange = '\\\\ud800-\\\\udfff',\n rsComboMarksRange = '\\\\u0300-\\\\u036f',\n reComboHalfMarksRange = '\\\\ufe20-\\\\ufe2f',\n rsComboSymbolsRange = '\\\\u20d0-\\\\u20ff',\n rsComboRange = rsComboMarksRange + reComboHalfMarksRange + rsComboSymbolsRange,\n rsVarRange = '\\\\ufe0e\\\\ufe0f';\n\n/** Used to compose unicode capture groups. */\nvar rsZWJ = '\\\\u200d';\n\n/** Used to detect strings with [zero-width joiners or code points from the astral planes](http://eev.ee/blog/2015/09/12/dark-corners-of-unicode/). */\nvar reHasUnicode = RegExp('[' + rsZWJ + rsAstralRange + rsComboRange + rsVarRange + ']');\n\n/**\n * Checks if `string` contains Unicode symbols.\n *\n * @private\n * @param {string} string The string to inspect.\n * @returns {boolean} Returns `true` if a symbol is found, else `false`.\n */\nfunction hasUnicode(string) {\n return reHasUnicode.test(string);\n}\n\nexport default hasUnicode;\n","/**\n * Converts an ASCII `string` to an array.\n *\n * @private\n * @param {string} string The string to convert.\n * @returns {Array} Returns the converted array.\n */\nfunction asciiToArray(string) {\n return string.split('');\n}\n\nexport default asciiToArray;\n","/** Used to compose unicode character classes. */\nvar rsAstralRange = '\\\\ud800-\\\\udfff',\n rsComboMarksRange = '\\\\u0300-\\\\u036f',\n reComboHalfMarksRange = '\\\\ufe20-\\\\ufe2f',\n rsComboSymbolsRange = '\\\\u20d0-\\\\u20ff',\n rsComboRange = rsComboMarksRange + reComboHalfMarksRange + rsComboSymbolsRange,\n rsVarRange = '\\\\ufe0e\\\\ufe0f';\n\n/** Used to compose unicode capture groups. */\nvar rsAstral = '[' + rsAstralRange + ']',\n rsCombo = '[' + rsComboRange + ']',\n rsFitz = '\\\\ud83c[\\\\udffb-\\\\udfff]',\n rsModifier = '(?:' + rsCombo + '|' + rsFitz + ')',\n rsNonAstral = '[^' + rsAstralRange + ']',\n rsRegional = '(?:\\\\ud83c[\\\\udde6-\\\\uddff]){2}',\n rsSurrPair = '[\\\\ud800-\\\\udbff][\\\\udc00-\\\\udfff]',\n rsZWJ = '\\\\u200d';\n\n/** Used to compose unicode regexes. */\nvar reOptMod = rsModifier + '?',\n rsOptVar = '[' + rsVarRange + ']?',\n rsOptJoin = '(?:' + rsZWJ + '(?:' + [rsNonAstral, rsRegional, rsSurrPair].join('|') + ')' + rsOptVar + reOptMod + ')*',\n rsSeq = rsOptVar + reOptMod + rsOptJoin,\n rsSymbol = '(?:' + [rsNonAstral + rsCombo + '?', rsCombo, rsRegional, rsSurrPair, rsAstral].join('|') + ')';\n\n/** Used to match [string symbols](https://mathiasbynens.be/notes/javascript-unicode). */\nvar reUnicode = RegExp(rsFitz + '(?=' + rsFitz + ')|' + rsSymbol + rsSeq, 'g');\n\n/**\n * Converts a Unicode `string` to an array.\n *\n * @private\n * @param {string} string The string to convert.\n * @returns {Array} Returns the converted array.\n */\nfunction unicodeToArray(string) {\n return string.match(reUnicode) || [];\n}\n\nexport default unicodeToArray;\n","import asciiToArray from './_asciiToArray.js';\nimport hasUnicode from './_hasUnicode.js';\nimport unicodeToArray from './_unicodeToArray.js';\n\n/**\n * Converts `string` to an array.\n *\n * @private\n * @param {string} string The string to convert.\n * @returns {Array} Returns the converted array.\n */\nfunction stringToArray(string) {\n return hasUnicode(string)\n ? unicodeToArray(string)\n : asciiToArray(string);\n}\n\nexport default stringToArray;\n","import createCaseFirst from './_createCaseFirst.js';\n\n/**\n * Converts the first character of `string` to upper case.\n *\n * @static\n * @memberOf _\n * @since 4.0.0\n * @category String\n * @param {string} [string=''] The string to convert.\n * @returns {string} Returns the converted string.\n * @example\n *\n * _.upperFirst('fred');\n * // => 'Fred'\n *\n * _.upperFirst('FRED');\n * // => 'FRED'\n */\nvar upperFirst = createCaseFirst('toUpperCase');\n\nexport default upperFirst;\n","import castSlice from './_castSlice.js';\nimport hasUnicode from './_hasUnicode.js';\nimport stringToArray from './_stringToArray.js';\nimport toString from './toString.js';\n\n/**\n * Creates a function like `_.lowerFirst`.\n *\n * @private\n * @param {string} methodName The name of the `String` case method to use.\n * @returns {Function} Returns the new case function.\n */\nfunction createCaseFirst(methodName) {\n return function(string) {\n string = toString(string);\n\n var strSymbols = hasUnicode(string)\n ? stringToArray(string)\n : undefined;\n\n var chr = strSymbols\n ? strSymbols[0]\n : string.charAt(0);\n\n var trailing = strSymbols\n ? castSlice(strSymbols, 1).join('')\n : string.slice(1);\n\n return chr[methodName]() + trailing;\n };\n}\n\nexport default createCaseFirst;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/utils\n */\n\n/* global window */\n\nimport type {\n\tDowncastConversionApi,\n\tElement,\n\tSchema,\n\tViewAttributeElement,\n\tViewNode,\n\tViewDocumentFragment\n} from 'ckeditor5/src/engine';\nimport type { LocaleTranslate } from 'ckeditor5/src/utils';\n\nimport type {\n\tLinkDecoratorAutomaticDefinition,\n\tLinkDecoratorDefinition,\n\tLinkDecoratorManualDefinition\n} from './linkconfig';\n\nimport { upperFirst } from 'lodash-es';\n\nconst ATTRIBUTE_WHITESPACES = /[\\u0000-\\u0020\\u00A0\\u1680\\u180E\\u2000-\\u2029\\u205f\\u3000]/g; // eslint-disable-line no-control-regex\nconst SAFE_URL = /^(?:(?:https?|ftps?|mailto):|[^a-z]|[a-z+.-]+(?:[^a-z+.:-]|$))/i;\n\n// Simplified email test - should be run over previously found URL.\nconst EMAIL_REG_EXP = /^[\\S]+@((?![-_])(?:[-\\w\\u00a1-\\uffff]{0,63}[^-_]\\.))+(?:[a-z\\u00a1-\\uffff]{2,})$/i;\n\n// The regex checks for the protocol syntax ('xxxx://' or 'xxxx:')\n// or non-word characters at the beginning of the link ('/', '#' etc.).\nconst PROTOCOL_REG_EXP = /^((\\w+:(\\/{2,})?)|(\\W))/i;\n\n/**\n * A keystroke used by the {@link module:link/linkui~LinkUI link UI feature}.\n */\nexport const LINK_KEYSTROKE = 'Ctrl+K';\n\n/**\n * Returns `true` if a given view node is the link element.\n */\nexport function isLinkElement( node: ViewNode | ViewDocumentFragment ): boolean {\n\treturn node.is( 'attributeElement' ) && !!node.getCustomProperty( 'link' );\n}\n\n/**\n * Creates a link {@link module:engine/view/attributeelement~AttributeElement} with the provided `href` attribute.\n */\nexport function createLinkElement( href: string, { writer }: DowncastConversionApi ): ViewAttributeElement {\n\t// Priority 5 - https://github.com/ckeditor/ckeditor5-link/issues/121.\n\tconst linkElement = writer.createAttributeElement( 'a', { href }, { priority: 5 } );\n\n\twriter.setCustomProperty( 'link', true, linkElement );\n\n\treturn linkElement;\n}\n\n/**\n * Returns a safe URL based on a given value.\n *\n * A URL is considered safe if it is safe for the user (does not contain any malicious code).\n *\n * If a URL is considered unsafe, a simple `\"#\"` is returned.\n *\n * @internal\n */\nexport function ensureSafeUrl( url: unknown ): string {\n\tconst urlString = String( url );\n\n\treturn isSafeUrl( urlString ) ? urlString : '#';\n}\n\n/**\n * Checks whether the given URL is safe for the user (does not contain any malicious code).\n */\nfunction isSafeUrl( url: string ): boolean {\n\tconst normalizedUrl = url.replace( ATTRIBUTE_WHITESPACES, '' );\n\n\treturn !!normalizedUrl.match( SAFE_URL );\n}\n\n/**\n * Returns the {@link module:link/linkconfig~LinkConfig#decorators `config.link.decorators`} configuration processed\n * to respect the locale of the editor, i.e. to display the {@link module:link/linkconfig~LinkDecoratorManualDefinition label}\n * in the correct language.\n *\n * **Note**: Only the few most commonly used labels are translated automatically. Other labels should be manually\n * translated in the {@link module:link/linkconfig~LinkConfig#decorators `config.link.decorators`} configuration.\n *\n * @param t Shorthand for {@link module:utils/locale~Locale#t Locale#t}.\n * @param decorators The decorator reference where the label values should be localized.\n */\nexport function getLocalizedDecorators(\n\tt: LocaleTranslate,\n\tdecorators: Array<NormalizedLinkDecoratorDefinition>\n): Array<NormalizedLinkDecoratorDefinition> {\n\tconst localizedDecoratorsLabels: Record<string, string> = {\n\t\t'Open in a new tab': t( 'Open in a new tab' ),\n\t\t'Downloadable': t( 'Downloadable' )\n\t};\n\n\tdecorators.forEach( decorator => {\n\t\tif ( 'label' in decorator && localizedDecoratorsLabels[ decorator.label ] ) {\n\t\t\tdecorator.label = localizedDecoratorsLabels[ decorator.label ];\n\t\t}\n\n\t\treturn decorator;\n\t} );\n\n\treturn decorators;\n}\n\n/**\n * Converts an object with defined decorators to a normalized array of decorators. The `id` key is added for each decorator and\n * is used as the attribute's name in the model.\n */\nexport function normalizeDecorators( decorators?: Record<string, LinkDecoratorDefinition> ): Array<NormalizedLinkDecoratorDefinition> {\n\tconst retArray: Array<NormalizedLinkDecoratorDefinition> = [];\n\n\tif ( decorators ) {\n\t\tfor ( const [ key, value ] of Object.entries( decorators ) ) {\n\t\t\tconst decorator = Object.assign(\n\t\t\t\t{},\n\t\t\t\tvalue,\n\t\t\t\t{ id: `link${ upperFirst( key ) }` }\n\t\t\t);\n\n\t\t\tretArray.push( decorator );\n\t\t}\n\t}\n\n\treturn retArray;\n}\n\n/**\n * Returns `true` if the specified `element` can be linked (the element allows the `linkHref` attribute).\n */\nexport function isLinkableElement( element: Element | null, schema: Schema ): element is Element {\n\tif ( !element ) {\n\t\treturn false;\n\t}\n\n\treturn schema.checkAttribute( element.name, 'linkHref' );\n}\n\n/**\n * Returns `true` if the specified `value` is an email.\n */\nexport function isEmail( value: string ): boolean {\n\treturn EMAIL_REG_EXP.test( value );\n}\n\n/**\n * Adds the protocol prefix to the specified `link` when:\n *\n * * it does not contain it already, and there is a {@link module:link/linkconfig~LinkConfig#defaultProtocol `defaultProtocol` }\n * configuration value provided,\n * * or the link is an email address.\n */\nexport function addLinkProtocolIfApplicable( link: string, defaultProtocol?: string ): string {\n\tconst protocol = isEmail( link ) ? 'mailto:' : defaultProtocol;\n\tconst isProtocolNeeded = !!protocol && !linkHasProtocol( link );\n\n\treturn link && isProtocolNeeded ? protocol + link : link;\n}\n\n/**\n * Checks if protocol is already included in the link.\n */\nexport function linkHasProtocol( link: string ): boolean {\n\treturn PROTOCOL_REG_EXP.test( link );\n}\n\n/**\n * Opens the link in a new browser tab.\n */\nexport function openLink( link: string ): void {\n\twindow.open( link, '_blank', 'noopener' );\n}\n\nexport type NormalizedLinkDecoratorAutomaticDefinition = LinkDecoratorAutomaticDefinition & { id: string };\nexport type NormalizedLinkDecoratorManualDefinition = LinkDecoratorManualDefinition & { id: string };\nexport type NormalizedLinkDecoratorDefinition = NormalizedLinkDecoratorAutomaticDefinition | NormalizedLinkDecoratorManualDefinition;\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/linkcommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport { findAttributeRange } from 'ckeditor5/src/typing';\nimport { Collection, first, toMap } from 'ckeditor5/src/utils';\nimport type { Range, DocumentSelection, Model, Writer } from 'ckeditor5/src/engine';\n\nimport AutomaticDecorators from './utils/automaticdecorators';\nimport { isLinkableElement } from './utils';\nimport type ManualDecorator from './utils/manualdecorator';\n\n/**\n * The link command. It is used by the {@link module:link/link~Link link feature}.\n */\nexport default class LinkCommand extends Command {\n\t/**\n\t * The value of the `'linkHref'` attribute if the start of the selection is located in a node with this attribute.\n\t *\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public value: string | undefined;\n\n\t/**\n\t * A collection of {@link module:link/utils/manualdecorator~ManualDecorator manual decorators}\n\t * corresponding to the {@link module:link/linkconfig~LinkConfig#decorators decorator configuration}.\n\t *\n\t * You can consider it a model with states of manual decorators added to the currently selected link.\n\t */\n\tpublic readonly manualDecorators = new Collection<ManualDecorator>();\n\n\t/**\n\t * An instance of the helper that ties together all {@link module:link/linkconfig~LinkDecoratorAutomaticDefinition}\n\t * that are used by the {@glink features/link link} and the {@glink features/images/images-linking linking images} features.\n\t */\n\tpublic readonly automaticDecorators = new AutomaticDecorators();\n\n\t/**\n\t * Synchronizes the state of {@link #manualDecorators} with the currently present elements in the model.\n\t */\n\tpublic restoreManualDecoratorStates(): void {\n\t\tfor ( const manualDecorator of this.manualDecorators ) {\n\t\t\tmanualDecorator.value = this._getDecoratorStateFromModel( manualDecorator.id );\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst selectedElement = selection.getSelectedElement() || first( selection.getSelectedBlocks() );\n\n\t\t// A check for any integration that allows linking elements (e.g. `LinkImage`).\n\t\t// Currently the selection reads attributes from text nodes only. See #7429 and #7465.\n\t\tif ( isLinkableElement( selectedElement, model.schema ) ) {\n\t\t\tthis.value = selectedElement.getAttribute( 'linkHref' ) as string | undefined;\n\t\t\tthis.isEnabled = model.schema.checkAttribute( selectedElement, 'linkHref' );\n\t\t} else {\n\t\t\tthis.value = selection.getAttribute( 'linkHref' ) as string | undefined;\n\t\t\tthis.isEnabled = model.schema.checkAttributeInSelection( selection, 'linkHref' );\n\t\t}\n\n\t\tfor ( const manualDecorator of this.manualDecorators ) {\n\t\t\tmanualDecorator.value = this._getDecoratorStateFromModel( manualDecorator.id );\n\t\t}\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * When the selection is non-collapsed, the `linkHref` attribute will be applied to nodes inside the selection, but only to\n\t * those nodes where the `linkHref` attribute is allowed (disallowed nodes will be omitted).\n\t *\n\t * When the selection is collapsed and is not inside the text with the `linkHref` attribute, a\n\t * new {@link module:engine/model/text~Text text node} with the `linkHref` attribute will be inserted in place of the caret, but\n\t * only if such element is allowed in this place. The `_data` of the inserted text will equal the `href` parameter.\n\t * The selection will be updated to wrap the just inserted text node.\n\t *\n\t * When the selection is collapsed and inside the text with the `linkHref` attribute, the attribute value will be updated.\n\t *\n\t * # Decorators and model attribute management\n\t *\n\t * There is an optional argument to this command that applies or removes model\n\t * {@glink framework/architecture/editing-engine#text-attributes text attributes} brought by\n\t * {@link module:link/utils/manualdecorator~ManualDecorator manual link decorators}.\n\t *\n\t * Text attribute names in the model correspond to the entries in the {@link module:link/linkconfig~LinkConfig#decorators\n\t * configuration}.\n\t * For every decorator configured, a model text attribute exists with the \"link\" prefix. For example, a `'linkMyDecorator'` attribute\n\t * corresponds to `'myDecorator'` in the configuration.\n\t *\n\t * To learn more about link decorators, check out the {@link module:link/linkconfig~LinkConfig#decorators `config.link.decorators`}\n\t * documentation.\n\t *\n\t * Here is how to manage decorator attributes with the link command:\n\t *\n\t * ```ts\n\t * const linkCommand = editor.commands.get( 'link' );\n\t *\n\t * // Adding a new decorator attribute.\n\t * linkCommand.execute( 'http://example.com', {\n\t * \tlinkIsExternal: true\n\t * } );\n\t *\n\t * // Removing a decorator attribute from the selection.\n\t * linkCommand.execute( 'http://example.com', {\n\t * \tlinkIsExternal: false\n\t * } );\n\t *\n\t * // Adding multiple decorator attributes at the same time.\n\t * linkCommand.execute( 'http://example.com', {\n\t * \tlinkIsExternal: true,\n\t * \tlinkIsDownloadable: true,\n\t * } );\n\t *\n\t * // Removing and adding decorator attributes at the same time.\n\t * linkCommand.execute( 'http://example.com', {\n\t * \tlinkIsExternal: false,\n\t * \tlinkFoo: true,\n\t * \tlinkIsDownloadable: false,\n\t * } );\n\t * ```\n\t *\n\t * **Note**: If the decorator attribute name is not specified, its state remains untouched.\n\t *\n\t * **Note**: {@link module:link/unlinkcommand~UnlinkCommand#execute `UnlinkCommand#execute()`} removes all\n\t * decorator attributes.\n\t *\n\t * @fires execute\n\t * @param href Link destination.\n\t * @param manualDecoratorIds The information about manual decorator attributes to be applied or removed upon execution.\n\t */\n\tpublic override execute( href: string, manualDecoratorIds: Record<string, boolean> = {} ): void {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\t// Stores information about manual decorators to turn them on/off when command is applied.\n\t\tconst truthyManualDecorators: Array<string> = [];\n\t\tconst falsyManualDecorators: Array<string> = [];\n\n\t\tfor ( const name in manualDecoratorIds ) {\n\t\t\tif ( manualDecoratorIds[ name ] ) {\n\t\t\t\ttruthyManualDecorators.push( name );\n\t\t\t} else {\n\t\t\t\tfalsyManualDecorators.push( name );\n\t\t\t}\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\t// If selection is collapsed then update selected link or insert new one at the place of caret.\n\t\t\tif ( selection.isCollapsed ) {\n\t\t\t\tconst position = selection.getFirstPosition()!;\n\n\t\t\t\t// When selection is inside text with `linkHref` attribute.\n\t\t\t\tif ( selection.hasAttribute( 'linkHref' ) ) {\n\t\t\t\t\tconst linkText = extractTextFromSelection( selection );\n\t\t\t\t\t// Then update `linkHref` value.\n\t\t\t\t\tlet linkRange = findAttributeRange( position, 'linkHref', selection.getAttribute( 'linkHref' ), model );\n\n\t\t\t\t\tif ( selection.getAttribute( 'linkHref' ) === linkText ) {\n\t\t\t\t\t\tlinkRange = this._updateLinkContent( model, writer, linkRange, href );\n\t\t\t\t\t}\n\n\t\t\t\t\twriter.setAttribute( 'linkHref', href, linkRange );\n\n\t\t\t\t\ttruthyManualDecorators.forEach( item => {\n\t\t\t\t\t\twriter.setAttribute( item, true, linkRange );\n\t\t\t\t\t} );\n\n\t\t\t\t\tfalsyManualDecorators.forEach( item => {\n\t\t\t\t\t\twriter.removeAttribute( item, linkRange );\n\t\t\t\t\t} );\n\n\t\t\t\t\t// Put the selection at the end of the updated link.\n\t\t\t\t\twriter.setSelection( writer.createPositionAfter( linkRange.end.nodeBefore! ) );\n\t\t\t\t}\n\t\t\t\t// If not then insert text node with `linkHref` attribute in place of caret.\n\t\t\t\t// However, since selection is collapsed, attribute value will be used as data for text node.\n\t\t\t\t// So, if `href` is empty, do not create text node.\n\t\t\t\telse if ( href !== '' ) {\n\t\t\t\t\tconst attributes = toMap( selection.getAttributes() );\n\n\t\t\t\t\tattributes.set( 'linkHref', href );\n\n\t\t\t\t\ttruthyManualDecorators.forEach( item => {\n\t\t\t\t\t\tattributes.set( item, true );\n\t\t\t\t\t} );\n\n\t\t\t\t\tconst { end: positionAfter } = model.insertContent( writer.createText( href, attributes ), position );\n\n\t\t\t\t\t// Put the selection at the end of the inserted link.\n\t\t\t\t\t// Using end of range returned from insertContent in case nodes with the same attributes got merged.\n\t\t\t\t\twriter.setSelection( positionAfter );\n\t\t\t\t}\n\n\t\t\t\t// Remove the `linkHref` attribute and all link decorators from the selection.\n\t\t\t\t// It stops adding a new content into the link element.\n\t\t\t\t[ 'linkHref', ...truthyManualDecorators, ...falsyManualDecorators ].forEach( item => {\n\t\t\t\t\twriter.removeSelectionAttribute( item );\n\t\t\t\t} );\n\t\t\t} else {\n\t\t\t\t// If selection has non-collapsed ranges, we change attribute on nodes inside those ranges\n\t\t\t\t// omitting nodes where the `linkHref` attribute is disallowed.\n\t\t\t\tconst ranges = model.schema.getValidRanges( selection.getRanges(), 'linkHref' );\n\n\t\t\t\t// But for the first, check whether the `linkHref` attribute is allowed on selected blocks (e.g. the \"image\" element).\n\t\t\t\tconst allowedRanges = [];\n\n\t\t\t\tfor ( const element of selection.getSelectedBlocks() ) {\n\t\t\t\t\tif ( model.schema.checkAttribute( element, 'linkHref' ) ) {\n\t\t\t\t\t\tallowedRanges.push( writer.createRangeOn( element ) );\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Ranges that accept the `linkHref` attribute. Since we will iterate over `allowedRanges`, let's clone it.\n\t\t\t\tconst rangesToUpdate = allowedRanges.slice();\n\n\t\t\t\t// For all selection ranges we want to check whether given range is inside an element that accepts the `linkHref` attribute.\n\t\t\t\t// If so, we don't want to propagate applying the attribute to its children.\n\t\t\t\tfor ( const range of ranges ) {\n\t\t\t\t\tif ( this._isRangeToUpdate( range, allowedRanges ) ) {\n\t\t\t\t\t\trangesToUpdate.push( range );\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tfor ( const range of rangesToUpdate ) {\n\t\t\t\t\tlet linkRange = range;\n\n\t\t\t\t\tif ( rangesToUpdate.length === 1 ) {\n\t\t\t\t\t\t// Current text of the link in the document.\n\t\t\t\t\t\tconst linkText = extractTextFromSelection( selection );\n\n\t\t\t\t\t\tif ( selection.getAttribute( 'linkHref' ) === linkText ) {\n\t\t\t\t\t\t\tlinkRange = this._updateLinkContent( model, writer, range, href );\n\t\t\t\t\t\t\twriter.setSelection( writer.createSelection( linkRange ) );\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\twriter.setAttribute( 'linkHref', href, linkRange );\n\n\t\t\t\t\ttruthyManualDecorators.forEach( item => {\n\t\t\t\t\t\twriter.setAttribute( item, true, linkRange );\n\t\t\t\t\t} );\n\n\t\t\t\t\tfalsyManualDecorators.forEach( item => {\n\t\t\t\t\t\twriter.removeAttribute( item, linkRange );\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Provides information whether a decorator with a given name is present in the currently processed selection.\n\t *\n\t * @param decoratorName The name of the manual decorator used in the model\n\t * @returns The information whether a given decorator is currently present in the selection.\n\t */\n\tprivate _getDecoratorStateFromModel( decoratorName: string ): boolean | undefined {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\t// A check for the `LinkImage` plugin. If the selection contains an element, get values from the element.\n\t\t// Currently the selection reads attributes from text nodes only. See #7429 and #7465.\n\t\tif ( isLinkableElement( selectedElement, model.schema ) ) {\n\t\t\treturn selectedElement.getAttribute( decoratorName ) as boolean | undefined;\n\t\t}\n\n\t\treturn selection.getAttribute( decoratorName ) as boolean | undefined;\n\t}\n\n\t/**\n\t * Checks whether specified `range` is inside an element that accepts the `linkHref` attribute.\n\t *\n\t * @param range A range to check.\n\t * @param allowedRanges An array of ranges created on elements where the attribute is accepted.\n\t */\n\tprivate _isRangeToUpdate( range: Range, allowedRanges: Array<Range> ): boolean {\n\t\tfor ( const allowedRange of allowedRanges ) {\n\t\t\t// A range is inside an element that will have the `linkHref` attribute. Do not modify its nodes.\n\t\t\tif ( allowedRange.containsRange( range ) ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Updates selected link with a new value as its content and as its href attribute.\n\t *\n\t * @param model Model is need to insert content.\n\t * @param writer Writer is need to create text element in model.\n\t * @param range A range where should be inserted content.\n\t * @param href A link value which should be in the href attribute and in the content.\n\t */\n\tprivate _updateLinkContent( model: Model, writer: Writer, range: Range, href: string ): Range {\n\t\tconst text = writer.createText( href, { linkHref: href } );\n\n\t\treturn model.insertContent( text, range );\n\t}\n}\n\n// Returns a text of a link under the collapsed selection or a selection that contains the entire link.\nfunction extractTextFromSelection( selection: DocumentSelection ): string | null {\n\tif ( selection.isCollapsed ) {\n\t\tconst firstPosition = selection.getFirstPosition();\n\n\t\treturn firstPosition!.textNode && firstPosition!.textNode.data;\n\t} else {\n\t\tconst rangeItems = Array.from( selection.getFirstRange()!.getItems() );\n\n\t\tif ( rangeItems.length > 1 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tconst firstNode = rangeItems[ 0 ];\n\n\t\tif ( firstNode.is( '$text' ) || firstNode.is( '$textProxy' ) ) {\n\t\t\treturn firstNode.data;\n\t\t}\n\n\t\treturn null;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/unlinkcommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport { findAttributeRange } from 'ckeditor5/src/typing';\n\nimport type LinkCommand from './linkcommand';\nimport { isLinkableElement } from './utils';\n\n/**\n * The unlink command. It is used by the {@link module:link/link~Link link plugin}.\n */\nexport default class UnlinkCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\t// A check for any integration that allows linking elements (e.g. `LinkImage`).\n\t\t// Currently the selection reads attributes from text nodes only. See #7429 and #7465.\n\t\tif ( isLinkableElement( selectedElement, model.schema ) ) {\n\t\t\tthis.isEnabled = model.schema.checkAttribute( selectedElement, 'linkHref' );\n\t\t} else {\n\t\t\tthis.isEnabled = model.schema.checkAttributeInSelection( selection, 'linkHref' );\n\t\t}\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * When the selection is collapsed, it removes the `linkHref` attribute from each node with the same `linkHref` attribute value.\n\t * When the selection is non-collapsed, it removes the `linkHref` attribute from each node in selected ranges.\n\t *\n\t * # Decorators\n\t *\n\t * If {@link module:link/linkconfig~LinkConfig#decorators `config.link.decorators`} is specified,\n\t * all configured decorators are removed together with the `linkHref` attribute.\n\t *\n\t * @fires execute\n\t */\n\tpublic override execute(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst linkCommand: LinkCommand | undefined = editor.commands.get( 'link' );\n\n\t\tmodel.change( writer => {\n\t\t\t// Get ranges to unlink.\n\t\t\tconst rangesToUnlink = selection.isCollapsed ?\n\t\t\t\t[ findAttributeRange(\n\t\t\t\t\tselection.getFirstPosition()!,\n\t\t\t\t\t'linkHref',\n\t\t\t\t\tselection.getAttribute( 'linkHref' ),\n\t\t\t\t\tmodel\n\t\t\t\t) ] :\n\t\t\t\tmodel.schema.getValidRanges( selection.getRanges(), 'linkHref' );\n\n\t\t\t// Remove `linkHref` attribute from specified ranges.\n\t\t\tfor ( const range of rangesToUnlink ) {\n\t\t\t\twriter.removeAttribute( 'linkHref', range );\n\t\t\t\t// If there are registered custom attributes, then remove them during unlink.\n\t\t\t\tif ( linkCommand ) {\n\t\t\t\t\tfor ( const manualDecorator of linkCommand.manualDecorators ) {\n\t\t\t\t\t\twriter.removeAttribute( manualDecorator.id, range );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/utils/manualdecorator\n */\n\nimport { ObservableMixin, type ArrayOrItem } from 'ckeditor5/src/utils';\nimport type { MatcherPattern, MatcherObjectPattern } from 'ckeditor5/src/engine';\nimport type { NormalizedLinkDecoratorManualDefinition } from '../utils';\n\n/**\n * Helper class that stores manual decorators with observable {@link module:link/utils/manualdecorator~ManualDecorator#value}\n * to support integration with the UI state. An instance of this class is a model with the state of individual manual decorators.\n * These decorators are kept as collections in {@link module:link/linkcommand~LinkCommand#manualDecorators}.\n */\nexport default class ManualDecorator extends ObservableMixin() {\n\t/**\n\t * An ID of a manual decorator which is the name of the attribute in the model, for example: 'linkManualDecorator0'.\n\t */\n\tpublic id: string;\n\n\t/**\n\t * The value of the current manual decorator. It reflects its state from the UI.\n\t *\n\t * @observable\n\t */\n\tdeclare public value: boolean | undefined;\n\n\t/**\n\t * The default value of manual decorator.\n\t */\n\tpublic defaultValue?: boolean;\n\n\t/**\n\t * The label used in the user interface to toggle the manual decorator.\n\t */\n\tpublic label: string;\n\n\t/**\n\t * A set of attributes added to downcasted data when the decorator is activated for a specific link.\n\t * Attributes should be added in a form of attributes defined in {@link module:engine/view/elementdefinition~ElementDefinition}.\n\t */\n\tpublic attributes?: Record<string, string>;\n\n\t/**\n\t * A set of classes added to downcasted data when the decorator is activated for a specific link.\n\t * Classes should be added in a form of classes defined in {@link module:engine/view/elementdefinition~ElementDefinition}.\n\t */\n\tpublic classes?: ArrayOrItem<string>;\n\n\t/**\n\t * A set of styles added to downcasted data when the decorator is activated for a specific link.\n\t * Styles should be added in a form of styles defined in {@link module:engine/view/elementdefinition~ElementDefinition}.\n\t */\n\tpublic styles?: Record<string, string>;\n\n\t/**\n\t * Creates a new instance of {@link module:link/utils/manualdecorator~ManualDecorator}.\n\t *\n\t * @param config.id The name of the attribute used in the model that represents a given manual decorator.\n\t * For example: `'linkIsExternal'`.\n\t * @param config.label The label used in the user interface to toggle the manual decorator.\n\t * @param config.attributes A set of attributes added to output data when the decorator is active for a specific link.\n\t * Attributes should keep the format of attributes defined in {@link module:engine/view/elementdefinition~ElementDefinition}.\n\t * @param [config.defaultValue] Controls whether the decorator is \"on\" by default.\n\t */\n\tconstructor( { id, label, attributes, classes, styles, defaultValue }: NormalizedLinkDecoratorManualDefinition ) {\n\t\tsuper();\n\n\t\tthis.id = id;\n\t\tthis.set( 'value', undefined );\n\t\tthis.defaultValue = defaultValue;\n\t\tthis.label = label;\n\t\tthis.attributes = attributes;\n\t\tthis.classes = classes;\n\t\tthis.styles = styles;\n\t}\n\n\t/**\n\t * Returns {@link module:engine/view/matcher~MatcherPattern} with decorator attributes.\n\t *\n\t * @internal\n\t */\n\tpublic _createPattern(): MatcherObjectPattern {\n\t\treturn {\n\t\t\tattributes: this.attributes,\n\t\t\tclasses: this.classes,\n\t\t\tstyles: this.styles\n\t\t};\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./link.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/linkediting\n */\n\nimport {\n\tPlugin,\n\ttype Editor\n} from 'ckeditor5/src/core';\nimport {\n\tMouseObserver,\n\ttype Model,\n\ttype Schema,\n\ttype Writer,\n\ttype ViewElement,\n\ttype ModelDeleteContentEvent,\n\ttype ModelInsertContentEvent,\n\ttype ViewDocumentKeyDownEvent,\n\ttype ViewDocumentMouseDownEvent,\n\ttype ViewDocumentClickEvent,\n\ttype ViewDocumentSelectionChangeEvent\n} from 'ckeditor5/src/engine';\nimport {\n\tInput,\n\tTwoStepCaretMovement,\n\tinlineHighlight,\n\tfindAttributeRange,\n\ttype ViewDocumentDeleteEvent\n} from 'ckeditor5/src/typing';\nimport {\n\tClipboardPipeline,\n\ttype ClipboardContentInsertionEvent\n} from 'ckeditor5/src/clipboard';\nimport { keyCodes, env } from 'ckeditor5/src/utils';\n\nimport LinkCommand from './linkcommand';\nimport UnlinkCommand from './unlinkcommand';\nimport ManualDecorator from './utils/manualdecorator';\nimport {\n\tcreateLinkElement,\n\tensureSafeUrl,\n\tgetLocalizedDecorators,\n\tnormalizeDecorators,\n\topenLink,\n\taddLinkProtocolIfApplicable,\n\ttype NormalizedLinkDecoratorAutomaticDefinition,\n\ttype NormalizedLinkDecoratorManualDefinition\n} from './utils';\n\nimport '../theme/link.css';\n\nconst HIGHLIGHT_CLASS = 'ck-link_selected';\nconst DECORATOR_AUTOMATIC = 'automatic';\nconst DECORATOR_MANUAL = 'manual';\nconst EXTERNAL_LINKS_REGEXP = /^(https?:)?\\/\\//;\n\n/**\n * The link engine feature.\n *\n * It introduces the `linkHref=\"url\"` attribute in the model which renders to the view as a `<a href=\"url\">` element\n * as well as `'link'` and `'unlink'` commands.\n */\nexport default class LinkEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'LinkEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\t// Clipboard is required for handling cut and paste events while typing over the link.\n\t\treturn [ TwoStepCaretMovement, Input, ClipboardPipeline ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\teditor.config.define( 'link', {\n\t\t\taddTargetToExternalLinks: false\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Allow link attribute on all inline nodes.\n\t\teditor.model.schema.extend( '$text', { allowAttributes: 'linkHref' } );\n\n\t\teditor.conversion.for( 'dataDowncast' )\n\t\t\t.attributeToElement( { model: 'linkHref', view: createLinkElement } );\n\n\t\teditor.conversion.for( 'editingDowncast' )\n\t\t\t.attributeToElement( { model: 'linkHref', view: ( href, conversionApi ) => {\n\t\t\t\treturn createLinkElement( ensureSafeUrl( href ), conversionApi );\n\t\t\t} } );\n\n\t\teditor.conversion.for( 'upcast' )\n\t\t\t.elementToAttribute( {\n\t\t\t\tview: {\n\t\t\t\t\tname: 'a',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\thref: true\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\tmodel: {\n\t\t\t\t\tkey: 'linkHref',\n\t\t\t\t\tvalue: ( viewElement: ViewElement ) => viewElement.getAttribute( 'href' )\n\t\t\t\t}\n\t\t\t} );\n\n\t\t// Create linking commands.\n\t\teditor.commands.add( 'link', new LinkCommand( editor ) );\n\t\teditor.commands.add( 'unlink', new UnlinkCommand( editor ) );\n\n\t\tconst linkDecorators = getLocalizedDecorators( editor.t, normalizeDecorators( editor.config.get( 'link.decorators' ) ) );\n\n\t\tthis._enableAutomaticDecorators( linkDecorators\n\t\t\t.filter( ( item ): item is NormalizedLinkDecoratorAutomaticDefinition => item.mode === DECORATOR_AUTOMATIC ) );\n\t\tthis._enableManualDecorators( linkDecorators\n\t\t\t.filter( ( item ): item is NormalizedLinkDecoratorManualDefinition => item.mode === DECORATOR_MANUAL ) );\n\n\t\t// Enable two-step caret movement for `linkHref` attribute.\n\t\tconst twoStepCaretMovementPlugin = editor.plugins.get( TwoStepCaretMovement );\n\t\ttwoStepCaretMovementPlugin.registerAttribute( 'linkHref' );\n\n\t\t// Setup highlight over selected link.\n\t\tinlineHighlight( editor, 'linkHref', 'a', HIGHLIGHT_CLASS );\n\n\t\t// Handle link following by CTRL+click or ALT+ENTER\n\t\tthis._enableLinkOpen();\n\n\t\t// Change the attributes of the selection in certain situations after the link was inserted into the document.\n\t\tthis._enableInsertContentSelectionAttributesFixer();\n\n\t\t// Handle a click at the beginning/end of a link element.\n\t\tthis._enableClickingAfterLink();\n\n\t\t// Handle typing over the link.\n\t\tthis._enableTypingOverLink();\n\n\t\t// Handle removing the content after the link element.\n\t\tthis._handleDeleteContentAfterLink();\n\n\t\t// Handle adding default protocol to pasted links.\n\t\tthis._enableClipboardIntegration();\n\t}\n\n\t/**\n\t * Processes an array of configured {@link module:link/linkconfig~LinkDecoratorAutomaticDefinition automatic decorators}\n\t * and registers a {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher downcast dispatcher}\n\t * for each one of them. Downcast dispatchers are obtained using the\n\t * {@link module:link/utils/automaticdecorators~AutomaticDecorators#getDispatcher} method.\n\t *\n\t * **Note**: This method also activates the automatic external link decorator if enabled with\n\t * {@link module:link/linkconfig~LinkConfig#addTargetToExternalLinks `config.link.addTargetToExternalLinks`}.\n\t */\n\tprivate _enableAutomaticDecorators( automaticDecoratorDefinitions: Array<NormalizedLinkDecoratorAutomaticDefinition> ): void {\n\t\tconst editor = this.editor;\n\t\t// Store automatic decorators in the command instance as we do the same with manual decorators.\n\t\t// Thanks to that, `LinkImageEditing` plugin can re-use the same definitions.\n\t\tconst command: LinkCommand = editor.commands.get( 'link' )!;\n\t\tconst automaticDecorators = command.automaticDecorators;\n\n\t\t// Adds a default decorator for external links.\n\t\tif ( editor.config.get( 'link.addTargetToExternalLinks' ) ) {\n\t\t\tautomaticDecorators.add( {\n\t\t\t\tid: 'linkIsExternal',\n\t\t\t\tmode: DECORATOR_AUTOMATIC,\n\t\t\t\tcallback: url => !!url && EXTERNAL_LINKS_REGEXP.test( url ),\n\t\t\t\tattributes: {\n\t\t\t\t\ttarget: '_blank',\n\t\t\t\t\trel: 'noopener noreferrer'\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\n\t\tautomaticDecorators.add( automaticDecoratorDefinitions );\n\n\t\tif ( automaticDecorators.length ) {\n\t\t\teditor.conversion.for( 'downcast' ).add( automaticDecorators.getDispatcher() );\n\t\t}\n\t}\n\n\t/**\n\t * Processes an array of configured {@link module:link/linkconfig~LinkDecoratorManualDefinition manual decorators},\n\t * transforms them into {@link module:link/utils/manualdecorator~ManualDecorator} instances and stores them in the\n\t * {@link module:link/linkcommand~LinkCommand#manualDecorators} collection (a model for manual decorators state).\n\t *\n\t * Also registers an {@link module:engine/conversion/downcasthelpers~DowncastHelpers#attributeToElement attribute-to-element}\n\t * converter for each manual decorator and extends the {@link module:engine/model/schema~Schema model's schema}\n\t * with adequate model attributes.\n\t */\n\tprivate _enableManualDecorators( manualDecoratorDefinitions: Array<NormalizedLinkDecoratorManualDefinition> ): void {\n\t\tif ( !manualDecoratorDefinitions.length ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst editor = this.editor;\n\t\tconst command: LinkCommand = editor.commands.get( 'link' )!;\n\t\tconst manualDecorators = command.manualDecorators;\n\n\t\tmanualDecoratorDefinitions.forEach( decoratorDefinition => {\n\t\t\teditor.model.schema.extend( '$text', { allowAttributes: decoratorDefinition.id } );\n\n\t\t\t// Keeps reference to manual decorator to decode its name to attributes during downcast.\n\t\t\tconst decorator = new ManualDecorator( decoratorDefinition );\n\n\t\t\tmanualDecorators.add( decorator );\n\n\t\t\teditor.conversion.for( 'downcast' ).attributeToElement( {\n\t\t\t\tmodel: decorator.id,\n\t\t\t\tview: ( manualDecoratorValue, { writer, schema }, { item } ) => {\n\t\t\t\t\t// Manual decorators for block links are handled e.g. in LinkImageEditing.\n\t\t\t\t\tif ( !( item.is( 'selection' ) || schema.isInline( item ) ) ) {\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\n\t\t\t\t\tif ( manualDecoratorValue ) {\n\t\t\t\t\t\tconst element = writer.createAttributeElement( 'a', decorator.attributes, { priority: 5 } );\n\n\t\t\t\t\t\tif ( decorator.classes ) {\n\t\t\t\t\t\t\twriter.addClass( decorator.classes, element );\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tfor ( const key in decorator.styles ) {\n\t\t\t\t\t\t\twriter.setStyle( key, decorator.styles[ key ], element );\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\twriter.setCustomProperty( 'link', true, element );\n\n\t\t\t\t\t\treturn element;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\teditor.conversion.for( 'upcast' ).elementToAttribute( {\n\t\t\t\tview: {\n\t\t\t\t\tname: 'a',\n\t\t\t\t\t...decorator._createPattern()\n\t\t\t\t},\n\t\t\t\tmodel: {\n\t\t\t\t\tkey: decorator.id\n\t\t\t\t}\n\t\t\t} );\n\t\t} );\n\t}\n\n\t/**\n\t * Attaches handlers for {@link module:engine/view/document~Document#event:enter} and\n\t * {@link module:engine/view/document~Document#event:click} to enable link following.\n\t */\n\tprivate _enableLinkOpen(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\tthis.listenTo<ViewDocumentClickEvent>( viewDocument, 'click', ( evt, data ) => {\n\t\t\tconst shouldOpen = env.isMac ? data.domEvent.metaKey : data.domEvent.ctrlKey;\n\n\t\t\tif ( !shouldOpen ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tlet clickedElement: Element | null = data.domTarget;\n\n\t\t\tif ( clickedElement.tagName.toLowerCase() != 'a' ) {\n\t\t\t\tclickedElement = clickedElement.closest( 'a' );\n\t\t\t}\n\n\t\t\tif ( !clickedElement ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst url = clickedElement.getAttribute( 'href' );\n\n\t\t\tif ( !url ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tevt.stop();\n\t\t\tdata.preventDefault();\n\n\t\t\topenLink( url );\n\t\t}, { context: '$capture' } );\n\n\t\t// Open link on Alt+Enter.\n\t\tthis.listenTo<ViewDocumentKeyDownEvent>( viewDocument, 'keydown', ( evt, data ) => {\n\t\t\tconst linkCommand: LinkCommand = editor.commands.get( 'link' )!;\n\t\t\tconst url = linkCommand!.value;\n\t\t\tconst shouldOpen = !!url && data.keyCode === keyCodes.enter && data.altKey;\n\n\t\t\tif ( !shouldOpen ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tevt.stop();\n\n\t\t\topenLink( url );\n\t\t} );\n\t}\n\n\t/**\n\t * Starts listening to {@link module:engine/model/model~Model#event:insertContent} and corrects the model\n\t * selection attributes if the selection is at the end of a link after inserting the content.\n\t *\n\t * The purpose of this action is to improve the overall UX because the user is no longer \"trapped\" by the\n\t * `linkHref` attribute of the selection and they can type a \"clean\" (`linkHref`less) text right away.\n\t *\n\t * See https://github.com/ckeditor/ckeditor5/issues/6053.\n\t */\n\tprivate _enableInsertContentSelectionAttributesFixer(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst selection = model.document.selection;\n\n\t\tthis.listenTo<ModelInsertContentEvent>( model, 'insertContent', () => {\n\t\t\tconst nodeBefore = selection.anchor!.nodeBefore;\n\t\t\tconst nodeAfter = selection.anchor!.nodeAfter;\n\n\t\t\t// NOTE: ↰ and ↱ represent the gravity of the selection.\n\n\t\t\t// The only truly valid case is:\n\t\t\t//\n\t\t\t//\t\t ↰\n\t\t\t//\t\t...<$text linkHref=\"foo\">INSERTED[]</$text>\n\t\t\t//\n\t\t\t// If the selection is not \"trapped\" by the `linkHref` attribute after inserting, there's nothing\n\t\t\t// to fix there.\n\t\t\tif ( !selection.hasAttribute( 'linkHref' ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Filter out the following case where a link with the same href (e.g. <a href=\"foo\">INSERTED</a>) is inserted\n\t\t\t// in the middle of an existing link:\n\t\t\t//\n\t\t\t// Before insertion:\n\t\t\t//\t\t ↰\n\t\t\t//\t\t<$text linkHref=\"foo\">l[]ink</$text>\n\t\t\t//\n\t\t\t// Expected after insertion:\n\t\t\t//\t\t ↰\n\t\t\t//\t\t<$text linkHref=\"foo\">lINSERTED[]ink</$text>\n\t\t\t//\n\t\t\tif ( !nodeBefore ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Filter out the following case where the selection has the \"linkHref\" attribute because the\n\t\t\t// gravity is overridden and some text with another attribute (e.g. <b>INSERTED</b>) is inserted:\n\t\t\t//\n\t\t\t// Before insertion:\n\t\t\t//\n\t\t\t//\t\t ↱\n\t\t\t//\t\t<$text linkHref=\"foo\">[]link</$text>\n\t\t\t//\n\t\t\t// Expected after insertion:\n\t\t\t//\n\t\t\t//\t\t ↱\n\t\t\t//\t\t<$text bold=\"true\">INSERTED</$text><$text linkHref=\"foo\">[]link</$text>\n\t\t\t//\n\t\t\tif ( !nodeBefore.hasAttribute( 'linkHref' ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Filter out the following case where a link is a inserted in the middle (or before) another link\n\t\t\t// (different URLs, so they will not merge). In this (let's say weird) case, we can leave the selection\n\t\t\t// attributes as they are because the user will end up writing in one link or another anyway.\n\t\t\t//\n\t\t\t// Before insertion:\n\t\t\t//\n\t\t\t//\t\t ↰\n\t\t\t//\t\t<$text linkHref=\"foo\">l[]ink</$text>\n\t\t\t//\n\t\t\t// Expected after insertion:\n\t\t\t//\n\t\t\t//\t\t ↰\n\t\t\t//\t\t<$text linkHref=\"foo\">l</$text><$text linkHref=\"bar\">INSERTED[]</$text><$text linkHref=\"foo\">ink</$text>\n\t\t\t//\n\t\t\tif ( nodeAfter && nodeAfter.hasAttribute( 'linkHref' ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tmodel.change( writer => {\n\t\t\t\tremoveLinkAttributesFromSelection( writer, getLinkAttributesAllowedOnText( model.schema ) );\n\t\t\t} );\n\t\t}, { priority: 'low' } );\n\t}\n\n\t/**\n\t * Starts listening to {@link module:engine/view/document~Document#event:mousedown} and\n\t * {@link module:engine/view/document~Document#event:selectionChange} and puts the selection before/after a link node\n\t * if clicked at the beginning/ending of the link.\n\t *\n\t * The purpose of this action is to allow typing around the link node directly after a click.\n\t *\n\t * See https://github.com/ckeditor/ckeditor5/issues/1016.\n\t */\n\tprivate _enableClickingAfterLink(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\n\t\teditor.editing.view.addObserver( MouseObserver );\n\n\t\tlet clicked = false;\n\n\t\t// Detect the click.\n\t\tthis.listenTo<ViewDocumentMouseDownEvent>( editor.editing.view.document, 'mousedown', () => {\n\t\t\tclicked = true;\n\t\t} );\n\n\t\t// When the selection has changed...\n\t\tthis.listenTo<ViewDocumentSelectionChangeEvent>( editor.editing.view.document, 'selectionChange', () => {\n\t\t\tif ( !clicked ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// ...and it was caused by the click...\n\t\t\tclicked = false;\n\n\t\t\tconst selection = model.document.selection;\n\n\t\t\t// ...and no text is selected...\n\t\t\tif ( !selection.isCollapsed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// ...and clicked text is the link...\n\t\t\tif ( !selection.hasAttribute( 'linkHref' ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst position = selection.getFirstPosition()!;\n\t\t\tconst linkRange = findAttributeRange( position, 'linkHref', selection.getAttribute( 'linkHref' ), model );\n\n\t\t\t// ...check whether clicked start/end boundary of the link.\n\t\t\t// If so, remove the `linkHref` attribute.\n\t\t\tif ( position.isTouching( linkRange.start ) || position.isTouching( linkRange.end ) ) {\n\t\t\t\tmodel.change( writer => {\n\t\t\t\t\tremoveLinkAttributesFromSelection( writer, getLinkAttributesAllowedOnText( model.schema ) );\n\t\t\t\t} );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Starts listening to {@link module:engine/model/model~Model#deleteContent} and {@link module:engine/model/model~Model#insertContent}\n\t * and checks whether typing over the link. If so, attributes of removed text are preserved and applied to the inserted text.\n\t *\n\t * The purpose of this action is to allow modifying a text without loosing the `linkHref` attribute (and other).\n\t *\n\t * See https://github.com/ckeditor/ckeditor5/issues/4762.\n\t */\n\tprivate _enableTypingOverLink(): void {\n\t\tconst editor = this.editor;\n\t\tconst view = editor.editing.view;\n\n\t\t// Selection attributes when started typing over the link.\n\t\tlet selectionAttributes: IterableIterator<[ string, unknown ]> | null = null;\n\n\t\t// Whether pressed `Backspace` or `Delete`. If so, attributes should not be preserved.\n\t\tlet deletedContent = false;\n\n\t\t// Detect pressing `Backspace` / `Delete`.\n\t\tthis.listenTo<ViewDocumentDeleteEvent>( view.document, 'delete', () => {\n\t\t\tdeletedContent = true;\n\t\t}, { priority: 'high' } );\n\n\t\t// Listening to `model#deleteContent` allows detecting whether selected content was a link.\n\t\t// If so, before removing the element, we will copy its attributes.\n\t\tthis.listenTo<ModelDeleteContentEvent>( editor.model, 'deleteContent', () => {\n\t\t\tconst selection = editor.model.document.selection;\n\n\t\t\t// Copy attributes only if anything is selected.\n\t\t\tif ( selection.isCollapsed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// When the content was deleted, do not preserve attributes.\n\t\t\tif ( deletedContent ) {\n\t\t\t\tdeletedContent = false;\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Enabled only when typing.\n\t\t\tif ( !isTyping( editor ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( shouldCopyAttributes( editor.model ) ) {\n\t\t\t\tselectionAttributes = selection.getAttributes();\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\n\t\t// Listening to `model#insertContent` allows detecting the content insertion.\n\t\t// We want to apply attributes that were removed while typing over the link.\n\t\tthis.listenTo( editor.model, 'insertContent', ( evt, [ element ] ) => {\n\t\t\tdeletedContent = false;\n\n\t\t\t// Enabled only when typing.\n\t\t\tif ( !isTyping( editor ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( !selectionAttributes ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\teditor.model.change( writer => {\n\t\t\t\tfor ( const [ attribute, value ] of selectionAttributes! ) {\n\t\t\t\t\twriter.setAttribute( attribute, value, element );\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tselectionAttributes = null;\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * Starts listening to {@link module:engine/model/model~Model#deleteContent} and checks whether\n\t * removing a content right after the \"linkHref\" attribute.\n\t *\n\t * If so, the selection should not preserve the `linkHref` attribute. However, if\n\t * the {@link module:typing/twostepcaretmovement~TwoStepCaretMovement} plugin is active and\n\t * the selection has the \"linkHref\" attribute due to overriden gravity (at the end), the `linkHref` attribute should stay untouched.\n\t *\n\t * The purpose of this action is to allow removing the link text and keep the selection outside the link.\n\t *\n\t * See https://github.com/ckeditor/ckeditor5/issues/7521.\n\t */\n\tprivate _handleDeleteContentAfterLink(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst view = editor.editing.view;\n\n\t\t// A flag whether attributes `linkHref` attribute should be preserved.\n\t\tlet shouldPreserveAttributes = false;\n\n\t\t// A flag whether the `Backspace` key was pressed.\n\t\tlet hasBackspacePressed = false;\n\n\t\t// Detect pressing `Backspace`.\n\t\tthis.listenTo<ViewDocumentDeleteEvent>( view.document, 'delete', ( evt, data ) => {\n\t\t\thasBackspacePressed = data.direction === 'backward';\n\t\t}, { priority: 'high' } );\n\n\t\t// Before removing the content, check whether the selection is inside a link or at the end of link but with 2-SCM enabled.\n\t\t// If so, we want to preserve link attributes.\n\t\tthis.listenTo<ModelDeleteContentEvent>( model, 'deleteContent', () => {\n\t\t\t// Reset the state.\n\t\t\tshouldPreserveAttributes = false;\n\n\t\t\tconst position = selection.getFirstPosition()!;\n\t\t\tconst linkHref = selection.getAttribute( 'linkHref' );\n\n\t\t\tif ( !linkHref ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst linkRange = findAttributeRange( position, 'linkHref', linkHref, model );\n\n\t\t\t// Preserve `linkHref` attribute if the selection is in the middle of the link or\n\t\t\t// the selection is at the end of the link and 2-SCM is activated.\n\t\t\tshouldPreserveAttributes = linkRange.containsPosition( position ) || linkRange.end.isEqual( position );\n\t\t}, { priority: 'high' } );\n\n\t\t// After removing the content, check whether the current selection should preserve the `linkHref` attribute.\n\t\tthis.listenTo<ModelDeleteContentEvent>( model, 'deleteContent', () => {\n\t\t\t// If didn't press `Backspace`.\n\t\t\tif ( !hasBackspacePressed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\thasBackspacePressed = false;\n\n\t\t\t// Disable the mechanism if inside a link (`<$text url=\"foo\">F[]oo</$text>` or <$text url=\"foo\">Foo[]</$text>`).\n\t\t\tif ( shouldPreserveAttributes ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Use `model.enqueueChange()` in order to execute the callback at the end of the changes process.\n\t\t\teditor.model.enqueueChange( writer => {\n\t\t\t\tremoveLinkAttributesFromSelection( writer, getLinkAttributesAllowedOnText( model.schema ) );\n\t\t\t} );\n\t\t}, { priority: 'low' } );\n\t}\n\n\t/**\n\t * Enables URL fixing on pasting.\n\t */\n\tprivate _enableClipboardIntegration(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst defaultProtocol = this.editor.config.get( 'link.defaultProtocol' );\n\n\t\tif ( !defaultProtocol ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis.listenTo<ClipboardContentInsertionEvent>( editor.plugins.get( 'ClipboardPipeline' ), 'contentInsertion', ( evt, data ) => {\n\t\t\tmodel.change( writer => {\n\t\t\t\tconst range = writer.createRangeIn( data.content );\n\n\t\t\t\tfor ( const item of range.getItems() ) {\n\t\t\t\t\tif ( item.hasAttribute( 'linkHref' ) ) {\n\t\t\t\t\t\tconst newLink = addLinkProtocolIfApplicable( item.getAttribute( 'linkHref' ) as string, defaultProtocol );\n\n\t\t\t\t\t\twriter.setAttribute( 'linkHref', newLink, item );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t} );\n\t\t} );\n\t}\n}\n\n/**\n * Make the selection free of link-related model attributes.\n * All link-related model attributes start with \"link\". That includes not only \"linkHref\"\n * but also all decorator attributes (they have dynamic names), or even custom plugins.\n */\nfunction removeLinkAttributesFromSelection( writer: Writer, linkAttributes: Array<string> ): void {\n\twriter.removeSelectionAttribute( 'linkHref' );\n\n\tfor ( const attribute of linkAttributes ) {\n\t\twriter.removeSelectionAttribute( attribute );\n\t}\n}\n\n/**\n * Checks whether selection's attributes should be copied to the new inserted text.\n */\nfunction shouldCopyAttributes( model: Model ): boolean {\n\tconst selection = model.document.selection;\n\tconst firstPosition = selection.getFirstPosition()!;\n\tconst lastPosition = selection.getLastPosition()!;\n\tconst nodeAtFirstPosition = firstPosition.nodeAfter;\n\n\t// The text link node does not exist...\n\tif ( !nodeAtFirstPosition ) {\n\t\treturn false;\n\t}\n\n\t// ...or it isn't the text node...\n\tif ( !nodeAtFirstPosition.is( '$text' ) ) {\n\t\treturn false;\n\t}\n\n\t// ...or isn't the link.\n\tif ( !nodeAtFirstPosition.hasAttribute( 'linkHref' ) ) {\n\t\treturn false;\n\t}\n\n\t// `textNode` = the position is inside the link element.\n\t// `nodeBefore` = the position is at the end of the link element.\n\tconst nodeAtLastPosition = lastPosition.textNode || lastPosition.nodeBefore;\n\n\t// If both references the same node selection contains a single text node.\n\tif ( nodeAtFirstPosition === nodeAtLastPosition ) {\n\t\treturn true;\n\t}\n\n\t// If nodes are not equal, maybe the link nodes has defined additional attributes inside.\n\t// First, we need to find the entire link range.\n\tconst linkRange = findAttributeRange( firstPosition, 'linkHref', nodeAtFirstPosition.getAttribute( 'linkHref' ), model );\n\n\t// Then we can check whether selected range is inside the found link range. If so, attributes should be preserved.\n\treturn linkRange.containsRange( model.createRange( firstPosition, lastPosition ), true );\n}\n\n/**\n * Checks whether provided changes were caused by typing.\n */\nfunction isTyping( editor: Editor ): boolean {\n\tconst currentBatch = editor.model.change( writer => writer.batch );\n\n\treturn currentBatch.isTyping;\n}\n\n/**\n * Returns an array containing names of the attributes allowed on `$text` that describes the link item.\n */\nfunction getLinkAttributesAllowedOnText( schema: Schema ): Array<string> {\n\tconst textAttributes = schema.getDefinition( '$text' )!.allowAttributes;\n\n\treturn textAttributes.filter( attribute => attribute.startsWith( 'link' ) );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module typing/utils/inlinehighlight\n */\n\nimport findAttributeRange from './findattributerange';\nimport type { Editor } from '@ckeditor/ckeditor5-core';\nimport type { ViewElement } from '@ckeditor/ckeditor5-engine';\n\n/**\n * Adds a visual highlight style to an attribute element in which the selection is anchored.\n * Together with two-step caret movement, they indicate that the user is typing inside the element.\n *\n * Highlight is turned on by adding the given class to the attribute element in the view:\n *\n * * The class is removed before the conversion has started, as callbacks added with the `'highest'` priority\n * to {@link module:engine/conversion/downcastdispatcher~DowncastDispatcher} events.\n * * The class is added in the view post fixer, after other changes in the model tree were converted to the view.\n *\n * This way, adding and removing the highlight does not interfere with conversion.\n *\n * Usage:\n *\n * ```ts\n * import inlineHighlight from '@ckeditor/ckeditor5-typing/src/utils/inlinehighlight';\n *\n * // Make `ck-link_selected` class be applied on an `a` element\n * // whenever the corresponding `linkHref` attribute element is selected.\n * inlineHighlight( editor, 'linkHref', 'a', 'ck-link_selected' );\n * ```\n *\n * @param editor The editor instance.\n * @param attributeName The attribute name to check.\n * @param tagName The tagName of a view item.\n * @param className The class name to apply in the view.\n */\nexport default function inlineHighlight(\n\teditor: Editor,\n\tattributeName: string,\n\ttagName: string,\n\tclassName: string\n): void {\n\tconst view = editor.editing.view;\n\tconst highlightedElements = new Set<ViewElement>();\n\n\t// Adding the class.\n\tview.document.registerPostFixer( writer => {\n\t\tconst selection = editor.model.document.selection;\n\t\tlet changed = false;\n\n\t\tif ( selection.hasAttribute( attributeName ) ) {\n\t\t\tconst modelRange = findAttributeRange(\n\t\t\t\tselection.getFirstPosition()!,\n\t\t\t\tattributeName,\n\t\t\t\tselection.getAttribute( attributeName ),\n\t\t\t\teditor.model\n\t\t\t);\n\t\t\tconst viewRange = editor.editing.mapper.toViewRange( modelRange );\n\n\t\t\t// There might be multiple view elements in the `viewRange`, for example, when the `a` element is\n\t\t\t// broken by a UIElement.\n\t\t\tfor ( const item of viewRange.getItems() ) {\n\t\t\t\tif ( item.is( 'element', tagName ) && !item.hasClass( className ) ) {\n\t\t\t\t\twriter.addClass( className, item );\n\t\t\t\t\thighlightedElements.add( item );\n\t\t\t\t\tchanged = true;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn changed;\n\t} );\n\n\t// Removing the class.\n\teditor.conversion.for( 'editingDowncast' ).add( dispatcher => {\n\t\t// Make sure the highlight is removed on every possible event, before conversion is started.\n\t\tdispatcher.on( 'insert', removeHighlight, { priority: 'highest' } );\n\t\tdispatcher.on( 'remove', removeHighlight, { priority: 'highest' } );\n\t\tdispatcher.on( 'attribute', removeHighlight, { priority: 'highest' } );\n\t\tdispatcher.on( 'selection', removeHighlight, { priority: 'highest' } );\n\n\t\tfunction removeHighlight() {\n\t\t\tview.change( writer => {\n\t\t\t\tfor ( const item of highlightedElements.values() ) {\n\t\t\t\t\twriter.removeClass( className, item );\n\t\t\t\t\thighlightedElements.delete( item );\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\t} );\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./linkform.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/ui/linkformview\n */\n\nimport {\n\tButtonView,\n\tFocusCycler,\n\tLabeledFieldView,\n\tSwitchButtonView,\n\tView,\n\tViewCollection,\n\tcreateLabeledInputText,\n\tsubmitHandler,\n\ttype InputTextView\n} from 'ckeditor5/src/ui';\nimport {\n\tFocusTracker,\n\tKeystrokeHandler,\n\ttype Collection,\n\ttype Locale\n} from 'ckeditor5/src/utils';\nimport { icons } from 'ckeditor5/src/core';\n\nimport type LinkCommand from '../linkcommand';\nimport type ManualDecorator from '../utils/manualdecorator';\n\n// See: #8833.\n// eslint-disable-next-line ckeditor5-rules/ckeditor-imports\nimport '@ckeditor/ckeditor5-ui/theme/components/responsive-form/responsiveform.css';\nimport '../../theme/linkform.css';\n\n/**\n * The link form view controller class.\n *\n * See {@link module:link/ui/linkformview~LinkFormView}.\n */\nexport default class LinkFormView extends View {\n\t/**\n\t * Tracks information about DOM focus in the form.\n\t */\n\tpublic readonly focusTracker = new FocusTracker();\n\n\t/**\n\t * An instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}.\n\t */\n\tpublic readonly keystrokes = new KeystrokeHandler();\n\n\t/**\n\t * The URL input view.\n\t */\n\tpublic urlInputView: LabeledFieldView<InputTextView>;\n\n\t/**\n\t * The Save button view.\n\t */\n\tpublic saveButtonView: ButtonView;\n\n\t/**\n\t * The Cancel button view.\n\t */\n\tpublic cancelButtonView: ButtonView;\n\n\t/**\n\t * A collection of {@link module:ui/button/switchbuttonview~SwitchButtonView},\n\t * which corresponds to {@link module:link/linkcommand~LinkCommand#manualDecorators manual decorators}\n\t * configured in the editor.\n\t */\n\tprivate readonly _manualDecoratorSwitches: ViewCollection;\n\n\t/**\n\t * A collection of child views in the form.\n\t */\n\tpublic readonly children: ViewCollection;\n\n\t/**\n\t * A collection of views that can be focused in the form.\n\t */\n\tprivate readonly _focusables = new ViewCollection();\n\n\t/**\n\t * Helps cycling over {@link #_focusables} in the form.\n\t */\n\tprivate readonly _focusCycler: FocusCycler;\n\n\t/**\n\t * Creates an instance of the {@link module:link/ui/linkformview~LinkFormView} class.\n\t *\n\t * Also see {@link #render}.\n\t *\n\t * @param locale The localization services instance.\n\t * @param linkCommand Reference to {@link module:link/linkcommand~LinkCommand}.\n\t */\n\tconstructor( locale: Locale, linkCommand: LinkCommand ) {\n\t\tsuper( locale );\n\n\t\tconst t = locale.t;\n\n\t\tthis.urlInputView = this._createUrlInput();\n\t\tthis.saveButtonView = this._createButton( t( 'Save' ), icons.check, 'ck-button-save' );\n\t\tthis.saveButtonView.type = 'submit';\n\t\tthis.cancelButtonView = this._createButton( t( 'Cancel' ), icons.cancel, 'ck-button-cancel', 'cancel' );\n\t\tthis._manualDecoratorSwitches = this._createManualDecoratorSwitches( linkCommand );\n\t\tthis.children = this._createFormChildren( linkCommand.manualDecorators );\n\n\t\tthis._focusCycler = new FocusCycler( {\n\t\t\tfocusables: this._focusables,\n\t\t\tfocusTracker: this.focusTracker,\n\t\t\tkeystrokeHandler: this.keystrokes,\n\t\t\tactions: {\n\t\t\t\t// Navigate form fields backwards using the Shift + Tab keystroke.\n\t\t\t\tfocusPrevious: 'shift + tab',\n\n\t\t\t\t// Navigate form fields forwards using the Tab key.\n\t\t\t\tfocusNext: 'tab'\n\t\t\t}\n\t\t} );\n\n\t\tconst classList = [ 'ck', 'ck-link-form', 'ck-responsive-form' ];\n\n\t\tif ( linkCommand.manualDecorators.length ) {\n\t\t\tclassList.push( 'ck-link-form_layout-vertical', 'ck-vertical-form' );\n\t\t}\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'form',\n\n\t\t\tattributes: {\n\t\t\t\tclass: classList,\n\n\t\t\t\t// https://github.com/ckeditor/ckeditor5-link/issues/90\n\t\t\t\ttabindex: '-1'\n\t\t\t},\n\n\t\t\tchildren: this.children\n\t\t} );\n\t}\n\n\t/**\n\t * Obtains the state of the {@link module:ui/button/switchbuttonview~SwitchButtonView switch buttons} representing\n\t * {@link module:link/linkcommand~LinkCommand#manualDecorators manual link decorators}\n\t * in the {@link module:link/ui/linkformview~LinkFormView}.\n\t *\n\t * @returns Key-value pairs, where the key is the name of the decorator and the value is its state.\n\t */\n\tpublic getDecoratorSwitchesState(): Record<string, boolean> {\n\t\treturn Array\n\t\t\t.from( this._manualDecoratorSwitches as Iterable<SwitchButtonView & { name: string }> )\n\t\t\t.reduce( ( accumulator, switchButton ) => {\n\t\t\t\taccumulator[ switchButton.name ] = switchButton.isOn;\n\t\t\t\treturn accumulator;\n\t\t\t}, {} as Record<string, boolean> );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tsubmitHandler( {\n\t\t\tview: this\n\t\t} );\n\n\t\tconst childViews = [\n\t\t\tthis.urlInputView,\n\t\t\t...this._manualDecoratorSwitches,\n\t\t\tthis.saveButtonView,\n\t\t\tthis.cancelButtonView\n\t\t];\n\n\t\tchildViews.forEach( v => {\n\t\t\t// Register the view as focusable.\n\t\t\tthis._focusables.add( v );\n\n\t\t\t// Register the view in the focus tracker.\n\t\t\tthis.focusTracker.add( v.element! );\n\t\t} );\n\n\t\t// Start listening for the keystrokes coming from #element.\n\t\tthis.keystrokes.listenTo( this.element! );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.focusTracker.destroy();\n\t\tthis.keystrokes.destroy();\n\t}\n\n\t/**\n\t * Focuses the fist {@link #_focusables} in the form.\n\t */\n\tpublic focus(): void {\n\t\tthis._focusCycler.focusFirst();\n\t}\n\n\t/**\n\t * Creates a labeled input view.\n\t *\n\t * @returns Labeled field view instance.\n\t */\n\tprivate _createUrlInput(): LabeledFieldView<InputTextView> {\n\t\tconst t = this.locale!.t;\n\t\tconst labeledInput = new LabeledFieldView( this.locale, createLabeledInputText );\n\n\t\tlabeledInput.label = t( 'Link URL' );\n\n\t\treturn labeledInput;\n\t}\n\n\t/**\n\t * Creates a button view.\n\t *\n\t * @param label The button label.\n\t * @param icon The button icon.\n\t * @param className The additional button CSS class name.\n\t * @param eventName An event name that the `ButtonView#execute` event will be delegated to.\n\t * @returns The button view instance.\n\t */\n\tprivate _createButton( label: string, icon: string, className: string, eventName?: string ): ButtonView {\n\t\tconst button = new ButtonView( this.locale );\n\n\t\tbutton.set( {\n\t\t\tlabel,\n\t\t\ticon,\n\t\t\ttooltip: true\n\t\t} );\n\n\t\tbutton.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: className\n\t\t\t}\n\t\t} );\n\n\t\tif ( eventName ) {\n\t\t\tbutton.delegate( 'execute' ).to( this, eventName );\n\t\t}\n\n\t\treturn button;\n\t}\n\n\t/**\n\t * Populates {@link module:ui/viewcollection~ViewCollection} of {@link module:ui/button/switchbuttonview~SwitchButtonView}\n\t * made based on {@link module:link/linkcommand~LinkCommand#manualDecorators}.\n\t *\n\t * @param linkCommand A reference to the link command.\n\t * @returns ViewCollection of switch buttons.\n\t */\n\tprivate _createManualDecoratorSwitches( linkCommand: LinkCommand ): ViewCollection {\n\t\tconst switches = this.createCollection();\n\n\t\tfor ( const manualDecorator of linkCommand.manualDecorators ) {\n\t\t\tconst switchButton: SwitchButtonView & { name?: string } = new SwitchButtonView( this.locale );\n\n\t\t\tswitchButton.set( {\n\t\t\t\tname: manualDecorator.id,\n\t\t\t\tlabel: manualDecorator.label,\n\t\t\t\twithText: true\n\t\t\t} );\n\n\t\t\tswitchButton.bind( 'isOn' ).toMany( [ manualDecorator, linkCommand ], 'value', ( decoratorValue, commandValue ) => {\n\t\t\t\treturn commandValue === undefined && decoratorValue === undefined ? !!manualDecorator.defaultValue : !!decoratorValue;\n\t\t\t} );\n\n\t\t\tswitchButton.on( 'execute', () => {\n\t\t\t\tmanualDecorator.set( 'value', !switchButton.isOn );\n\t\t\t} );\n\n\t\t\tswitches.add( switchButton );\n\t\t}\n\n\t\treturn switches;\n\t}\n\n\t/**\n\t * Populates the {@link #children} collection of the form.\n\t *\n\t * If {@link module:link/linkcommand~LinkCommand#manualDecorators manual decorators} are configured in the editor, it creates an\n\t * additional `View` wrapping all {@link #_manualDecoratorSwitches} switch buttons corresponding\n\t * to these decorators.\n\t *\n\t * @param manualDecorators A reference to\n\t * the collection of manual decorators stored in the link command.\n\t * @returns The children of link form view.\n\t */\n\tprivate _createFormChildren( manualDecorators: Collection<ManualDecorator> ): ViewCollection {\n\t\tconst children = this.createCollection();\n\n\t\tchildren.add( this.urlInputView );\n\n\t\tif ( manualDecorators.length ) {\n\t\t\tconst additionalButtonsView = new View();\n\n\t\t\tadditionalButtonsView.setTemplate( {\n\t\t\t\ttag: 'ul',\n\t\t\t\tchildren: this._manualDecoratorSwitches.map( switchButton => ( {\n\t\t\t\t\ttag: 'li',\n\t\t\t\t\tchildren: [ switchButton ],\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: [\n\t\t\t\t\t\t\t'ck',\n\t\t\t\t\t\t\t'ck-list__item'\n\t\t\t\t\t\t]\n\t\t\t\t\t}\n\t\t\t\t} ) ),\n\t\t\t\tattributes: {\n\t\t\t\t\tclass: [\n\t\t\t\t\t\t'ck',\n\t\t\t\t\t\t'ck-reset',\n\t\t\t\t\t\t'ck-list'\n\t\t\t\t\t]\n\t\t\t\t}\n\t\t\t} );\n\t\t\tchildren.add( additionalButtonsView );\n\t\t}\n\n\t\tchildren.add( this.saveButtonView );\n\t\tchildren.add( this.cancelButtonView );\n\n\t\treturn children;\n\t}\n}\n\n/**\n * Fired when the form view is submitted (when one of the children triggered the submit event),\n * for example with a click on {@link ~LinkFormView#saveButtonView}.\n *\n * @eventName ~LinkFormView#submit\n */\nexport type SubmitEvent = {\n\tname: 'submit';\n\targs: [];\n};\n\n/**\n * Fired when the form view is canceled, for example with a click on {@link ~LinkFormView#cancelButtonView}.\n *\n * @eventName ~LinkFormView#cancel\n */\nexport type CancelEvent = {\n\tname: 'cancel';\n\targs: [];\n};\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./linkactions.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/ui/linkactionsview\n */\n\nimport { ButtonView, View, ViewCollection, FocusCycler } from 'ckeditor5/src/ui';\nimport { FocusTracker, KeystrokeHandler, type LocaleTranslate, type Locale } from 'ckeditor5/src/utils';\nimport { icons } from 'ckeditor5/src/core';\n\nimport { ensureSafeUrl } from '../utils';\n\n// See: #8833.\n// eslint-disable-next-line ckeditor5-rules/ckeditor-imports\nimport '@ckeditor/ckeditor5-ui/theme/components/responsive-form/responsiveform.css';\nimport '../../theme/linkactions.css';\n\nimport unlinkIcon from '../../theme/icons/unlink.svg';\n\n/**\n * The link actions view class. This view displays the link preview, allows\n * unlinking or editing the link.\n */\nexport default class LinkActionsView extends View {\n\t/**\n\t * Tracks information about DOM focus in the actions.\n\t */\n\tpublic readonly focusTracker = new FocusTracker();\n\n\t/**\n\t * An instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}.\n\t */\n\tpublic readonly keystrokes = new KeystrokeHandler();\n\n\t/**\n\t * The href preview view.\n\t */\n\tpublic previewButtonView: View;\n\n\t/**\n\t * The unlink button view.\n\t */\n\tpublic unlinkButtonView: ButtonView;\n\n\t/**\n\t * The edit link button view.\n\t */\n\tpublic editButtonView: ButtonView;\n\n\t/**\n\t * The value of the \"href\" attribute of the link to use in the {@link #previewButtonView}.\n\t *\n\t * @observable\n\t */\n\tdeclare public href: string | undefined;\n\n\t/**\n\t * A collection of views that can be focused in the view.\n\t */\n\tprivate readonly _focusables = new ViewCollection();\n\n\t/**\n\t * Helps cycling over {@link #_focusables} in the view.\n\t */\n\tprivate readonly _focusCycler: FocusCycler;\n\n\tdeclare public t: LocaleTranslate;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst t = locale.t;\n\n\t\tthis.previewButtonView = this._createPreviewButton();\n\t\tthis.unlinkButtonView = this._createButton( t( 'Unlink' ), unlinkIcon, 'unlink' );\n\t\tthis.editButtonView = this._createButton( t( 'Edit link' ), icons.pencil, 'edit' );\n\n\t\tthis.set( 'href', undefined );\n\n\t\tthis._focusCycler = new FocusCycler( {\n\t\t\tfocusables: this._focusables,\n\t\t\tfocusTracker: this.focusTracker,\n\t\t\tkeystrokeHandler: this.keystrokes,\n\t\t\tactions: {\n\t\t\t\t// Navigate fields backwards using the Shift + Tab keystroke.\n\t\t\t\tfocusPrevious: 'shift + tab',\n\n\t\t\t\t// Navigate fields forwards using the Tab key.\n\t\t\t\tfocusNext: 'tab'\n\t\t\t}\n\t\t} );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-link-actions',\n\t\t\t\t\t'ck-responsive-form'\n\t\t\t\t],\n\n\t\t\t\t// https://github.com/ckeditor/ckeditor5-link/issues/90\n\t\t\t\ttabindex: '-1'\n\t\t\t},\n\n\t\t\tchildren: [\n\t\t\t\tthis.previewButtonView,\n\t\t\t\tthis.editButtonView,\n\t\t\t\tthis.unlinkButtonView\n\t\t\t]\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tconst childViews = [\n\t\t\tthis.previewButtonView,\n\t\t\tthis.editButtonView,\n\t\t\tthis.unlinkButtonView\n\t\t];\n\n\t\tchildViews.forEach( v => {\n\t\t\t// Register the view as focusable.\n\t\t\tthis._focusables.add( v );\n\n\t\t\t// Register the view in the focus tracker.\n\t\t\tthis.focusTracker.add( v.element! );\n\t\t} );\n\n\t\t// Start listening for the keystrokes coming from #element.\n\t\tthis.keystrokes.listenTo( this.element! );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.focusTracker.destroy();\n\t\tthis.keystrokes.destroy();\n\t}\n\n\t/**\n\t * Focuses the fist {@link #_focusables} in the actions.\n\t */\n\tpublic focus(): void {\n\t\tthis._focusCycler.focusFirst();\n\t}\n\n\t/**\n\t * Creates a button view.\n\t *\n\t * @param label The button label.\n\t * @param icon The button icon.\n\t * @param eventName An event name that the `ButtonView#execute` event will be delegated to.\n\t * @returns The button view instance.\n\t */\n\tprivate _createButton( label: string, icon: string, eventName?: string ): ButtonView {\n\t\tconst button = new ButtonView( this.locale );\n\n\t\tbutton.set( {\n\t\t\tlabel,\n\t\t\ticon,\n\t\t\ttooltip: true\n\t\t} );\n\n\t\tbutton.delegate( 'execute' ).to( this, eventName );\n\n\t\treturn button;\n\t}\n\n\t/**\n\t * Creates a link href preview button.\n\t *\n\t * @returns The button view instance.\n\t */\n\tprivate _createPreviewButton(): ButtonView {\n\t\tconst button = new ButtonView( this.locale );\n\t\tconst bind = this.bindTemplate;\n\t\tconst t = this.t;\n\n\t\tbutton.set( {\n\t\t\twithText: true,\n\t\t\ttooltip: t( 'Open link in new tab' )\n\t\t} );\n\n\t\tbutton.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-link-actions__preview'\n\t\t\t\t],\n\t\t\t\thref: bind.to( 'href', href => href && ensureSafeUrl( href ) ),\n\t\t\t\ttarget: '_blank',\n\t\t\t\trel: 'noopener noreferrer'\n\t\t\t}\n\t\t} );\n\n\t\tbutton.bind( 'label' ).to( this, 'href', href => {\n\t\t\treturn href || t( 'This link has no URL' );\n\t\t} );\n\n\t\tbutton.bind( 'isEnabled' ).to( this, 'href', href => !!href );\n\n\t\tbutton.template!.tag = 'a';\n\t\tbutton.template!.eventListeners = {};\n\n\t\treturn button;\n\t}\n}\n\n/**\n * Fired when the {@link ~LinkActionsView#editButtonView} is clicked.\n *\n * @eventName ~LinkActionsView#edit\n */\nexport type EditEvent = {\n\tname: 'edit';\n\targs: [];\n};\n\n/**\n * Fired when the {@link ~LinkActionsView#unlinkButtonView} is clicked.\n *\n * @eventName ~LinkActionsView#unlink\n */\nexport type UnlinkEvent = {\n\tname: 'unlink';\n\targs: [];\n};\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m11.077 15 .991-1.416a.75.75 0 1 1 1.229.86l-1.148 1.64a.748.748 0 0 1-.217.206 5.251 5.251 0 0 1-8.503-5.955.741.741 0 0 1 .12-.274l1.147-1.639a.75.75 0 1 1 1.228.86L4.933 10.7l.006.003a3.75 3.75 0 0 0 6.132 4.294l.006.004zm5.494-5.335a.748.748 0 0 1-.12.274l-1.147 1.639a.75.75 0 1 1-1.228-.86l.86-1.23a3.75 3.75 0 0 0-6.144-4.301l-.86 1.229a.75.75 0 0 1-1.229-.86l1.148-1.64a.748.748 0 0 1 .217-.206 5.251 5.251 0 0 1 8.503 5.955zm-4.563-2.532a.75.75 0 0 1 .184 1.045l-3.155 4.505a.75.75 0 1 1-1.229-.86l3.155-4.506a.75.75 0 0 1 1.045-.184zm4.919 10.562-1.414 1.414a.75.75 0 1 1-1.06-1.06l1.414-1.415-1.415-1.414a.75.75 0 0 1 1.061-1.06l1.414 1.414 1.414-1.415a.75.75 0 0 1 1.061 1.061l-1.414 1.414 1.414 1.415a.75.75 0 0 1-1.06 1.06l-1.415-1.414z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"m11.077 15 .991-1.416a.75.75 0 1 1 1.229.86l-1.148 1.64a.748.748 0 0 1-.217.206 5.251 5.251 0 0 1-8.503-5.955.741.741 0 0 1 .12-.274l1.147-1.639a.75.75 0 1 1 1.228.86L4.933 10.7l.006.003a3.75 3.75 0 0 0 6.132 4.294l.006.004zm5.494-5.335a.748.748 0 0 1-.12.274l-1.147 1.639a.75.75 0 1 1-1.228-.86l.86-1.23a3.75 3.75 0 0 0-6.144-4.301l-.86 1.229a.75.75 0 0 1-1.229-.86l1.148-1.64a.748.748 0 0 1 .217-.206 5.251 5.251 0 0 1 8.503 5.955zm-4.563-2.532a.75.75 0 0 1 .184 1.045l-3.155 4.505a.75.75 0 1 1-1.229-.86l3.155-4.506a.75.75 0 0 1 1.045-.184z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/linkui\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport {\n\tClickObserver,\n\ttype ViewAttributeElement,\n\ttype ViewDocumentClickEvent,\n\ttype ViewElement,\n\ttype ViewPosition\n} from 'ckeditor5/src/engine';\nimport {\n\tButtonView,\n\tContextualBalloon,\n\tclickOutsideHandler,\n\tCssTransitionDisablerMixin,\n\ttype ViewWithCssTransitionDisabler\n} from 'ckeditor5/src/ui';\nimport type { PositionOptions } from 'ckeditor5/src/utils';\nimport { isWidget } from 'ckeditor5/src/widget';\n\nimport LinkFormView from './ui/linkformview';\nimport LinkActionsView from './ui/linkactionsview';\nimport type LinkCommand from './linkcommand';\nimport type UnlinkCommand from './unlinkcommand';\nimport { addLinkProtocolIfApplicable, isLinkElement, LINK_KEYSTROKE } from './utils';\n\nimport linkIcon from '../theme/icons/link.svg';\n\nconst VISUAL_SELECTION_MARKER_NAME = 'link-ui';\n\n/**\n * The link UI plugin. It introduces the `'link'` and `'unlink'` buttons and support for the <kbd>Ctrl+K</kbd> keystroke.\n *\n * It uses the\n * {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon contextual balloon plugin}.\n */\nexport default class LinkUI extends Plugin {\n\t/**\n\t * The actions view displayed inside of the balloon.\n\t */\n\tpublic actionsView: LinkActionsView | null = null;\n\n\t/**\n\t * The form view displayed inside the balloon.\n\t */\n\tpublic formView: LinkFormView & ViewWithCssTransitionDisabler | null = null;\n\n\t/**\n\t * The contextual balloon plugin instance.\n\t */\n\tprivate _balloon!: ContextualBalloon;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ContextualBalloon ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'LinkUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\teditor.editing.view.addObserver( ClickObserver );\n\n\t\tthis._balloon = editor.plugins.get( ContextualBalloon );\n\n\t\t// Create toolbar buttons.\n\t\tthis._createToolbarLinkButton();\n\t\tthis._enableBalloonActivators();\n\n\t\t// Renders a fake visual selection marker on an expanded selection.\n\t\teditor.conversion.for( 'editingDowncast' ).markerToHighlight( {\n\t\t\tmodel: VISUAL_SELECTION_MARKER_NAME,\n\t\t\tview: {\n\t\t\t\tclasses: [ 'ck-fake-link-selection' ]\n\t\t\t}\n\t\t} );\n\n\t\t// Renders a fake visual selection marker on a collapsed selection.\n\t\teditor.conversion.for( 'editingDowncast' ).markerToElement( {\n\t\t\tmodel: VISUAL_SELECTION_MARKER_NAME,\n\t\t\tview: {\n\t\t\t\tname: 'span',\n\t\t\t\tclasses: [ 'ck-fake-link-selection', 'ck-fake-link-selection_collapsed' ]\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\t// Destroy created UI components as they are not automatically destroyed (see ckeditor5#1341).\n\t\tif ( this.formView ) {\n\t\t\tthis.formView.destroy();\n\t\t}\n\n\t\tif ( this.actionsView ) {\n\t\t\tthis.actionsView.destroy();\n\t\t}\n\t}\n\n\t/**\n\t * Creates views.\n\t */\n\tprivate _createViews() {\n\t\tthis.actionsView = this._createActionsView();\n\t\tthis.formView = this._createFormView();\n\n\t\t// Attach lifecycle actions to the the balloon.\n\t\tthis._enableUserBalloonInteractions();\n\t}\n\n\t/**\n\t * Creates the {@link module:link/ui/linkactionsview~LinkActionsView} instance.\n\t */\n\tprivate _createActionsView(): LinkActionsView {\n\t\tconst editor = this.editor;\n\t\tconst actionsView = new LinkActionsView( editor.locale );\n\t\tconst linkCommand: LinkCommand = editor.commands.get( 'link' )!;\n\t\tconst unlinkCommand: UnlinkCommand = editor.commands.get( 'unlink' )!;\n\n\t\tactionsView.bind( 'href' ).to( linkCommand, 'value' );\n\t\tactionsView.editButtonView.bind( 'isEnabled' ).to( linkCommand );\n\t\tactionsView.unlinkButtonView.bind( 'isEnabled' ).to( unlinkCommand );\n\n\t\t// Execute unlink command after clicking on the \"Edit\" button.\n\t\tthis.listenTo( actionsView, 'edit', () => {\n\t\t\tthis._addFormView();\n\t\t} );\n\n\t\t// Execute unlink command after clicking on the \"Unlink\" button.\n\t\tthis.listenTo( actionsView, 'unlink', () => {\n\t\t\teditor.execute( 'unlink' );\n\t\t\tthis._hideUI();\n\t\t} );\n\n\t\t// Close the panel on esc key press when the **actions have focus**.\n\t\tactionsView.keystrokes.set( 'Esc', ( data, cancel ) => {\n\t\t\tthis._hideUI();\n\t\t\tcancel();\n\t\t} );\n\n\t\t// Open the form view on Ctrl+K when the **actions have focus**..\n\t\tactionsView.keystrokes.set( LINK_KEYSTROKE, ( data, cancel ) => {\n\t\t\tthis._addFormView();\n\t\t\tcancel();\n\t\t} );\n\n\t\treturn actionsView;\n\t}\n\n\t/**\n\t * Creates the {@link module:link/ui/linkformview~LinkFormView} instance.\n\t */\n\tprivate _createFormView(): LinkFormView & ViewWithCssTransitionDisabler {\n\t\tconst editor = this.editor;\n\t\tconst linkCommand: LinkCommand = editor.commands.get( 'link' )!;\n\t\tconst defaultProtocol = editor.config.get( 'link.defaultProtocol' );\n\n\t\tconst formView = new ( CssTransitionDisablerMixin( LinkFormView ) )( editor.locale, linkCommand );\n\n\t\tformView.urlInputView.fieldView.bind( 'value' ).to( linkCommand, 'value' );\n\n\t\t// Form elements should be read-only when corresponding commands are disabled.\n\t\tformView.urlInputView.bind( 'isEnabled' ).to( linkCommand, 'isEnabled' );\n\t\tformView.saveButtonView.bind( 'isEnabled' ).to( linkCommand );\n\n\t\t// Execute link command after clicking the \"Save\" button.\n\t\tthis.listenTo( formView, 'submit', () => {\n\t\t\tconst { value } = formView.urlInputView.fieldView.element!;\n\t\t\tconst parsedUrl = addLinkProtocolIfApplicable( value, defaultProtocol );\n\t\t\teditor.execute( 'link', parsedUrl, formView.getDecoratorSwitchesState() );\n\t\t\tthis._closeFormView();\n\t\t} );\n\n\t\t// Hide the panel after clicking the \"Cancel\" button.\n\t\tthis.listenTo( formView, 'cancel', () => {\n\t\t\tthis._closeFormView();\n\t\t} );\n\n\t\t// Close the panel on esc key press when the **form has focus**.\n\t\tformView.keystrokes.set( 'Esc', ( data, cancel ) => {\n\t\t\tthis._closeFormView();\n\t\t\tcancel();\n\t\t} );\n\n\t\treturn formView;\n\t}\n\n\t/**\n\t * Creates a toolbar Link button. Clicking this button will show\n\t * a {@link #_balloon} attached to the selection.\n\t */\n\tprivate _createToolbarLinkButton(): void {\n\t\tconst editor = this.editor;\n\t\tconst linkCommand: LinkCommand = editor.commands.get( 'link' )!;\n\t\tconst t = editor.t;\n\n\t\teditor.ui.componentFactory.add( 'link', locale => {\n\t\t\tconst button = new ButtonView( locale );\n\n\t\t\tbutton.isEnabled = true;\n\t\t\tbutton.label = t( 'Link' );\n\t\t\tbutton.icon = linkIcon;\n\t\t\tbutton.keystroke = LINK_KEYSTROKE;\n\t\t\tbutton.tooltip = true;\n\t\t\tbutton.isToggleable = true;\n\n\t\t\t// Bind button to the command.\n\t\t\tbutton.bind( 'isEnabled' ).to( linkCommand, 'isEnabled' );\n\t\t\tbutton.bind( 'isOn' ).to( linkCommand, 'value', value => !!value );\n\n\t\t\t// Show the panel on button click.\n\t\t\tthis.listenTo( button, 'execute', () => this._showUI( true ) );\n\n\t\t\treturn button;\n\t\t} );\n\t}\n\n\t/**\n\t * Attaches actions that control whether the balloon panel containing the\n\t * {@link #formView} should be displayed.\n\t */\n\tprivate _enableBalloonActivators(): void {\n\t\tconst editor = this.editor;\n\t\tconst viewDocument = editor.editing.view.document;\n\n\t\t// Handle click on view document and show panel when selection is placed inside the link element.\n\t\t// Keep panel open until selection will be inside the same link element.\n\t\tthis.listenTo<ViewDocumentClickEvent>( viewDocument, 'click', () => {\n\t\t\tconst parentLink = this._getSelectedLinkElement();\n\n\t\t\tif ( parentLink ) {\n\t\t\t\t// Then show panel but keep focus inside editor editable.\n\t\t\t\tthis._showUI();\n\t\t\t}\n\t\t} );\n\n\t\t// Handle the `Ctrl+K` keystroke and show the panel.\n\t\teditor.keystrokes.set( LINK_KEYSTROKE, ( keyEvtData, cancel ) => {\n\t\t\t// Prevent focusing the search bar in FF, Chrome and Edge. See https://github.com/ckeditor/ckeditor5/issues/4811.\n\t\t\tcancel();\n\n\t\t\tif ( editor.commands.get( 'link' )!.isEnabled ) {\n\t\t\t\tthis._showUI( true );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Attaches actions that control whether the balloon panel containing the\n\t * {@link #formView} is visible or not.\n\t */\n\tprivate _enableUserBalloonInteractions(): void {\n\t\t// Focus the form if the balloon is visible and the Tab key has been pressed.\n\t\tthis.editor.keystrokes.set( 'Tab', ( data, cancel ) => {\n\t\t\tif ( this._areActionsVisible && !this.actionsView!.focusTracker.isFocused ) {\n\t\t\t\tthis.actionsView!.focus();\n\t\t\t\tcancel();\n\t\t\t}\n\t\t}, {\n\t\t\t// Use the high priority because the link UI navigation is more important\n\t\t\t// than other feature's actions, e.g. list indentation.\n\t\t\t// https://github.com/ckeditor/ckeditor5-link/issues/146\n\t\t\tpriority: 'high'\n\t\t} );\n\n\t\t// Close the panel on the Esc key press when the editable has focus and the balloon is visible.\n\t\tthis.editor.keystrokes.set( 'Esc', ( data, cancel ) => {\n\t\t\tif ( this._isUIVisible ) {\n\t\t\t\tthis._hideUI();\n\t\t\t\tcancel();\n\t\t\t}\n\t\t} );\n\n\t\t// Close on click outside of balloon panel element.\n\t\tclickOutsideHandler( {\n\t\t\temitter: this.formView!,\n\t\t\tactivator: () => this._isUIInPanel,\n\t\t\tcontextElements: () => [ this._balloon.view.element! ],\n\t\t\tcallback: () => this._hideUI()\n\t\t} );\n\t}\n\n\t/**\n\t * Adds the {@link #actionsView} to the {@link #_balloon}.\n\t *\n\t * @internal\n\t */\n\tpublic _addActionsView(): void {\n\t\tif ( !this.actionsView ) {\n\t\t\tthis._createViews();\n\t\t}\n\n\t\tif ( this._areActionsInPanel ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._balloon.add( {\n\t\t\tview: this.actionsView!,\n\t\t\tposition: this._getBalloonPositionData()\n\t\t} );\n\t}\n\n\t/**\n\t * Adds the {@link #formView} to the {@link #_balloon}.\n\t */\n\tprivate _addFormView(): void {\n\t\tif ( !this.formView ) {\n\t\t\tthis._createViews();\n\t\t}\n\n\t\tif ( this._isFormInPanel ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst editor = this.editor;\n\t\tconst linkCommand: LinkCommand = editor.commands.get( 'link' )!;\n\n\t\tthis.formView!.disableCssTransitions();\n\n\t\tthis._balloon.add( {\n\t\t\tview: this.formView!,\n\t\t\tposition: this._getBalloonPositionData()\n\t\t} );\n\n\t\t// Select input when form view is currently visible.\n\t\tif ( this._balloon.visibleView === this.formView ) {\n\t\t\tthis.formView!.urlInputView.fieldView.select();\n\t\t}\n\n\t\tthis.formView!.enableCssTransitions();\n\n\t\t// Make sure that each time the panel shows up, the URL field remains in sync with the value of\n\t\t// the command. If the user typed in the input, then canceled the balloon (`urlInputView.fieldView#value` stays\n\t\t// unaltered) and re-opened it without changing the value of the link command (e.g. because they\n\t\t// clicked the same link), they would see the old value instead of the actual value of the command.\n\t\t// https://github.com/ckeditor/ckeditor5-link/issues/78\n\t\t// https://github.com/ckeditor/ckeditor5-link/issues/123\n\t\tthis.formView!.urlInputView.fieldView.element!.value = linkCommand.value || '';\n\t}\n\n\t/**\n\t * Closes the form view. Decides whether the balloon should be hidden completely or if the action view should be shown. This is\n\t * decided upon the link command value (which has a value if the document selection is in the link).\n\t *\n\t * Additionally, if any {@link module:link/linkconfig~LinkConfig#decorators} are defined in the editor configuration, the state of\n\t * switch buttons responsible for manual decorator handling is restored.\n\t */\n\tprivate _closeFormView(): void {\n\t\tconst linkCommand: LinkCommand = this.editor.commands.get( 'link' )!;\n\n\t\t// Restore manual decorator states to represent the current model state. This case is important to reset the switch buttons\n\t\t// when the user cancels the editing form.\n\t\tlinkCommand.restoreManualDecoratorStates();\n\n\t\tif ( linkCommand.value !== undefined ) {\n\t\t\tthis._removeFormView();\n\t\t} else {\n\t\t\tthis._hideUI();\n\t\t}\n\t}\n\n\t/**\n\t * Removes the {@link #formView} from the {@link #_balloon}.\n\t */\n\tprivate _removeFormView(): void {\n\t\tif ( this._isFormInPanel ) {\n\t\t\t// Blur the input element before removing it from DOM to prevent issues in some browsers.\n\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/1501.\n\t\t\tthis.formView!.saveButtonView.focus();\n\n\t\t\tthis._balloon.remove( this.formView! );\n\n\t\t\t// Because the form has an input which has focus, the focus must be brought back\n\t\t\t// to the editor. Otherwise, it would be lost.\n\t\t\tthis.editor.editing.view.focus();\n\n\t\t\tthis._hideFakeVisualSelection();\n\t\t}\n\t}\n\n\t/**\n\t * Shows the correct UI type. It is either {@link #formView} or {@link #actionsView}.\n\t *\n\t * @internal\n\t */\n\tpublic _showUI( forceVisible: boolean = false ): void {\n\t\tif ( !this.formView ) {\n\t\t\tthis._createViews();\n\t\t}\n\n\t\t// When there's no link under the selection, go straight to the editing UI.\n\t\tif ( !this._getSelectedLinkElement() ) {\n\t\t\t// Show visual selection on a text without a link when the contextual balloon is displayed.\n\t\t\t// See https://github.com/ckeditor/ckeditor5/issues/4721.\n\t\t\tthis._showFakeVisualSelection();\n\n\t\t\tthis._addActionsView();\n\n\t\t\t// Be sure panel with link is visible.\n\t\t\tif ( forceVisible ) {\n\t\t\t\tthis._balloon.showStack( 'main' );\n\t\t\t}\n\n\t\t\tthis._addFormView();\n\t\t}\n\t\t// If there's a link under the selection...\n\t\telse {\n\t\t\t// Go to the editing UI if actions are already visible.\n\t\t\tif ( this._areActionsVisible ) {\n\t\t\t\tthis._addFormView();\n\t\t\t}\n\t\t\t// Otherwise display just the actions UI.\n\t\t\telse {\n\t\t\t\tthis._addActionsView();\n\t\t\t}\n\n\t\t\t// Be sure panel with link is visible.\n\t\t\tif ( forceVisible ) {\n\t\t\t\tthis._balloon.showStack( 'main' );\n\t\t\t}\n\t\t}\n\n\t\t// Begin responding to ui#update once the UI is added.\n\t\tthis._startUpdatingUI();\n\t}\n\n\t/**\n\t * Removes the {@link #formView} from the {@link #_balloon}.\n\t *\n\t * See {@link #_addFormView}, {@link #_addActionsView}.\n\t */\n\tprivate _hideUI(): void {\n\t\tif ( !this._isUIInPanel ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst editor = this.editor;\n\n\t\tthis.stopListening( editor.ui, 'update' );\n\t\tthis.stopListening( this._balloon, 'change:visibleView' );\n\n\t\t// Make sure the focus always gets back to the editable _before_ removing the focused form view.\n\t\t// Doing otherwise causes issues in some browsers. See https://github.com/ckeditor/ckeditor5-link/issues/193.\n\t\teditor.editing.view.focus();\n\n\t\t// Remove form first because it's on top of the stack.\n\t\tthis._removeFormView();\n\n\t\t// Then remove the actions view because it's beneath the form.\n\t\tthis._balloon.remove( this.actionsView! );\n\n\t\tthis._hideFakeVisualSelection();\n\t}\n\n\t/**\n\t * Makes the UI react to the {@link module:ui/editorui/editorui~EditorUI#event:update} event to\n\t * reposition itself when the editor UI should be refreshed.\n\t *\n\t * See: {@link #_hideUI} to learn when the UI stops reacting to the `update` event.\n\t */\n\tprivate _startUpdatingUI(): void {\n\t\tconst editor = this.editor;\n\t\tconst viewDocument = editor.editing.view.document;\n\n\t\tlet prevSelectedLink = this._getSelectedLinkElement();\n\t\tlet prevSelectionParent = getSelectionParent();\n\n\t\tconst update = () => {\n\t\t\tconst selectedLink = this._getSelectedLinkElement();\n\t\t\tconst selectionParent = getSelectionParent();\n\n\t\t\t// Hide the panel if:\n\t\t\t//\n\t\t\t// * the selection went out of the EXISTING link element. E.g. user moved the caret out\n\t\t\t// of the link,\n\t\t\t// * the selection went to a different parent when creating a NEW link. E.g. someone\n\t\t\t// else modified the document.\n\t\t\t// * the selection has expanded (e.g. displaying link actions then pressing SHIFT+Right arrow).\n\t\t\t//\n\t\t\t// Note: #_getSelectedLinkElement will return a link for a non-collapsed selection only\n\t\t\t// when fully selected.\n\t\t\tif ( ( prevSelectedLink && !selectedLink ) ||\n\t\t\t\t( !prevSelectedLink && selectionParent !== prevSelectionParent ) ) {\n\t\t\t\tthis._hideUI();\n\t\t\t}\n\t\t\t// Update the position of the panel when:\n\t\t\t// * link panel is in the visible stack\n\t\t\t// * the selection remains in the original link element,\n\t\t\t// * there was no link element in the first place, i.e. creating a new link\n\t\t\telse if ( this._isUIVisible ) {\n\t\t\t\t// If still in a link element, simply update the position of the balloon.\n\t\t\t\t// If there was no link (e.g. inserting one), the balloon must be moved\n\t\t\t\t// to the new position in the editing view (a new native DOM range).\n\t\t\t\tthis._balloon.updatePosition( this._getBalloonPositionData() );\n\t\t\t}\n\n\t\t\tprevSelectedLink = selectedLink;\n\t\t\tprevSelectionParent = selectionParent;\n\t\t};\n\n\t\tfunction getSelectionParent() {\n\t\t\treturn viewDocument.selection.focus!.getAncestors()\n\t\t\t\t.reverse()\n\t\t\t\t.find( ( node ): node is ViewElement => node.is( 'element' ) );\n\t\t}\n\n\t\tthis.listenTo( editor.ui, 'update', update );\n\t\tthis.listenTo( this._balloon, 'change:visibleView', update );\n\t}\n\n\t/**\n\t * Returns `true` when {@link #formView} is in the {@link #_balloon}.\n\t */\n\tprivate get _isFormInPanel(): boolean {\n\t\treturn !!this.formView && this._balloon.hasView( this.formView );\n\t}\n\n\t/**\n\t * Returns `true` when {@link #actionsView} is in the {@link #_balloon}.\n\t */\n\tprivate get _areActionsInPanel(): boolean {\n\t\treturn !!this.actionsView && this._balloon.hasView( this.actionsView );\n\t}\n\n\t/**\n\t * Returns `true` when {@link #actionsView} is in the {@link #_balloon} and it is\n\t * currently visible.\n\t */\n\tprivate get _areActionsVisible(): boolean {\n\t\treturn !!this.actionsView && this._balloon.visibleView === this.actionsView;\n\t}\n\n\t/**\n\t * Returns `true` when {@link #actionsView} or {@link #formView} is in the {@link #_balloon}.\n\t */\n\tprivate get _isUIInPanel(): boolean {\n\t\treturn this._isFormInPanel || this._areActionsInPanel;\n\t}\n\n\t/**\n\t * Returns `true` when {@link #actionsView} or {@link #formView} is in the {@link #_balloon} and it is\n\t * currently visible.\n\t */\n\tprivate get _isUIVisible(): boolean {\n\t\tconst visibleView = this._balloon.visibleView;\n\n\t\treturn !!this.formView && visibleView == this.formView || this._areActionsVisible;\n\t}\n\n\t/**\n\t * Returns positioning options for the {@link #_balloon}. They control the way the balloon is attached\n\t * to the target element or selection.\n\t *\n\t * If the selection is collapsed and inside a link element, the panel will be attached to the\n\t * entire link element. Otherwise, it will be attached to the selection.\n\t */\n\tprivate _getBalloonPositionData(): Partial<PositionOptions> {\n\t\tconst view = this.editor.editing.view;\n\t\tconst model = this.editor.model;\n\t\tconst viewDocument = view.document;\n\t\tlet target: PositionOptions[ 'target' ];\n\n\t\tif ( model.markers.has( VISUAL_SELECTION_MARKER_NAME ) ) {\n\t\t\t// There are cases when we highlight selection using a marker (#7705, #4721).\n\t\t\tconst markerViewElements = Array.from( this.editor.editing.mapper.markerNameToElements( VISUAL_SELECTION_MARKER_NAME )! );\n\t\t\tconst newRange = view.createRange(\n\t\t\t\tview.createPositionBefore( markerViewElements[ 0 ] ),\n\t\t\t\tview.createPositionAfter( markerViewElements[ markerViewElements.length - 1 ] )\n\t\t\t);\n\n\t\t\ttarget = view.domConverter.viewRangeToDom( newRange );\n\t\t} else {\n\t\t\t// Make sure the target is calculated on demand at the last moment because a cached DOM range\n\t\t\t// (which is very fragile) can desynchronize with the state of the editing view if there was\n\t\t\t// any rendering done in the meantime. This can happen, for instance, when an inline widget\n\t\t\t// gets unlinked.\n\t\t\ttarget = () => {\n\t\t\t\tconst targetLink = this._getSelectedLinkElement();\n\n\t\t\t\treturn targetLink ?\n\t\t\t\t\t// When selection is inside link element, then attach panel to this element.\n\t\t\t\t\tview.domConverter.mapViewToDom( targetLink )! :\n\t\t\t\t\t// Otherwise attach panel to the selection.\n\t\t\t\t\tview.domConverter.viewRangeToDom( viewDocument.selection.getFirstRange()! );\n\t\t\t};\n\t\t}\n\n\t\treturn { target };\n\t}\n\n\t/**\n\t * Returns the link {@link module:engine/view/attributeelement~AttributeElement} under\n\t * the {@link module:engine/view/document~Document editing view's} selection or `null`\n\t * if there is none.\n\t *\n\t * **Note**: For a noncollapsed selection, the link element is returned when **fully**\n\t * selected and the **only** element within the selection boundaries, or when\n\t * a linked widget is selected.\n\t */\n\tprivate _getSelectedLinkElement(): ViewAttributeElement | null {\n\t\tconst view = this.editor.editing.view;\n\t\tconst selection = view.document.selection;\n\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\t// The selection is collapsed or some widget is selected (especially inline widget).\n\t\tif ( selection.isCollapsed || selectedElement && isWidget( selectedElement ) ) {\n\t\t\treturn findLinkElementAncestor( selection.getFirstPosition()! );\n\t\t} else {\n\t\t\t// The range for fully selected link is usually anchored in adjacent text nodes.\n\t\t\t// Trim it to get closer to the actual link element.\n\t\t\tconst range = selection.getFirstRange()!.getTrimmed();\n\t\t\tconst startLink = findLinkElementAncestor( range.start );\n\t\t\tconst endLink = findLinkElementAncestor( range.end );\n\n\t\t\tif ( !startLink || startLink != endLink ) {\n\t\t\t\treturn null;\n\t\t\t}\n\n\t\t\t// Check if the link element is fully selected.\n\t\t\tif ( view.createRangeIn( startLink ).getTrimmed().isEqual( range ) ) {\n\t\t\t\treturn startLink;\n\t\t\t} else {\n\t\t\t\treturn null;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Displays a fake visual selection when the contextual balloon is displayed.\n\t *\n\t * This adds a 'link-ui' marker into the document that is rendered as a highlight on selected text fragment.\n\t */\n\tprivate _showFakeVisualSelection(): void {\n\t\tconst model = this.editor.model;\n\n\t\tmodel.change( writer => {\n\t\t\tconst range = model.document.selection.getFirstRange()!;\n\n\t\t\tif ( model.markers.has( VISUAL_SELECTION_MARKER_NAME ) ) {\n\t\t\t\twriter.updateMarker( VISUAL_SELECTION_MARKER_NAME, { range } );\n\t\t\t} else {\n\t\t\t\tif ( range.start.isAtEnd ) {\n\t\t\t\t\tconst startPosition = range.start.getLastMatchingPosition(\n\t\t\t\t\t\t( { item } ) => !model.schema.isContent( item ),\n\t\t\t\t\t\t{ boundaries: range }\n\t\t\t\t\t);\n\n\t\t\t\t\twriter.addMarker( VISUAL_SELECTION_MARKER_NAME, {\n\t\t\t\t\t\tusingOperation: false,\n\t\t\t\t\t\taffectsData: false,\n\t\t\t\t\t\trange: writer.createRange( startPosition, range.end )\n\t\t\t\t\t} );\n\t\t\t\t} else {\n\t\t\t\t\twriter.addMarker( VISUAL_SELECTION_MARKER_NAME, {\n\t\t\t\t\t\tusingOperation: false,\n\t\t\t\t\t\taffectsData: false,\n\t\t\t\t\t\trange\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Hides the fake visual selection created in {@link #_showFakeVisualSelection}.\n\t */\n\tprivate _hideFakeVisualSelection(): void {\n\t\tconst model = this.editor.model;\n\n\t\tif ( model.markers.has( VISUAL_SELECTION_MARKER_NAME ) ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\twriter.removeMarker( VISUAL_SELECTION_MARKER_NAME );\n\t\t\t} );\n\t\t}\n\t}\n}\n\n/**\n * Returns a link element if there's one among the ancestors of the provided `Position`.\n *\n * @param View position to analyze.\n * @returns Link element at the position or null.\n */\nfunction findLinkElementAncestor( position: ViewPosition ): ViewAttributeElement | null {\n\treturn position.getAncestors().find( ( ancestor ): ancestor is ViewAttributeElement => isLinkElement( ancestor ) ) || null;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/autolink\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport type { DocumentSelectionChangeEvent, Element, Model, Range } from 'ckeditor5/src/engine';\nimport { Delete, TextWatcher, getLastTextLine, type TextWatcherMatchedDataEvent } from 'ckeditor5/src/typing';\nimport type { EnterCommand, ShiftEnterCommand } from 'ckeditor5/src/enter';\n\nimport { addLinkProtocolIfApplicable, linkHasProtocol } from './utils';\n\nconst MIN_LINK_LENGTH_WITH_SPACE_AT_END = 4; // Ie: \"t.co \" (length 5).\n\n// This was a tweak from https://gist.github.com/dperini/729294.\nconst URL_REG_EXP = new RegExp(\n\t// Group 1: Line start or after a space.\n\t'(^|\\\\s)' +\n\t// Group 2: Detected URL (or e-mail).\n\t'(' +\n\t\t// Protocol identifier or short syntax \"//\"\n\t\t// a. Full form http://user@foo.bar.baz:8080/foo/bar.html#baz?foo=bar\n\t\t'(' +\n\t\t\t'(?:(?:(?:https?|ftp):)?\\\\/\\\\/)' +\n\t\t\t// BasicAuth using user:pass (optional)\n\t\t\t'(?:\\\\S+(?::\\\\S*)?@)?' +\n\t\t\t'(?:' +\n\t\t\t\t// IP address dotted notation octets\n\t\t\t\t// excludes loopback network 0.0.0.0\n\t\t\t\t// excludes reserved space >= 224.0.0.0\n\t\t\t\t// excludes network & broadcast addresses\n\t\t\t\t// (first & last IP address of each class)\n\t\t\t\t'(?:[1-9]\\\\d?|1\\\\d\\\\d|2[01]\\\\d|22[0-3])' +\n\t\t\t\t'(?:\\\\.(?:1?\\\\d{1,2}|2[0-4]\\\\d|25[0-5])){2}' +\n\t\t\t\t'(?:\\\\.(?:[1-9]\\\\d?|1\\\\d\\\\d|2[0-4]\\\\d|25[0-4]))' +\n\t\t\t\t'|' +\n\t\t\t\t'(' +\n\t\t\t\t\t// Do not allow `www.foo` - see https://github.com/ckeditor/ckeditor5/issues/8050.\n\t\t\t\t\t'((?!www\\\\.)|(www\\\\.))' +\n\t\t\t\t\t// Host & domain names.\n\t\t\t\t\t'(?![-_])(?:[-_a-z0-9\\\\u00a1-\\\\uffff]{1,63}\\\\.)+' +\n\t\t\t\t\t// TLD identifier name.\n\t\t\t\t\t'(?:[a-z\\\\u00a1-\\\\uffff]{2,63})' +\n\t\t\t\t')' +\n\t\t\t')' +\n\t\t\t// port number (optional)\n\t\t\t'(?::\\\\d{2,5})?' +\n\t\t\t// resource path (optional)\n\t\t\t'(?:[/?#]\\\\S*)?' +\n\t\t')' +\n\t\t'|' +\n\t\t// b. Short form (either www.example.com or example@example.com)\n\t\t'(' +\n\t\t\t'(www.|(\\\\S+@))' +\n\t\t\t// Host & domain names.\n\t\t\t'((?![-_])(?:[-_a-z0-9\\\\u00a1-\\\\uffff]{1,63}\\\\.))+' +\n\t\t\t// TLD identifier name.\n\t\t\t'(?:[a-z\\\\u00a1-\\\\uffff]{2,63})' +\n\t\t')' +\n\t')$', 'i' );\n\nconst URL_GROUP_IN_MATCH = 2;\n\n/**\n * The autolink plugin.\n */\nexport default class AutoLink extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Delete ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'AutoLink' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\n\t\tselection.on<DocumentSelectionChangeEvent>( 'change:range', () => {\n\t\t\t// Disable plugin when selection is inside a code block.\n\t\t\tthis.isEnabled = !selection.anchor!.parent.is( 'element', 'codeBlock' );\n\t\t} );\n\n\t\tthis._enableTypingHandling();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\tthis._enableEnterHandling();\n\t\tthis._enableShiftEnterHandling();\n\t}\n\n\t/**\n\t * Enables autolinking on typing.\n\t */\n\tprivate _enableTypingHandling(): void {\n\t\tconst editor = this.editor;\n\n\t\tconst watcher = new TextWatcher( editor.model, text => {\n\t\t\t// 1. Detect <kbd>Space</kbd> after a text with a potential link.\n\t\t\tif ( !isSingleSpaceAtTheEnd( text ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// 2. Check text before last typed <kbd>Space</kbd>.\n\t\t\tconst url = getUrlAtTextEnd( text.substr( 0, text.length - 1 ) );\n\n\t\t\tif ( url ) {\n\t\t\t\treturn { url };\n\t\t\t}\n\t\t} );\n\n\t\twatcher.on<TextWatcherMatchedDataEvent<{ url: string }>>( 'matched:data', ( evt, data ) => {\n\t\t\tconst { batch, range, url } = data;\n\n\t\t\tif ( !batch.isTyping ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst linkEnd = range.end.getShiftedBy( -1 ); // Executed after a space character.\n\t\t\tconst linkStart = linkEnd.getShiftedBy( -url.length );\n\n\t\t\tconst linkRange = editor.model.createRange( linkStart, linkEnd );\n\n\t\t\tthis._applyAutoLink( url, linkRange );\n\t\t} );\n\n\t\twatcher.bind( 'isEnabled' ).to( this );\n\t}\n\n\t/**\n\t * Enables autolinking on the <kbd>Enter</kbd> key.\n\t */\n\tprivate _enableEnterHandling(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst enterCommand: EnterCommand | undefined = editor.commands.get( 'enter' );\n\n\t\tif ( !enterCommand ) {\n\t\t\treturn;\n\t\t}\n\n\t\tenterCommand.on( 'execute', () => {\n\t\t\tconst position = model.document.selection.getFirstPosition()!;\n\n\t\t\tif ( !position.parent.previousSibling ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst rangeToCheck = model.createRangeIn( position.parent.previousSibling as Element );\n\n\t\t\tthis._checkAndApplyAutoLinkOnRange( rangeToCheck );\n\t\t} );\n\t}\n\n\t/**\n\t * Enables autolinking on the <kbd>Shift</kbd>+<kbd>Enter</kbd> keyboard shortcut.\n\t */\n\tprivate _enableShiftEnterHandling(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\n\t\tconst shiftEnterCommand: ShiftEnterCommand | undefined = editor.commands.get( 'shiftEnter' );\n\n\t\tif ( !shiftEnterCommand ) {\n\t\t\treturn;\n\t\t}\n\n\t\tshiftEnterCommand.on( 'execute', () => {\n\t\t\tconst position = model.document.selection.getFirstPosition()!;\n\n\t\t\tconst rangeToCheck = model.createRange(\n\t\t\t\tmodel.createPositionAt( position.parent, 0 ),\n\t\t\t\tposition.getShiftedBy( -1 )\n\t\t\t);\n\n\t\t\tthis._checkAndApplyAutoLinkOnRange( rangeToCheck );\n\t\t} );\n\t}\n\n\t/**\n\t * Checks if the passed range contains a linkable text.\n\t */\n\tprivate _checkAndApplyAutoLinkOnRange( rangeToCheck: Range ): void {\n\t\tconst model = this.editor.model;\n\t\tconst { text, range } = getLastTextLine( rangeToCheck, model );\n\n\t\tconst url = getUrlAtTextEnd( text );\n\n\t\tif ( url ) {\n\t\t\tconst linkRange = model.createRange(\n\t\t\t\trange.end.getShiftedBy( -url.length ),\n\t\t\t\trange.end\n\t\t\t);\n\n\t\t\tthis._applyAutoLink( url, linkRange );\n\t\t}\n\t}\n\n\t/**\n\t * Applies a link on a given range if the link should be applied.\n\t *\n\t * @param url The URL to link.\n\t * @param range The text range to apply the link attribute to.\n\t */\n\tprivate _applyAutoLink( url: string, range: Range ): void {\n\t\tconst model = this.editor.model;\n\n\t\tconst defaultProtocol = this.editor.config.get( 'link.defaultProtocol' );\n\t\tconst fullUrl = addLinkProtocolIfApplicable( url, defaultProtocol );\n\n\t\tif ( !this.isEnabled || !isLinkAllowedOnRange( range, model ) || !linkHasProtocol( fullUrl ) || linkIsAlreadySet( range ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tthis._persistAutoLink( fullUrl, range );\n\t}\n\n\t/**\n\t * Enqueues autolink changes in the model.\n\t *\n\t * @param url The URL to link.\n\t * @param range The text range to apply the link attribute to.\n\t */\n\tprivate _persistAutoLink( url: string, range: Range ): void {\n\t\tconst model = this.editor.model;\n\t\tconst deletePlugin = this.editor.plugins.get( 'Delete' );\n\n\t\t// Enqueue change to make undo step.\n\t\tmodel.enqueueChange( writer => {\n\t\t\twriter.setAttribute( 'linkHref', url, range );\n\n\t\t\tmodel.enqueueChange( () => {\n\t\t\t\tdeletePlugin.requestUndoOnBackspace();\n\t\t\t} );\n\t\t} );\n\t}\n}\n\n// Check if text should be evaluated by the plugin in order to reduce number of RegExp checks on whole text.\nfunction isSingleSpaceAtTheEnd( text: string ): boolean {\n\treturn text.length > MIN_LINK_LENGTH_WITH_SPACE_AT_END && text[ text.length - 1 ] === ' ' && text[ text.length - 2 ] !== ' ';\n}\n\nfunction getUrlAtTextEnd( text: string ): string | null {\n\tconst match = URL_REG_EXP.exec( text );\n\n\treturn match ? match[ URL_GROUP_IN_MATCH ] : null;\n}\n\nfunction isLinkAllowedOnRange( range: Range, model: Model ): boolean {\n\treturn model.schema.checkAttributeInSelection( model.createSelection( range ), 'linkHref' );\n}\n\nfunction linkIsAlreadySet( range: Range ): boolean {\n\tconst item = range.start.nodeAfter;\n\treturn !!item && item.hasAttribute( 'linkHref' );\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./linkimage.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/documentlist/utils/listwalker\n */\n\nimport { first, toArray, type ArrayOrItem } from 'ckeditor5/src/utils';\nimport { isListItemBlock, type ListElement } from './model';\n\nimport type { DocumentFragment, Element, Node } from 'ckeditor5/src/engine';\n\n/**\n * Document list blocks iterator.\n */\nexport default class ListWalker {\n\t/**\n\t * The start list item block element.\n\t */\n\tprivate _startElement: Node;\n\n\t/**\n\t * The reference indent. Initialized by the indent of the start block.\n\t */\n\tprivate _referenceIndent: number;\n\n\t/**\n\t * The iterating direction.\n\t */\n\tprivate _isForward: boolean;\n\n\t/**\n\t * Whether start block should be included in the result (if it's matching other criteria).\n\t */\n\tprivate _includeSelf: boolean;\n\n\t/**\n\t * Additional attributes that must be the same for each block.\n\t */\n\tprivate _sameAttributes: Array<string>;\n\n\t/**\n\t * Whether blocks with the same indent level as the start block should be included in the result.\n\t */\n\tprivate _sameIndent: boolean;\n\n\t/**\n\t * Whether blocks with a lower indent level than the start block should be included in the result.\n\t */\n\tprivate _lowerIndent: boolean;\n\n\t/**\n\t * Whether blocks with a higher indent level than the start block should be included in the result.\n\t */\n\tprivate _higherIndent: boolean;\n\n\t/**\n\t * Creates a document list iterator.\n\t *\n\t * @param startElement The start list item block element.\n\t * @param options.direction The iterating direction.\n\t * @param options.includeSelf Whether start block should be included in the result (if it's matching other criteria).\n\t * @param options.sameAttributes Additional attributes that must be the same for each block.\n\t * @param options.sameIndent Whether blocks with the same indent level as the start block should be included\n\t * in the result.\n\t * @param options.lowerIndent Whether blocks with a lower indent level than the start block should be included\n\t * in the result.\n\t * @param options.higherIndent Whether blocks with a higher indent level than the start block should be included\n\t * in the result.\n\t */\n\tconstructor(\n\t\tstartElement: Node,\n\t\toptions: {\n\t\t\tdirection?: 'forward' | 'backward';\n\t\t\tincludeSelf?: boolean;\n\t\t\tsameAttributes?: ArrayOrItem<string>;\n\t\t\tsameIndent?: boolean;\n\t\t\tlowerIndent?: boolean;\n\t\t\thigherIndent?: boolean;\n\t\t}\n\t) {\n\t\tthis._startElement = startElement;\n\t\tthis._referenceIndent = startElement.getAttribute( 'listIndent' ) as number;\n\t\tthis._isForward = options.direction == 'forward';\n\t\tthis._includeSelf = !!options.includeSelf;\n\t\tthis._sameAttributes = toArray( options.sameAttributes || [] );\n\t\tthis._sameIndent = !!options.sameIndent;\n\t\tthis._lowerIndent = !!options.lowerIndent;\n\t\tthis._higherIndent = !!options.higherIndent;\n\t}\n\n\t/**\n\t * Performs only first step of iteration and returns the result.\n\t *\n\t * @param startElement The start list item block element.\n\t * @param options.direction The iterating direction.\n\t * @param options.includeSelf Whether start block should be included in the result (if it's matching other criteria).\n\t * @param options.sameAttributes Additional attributes that must be the same for each block.\n\t * @param options.sameIndent Whether blocks with the same indent level as the start block should be included\n\t * in the result.\n\t * @param options.lowerIndent Whether blocks with a lower indent level than the start block should be included\n\t * in the result.\n\t * @param options.higherIndent Whether blocks with a higher indent level than the start block should be included\n\t * in the result.\n\t */\n\tpublic static first(\n\t\tstartElement: Node,\n\t\toptions: {\n\t\t\tdirection?: 'forward' | 'backward';\n\t\t\tincludeSelf?: boolean;\n\t\t\tsameAttributes?: ArrayOrItem<string>;\n\t\t\tsameIndent?: boolean;\n\t\t\tlowerIndent?: boolean;\n\t\t\thigherIndent?: boolean;\n\t\t}\n\t): ListElement | null {\n\t\tconst walker = new this( startElement, options );\n\t\tconst iterator = walker[ Symbol.iterator ]();\n\n\t\treturn first( iterator );\n\t}\n\n\t/**\n\t * Iterable interface.\n\t */\n\tpublic* [ Symbol.iterator ](): Iterator<ListElement> {\n\t\tconst nestedItems: Array<ListElement> = [];\n\n\t\tfor ( const { node } of iterateSiblingListBlocks( this._getStartNode(), this._isForward ? 'forward' : 'backward' ) ) {\n\t\t\tconst indent = node.getAttribute( 'listIndent' );\n\n\t\t\t// Leaving a nested list.\n\t\t\tif ( indent < this._referenceIndent ) {\n\t\t\t\t// Abort searching blocks.\n\t\t\t\tif ( !this._lowerIndent ) {\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\t// While searching for lower indents, update the reference indent to find another parent in the next step.\n\t\t\t\tthis._referenceIndent = indent;\n\t\t\t}\n\t\t\t// Entering a nested list.\n\t\t\telse if ( indent > this._referenceIndent ) {\n\t\t\t\t// Ignore nested blocks.\n\t\t\t\tif ( !this._higherIndent ) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\t// Collect nested blocks to verify if they are really nested, or it's a different item.\n\t\t\t\tif ( !this._isForward ) {\n\t\t\t\t\tnestedItems.push( node );\n\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\t\t\t}\n\t\t\t// Same indent level block.\n\t\t\telse {\n\t\t\t\t// Ignore same indent block.\n\t\t\t\tif ( !this._sameIndent ) {\n\t\t\t\t\t// While looking for nested blocks, stop iterating while encountering first same indent block.\n\t\t\t\t\tif ( this._higherIndent ) {\n\t\t\t\t\t\t// No more nested blocks so yield nested items.\n\t\t\t\t\t\tif ( nestedItems.length ) {\n\t\t\t\t\t\t\tyield* nestedItems;\n\t\t\t\t\t\t\tnestedItems.length = 0;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tbreak;\n\t\t\t\t\t}\n\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\t// Abort if item has any additionally specified attribute different.\n\t\t\t\tif ( this._sameAttributes.some( attr => node.getAttribute( attr ) !== this._startElement.getAttribute( attr ) ) ) {\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// There is another block for the same list item so the nested items were in the same list item.\n\t\t\tif ( nestedItems.length ) {\n\t\t\t\tyield* nestedItems;\n\t\t\t\tnestedItems.length = 0;\n\t\t\t}\n\n\t\t\tyield node;\n\t\t}\n\t}\n\n\t/**\n\t * Returns the model element to start iterating.\n\t */\n\tprivate _getStartNode() {\n\t\tif ( this._includeSelf ) {\n\t\t\treturn this._startElement;\n\t\t}\n\n\t\treturn this._isForward ?\n\t\t\tthis._startElement.nextSibling :\n\t\t\tthis._startElement.previousSibling;\n\t}\n}\n\n/**\n * Iterates sibling list blocks starting from the given node.\n *\n * @internal\n * @param node The model node.\n * @param direction Iteration direction.\n * @returns The object with `node` and `previous` {@link module:engine/model/element~Element blocks}.\n */\nexport function* iterateSiblingListBlocks(\n\tnode: Node | null,\n\tdirection: 'forward' | 'backward' = 'forward'\n): IterableIterator<ListIteratorValue> {\n\tconst isForward = direction == 'forward';\n\tlet previous = null;\n\n\twhile ( isListItemBlock( node ) ) {\n\t\tyield { node, previous };\n\n\t\tprevious = node;\n\t\tnode = isForward ? node.nextSibling : node.previousSibling;\n\t}\n}\n\n/**\n * The iterable protocol over the list elements.\n *\n * @internal\n */\nexport class ListBlocksIterable {\n\tprivate _listHead: Element;\n\n\t/**\n\t * @param listHead The head element of a list.\n\t */\n\tconstructor( listHead: Element ) {\n\t\tthis._listHead = listHead;\n\t}\n\n\t/**\n\t * List blocks iterator.\n\t *\n\t * Iterates over all blocks of a list.\n\t */\n\tpublic [ Symbol.iterator ](): Iterator<ListIteratorValue> {\n\t\treturn iterateSiblingListBlocks( this._listHead, 'forward' );\n\t}\n}\n\n/**\n * Object returned by `iterateSiblingListBlocks()` when traversing a list.\n *\n * @internal\n */\nexport interface ListIteratorValue {\n\n\t/**\n\t * The current list node.\n\t */\n\tnode: ListElement;\n\n\t/**\n\t * The previous list node.\n\t */\n\tprevious: ListElement | null;\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./documentlist.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./list.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/list/utils\n */\n\nimport type { Editor } from 'ckeditor5/src/core';\n\nimport {\n\tTreeWalker,\n\tgetFillerOffset,\n\ttype DowncastConversionApi,\n\ttype DowncastWriter,\n\ttype Element,\n\ttype Item,\n\ttype Model,\n\ttype Position,\n\ttype ViewContainerElement,\n\ttype ViewElement,\n\ttype ViewItem,\n\ttype ViewPosition\n} from 'ckeditor5/src/engine';\n\nimport { ButtonView, type ButtonExecuteEvent } from 'ckeditor5/src/ui';\n\n/**\n * Creates a list item {@link module:engine/view/containerelement~ContainerElement}.\n *\n * @param writer The writer instance.\n */\nexport function createViewListItemElement( writer: DowncastWriter ): ViewContainerElement {\n\tconst viewItem = writer.createContainerElement( 'li' );\n\n\tviewItem.getFillerOffset = getListItemFillerOffset;\n\n\treturn viewItem;\n}\n\n/**\n * Helper function that creates a `<ul><li></li></ul>` or (`<ol>`) structure out of the given `modelItem` model `listItem` element.\n * Then, it binds the created view list item (`<li>`) with the model `listItem` element.\n * The function then returns the created view list item (`<li>`).\n *\n * @param modelItem Model list item.\n * @param conversionApi Conversion interface.\n * @returns View list element.\n */\nexport function generateLiInUl( modelItem: Item, conversionApi: DowncastConversionApi ): ViewContainerElement {\n\tconst mapper = conversionApi.mapper;\n\tconst viewWriter = conversionApi.writer;\n\tconst listType = modelItem.getAttribute( 'listType' ) == 'numbered' ? 'ol' : 'ul';\n\tconst viewItem = createViewListItemElement( viewWriter );\n\n\tconst viewList = viewWriter.createContainerElement( listType, null );\n\n\tviewWriter.insert( viewWriter.createPositionAt( viewList, 0 ), viewItem );\n\n\tmapper.bindElements( modelItem as any, viewItem );\n\n\treturn viewItem;\n}\n\n/**\n * Helper function that inserts a view list at a correct place and merges it with its siblings.\n * It takes a model list item element (`modelItem`) and a corresponding view list item element (`injectedItem`). The view list item\n * should be in a view list element (`<ul>` or `<ol>`) and should be its only child.\n * See comments below to better understand the algorithm.\n *\n * @param modelItem Model list item.\n * @param injectedItem\n * @param conversionApi Conversion interface.\n * @param model The model instance.\n */\nexport function injectViewList(\n\tmodelItem: Element,\n\tinjectedItem: ViewContainerElement,\n\tconversionApi: DowncastConversionApi,\n\tmodel: Model\n): void {\n\tconst injectedList = injectedItem.parent as ViewElement;\n\tconst mapper = conversionApi.mapper;\n\tconst viewWriter = conversionApi.writer;\n\n\t// The position where the view list will be inserted.\n\tlet insertPosition = mapper.toViewPosition( model.createPositionBefore( modelItem ) );\n\n\t// 1. Find the previous list item that has the same or smaller indent. Basically we are looking for the first model item\n\t// that is a \"parent\" or \"sibling\" of the injected model item.\n\t// If there is no such list item, it means that the injected list item is the first item in \"its list\".\n\tconst refItem = getSiblingListItem( modelItem.previousSibling, {\n\t\tsameIndent: true,\n\t\tsmallerIndent: true,\n\t\tlistIndent: modelItem.getAttribute( 'listIndent' ) as number\n\t} );\n\tconst prevItem = modelItem.previousSibling as Element | null;\n\n\tif ( refItem && refItem.getAttribute( 'listIndent' ) == modelItem.getAttribute( 'listIndent' ) ) {\n\t\t// There is a list item with the same indent - we found the same-level sibling.\n\t\t// Break the list after it. The inserted view item will be added in the broken space.\n\t\tconst viewItem = mapper.toViewElement( refItem )!;\n\t\tinsertPosition = viewWriter.breakContainer( viewWriter.createPositionAfter( viewItem ) );\n\t} else {\n\t\t// There is no list item with the same indent. Check the previous model item.\n\t\tif ( prevItem && prevItem.name == 'listItem' ) {\n\t\t\t// If it is a list item, it has to have a lower indent.\n\t\t\t// It means that the inserted item should be added to it as its nested item.\n\t\t\tinsertPosition = mapper.toViewPosition( model.createPositionAt( prevItem, 'end' ) );\n\n\t\t\t// There could be some not mapped elements (eg. span in to-do list) but we need to insert\n\t\t\t// a nested list directly inside the li element.\n\t\t\tconst mappedViewAncestor = mapper.findMappedViewAncestor( insertPosition );\n\t\t\tconst nestedList = findNestedList( mappedViewAncestor );\n\n\t\t\t// If there already is some nested list, then use it's position.\n\t\t\tif ( nestedList ) {\n\t\t\t\tinsertPosition = viewWriter.createPositionBefore( nestedList );\n\t\t\t} else {\n\t\t\t\t// Else just put new list on the end of list item content.\n\t\t\t\tinsertPosition = viewWriter.createPositionAt( mappedViewAncestor, 'end' );\n\t\t\t}\n\t\t} else {\n\t\t\t// The previous item is not a list item (or does not exist at all).\n\t\t\t// Just map the position and insert the view item at the mapped position.\n\t\t\tinsertPosition = mapper.toViewPosition( model.createPositionBefore( modelItem ) );\n\t\t}\n\t}\n\n\tinsertPosition = positionAfterUiElements( insertPosition );\n\n\t// Insert the view item.\n\tviewWriter.insert( insertPosition, injectedList );\n\n\t// 2. Handle possible children of the injected model item.\n\tif ( prevItem && prevItem.name == 'listItem' ) {\n\t\tconst prevView = mapper.toViewElement( prevItem )!;\n\n\t\tconst walkerBoundaries = viewWriter.createRange( viewWriter.createPositionAt( prevView, 0 ), insertPosition );\n\t\tconst walker = walkerBoundaries.getWalker( { ignoreElementEnd: true } );\n\n\t\tfor ( const value of walker ) {\n\t\t\tif ( value.item.is( 'element', 'li' ) ) {\n\t\t\t\tconst breakPosition = viewWriter.breakContainer( viewWriter.createPositionBefore( value.item ) );\n\t\t\t\tconst viewList = value.item.parent as ViewElement;\n\n\t\t\t\tconst targetPosition = viewWriter.createPositionAt( injectedItem, 'end' );\n\t\t\t\tmergeViewLists( viewWriter, targetPosition.nodeBefore!, targetPosition.nodeAfter! );\n\t\t\t\tviewWriter.move( viewWriter.createRangeOn( viewList ), targetPosition );\n\n\t\t\t\t// This is bad, but those lists will be removed soon anyway.\n\t\t\t\t( walker as any )._position = breakPosition;\n\t\t\t}\n\t\t}\n\t} else {\n\t\tconst nextViewList = injectedList.nextSibling;\n\n\t\tif ( nextViewList && ( nextViewList.is( 'element', 'ul' ) || nextViewList.is( 'element', 'ol' ) ) ) {\n\t\t\tlet lastSubChild = null;\n\n\t\t\tfor ( const child of nextViewList.getChildren() as Iterable<ViewElement> ) {\n\t\t\t\tconst modelChild = mapper.toModelElement( child );\n\n\t\t\t\tif (\n\t\t\t\t\tmodelChild &&\n\t\t\t\t\t( modelChild.getAttribute( 'listIndent' ) as number ) > ( modelItem.getAttribute( 'listIndent' ) as number )\n\t\t\t\t) {\n\t\t\t\t\tlastSubChild = child;\n\t\t\t\t} else {\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif ( lastSubChild ) {\n\t\t\t\tviewWriter.breakContainer( viewWriter.createPositionAfter( lastSubChild ) );\n\t\t\t\tviewWriter.move(\n\t\t\t\t\tviewWriter.createRangeOn( lastSubChild.parent as any ),\n\t\t\t\t\tviewWriter.createPositionAt( injectedItem, 'end' )\n\t\t\t\t);\n\t\t\t}\n\t\t}\n\t}\n\n\t// Merge the inserted view list with its possible neighbor lists.\n\tmergeViewLists( viewWriter, injectedList, injectedList.nextSibling! );\n\tmergeViewLists( viewWriter, injectedList.previousSibling!, injectedList );\n}\n\n/**\n * Helper function that takes two parameters that are expected to be view list elements, and merges them.\n * The merge happens only if both parameters are list elements of the same type (the same element name and the same class attributes).\n *\n * @param viewWriter The writer instance.\n * @param firstList The first element to compare.\n * @param secondList The second element to compare.\n * @returns The position after merge or `null` when there was no merge.\n */\nexport function mergeViewLists(\n\tviewWriter: DowncastWriter,\n\tfirstList: ViewItem,\n\tsecondList: ViewItem\n): ViewPosition | null;\n\nexport function mergeViewLists(\n\tviewWriter: DowncastWriter,\n\tfirstList: any,\n\tsecondList: any\n): ViewPosition | null {\n\t// Check if two lists are going to be merged.\n\tif ( !firstList || !secondList || ( firstList.name != 'ul' && firstList.name != 'ol' ) ) {\n\t\treturn null;\n\t}\n\n\t// Both parameters are list elements, so compare types now.\n\tif ( firstList.name != secondList.name || firstList.getAttribute( 'class' ) !== secondList.getAttribute( 'class' ) ) {\n\t\treturn null;\n\t}\n\n\treturn viewWriter.mergeContainers( viewWriter.createPositionAfter( firstList ) );\n}\n\n/**\n * Helper function that for a given `view.Position`, returns a `view.Position` that is after all `view.UIElement`s that\n * are after the given position.\n *\n * For example:\n * `<container:p>foo^<ui:span></ui:span><ui:span></ui:span>bar</container:p>`\n * For position ^, the position before \"bar\" will be returned.\n *\n */\nexport function positionAfterUiElements( viewPosition: ViewPosition ): ViewPosition {\n\treturn viewPosition.getLastMatchingPosition( value => value.item.is( 'uiElement' ) );\n}\n\n/**\n * Helper function that searches for a previous list item sibling of a given model item that meets the given criteria\n * passed by the options object.\n *\n * @param options Search criteria.\n * @param options.sameIndent Whether the sought sibling should have the same indentation.\n * @param options.smallerIndent Whether the sought sibling should have a smaller indentation.\n * @param options.listIndent The reference indentation.\n * @param options.direction Walking direction.\n */\nexport function getSiblingListItem(\n\tmodelItem: Item | null,\n\toptions: {\n\t\tsameIndent?: boolean;\n\t\tsmallerIndent?: boolean;\n\t\tlistIndent?: number;\n\t\tdirection?: 'forward' | 'backward';\n\t}\n): Element | null {\n\tconst sameIndent = !!options.sameIndent;\n\tconst smallerIndent = !!options.smallerIndent;\n\tconst indent = options.listIndent;\n\n\tlet item: any = modelItem;\n\n\twhile ( item && item.name == 'listItem' ) {\n\t\tconst itemIndent = item.getAttribute( 'listIndent' ) as number;\n\n\t\tif ( ( sameIndent && indent == itemIndent ) || ( smallerIndent && indent as number > itemIndent ) ) {\n\t\t\treturn item;\n\t\t}\n\n\t\tif ( options.direction === 'forward' ) {\n\t\t\titem = item.nextSibling;\n\t\t} else {\n\t\t\titem = item.previousSibling;\n\t\t}\n\t}\n\n\treturn null;\n}\n\n/**\n * Helper method for creating a UI button and linking it with an appropriate command.\n *\n * @internal\n * @param editor The editor instance to which the UI component will be added.\n * @param commandName The name of the command.\n * @param label The button label.\n * @param icon The source of the icon.\n */\nexport function createUIComponent(\n\teditor: Editor,\n\tcommandName: 'bulletedList' | 'numberedList' | 'todoList',\n\tlabel: string,\n\ticon: string\n): void {\n\teditor.ui.componentFactory.add( commandName, locale => {\n\t\tconst command = editor.commands.get( commandName )!;\n\t\tconst buttonView = new ButtonView( locale );\n\n\t\tbuttonView.set( {\n\t\t\tlabel,\n\t\t\ticon,\n\t\t\ttooltip: true,\n\t\t\tisToggleable: true\n\t\t} );\n\n\t\t// Bind button model to command.\n\t\tbuttonView.bind( 'isOn', 'isEnabled' ).to( command, 'value', 'isEnabled' );\n\n\t\t// Execute command.\n\t\tbuttonView.on<ButtonExecuteEvent>( 'execute', () => {\n\t\t\teditor.execute( commandName );\n\t\t\teditor.editing.view.focus();\n\t\t} );\n\n\t\treturn buttonView;\n\t} );\n}\n\n/**\n * Returns a first list view element that is direct child of the given view element.\n */\nexport function findNestedList( viewElement: ViewElement ): ViewElement | null {\n\tfor ( const node of ( viewElement.getChildren() as Iterable<ViewElement> ) ) {\n\t\tif ( node.name == 'ul' || node.name == 'ol' ) {\n\t\t\treturn node;\n\t\t}\n\t}\n\n\treturn null;\n}\n\n/**\n * Returns an array with all `listItem` elements that represent the same list.\n *\n * It means that values of `listIndent`, `listType`, `listStyle`, `listReversed` and `listStart` for all items are equal.\n *\n * Additionally, if the `position` is inside a list item, that list item will be returned as well.\n *\n * @param position Starting position.\n * @param direction Walking direction.\n */\nexport function getSiblingNodes( position: Position, direction: 'forward' | 'backward' ): Array<Element> {\n\tconst items: Array<Element> = [];\n\tconst listItem = position.parent as Element;\n\tconst walkerOptions = {\n\t\tignoreElementEnd: false,\n\t\tstartPosition: position,\n\t\tshallow: true,\n\t\tdirection\n\t};\n\tconst limitIndent = listItem.getAttribute( 'listIndent' ) as number;\n\tconst nodes = [ ...new TreeWalker( walkerOptions ) ]\n\t\t.filter( value => value.item.is( 'element' ) )\n\t\t.map( value => value.item );\n\n\tfor ( const element of nodes ) {\n\t\t// If found something else than `listItem`, we're out of the list scope.\n\t\tif ( !element.is( 'element', 'listItem' ) ) {\n\t\t\tbreak;\n\t\t}\n\n\t\t// If current parsed item has lower indent that element that the element that was a starting point,\n\t\t// it means we left a nested list. Abort searching items.\n\t\t//\n\t\t// ■ List item 1. [listIndent=0]\n\t\t// ○ List item 2.[] [listIndent=1], limitIndent = 1,\n\t\t// ○ List item 3. [listIndent=1]\n\t\t// ■ List item 4. [listIndent=0]\n\t\t//\n\t\t// Abort searching when leave nested list.\n\t\tif ( ( element.getAttribute( 'listIndent' ) as number ) < limitIndent ) {\n\t\t\tbreak;\n\t\t}\n\n\t\t// ■ List item 1.[] [listIndent=0] limitIndent = 0,\n\t\t// ○ List item 2. [listIndent=1]\n\t\t// ○ List item 3. [listIndent=1]\n\t\t// ■ List item 4. [listIndent=0]\n\t\t//\n\t\t// Ignore nested lists.\n\t\tif ( ( element.getAttribute( 'listIndent' ) as number ) > limitIndent ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\t// ■ List item 1.[] [listType=bulleted]\n\t\t// 1. List item 2. [listType=numbered]\n\t\t// 2.List item 3. [listType=numbered]\n\t\t//\n\t\t// Abort searching when found a different kind of a list.\n\t\tif ( element.getAttribute( 'listType' ) !== listItem.getAttribute( 'listType' ) ) {\n\t\t\tbreak;\n\t\t}\n\n\t\t// ■ List item 1.[] [listType=bulleted]\n\t\t// ■ List item 2. [listType=bulleted]\n\t\t// ○ List item 3. [listType=bulleted]\n\t\t// ○ List item 4. [listType=bulleted]\n\t\t//\n\t\t// Abort searching when found a different list style,\n\t\tif ( element.getAttribute( 'listStyle' ) !== listItem.getAttribute( 'listStyle' ) ) {\n\t\t\tbreak;\n\t\t}\n\n\t\t// ... different direction\n\t\tif ( element.getAttribute( 'listReversed' ) !== listItem.getAttribute( 'listReversed' ) ) {\n\t\t\tbreak;\n\t\t}\n\n\t\t// ... and different start index\n\t\tif ( element.getAttribute( 'listStart' ) !== listItem.getAttribute( 'listStart' ) ) {\n\t\t\tbreak;\n\t\t}\n\n\t\tif ( direction === 'backward' ) {\n\t\t\titems.unshift( element );\n\t\t} else {\n\t\t\titems.push( element );\n\t\t}\n\t}\n\n\treturn items;\n}\n\n/**\n * Returns an array with all `listItem` elements in the model selection.\n *\n * It returns all the items even if only a part of the list is selected, including items that belong to nested lists.\n * If no list is selected, it returns an empty array.\n * The order of the elements is not specified.\n *\n * @internal\n */\nexport function getSelectedListItems( model: Model ): Array<Element> {\n\tconst document = model.document;\n\n\t// For all selected blocks find all list items that are being selected\n\t// and update the `listStyle` attribute in those lists.\n\tlet listItems = [ ...document.selection.getSelectedBlocks() ]\n\t\t.filter( element => element.is( 'element', 'listItem' ) )\n\t\t.map( element => {\n\t\t\tconst position = model.change( writer => writer.createPositionAt( element, 0 ) );\n\n\t\t\treturn [\n\t\t\t\t...getSiblingNodes( position, 'backward' ),\n\t\t\t\t...getSiblingNodes( position, 'forward' )\n\t\t\t];\n\t\t} )\n\t\t.flat();\n\n\t// Since `getSelectedBlocks()` can return items that belong to the same list, and\n\t// `getSiblingNodes()` returns the entire list, we need to remove duplicated items.\n\tlistItems = [ ...new Set( listItems ) ];\n\n\treturn listItems;\n}\n\nconst BULLETED_LIST_STYLE_TYPES = [ 'disc', 'circle', 'square' ];\n\n// There's a lot of them (https://www.w3.org/TR/css-counter-styles-3/#typedef-counter-style).\n// Let's support only those that can be selected by ListPropertiesUI.\nconst NUMBERED_LIST_STYLE_TYPES = [\n\t'decimal',\n\t'decimal-leading-zero',\n\t'lower-roman',\n\t'upper-roman',\n\t'lower-latin',\n\t'upper-latin'\n];\n\n/**\n * Checks whether the given list-style-type is supported by numbered or bulleted list.\n */\nexport function getListTypeFromListStyleType( listStyleType: string ): 'bulleted' | 'numbered' | null {\n\tif ( BULLETED_LIST_STYLE_TYPES.includes( listStyleType ) ) {\n\t\treturn 'bulleted';\n\t}\n\n\tif ( NUMBERED_LIST_STYLE_TYPES.includes( listStyleType ) ) {\n\t\treturn 'numbered';\n\t}\n\n\treturn null;\n}\n\n/**\n * Implementation of getFillerOffset for view list item element.\n *\n * @returns Block filler offset or `null` if block filler is not needed.\n */\nfunction getListItemFillerOffset( this: any ): number | null {\n\tconst hasOnlyLists = !this.isEmpty && ( this.getChild( 0 ).name == 'ul' || this.getChild( 0 ).name == 'ol' );\n\n\tif ( this.isEmpty || hasOnlyLists ) {\n\t\treturn 0;\n\t}\n\n\treturn getFillerOffset.call( this );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/list/listui\n */\n\nimport { createUIComponent } from './utils';\n\nimport numberedListIcon from '../../theme/icons/numberedlist.svg';\nimport bulletedListIcon from '../../theme/icons/bulletedlist.svg';\n\nimport { Plugin } from 'ckeditor5/src/core';\n\n/**\n * The list UI feature. It introduces the `'numberedList'` and `'bulletedList'` buttons that\n * allow to convert paragraphs to and from list items and indent or outdent them.\n */\nexport default class ListUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ListUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst t = this.editor.t;\n\n\t\t// Create two buttons and link them with numberedList and bulletedList commands.\n\t\tcreateUIComponent( this.editor, 'numberedList', t( 'Numbered List' ), numberedListIcon );\n\t\tcreateUIComponent( this.editor, 'bulletedList', t( 'Bulleted List' ), bulletedListIcon );\n\t}\n}\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M7 5.75c0 .414.336.75.75.75h9.5a.75.75 0 1 0 0-1.5h-9.5a.75.75 0 0 0-.75.75zM3.5 3v5H2V3.7H1v-1h2.5V3zM.343 17.857l2.59-3.257H2.92a.6.6 0 1 0-1.04 0H.302a2 2 0 1 1 3.995 0h-.001c-.048.405-.16.734-.333.988-.175.254-.59.692-1.244 1.312H4.3v1h-4l.043-.043zM7 14.75a.75.75 0 0 1 .75-.75h9.5a.75.75 0 1 1 0 1.5h-9.5a.75.75 0 0 1-.75-.75z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M7 5.75c0 .414.336.75.75.75h9.5a.75.75 0 1 0 0-1.5h-9.5a.75.75 0 0 0-.75.75zm-6 0C1 4.784 1.777 4 2.75 4c.966 0 1.75.777 1.75 1.75 0 .966-.777 1.75-1.75 1.75C1.784 7.5 1 6.723 1 5.75zm6 9c0 .414.336.75.75.75h9.5a.75.75 0 1 0 0-1.5h-9.5a.75.75 0 0 0-.75.75zm-6 0c0-.966.777-1.75 1.75-1.75.966 0 1.75.777 1.75 1.75 0 .966-.777 1.75-1.75 1.75-.966 0-1.75-.777-1.75-1.75z\\\"/></svg>\";","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n* @module list/documentlistproperties/utils/style\n*/\n\nconst LIST_STYLE_TO_LIST_TYPE: Record<string, 'bulleted' | 'numbered' | undefined> = {};\nconst LIST_STYLE_TO_TYPE_ATTRIBUTE: Record<string, string | null | undefined> = {};\nconst TYPE_ATTRIBUTE_TO_LIST_STYLE: Record<string, string | undefined> = {};\n\nconst LIST_STYLE_TYPES: Array<{ listStyle: string; typeAttribute: string | null; listType: 'bulleted' | 'numbered' }> = [\n\t{ listStyle: 'disc', typeAttribute: 'disc', listType: 'bulleted' },\n\t{ listStyle: 'circle', typeAttribute: 'circle', listType: 'bulleted' },\n\t{ listStyle: 'square', typeAttribute: 'square', listType: 'bulleted' },\n\t{ listStyle: 'decimal', typeAttribute: '1', listType: 'numbered' },\n\t{ listStyle: 'decimal-leading-zero', typeAttribute: null, listType: 'numbered' },\n\t{ listStyle: 'lower-roman', typeAttribute: 'i', listType: 'numbered' },\n\t{ listStyle: 'upper-roman', typeAttribute: 'I', listType: 'numbered' },\n\t{ listStyle: 'lower-alpha', typeAttribute: 'a', listType: 'numbered' },\n\t{ listStyle: 'upper-alpha', typeAttribute: 'A', listType: 'numbered' },\n\t{ listStyle: 'lower-latin', typeAttribute: 'a', listType: 'numbered' },\n\t{ listStyle: 'upper-latin', typeAttribute: 'A', listType: 'numbered' }\n];\n\nfor ( const { listStyle, typeAttribute, listType } of LIST_STYLE_TYPES ) {\n\tLIST_STYLE_TO_LIST_TYPE[ listStyle ] = listType;\n\tLIST_STYLE_TO_TYPE_ATTRIBUTE[ listStyle ] = typeAttribute;\n\n\tif ( typeAttribute ) {\n\t\tTYPE_ATTRIBUTE_TO_LIST_STYLE[ typeAttribute ] = listStyle;\n\t}\n}\n\n/**\n * Gets all the style types supported by given list type.\n */\nexport function getAllSupportedStyleTypes(): Array<string> {\n\treturn LIST_STYLE_TYPES.map( x => x.listStyle );\n}\n\n/**\n * Checks whether the given list-style-type is supported by numbered or bulleted list.\n */\nexport function getListTypeFromListStyleType( listStyleType: string ): 'bulleted' | 'numbered' | null {\n\treturn LIST_STYLE_TO_LIST_TYPE[ listStyleType ] || null;\n}\n\n/**\n * Converts `type` attribute of `<ul>` or `<ol>` elements to `list-style-type` equivalent.\n */\nexport function getListStyleTypeFromTypeAttribute( value: string ): string | null {\n\treturn TYPE_ATTRIBUTE_TO_LIST_STYLE[ value ] || null;\n}\n\n/**\n * Converts `list-style-type` style to `type` attribute of `<ul>` or `<ol>` elements.\n */\nexport function getTypeAttributeFromListStyleType( value: string ): string | null {\n\treturn LIST_STYLE_TO_TYPE_ATTRIBUTE[ value ] || null;\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./collapsible.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./listproperties.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./liststyles.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/list/indentcommand\n */\n\nimport type { Element } from 'ckeditor5/src/engine';\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport { first } from 'ckeditor5/src/utils';\n\n/**\n * The list indent command. It is used by the {@link module:list/list~List list feature}.\n */\nexport default class IndentCommand extends Command {\n\t/**\n\t * Determines by how much the command will change the list item's indent attribute.\n\t */\n\tprivate readonly _indentBy: 1 | -1;\n\n\t/**\n\t * Creates an instance of the command.\n\t *\n\t * @param editor The editor instance.\n\t * @param indentDirection The direction of indent. If it is equal to `backward`, the command will outdent a list item.\n\t */\n\tconstructor( editor: Editor, indentDirection: 'forward' | 'backward' ) {\n\t\tsuper( editor );\n\n\t\tthis._indentBy = indentDirection == 'forward' ? 1 : -1;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tthis.isEnabled = this._checkEnabled();\n\t}\n\n\t/**\n\t * Indents or outdents (depending on the {@link #constructor}'s `indentDirection` parameter) selected list items.\n\t *\n\t * @fires execute\n\t */\n\tpublic override execute(): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\t\tlet itemsToChange = Array.from( doc.selection.getSelectedBlocks() );\n\n\t\tmodel.change( writer => {\n\t\t\tconst lastItem = itemsToChange[ itemsToChange.length - 1 ];\n\n\t\t\t// Indenting a list item should also indent all the items that are already sub-items of indented item.\n\t\t\tlet next = lastItem.nextSibling as Element | null;\n\n\t\t\t// Check all items after last indented item, as long as their indent is bigger than indent of that item.\n\t\t\twhile (\n\t\t\t\tnext && next.name == 'listItem' &&\n\t\t\t\t( next.getAttribute( 'listIndent' ) as number ) > ( lastItem.getAttribute( 'listIndent' ) as number )\n\t\t\t) {\n\t\t\t\titemsToChange.push( next );\n\n\t\t\t\tnext = next.nextSibling as Element | null;\n\t\t\t}\n\n\t\t\t// We need to be sure to keep model in correct state after each small change, because converters\n\t\t\t// bases on that state and assumes that model is correct.\n\t\t\t// Because of that, if the command outdents items, we will outdent them starting from the last item, as\n\t\t\t// it is safer.\n\t\t\tif ( this._indentBy < 0 ) {\n\t\t\t\titemsToChange = itemsToChange.reverse();\n\t\t\t}\n\n\t\t\tfor ( const item of itemsToChange ) {\n\t\t\t\tconst indent = ( item.getAttribute( 'listIndent' ) as number ) + this._indentBy;\n\n\t\t\t\t// If indent is lower than 0, it means that the item got outdented when it was not indented.\n\t\t\t\t// This means that we need to convert that list item to paragraph.\n\t\t\t\tif ( indent < 0 ) {\n\t\t\t\t\t// To keep the model as correct as possible, first rename listItem, then remove attributes,\n\t\t\t\t\t// as listItem without attributes is very incorrect and will cause problems in converters.\n\t\t\t\t\t// No need to remove attributes, will be removed by post fixer.\n\t\t\t\t\twriter.rename( item, 'paragraph' );\n\t\t\t\t}\n\t\t\t\t// If indent is >= 0, change the attribute value.\n\t\t\t\telse {\n\t\t\t\t\twriter.setAttribute( 'listIndent', indent, item );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// It allows to execute an action after executing the `~IndentCommand#execute` method, for example adjusting\n\t\t\t// attributes of changed list items.\n\t\t\tthis.fire( '_executeCleanup', itemsToChange );\n\t\t} );\n\t}\n\n\t/**\n\t * Checks whether the command can be enabled in the current context.\n\t *\n\t * @returns Whether the command should be enabled.\n\t */\n\tprivate _checkEnabled() {\n\t\t// Check whether any of position's ancestor is a list item.\n\t\tconst listItem = first( this.editor.model.document.selection.getSelectedBlocks() );\n\n\t\t// If selection is not in a list item, the command is disabled.\n\t\tif ( !listItem || !listItem.is( 'element', 'listItem' ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( this._indentBy > 0 ) {\n\t\t\t// Cannot indent first item in it's list. Check if before `listItem` is a list item that is in same list.\n\t\t\t// To be in the same list, the item has to have same attributes and cannot be \"split\" by an item with lower indent.\n\t\t\tconst indent = listItem.getAttribute( 'listIndent' ) as number;\n\t\t\tconst type = listItem.getAttribute( 'listType' ) as string;\n\n\t\t\tlet prev = listItem.previousSibling;\n\n\t\t\twhile ( prev && prev.is( 'element', 'listItem' ) && ( prev.getAttribute( 'listIndent' ) as number ) >= indent ) {\n\t\t\t\tif ( prev.getAttribute( 'listIndent' ) == indent ) {\n\t\t\t\t\t// The item is on the same level.\n\t\t\t\t\t// If it has same type, it means that we found a preceding sibling from the same list.\n\t\t\t\t\t// If it does not have same type, it means that `listItem` is on different list (this can happen only\n\t\t\t\t\t// on top level lists, though).\n\t\t\t\t\treturn prev.getAttribute( 'listType' ) == type;\n\t\t\t\t}\n\n\t\t\t\tprev = prev.previousSibling;\n\t\t\t}\n\n\t\t\t// Could not find similar list item, this means that `listItem` is first in its list.\n\t\t\treturn false;\n\t\t}\n\n\t\t// If we are outdenting it is enough to be in list item. Every list item can always be outdented.\n\t\treturn true;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/list/listcommand\n */\n\nimport type { Element, Node, Schema } from 'ckeditor5/src/engine';\nimport { Command, type Editor } from 'ckeditor5/src/core';\n\nimport { first } from 'ckeditor5/src/utils';\n\n/**\n * The list command. It is used by the {@link module:list/list~List list feature}.\n */\nexport default class ListCommand extends Command {\n\t/**\n\t * The type of the list created by the command.\n\t */\n\tpublic readonly type: 'numbered' | 'bulleted' | 'todo';\n\n\t/**\n\t * A flag indicating whether the command is active, which means that the selection starts in a list of the same type.\n\t *\n\t * @readonly\n\t */\n\tdeclare public value: boolean;\n\n\t/**\n\t * Creates an instance of the command.\n\t *\n\t * @param editor The editor instance.\n\t * @param type List type that will be handled by this command.\n\t */\n\tconstructor( editor: Editor, type: 'numbered' | 'bulleted' | 'todo' ) {\n\t\tsuper( editor );\n\n\t\tthis.type = type;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tthis.value = this._getValue();\n\t\tthis.isEnabled = this._checkEnabled();\n\t}\n\n\t/**\n\t * Executes the list command.\n\t *\n\t * @fires execute\n\t * @param options Command options.\n\t * @param options.forceValue If set, it will force the command behavior. If `true`, the command will try to convert the\n\t * selected items and potentially the neighbor elements to the proper list items. If set to `false`, it will convert selected elements\n\t * to paragraphs. If not set, the command will toggle selected elements to list items or paragraphs, depending on the selection.\n\t */\n\tpublic override execute( options: { forceValue?: boolean } = {} ): void {\n\t\tconst model = this.editor.model;\n\t\tconst document = model.document;\n\t\tconst blocks = Array.from( document.selection.getSelectedBlocks() )\n\t\t\t.filter( block => checkCanBecomeListItem( block, model.schema ) );\n\n\t\t// Whether we are turning off some items.\n\t\tconst turnOff = options.forceValue !== undefined ? !options.forceValue : this.value;\n\n\t\t// If we are turning off items, we are going to rename them to paragraphs.\n\n\t\tmodel.change( writer => {\n\t\t\t// If part of a list got turned off, we need to handle (outdent) all of sub-items of the last turned-off item.\n\t\t\t// To be sure that model is all the time in a good state, we first fix items below turned-off item.\n\t\t\tif ( turnOff ) {\n\t\t\t\t// Start from the model item that is just after the last turned-off item.\n\t\t\t\tlet next = blocks[ blocks.length - 1 ].nextSibling as Element;\n\t\t\t\tlet currentIndent = Number.POSITIVE_INFINITY;\n\t\t\t\tlet changes = [];\n\n\t\t\t\t// Correct indent of all items after the last turned off item.\n\t\t\t\t// Rules that should be followed:\n\t\t\t\t// 1. All direct sub-items of turned-off item should become indent 0, because the first item after it\n\t\t\t\t// will be the first item of a new list. Other items are at the same level, so should have same 0 index.\n\t\t\t\t// 2. All items with indent lower than indent of turned-off item should become indent 0, because they\n\t\t\t\t// should not end up as a child of any of list items that they were not children of before.\n\t\t\t\t// 3. All other items should have their indent changed relatively to it's parent.\n\t\t\t\t//\n\t\t\t\t// For example:\n\t\t\t\t// 1 * --------\n\t\t\t\t// 2 * --------\n\t\t\t\t// 3 * --------\t\t\t<-- this is turned off.\n\t\t\t\t// 4 * --------\t\t<-- this has to become indent = 0, because it will be first item on a new list.\n\t\t\t\t// 5 * --------\t<-- this should be still be a child of item above, so indent = 1.\n\t\t\t\t// 6 * --------\t\t\t<-- this has to become indent = 0, because it should not be a child of any of items above.\n\t\t\t\t// 7 * --------\t\t<-- this should be still be a child of item above, so indent = 1.\n\t\t\t\t// 8 * --------\t\t\t\t<-- this has to become indent = 0.\n\t\t\t\t// 9 * --------\t\t\t<-- this should still be a child of item above, so indent = 1.\n\t\t\t\t// 10 * --------\t\t<-- this should still be a child of item above, so indent = 2.\n\t\t\t\t// 11 * --------\t\t<-- this should still be at the same level as item above, so indent = 2.\n\t\t\t\t// 12 * --------\t\t\t\t<-- this and all below are left unchanged.\n\t\t\t\t// 13 * --------\n\t\t\t\t// 14 * --------\n\t\t\t\t//\n\t\t\t\t// After turning off 3 the list becomes:\n\t\t\t\t//\n\t\t\t\t// 1 * --------\n\t\t\t\t// 2 * --------\n\t\t\t\t//\n\t\t\t\t// 3 --------\n\t\t\t\t//\n\t\t\t\t// 4 * --------\n\t\t\t\t// 5 * --------\n\t\t\t\t// 6 * --------\n\t\t\t\t// 7 * --------\n\t\t\t\t// 8 * --------\n\t\t\t\t// 9 * --------\n\t\t\t\t// 10 * --------\n\t\t\t\t// 11 * --------\n\t\t\t\t// 12 * --------\n\t\t\t\t// 13 * --------\n\t\t\t\t// 14 * --------\n\t\t\t\t//\n\t\t\t\t// Thanks to this algorithm no lists are mismatched and no items get unexpected children/parent, while\n\t\t\t\t// those parent-child connection which are possible to maintain are still maintained. It's worth noting\n\t\t\t\t// that this is the same effect that we would be get by multiple use of outdent command. However doing\n\t\t\t\t// it like this is much more efficient because it's less operation (less memory usage, easier OT) and\n\t\t\t\t// less conversion (faster).\n\t\t\t\twhile ( next && next.name == 'listItem' && next.getAttribute( 'listIndent' ) !== 0 ) {\n\t\t\t\t\t// Check each next list item, as long as its indent is bigger than 0.\n\t\t\t\t\t// If the indent is 0 we are not going to change anything anyway.\n\t\t\t\t\tconst indent = next.getAttribute( 'listIndent' ) as number;\n\n\t\t\t\t\t// We check if that's item indent is lower as current relative indent.\n\t\t\t\t\tif ( indent < currentIndent ) {\n\t\t\t\t\t\t// If it is, current relative indent becomes that indent.\n\t\t\t\t\t\tcurrentIndent = indent;\n\t\t\t\t\t}\n\n\t\t\t\t\t// Fix indent relatively to current relative indent.\n\t\t\t\t\t// Note, that if we just changed the current relative indent, the newIndent will be equal to 0.\n\t\t\t\t\tconst newIndent = indent - currentIndent;\n\n\t\t\t\t\t// Save the entry in changes array. We do not apply it at the moment, because we will need to\n\t\t\t\t\t// reverse the changes so the last item is changed first.\n\t\t\t\t\t// This is to keep model in correct state all the time.\n\t\t\t\t\tchanges.push( { element: next, listIndent: newIndent } );\n\n\t\t\t\t\t// Find next item.\n\t\t\t\t\tnext = next.nextSibling as Element;\n\t\t\t\t}\n\n\t\t\t\tchanges = changes.reverse();\n\n\t\t\t\tfor ( const item of changes ) {\n\t\t\t\t\twriter.setAttribute( 'listIndent', item.listIndent, item.element );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// If we are turning on, we might change some items that are already `listItem`s but with different type.\n\t\t\t// Changing one nested list item to other type should also trigger changing all its siblings so the\n\t\t\t// whole nested list is of the same type.\n\t\t\t// Example (assume changing to numbered list):\n\t\t\t// * ------\t\t\t\t<-- do not fix, top level item\n\t\t\t// * ------\t\t\t<-- fix, because latter list item of this item's list is changed\n\t\t\t// * ------\t\t<-- do not fix, item is not affected (different list)\n\t\t\t// * ------\t\t\t<-- fix, because latter list item of this item's list is changed\n\t\t\t// * ------\t\t<-- fix, because latter list item of this item's list is changed\n\t\t\t// * ---[--\t\t<-- already in selection\n\t\t\t// * ------\t\t\t<-- already in selection\n\t\t\t// * ------\t\t\t<-- already in selection\n\t\t\t// * ------\t\t\t\t<-- already in selection, but does not cause other list items to change because is top-level\n\t\t\t// * ---]--\t\t\t<-- already in selection\n\t\t\t// * ------\t\t\t<-- fix, because preceding list item of this item's list is changed\n\t\t\t// * ------\t\t<-- do not fix, item is not affected (different list)\n\t\t\t// * ------\t\t\t\t<-- do not fix, top level item\n\t\t\tif ( !turnOff ) {\n\t\t\t\t// Find lowest indent among selected items. This will be indicator what is the indent of\n\t\t\t\t// top-most list affected by the command.\n\t\t\t\tlet lowestIndent = Number.POSITIVE_INFINITY;\n\n\t\t\t\tfor ( const item of blocks ) {\n\t\t\t\t\tif ( item.is( 'element', 'listItem' ) && ( item.getAttribute( 'listIndent' ) as number ) < lowestIndent ) {\n\t\t\t\t\t\tlowestIndent = item.getAttribute( 'listIndent' ) as number;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Do not execute the fix for top-level lists.\n\t\t\t\tlowestIndent = lowestIndent === 0 ? 1 : lowestIndent;\n\n\t\t\t\t// Fix types of list items that are \"before\" the selected blocks.\n\t\t\t\t_fixType( blocks, true, lowestIndent );\n\n\t\t\t\t// Fix types of list items that are \"after\" the selected blocks.\n\t\t\t\t_fixType( blocks, false, lowestIndent );\n\t\t\t}\n\n\t\t\t// Phew! Now it will be easier :).\n\t\t\t// For each block element that was in the selection, we will either: turn it to list item,\n\t\t\t// turn it to paragraph, or change it's type. Or leave it as it is.\n\t\t\t// Do it in reverse as there might be multiple blocks (same as with changing indents).\n\t\t\tfor ( const element of blocks.reverse() ) {\n\t\t\t\tif ( turnOff && element.name == 'listItem' ) {\n\t\t\t\t\t// We are turning off and the element is a `listItem` - it should be converted to `paragraph`.\n\t\t\t\t\t// List item specific attributes are removed by post fixer.\n\t\t\t\t\twriter.rename( element, 'paragraph' );\n\t\t\t\t} else if ( !turnOff && element.name != 'listItem' ) {\n\t\t\t\t\t// We are turning on and the element is not a `listItem` - it should be converted to `listItem`.\n\t\t\t\t\t// The order of operations is important to keep model in correct state.\n\t\t\t\t\twriter.setAttributes( { listType: this.type, listIndent: 0 }, element );\n\t\t\t\t\twriter.rename( element, 'listItem' );\n\t\t\t\t} else if ( !turnOff && element.name == 'listItem' && element.getAttribute( 'listType' ) != this.type ) {\n\t\t\t\t\t// We are turning on and the element is a `listItem` but has different type - change it's type and\n\t\t\t\t\t// type of it's all siblings that have same indent.\n\t\t\t\t\twriter.setAttribute( 'listType', this.type, element );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t/**\n\t\t\t * Event fired by the {@link #execute} method.\n\t\t\t *\n\t\t\t * It allows to execute an action after executing the {@link ~ListCommand#execute} method, for example adjusting\n\t\t\t * attributes of changed blocks.\n\t\t\t *\n\t\t\t * @protected\n\t\t\t * @event _executeCleanup\n\t\t\t */\n\t\t\tthis.fire( '_executeCleanup', blocks );\n\t\t} );\n\t}\n\n\t/**\n\t * Checks the command's {@link #value}.\n\t *\n\t * @returns The current value.\n\t */\n\tprivate _getValue(): boolean {\n\t\t// Check whether closest `listItem` ancestor of the position has a correct type.\n\t\tconst listItem = first( this.editor.model.document.selection.getSelectedBlocks() );\n\n\t\treturn !!listItem && listItem.is( 'element', 'listItem' ) && listItem.getAttribute( 'listType' ) == this.type;\n\t}\n\n\t/**\n\t * Checks whether the command can be enabled in the current context.\n\t *\n\t * @returns Whether the command should be enabled.\n\t */\n\tprivate _checkEnabled() {\n\t\t// If command value is true it means that we are in list item, so the command should be enabled.\n\t\tif ( this.value ) {\n\t\t\treturn true;\n\t\t}\n\n\t\tconst selection = this.editor.model.document.selection;\n\t\tconst schema = this.editor.model.schema;\n\n\t\tconst firstBlock = first( selection.getSelectedBlocks() );\n\n\t\tif ( !firstBlock ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// Otherwise, check if list item can be inserted at the position start.\n\t\treturn checkCanBecomeListItem( firstBlock, schema );\n\t}\n}\n\n/**\n * Helper function used when one or more list item have their type changed. Fixes type of other list items\n * that are affected by the change (are in same lists) but are not directly in selection. The function got extracted\n * not to duplicated code, as same fix has to be performed before and after selection.\n *\n * @param blocks Blocks that are in selection.\n * @param isBackward Specified whether fix will be applied for blocks before first selected block (`true`)\n * or blocks after last selected block (`false`).\n * @param lowestIndent Lowest indent among selected blocks.\n */\nfunction _fixType(\n\tblocks: Array<Node>,\n\tisBackward: boolean,\n\tlowestIndent: number\n) {\n\t// We need to check previous sibling of first changed item and next siblings of last changed item.\n\tconst startingItem = isBackward ? blocks[ 0 ] : blocks[ blocks.length - 1 ];\n\n\tif ( startingItem.is( 'element', 'listItem' ) ) {\n\t\tlet item = startingItem[ isBackward ? 'previousSibling' : 'nextSibling' ];\n\t\t// During processing items, keeps the lowest indent of already processed items.\n\t\t// This saves us from changing too many items.\n\t\t// Following example is for going forward as it is easier to read, however same applies to going backward.\n\t\t// * ------\n\t\t// * ------\n\t\t// * --[---\n\t\t// * ------\t\t<-- `lowestIndent` should be 1\n\t\t// * --]---\t\t<-- `startingItem`, `currentIndent` = 2, `lowestIndent` == 1\n\t\t// * ------\t\t<-- should be fixed, `indent` == 2 == `currentIndent`\n\t\t// * ------\t\t<-- should be fixed, set `currentIndent` to 1, `indent` == 1 == `currentIndent`\n\t\t// * ------\t\t<-- should not be fixed, item is in different list, `indent` = 2, `indent` != `currentIndent`\n\t\t// * ------\t\t<-- should be fixed, `indent` == 1 == `currentIndent`\n\t\t// * ------\t\t\t<-- break loop (`indent` < `lowestIndent`)\n\t\tlet currentIndent = startingItem.getAttribute( 'listIndent' ) as number;\n\n\t\t// Look back until a list item with indent lower than reference `lowestIndent`.\n\t\t// That would be the parent of nested sublist which contains item having `lowestIndent`.\n\t\twhile ( item && item.is( 'element', 'listItem' ) && ( item.getAttribute( 'listIndent' ) as number ) >= lowestIndent ) {\n\t\t\tif ( currentIndent > ( item.getAttribute( 'listIndent' ) as number ) ) {\n\t\t\t\tcurrentIndent = item.getAttribute( 'listIndent' ) as number;\n\t\t\t}\n\n\t\t\t// Found an item that is in the same nested sublist.\n\t\t\tif ( item.getAttribute( 'listIndent' ) == currentIndent ) {\n\t\t\t\t// Just add the item to selected blocks like it was selected by the user.\n\t\t\t\tblocks[ isBackward ? 'unshift' : 'push' ]( item );\n\t\t\t}\n\n\t\t\titem = item[ isBackward ? 'previousSibling' : 'nextSibling' ];\n\t\t}\n\t}\n}\n\n/**\n * Checks whether the given block can be replaced by a listItem.\n *\n * @param block A block to be tested.\n * @param schema The schema of the document.\n */\nfunction checkCanBecomeListItem( block: Element, schema: Schema ) {\n\treturn schema.checkChild( block.parent as any, 'listItem' ) && !schema.isObject( block );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/list/listutils\n */\n\nimport type { Element, Model, Position } from 'ckeditor5/src/engine';\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport {\n\tgetListTypeFromListStyleType,\n\tgetSelectedListItems,\n\tgetSiblingNodes\n} from './utils';\n\n/**\n * A set of helpers related to document lists.\n */\nexport default class ListUtils extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ListUtils' as const;\n\t}\n\n\t/**\n\t * Checks whether the given list-style-type is supported by numbered or bulleted list.\n\t */\n\tpublic getListTypeFromListStyleType( listStyleType: string ): 'bulleted' | 'numbered' | null {\n\t\treturn getListTypeFromListStyleType( listStyleType );\n\t}\n\n\t/**\n\t * Returns an array with all `listItem` elements in the model selection.\n\t *\n\t * It returns all the items even if only a part of the list is selected, including items that belong to nested lists.\n\t * If no list is selected, it returns an empty array.\n\t * The order of the elements is not specified.\n\t */\n\tpublic getSelectedListItems( model: Model ): Array<Element> {\n\t\treturn getSelectedListItems( model );\n\t}\n\n\t/**\n\t * Returns an array with all `listItem` elements that represent the same list.\n\t *\n\t * It means that values of `listIndent`, `listType`, `listStyle`, `listReversed` and `listStart` for all items are equal.\n\t *\n\t * Additionally, if the `position` is inside a list item, that list item will be returned as well.\n\t *\n\t * @param position Starting position.\n\t * @param direction Walking direction.\n\t */\n\tpublic getSiblingNodes( position: Position, direction: 'forward' | 'backward' ): Array<Element> {\n\t\treturn getSiblingNodes( position, direction );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/list/converters\n */\n\nimport {\n\tTreeWalker,\n\ttype DowncastAttributeEvent,\n\ttype DowncastConversionApi,\n\ttype DowncastInsertEvent,\n\ttype DowncastRemoveEvent,\n\ttype Element,\n\ttype MapperModelToViewPositionEvent,\n\ttype MapperViewToModelPositionEvent,\n\ttype Model,\n\ttype ModelInsertContentEvent,\n\ttype Node,\n\ttype Position,\n\ttype UpcastConversionApi,\n\ttype UpcastElementEvent,\n\ttype View,\n\ttype ViewContainerElement,\n\ttype ViewElement,\n\ttype ViewNode,\n\ttype ViewPosition,\n\ttype ViewTypeCheckable,\n\ttype Writer\n} from 'ckeditor5/src/engine';\n\nimport type { GetCallback } from 'ckeditor5/src/utils';\n\nimport {\n\tgenerateLiInUl,\n\tinjectViewList,\n\tmergeViewLists,\n\tgetSiblingListItem,\n\tpositionAfterUiElements\n} from './utils';\n\n/**\n * A model-to-view converter for the `listItem` model element insertion.\n *\n * It creates a `<ul><li></li><ul>` (or `<ol>`) view structure out of a `listItem` model element, inserts it at the correct\n * position, and merges the list with surrounding lists (if available).\n *\n * @see module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:insert\n * @param model Model instance.\n */\nexport function modelViewInsertion( model: Model ): GetCallback<DowncastInsertEvent<Element>> {\n\treturn ( evt, data, conversionApi ) => {\n\t\tconst consumable = conversionApi.consumable;\n\n\t\tif ( !consumable.test( data.item, 'insert' ) ||\n\t\t\t!consumable.test( data.item, 'attribute:listType' ) ||\n\t\t\t!consumable.test( data.item, 'attribute:listIndent' )\n\t\t) {\n\t\t\treturn;\n\t\t}\n\n\t\tconsumable.consume( data.item, 'insert' );\n\t\tconsumable.consume( data.item, 'attribute:listType' );\n\t\tconsumable.consume( data.item, 'attribute:listIndent' );\n\n\t\tconst modelItem = data.item;\n\t\tconst viewItem = generateLiInUl( modelItem, conversionApi );\n\n\t\tinjectViewList( modelItem, viewItem, conversionApi, model );\n\t};\n}\n\n/**\n * A model-to-view converter for the `listItem` model element removal.\n *\n * @see module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:remove\n * @param model Model instance.\n * @returns Returns a conversion callback.\n */\nexport function modelViewRemove( model: Model ): GetCallback<DowncastRemoveEvent> {\n\treturn ( evt, data, conversionApi ) => {\n\t\tconst viewPosition = conversionApi.mapper.toViewPosition( data.position );\n\t\tconst viewStart = viewPosition.getLastMatchingPosition( value => !value.item.is( 'element', 'li' ) );\n\t\tconst viewItem = viewStart.nodeAfter as ViewElement;\n\t\tconst viewWriter = conversionApi.writer;\n\n\t\t// 1. Break the container after and before the list item.\n\t\t// This will create a view list with one view list item - the one to remove.\n\t\tviewWriter.breakContainer( viewWriter.createPositionBefore( viewItem ) );\n\t\tviewWriter.breakContainer( viewWriter.createPositionAfter( viewItem ) );\n\n\t\t// 2. Remove the list with the item to remove.\n\t\tconst viewList = viewItem.parent as ViewElement;\n\t\tconst viewListPrev = viewList.previousSibling;\n\t\tconst removeRange = viewWriter.createRangeOn( viewList );\n\t\tconst removed = viewWriter.remove( removeRange );\n\n\t\t// 3. Merge the whole created by breaking and removing the list.\n\t\tif ( viewListPrev && viewListPrev.nextSibling ) {\n\t\t\tmergeViewLists( viewWriter, viewListPrev, viewListPrev.nextSibling );\n\t\t}\n\n\t\t// 4. Bring back nested list that was in the removed <li>.\n\t\tconst modelItem = conversionApi.mapper.toModelElement( viewItem )!;\n\n\t\thoistNestedLists(\n\t\t\tmodelItem.getAttribute( 'listIndent' ) as number + 1,\n\t\t\tdata.position,\n\t\t\tremoveRange.start,\n\t\t\tviewItem,\n\t\t\tconversionApi,\n\t\t\tmodel\n\t\t);\n\n\t\t// 5. Unbind removed view item and all children.\n\t\tfor ( const child of viewWriter.createRangeIn( removed ).getItems() ) {\n\t\t\tconversionApi.mapper.unbindViewElement( child as ViewElement );\n\t\t}\n\n\t\tevt.stop();\n\t};\n}\n\n/**\n * A model-to-view converter for the `type` attribute change on the `listItem` model element.\n *\n * This change means that the `<li>` element parent changes from `<ul>` to `<ol>` (or vice versa). This is accomplished\n * by breaking view elements and changing their name. The next {@link module:list/list/converters~modelViewMergeAfterChangeType}\n * converter will attempt to merge split nodes.\n *\n * Splitting this conversion into 2 steps makes it possible to add an additional conversion in the middle.\n * Check {@link module:list/todolist/todolistconverters~modelViewChangeType} to see an example of it.\n *\n * @see module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute\n */\nexport const modelViewChangeType: GetCallback<DowncastAttributeEvent<Element>> = ( evt, data, conversionApi ) => {\n\tif ( !conversionApi.consumable.test( data.item, evt.name ) ) {\n\t\treturn;\n\t}\n\n\tconst viewItem = conversionApi.mapper.toViewElement( data.item )!;\n\tconst viewWriter = conversionApi.writer;\n\n\t// Break the container after and before the list item.\n\t// This will create a view list with one view list item -- the one that changed type.\n\tviewWriter.breakContainer( viewWriter.createPositionBefore( viewItem ) );\n\tviewWriter.breakContainer( viewWriter.createPositionAfter( viewItem ) );\n\n\t// Change name of the view list that holds the changed view item.\n\t// We cannot just change name property, because that would not render properly.\n\tconst viewList = viewItem.parent as ViewContainerElement;\n\tconst listName = data.attributeNewValue == 'numbered' ? 'ol' : 'ul';\n\n\tviewWriter.rename( listName, viewList );\n};\n\n/**\n * A model-to-view converter that attempts to merge nodes split by {@link module:list/list/converters~modelViewChangeType}.\n *\n * @see module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute\n */\nexport const modelViewMergeAfterChangeType: GetCallback<DowncastAttributeEvent<Element>> = ( evt, data, conversionApi ) => {\n\tconversionApi.consumable.consume( data.item, evt.name );\n\n\tconst viewItem = conversionApi.mapper.toViewElement( data.item )!;\n\tconst viewList = viewItem.parent as ViewContainerElement;\n\tconst viewWriter = conversionApi.writer;\n\n\t// Merge the changed view list with other lists, if possible.\n\tmergeViewLists( viewWriter, viewList, viewList.nextSibling! );\n\tmergeViewLists( viewWriter, viewList.previousSibling!, viewList );\n};\n\n/**\n * A model-to-view converter for the `listIndent` attribute change on the `listItem` model element.\n *\n * @see module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:attribute\n * @param model Model instance.\n * @returns Returns a conversion callback.\n */\nexport function modelViewChangeIndent( model: Model ): GetCallback<DowncastAttributeEvent<Element>> {\n\treturn ( evt, data, conversionApi ) => {\n\t\tif ( !conversionApi.consumable.consume( data.item, 'attribute:listIndent' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst viewItem = conversionApi.mapper.toViewElement( data.item ) as ViewContainerElement;\n\t\tconst viewWriter = conversionApi.writer;\n\n\t\t// 1. Break the container after and before the list item.\n\t\t// This will create a view list with one view list item -- the one that changed type.\n\t\tviewWriter.breakContainer( viewWriter.createPositionBefore( viewItem ) );\n\t\tviewWriter.breakContainer( viewWriter.createPositionAfter( viewItem ) );\n\n\t\t// 2. Extract view list with changed view list item and merge \"hole\" possibly created by breaking and removing elements.\n\t\tconst viewList = viewItem.parent as ViewElement;\n\t\tconst viewListPrev = viewList.previousSibling;\n\t\tconst removeRange = viewWriter.createRangeOn( viewList );\n\t\tviewWriter.remove( removeRange );\n\n\t\tif ( viewListPrev && viewListPrev.nextSibling ) {\n\t\t\tmergeViewLists( viewWriter, viewListPrev, viewListPrev.nextSibling );\n\t\t}\n\n\t\t// 3. Bring back nested list that was in the removed <li>.\n\t\thoistNestedLists(\n\t\t\tdata.attributeOldValue as number + 1,\n\t\t\tdata.range.start,\n\t\t\tremoveRange.start,\n\t\t\tviewItem,\n\t\t\tconversionApi,\n\t\t\tmodel\n\t\t);\n\n\t\t// 4. Inject view list like it is newly inserted.\n\t\tinjectViewList( data.item, viewItem, conversionApi, model );\n\n\t\t// 5. Consume insertion of children inside the item. They are already handled by re-building the item in view.\n\t\tfor ( const child of data.item.getChildren() ) {\n\t\t\tconversionApi.consumable.consume( child, 'insert' );\n\t\t}\n\t};\n}\n\n/**\n * A special model-to-view converter introduced by the {@link module:list/list~List list feature}. This converter is fired for\n * insert change of every model item, and should be fired before the actual converter. The converter checks whether the inserted\n * model item is a non-`listItem` element. If it is, and it is inserted inside a view list, the converter breaks the\n * list so the model element is inserted to the view parent element corresponding to its model parent element.\n *\n * The converter prevents such situations:\n *\n * ```xml\n * // Model: // View:\n * <listItem>foo</listItem> <ul>\n * <listItem>bar</listItem> <li>foo</li>\n * <li>bar</li>\n * </ul>\n *\n * // After change: // Correct view guaranteed by this converter:\n * <listItem>foo</listItem> <ul><li>foo</li></ul><p>xxx</p><ul><li>bar</li></ul>\n * <paragraph>xxx</paragraph> // Instead of this wrong view state:\n * <listItem>bar</listItem> <ul><li>foo</li><p>xxx</p><li>bar</li></ul>\n * ```\n *\n * @see module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:insert\n */\nexport const modelViewSplitOnInsert: GetCallback<DowncastInsertEvent<Element>> = ( evt, data, conversionApi ) => {\n\tif ( !conversionApi.consumable.test( data.item, evt.name ) ) {\n\t\treturn;\n\t}\n\n\tif ( data.item.name != 'listItem' ) {\n\t\tlet viewPosition = conversionApi.mapper.toViewPosition( data.range.start );\n\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst lists = [];\n\n\t\t// Break multiple ULs/OLs if there are.\n\t\t//\n\t\t// Imagine following list:\n\t\t//\n\t\t// 1 --------\n\t\t// 1.1 --------\n\t\t// 1.1.1 --------\n\t\t// 1.1.2 --------\n\t\t// 1.1.3 --------\n\t\t// 1.1.3.1 --------\n\t\t// 1.2 --------\n\t\t// 1.2.1 --------\n\t\t// 2 --------\n\t\t//\n\t\t// Insert paragraph after item 1.1.1:\n\t\t//\n\t\t// 1 --------\n\t\t// 1.1 --------\n\t\t// 1.1.1 --------\n\t\t//\n\t\t// Lorem ipsum.\n\t\t//\n\t\t// 1.1.2 --------\n\t\t// 1.1.3 --------\n\t\t// 1.1.3.1 --------\n\t\t// 1.2 --------\n\t\t// 1.2.1 --------\n\t\t// 2 --------\n\t\t//\n\t\t// In this case 1.1.2 has to become beginning of a new list.\n\t\t// We need to break list before 1.1.2 (obvious), then we need to break list also before 1.2.\n\t\t// Then we need to move those broken pieces one after another and merge:\n\t\t//\n\t\t// 1 --------\n\t\t// 1.1 --------\n\t\t// 1.1.1 --------\n\t\t//\n\t\t// Lorem ipsum.\n\t\t//\n\t\t// 1.1.2 --------\n\t\t// 1.1.3 --------\n\t\t// 1.1.3.1 --------\n\t\t// 1.2 --------\n\t\t// 1.2.1 --------\n\t\t// 2 --------\n\t\t//\n\t\twhile ( ( viewPosition.parent as any ).name == 'ul' || ( viewPosition.parent as any ).name == 'ol' ) {\n\t\t\tviewPosition = viewWriter.breakContainer( viewPosition );\n\n\t\t\tif ( ( viewPosition.parent as any ).name != 'li' ) {\n\t\t\t\tbreak;\n\t\t\t}\n\n\t\t\t// Remove lists that are after inserted element.\n\t\t\t// They will be brought back later, below the inserted element.\n\t\t\tconst removeStart = viewPosition;\n\t\t\tconst removeEnd = viewWriter.createPositionAt( viewPosition.parent as any, 'end' );\n\n\t\t\t// Don't remove if there is nothing to remove.\n\t\t\tif ( !removeStart.isEqual( removeEnd ) ) {\n\t\t\t\tconst removed = viewWriter.remove( viewWriter.createRange( removeStart, removeEnd ) );\n\t\t\t\tlists.push( removed );\n\t\t\t}\n\n\t\t\tviewPosition = viewWriter.createPositionAfter( viewPosition.parent as any );\n\t\t}\n\n\t\t// Bring back removed lists.\n\t\tif ( lists.length > 0 ) {\n\t\t\tfor ( let i = 0; i < lists.length; i++ ) {\n\t\t\t\tconst previousList = viewPosition.nodeBefore;\n\t\t\t\tconst insertedRange = viewWriter.insert( viewPosition, lists[ i ] );\n\t\t\t\tviewPosition = insertedRange.end;\n\n\t\t\t\t// Don't merge first list! We want a split in that place (this is why this converter is introduced).\n\t\t\t\tif ( i > 0 ) {\n\t\t\t\t\tconst mergePos = mergeViewLists( viewWriter, previousList!, previousList!.nextSibling! );\n\n\t\t\t\t\t// If `mergePos` is in `previousList` it means that the lists got merged.\n\t\t\t\t\t// In this case, we need to fix insert position.\n\t\t\t\t\tif ( mergePos && mergePos.parent == previousList ) {\n\t\t\t\t\t\tviewPosition.offset--;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Merge last inserted list with element after it.\n\t\t\tmergeViewLists( viewWriter, viewPosition.nodeBefore!, viewPosition.nodeAfter! );\n\t\t}\n\t}\n};\n\n/**\n * A special model-to-view converter introduced by the {@link module:list/list~List list feature}. This converter takes care of\n * merging view lists after something is removed or moved from near them.\n *\n * Example:\n *\n * ```xml\n * // Model: // View:\n * <listItem>foo</listItem> <ul><li>foo</li></ul>\n * <paragraph>xxx</paragraph> <p>xxx</p>\n * <listItem>bar</listItem> <ul><li>bar</li></ul>\n *\n * // After change: // Correct view guaranteed by this converter:\n * <listItem>foo</listItem> <ul>\n * <listItem>bar</listItem> <li>foo</li>\n * <li>bar</li>\n * </ul>\n * ```\n *\n * @see module:engine/conversion/downcastdispatcher~DowncastDispatcher#event:remove\n */\nexport const modelViewMergeAfter: GetCallback<DowncastRemoveEvent> = ( evt, data, conversionApi ) => {\n\tconst viewPosition = conversionApi.mapper.toViewPosition( data.position );\n\tconst viewItemPrev = viewPosition.nodeBefore;\n\tconst viewItemNext = viewPosition.nodeAfter;\n\n\t// Merge lists if something (remove, move) was done from inside of list.\n\t// Merging will be done only if both items are view lists of the same type.\n\t// The check is done inside the helper function.\n\tmergeViewLists( conversionApi.writer, viewItemPrev!, viewItemNext! );\n};\n\n/**\n * A view-to-model converter that converts the `<li>` view elements into the `listItem` model elements.\n *\n * To set correct values of the `listType` and `listIndent` attributes the converter:\n * * checks `<li>`'s parent,\n * * stores and increases the `conversionApi.store.indent` value when `<li>`'s sub-items are converted.\n *\n * @see module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:element\n */\nexport const viewModelConverter: GetCallback<UpcastElementEvent> = ( evt, data, conversionApi ) => {\n\tif ( conversionApi.consumable.consume( data.viewItem, { name: true } ) ) {\n\t\tconst writer = conversionApi.writer;\n\n\t\t// 1. Create `listItem` model element.\n\t\tconst listItem = writer.createElement( 'listItem' );\n\n\t\t// 2. Handle `listItem` model element attributes.\n\t\tconst indent = getIndent( data.viewItem );\n\n\t\twriter.setAttribute( 'listIndent', indent, listItem );\n\n\t\t// Set 'bulleted' as default. If this item is pasted into a context,\n\t\tconst type = data.viewItem.parent && ( data.viewItem.parent as any ).name == 'ol' ? 'numbered' : 'bulleted';\n\t\twriter.setAttribute( 'listType', type, listItem );\n\n\t\tif ( !conversionApi.safeInsert( listItem, data.modelCursor ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst nextPosition = viewToModelListItemChildrenConverter( listItem, data.viewItem.getChildren(), conversionApi );\n\n\t\t// Result range starts before the first item and ends after the last.\n\t\tdata.modelRange = writer.createRange( data.modelCursor, nextPosition );\n\n\t\tconversionApi.updateConversionResult( listItem, data );\n\t}\n};\n\n/**\n * A view-to-model converter for the `<ul>` and `<ol>` view elements that cleans the input view of garbage.\n * This is mostly to clean whitespaces from between the `<li>` view elements inside the view list element, however, also\n * incorrect data can be cleared if the view was incorrect.\n *\n * @see module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:element\n */\nexport const cleanList: GetCallback<UpcastElementEvent> = ( evt, data, conversionApi ) => {\n\tif ( conversionApi.consumable.test( data.viewItem, { name: true } ) ) {\n\t\t// Caching children because when we start removing them iterating fails.\n\t\tconst children = Array.from( data.viewItem.getChildren() );\n\n\t\tfor ( const child of children ) {\n\t\t\tconst isWrongElement = !( child.is( 'element', 'li' ) || isList( child ) );\n\n\t\t\tif ( isWrongElement ) {\n\t\t\t\tchild._remove();\n\t\t\t}\n\t\t}\n\t}\n};\n\n/**\n * A view-to-model converter for the `<li>` elements that cleans whitespace formatting from the input view.\n *\n * @see module:engine/conversion/upcastdispatcher~UpcastDispatcher#event:element\n */\nexport const cleanListItem: GetCallback<UpcastElementEvent> = ( evt, data, conversionApi ) => {\n\tif ( conversionApi.consumable.test( data.viewItem, { name: true } ) ) {\n\t\tif ( data.viewItem.childCount === 0 ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst children = [ ...data.viewItem.getChildren() ];\n\n\t\tlet foundList = false;\n\n\t\tfor ( const child of children ) {\n\t\t\tif ( foundList && !isList( child ) ) {\n\t\t\t\tchild._remove();\n\t\t\t}\n\n\t\t\tif ( isList( child ) ) {\n\t\t\t\t// If this is a <ul> or <ol>, do not process it, just mark that we already visited list element.\n\t\t\t\tfoundList = true;\n\t\t\t}\n\t\t}\n\t}\n};\n\n/**\n * Returns a callback for model position to view position mapping for {@link module:engine/conversion/mapper~Mapper}. The callback fixes\n * positions between the `listItem` elements that would be incorrectly mapped because of how list items are represented in the model\n * and in the view.\n */\nexport function modelToViewPosition( view: View ): GetCallback<MapperModelToViewPositionEvent> {\n\treturn ( evt, data ) => {\n\t\tif ( data.isPhantom ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst modelItem = data.modelPosition.nodeBefore;\n\n\t\tif ( modelItem && modelItem.is( 'element', 'listItem' ) ) {\n\t\t\tconst viewItem = data.mapper.toViewElement( modelItem )!;\n\t\t\tconst topmostViewList = viewItem.getAncestors().find( isList );\n\t\t\tconst walker = view.createPositionAt( viewItem, 0 ).getWalker();\n\n\t\t\tfor ( const value of walker ) {\n\t\t\t\tif ( value.type == 'elementStart' && value.item.is( 'element', 'li' ) ) {\n\t\t\t\t\tdata.viewPosition = value.previousPosition;\n\n\t\t\t\t\tbreak;\n\t\t\t\t} else if ( value.type == 'elementEnd' && value.item == topmostViewList ) {\n\t\t\t\t\tdata.viewPosition = value.nextPosition;\n\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t};\n}\n\n/**\n * The callback for view position to model position mapping for {@link module:engine/conversion/mapper~Mapper}. The callback fixes\n * positions between the `<li>` elements that would be incorrectly mapped because of how list items are represented in the model\n * and in the view.\n *\n * @see module:engine/conversion/mapper~Mapper#event:viewToModelPosition\n * @param model Model instance.\n * @returns Returns a conversion callback.\n */\nexport function viewToModelPosition( model: Model ): GetCallback<MapperViewToModelPositionEvent> {\n\treturn ( evt, data ) => {\n\t\tconst viewPos = data.viewPosition;\n\t\tconst viewParent = viewPos.parent as ViewElement;\n\t\tconst mapper = data.mapper;\n\n\t\tif ( viewParent.name == 'ul' || viewParent.name == 'ol' ) {\n\t\t\t// Position is directly in <ul> or <ol>.\n\t\t\tif ( !viewPos.isAtEnd ) {\n\t\t\t\t// If position is not at the end, it must be before <li>.\n\t\t\t\t// Get that <li>, map it to `listItem` and set model position before that `listItem`.\n\t\t\t\tconst modelNode = mapper.toModelElement( viewPos.nodeAfter as any )!;\n\n\t\t\t\tdata.modelPosition = model.createPositionBefore( modelNode );\n\t\t\t} else {\n\t\t\t\t// Position is at the end of <ul> or <ol>, so there is no <li> after it to be mapped.\n\t\t\t\t// There is <li> before the position, but we cannot just map it to `listItem` and set model position after it,\n\t\t\t\t// because that <li> may contain nested items.\n\t\t\t\t// We will check \"model length\" of that <li>, in other words - how many `listItem`s are in that <li>.\n\t\t\t\tconst modelNode = mapper.toModelElement( viewPos.nodeBefore as any );\n\t\t\t\tconst modelLength = mapper.getModelLength( viewPos.nodeBefore! );\n\n\t\t\t\t// Then we get model position before mapped `listItem` and shift it accordingly.\n\t\t\t\tdata.modelPosition = model.createPositionBefore( modelNode! ).getShiftedBy( modelLength );\n\t\t\t}\n\n\t\t\tevt.stop();\n\t\t} else if (\n\t\t\tviewParent.name == 'li' &&\n\t\t\tviewPos.nodeBefore &&\n\t\t\t( ( viewPos.nodeBefore as any ).name == 'ul' || ( viewPos.nodeBefore as any ).name == 'ol' )\n\t\t) {\n\t\t\t// In most cases when view position is in <li> it is in text and this is a correct position.\n\t\t\t// However, if position is after <ul> or <ol> we have to fix it -- because in model <ul>/<ol> are not in the `listItem`.\n\t\t\tconst modelNode = mapper.toModelElement( viewParent );\n\n\t\t\t// Check all <ul>s and <ol>s that are in the <li> but before mapped position.\n\t\t\t// Get model length of those elements and then add it to the offset of `listItem` mapped to the original <li>.\n\t\t\tlet modelLength = 1; // Starts from 1 because the original <li> has to be counted in too.\n\t\t\tlet viewList: ViewNode | null = viewPos.nodeBefore;\n\n\t\t\twhile ( viewList && isList( viewList ) ) {\n\t\t\t\tmodelLength += mapper.getModelLength( viewList );\n\n\t\t\t\tviewList = viewList.previousSibling;\n\t\t\t}\n\n\t\t\tdata.modelPosition = model.createPositionBefore( modelNode! ).getShiftedBy( modelLength );\n\n\t\t\tevt.stop();\n\t\t}\n\t};\n}\n\n/**\n * Post-fixer that reacts to changes on document and fixes incorrect model states.\n *\n * In the example below, there is a correct list structure.\n * Then the middle element is removed so the list structure will become incorrect:\n *\n * ```xml\n * <listItem listType=\"bulleted\" listIndent=0>Item 1</listItem>\n * <listItem listType=\"bulleted\" listIndent=1>Item 2</listItem> <--- this is removed.\n * <listItem listType=\"bulleted\" listIndent=2>Item 3</listItem>\n * ```\n *\n * The list structure after the middle element is removed:\n *\n * ```xml\n * <listItem listType=\"bulleted\" listIndent=0>Item 1</listItem>\n * <listItem listType=\"bulleted\" listIndent=2>Item 3</listItem>\n * ```\n *\n * Should become:\n *\n * ```xml\n * <listItem listType=\"bulleted\" listIndent=0>Item 1</listItem>\n * <listItem listType=\"bulleted\" listIndent=1>Item 3</listItem> <--- note that indent got post-fixed.\n * ```\n *\n * @param model The data model.\n * @param writer The writer to do changes with.\n * @returns `true` if any change has been applied, `false` otherwise.\n */\nexport function modelChangePostFixer( model: Model, writer: Writer ): boolean {\n\tconst changes = model.document.differ.getChanges();\n\tconst itemToListHead = new Map<Element, Element>();\n\n\tlet applied = false;\n\n\tfor ( const entry of changes ) {\n\t\tif ( entry.type == 'insert' && entry.name == 'listItem' ) {\n\t\t\t_addListToFix( entry.position );\n\t\t} else if ( entry.type == 'insert' && entry.name != 'listItem' ) {\n\t\t\tif ( entry.name != '$text' ) {\n\t\t\t\t// In case of renamed element.\n\t\t\t\tconst item = entry.position.nodeAfter as Element;\n\n\t\t\t\tif ( item.hasAttribute( 'listIndent' ) ) {\n\t\t\t\t\twriter.removeAttribute( 'listIndent', item );\n\n\t\t\t\t\tapplied = true;\n\t\t\t\t}\n\n\t\t\t\tif ( item.hasAttribute( 'listType' ) ) {\n\t\t\t\t\twriter.removeAttribute( 'listType', item );\n\n\t\t\t\t\tapplied = true;\n\t\t\t\t}\n\n\t\t\t\tif ( item.hasAttribute( 'listStyle' ) ) {\n\t\t\t\t\twriter.removeAttribute( 'listStyle', item );\n\n\t\t\t\t\tapplied = true;\n\t\t\t\t}\n\n\t\t\t\tif ( item.hasAttribute( 'listReversed' ) ) {\n\t\t\t\t\twriter.removeAttribute( 'listReversed', item );\n\n\t\t\t\t\tapplied = true;\n\t\t\t\t}\n\n\t\t\t\tif ( item.hasAttribute( 'listStart' ) ) {\n\t\t\t\t\twriter.removeAttribute( 'listStart', item );\n\n\t\t\t\t\tapplied = true;\n\t\t\t\t}\n\n\t\t\t\tfor ( const innerItem of Array.from( model.createRangeIn( item ) ).filter( e => e.item.is( 'element', 'listItem' ) ) ) {\n\t\t\t\t\t_addListToFix( innerItem.previousPosition );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tconst posAfter = entry.position.getShiftedBy( entry.length );\n\n\t\t\t_addListToFix( posAfter );\n\t\t} else if ( entry.type == 'remove' && entry.name == 'listItem' ) {\n\t\t\t_addListToFix( entry.position );\n\t\t} else if ( entry.type == 'attribute' && entry.attributeKey == 'listIndent' ) {\n\t\t\t_addListToFix( entry.range.start );\n\t\t} else if ( entry.type == 'attribute' && entry.attributeKey == 'listType' ) {\n\t\t\t_addListToFix( entry.range.start );\n\t\t}\n\t}\n\n\tfor ( const listHead of itemToListHead.values() ) {\n\t\t_fixListIndents( listHead );\n\t\t_fixListTypes( listHead );\n\t}\n\n\treturn applied;\n\n\tfunction _addListToFix( position: Position ) {\n\t\tconst previousNode = position.nodeBefore;\n\n\t\tif ( !previousNode || !previousNode.is( 'element', 'listItem' ) ) {\n\t\t\tconst item = position.nodeAfter;\n\n\t\t\tif ( item && item.is( 'element', 'listItem' ) ) {\n\t\t\t\titemToListHead.set( item, item );\n\t\t\t}\n\t\t} else {\n\t\t\tlet listHead = previousNode;\n\n\t\t\tif ( itemToListHead.has( listHead ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tfor (\n\t\t\t\t// Cache previousSibling and reuse for performance reasons. See #6581.\n\t\t\t\tlet previousSibling = listHead.previousSibling;\n\t\t\t\tpreviousSibling && previousSibling.is( 'element', 'listItem' );\n\t\t\t\tpreviousSibling = listHead.previousSibling\n\t\t\t) {\n\t\t\t\tlistHead = previousSibling;\n\n\t\t\t\tif ( itemToListHead.has( listHead ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\titemToListHead.set( previousNode, listHead );\n\t\t}\n\t}\n\n\tfunction _fixListIndents( item: Node | null ) {\n\t\tlet maxIndent = 0;\n\t\tlet fixBy = null;\n\n\t\twhile ( item && item.is( 'element', 'listItem' ) ) {\n\t\t\tconst itemIndent = item.getAttribute( 'listIndent' ) as number;\n\n\t\t\tif ( itemIndent > maxIndent ) {\n\t\t\t\tlet newIndent;\n\n\t\t\t\tif ( fixBy === null ) {\n\t\t\t\t\tfixBy = itemIndent - maxIndent;\n\t\t\t\t\tnewIndent = maxIndent;\n\t\t\t\t} else {\n\t\t\t\t\tif ( fixBy > itemIndent ) {\n\t\t\t\t\t\tfixBy = itemIndent;\n\t\t\t\t\t}\n\n\t\t\t\t\tnewIndent = itemIndent - fixBy;\n\t\t\t\t}\n\n\t\t\t\twriter.setAttribute( 'listIndent', newIndent, item );\n\n\t\t\t\tapplied = true;\n\t\t\t} else {\n\t\t\t\tfixBy = null;\n\t\t\t\tmaxIndent = item.getAttribute( 'listIndent' ) as number + 1;\n\t\t\t}\n\n\t\t\titem = item.nextSibling;\n\t\t}\n\t}\n\n\tfunction _fixListTypes( item: Node | null ) {\n\t\tlet typesStack: Array<string> = [];\n\t\tlet prev = null;\n\n\t\twhile ( item && item.is( 'element', 'listItem' ) ) {\n\t\t\tconst itemIndent = item.getAttribute( 'listIndent' ) as number;\n\n\t\t\tif ( prev && prev.getAttribute( 'listIndent' ) as number > itemIndent ) {\n\t\t\t\ttypesStack = typesStack.slice( 0, itemIndent + 1 );\n\t\t\t}\n\n\t\t\tif ( itemIndent != 0 ) {\n\t\t\t\tif ( typesStack[ itemIndent ] ) {\n\t\t\t\t\tconst type = typesStack[ itemIndent ];\n\n\t\t\t\t\tif ( item.getAttribute( 'listType' ) != type ) {\n\t\t\t\t\t\twriter.setAttribute( 'listType', type, item );\n\n\t\t\t\t\t\tapplied = true;\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\ttypesStack[ itemIndent ] = item.getAttribute( 'listType' ) as string;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tprev = item;\n\t\t\titem = item.nextSibling;\n\t\t}\n\t}\n}\n\n/**\n * A fixer for pasted content that includes list items.\n *\n * It fixes indentation of pasted list items so the pasted items match correctly to the context they are pasted into.\n *\n * Example:\n *\n * ```xml\n * <listItem listType=\"bulleted\" listIndent=0>A</listItem>\n * <listItem listType=\"bulleted\" listIndent=1>B^</listItem>\n * // At ^ paste: <listItem listType=\"bulleted\" listIndent=4>X</listItem>\n * // <listItem listType=\"bulleted\" listIndent=5>Y</listItem>\n * <listItem listType=\"bulleted\" listIndent=2>C</listItem>\n * ```\n *\n * Should become:\n *\n * ```xml\n * <listItem listType=\"bulleted\" listIndent=0>A</listItem>\n * <listItem listType=\"bulleted\" listIndent=1>BX</listItem>\n * <listItem listType=\"bulleted\" listIndent=2>Y/listItem>\n * <listItem listType=\"bulleted\" listIndent=2>C</listItem>\n * ```\n */\nexport const modelIndentPasteFixer: GetCallback<ModelInsertContentEvent> = function( evt, [ content, selectable ] ) {\n\tconst model = this as Model;\n\n\t// Check whether inserted content starts from a `listItem`. If it does not, it means that there are some other\n\t// elements before it and there is no need to fix indents, because even if we insert that content into a list,\n\t// that list will be broken.\n\t// Note: we also need to handle singular elements because inserting item with indent 0 into 0,1,[],2\n\t// would create incorrect model.\n\tlet item = content.is( 'documentFragment' ) ? content.getChild( 0 ) : content;\n\n\tlet selection;\n\n\tif ( !selectable ) {\n\t\tselection = model.document.selection;\n\t} else {\n\t\tselection = model.createSelection( selectable );\n\t}\n\n\tif ( item && item.is( 'element', 'listItem' ) ) {\n\t\t// Get a reference list item. Inserted list items will be fixed according to that item.\n\t\tconst pos = selection.getFirstPosition()!;\n\t\tlet refItem = null;\n\n\t\tif ( pos.parent.is( 'element', 'listItem' ) ) {\n\t\t\trefItem = pos.parent;\n\t\t} else if ( pos.nodeBefore && pos.nodeBefore.is( 'element', 'listItem' ) ) {\n\t\t\trefItem = pos.nodeBefore;\n\t\t}\n\n\t\t// If there is `refItem` it means that we do insert list items into an existing list.\n\t\tif ( refItem ) {\n\t\t\t// First list item in `data` has indent equal to 0 (it is a first list item). It should have indent equal\n\t\t\t// to the indent of reference item. We have to fix the first item and all of it's children and following siblings.\n\t\t\t// Indent of all those items has to be adjusted to reference item.\n\t\t\tconst indentChange = refItem.getAttribute( 'listIndent' ) as number;\n\n\t\t\t// Fix only if there is anything to fix.\n\t\t\tif ( indentChange > 0 ) {\n\t\t\t\t// Adjust indent of all \"first\" list items in inserted data.\n\t\t\t\twhile ( item && item.is( 'element', 'listItem' ) ) {\n\t\t\t\t\titem._setAttribute( 'listIndent', item.getAttribute( 'listIndent' ) as number + indentChange );\n\n\t\t\t\t\titem = item.nextSibling;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n};\n\n/**\n * Helper function that converts children of a given `<li>` view element into corresponding model elements.\n * The function maintains proper order of elements if model `listItem` is split during the conversion\n * due to block children conversion.\n *\n * @param listItemModel List item model element to which converted children will be inserted.\n * @param viewChildren View elements which will be converted.\n * @param conversionApi Conversion interface to be used by the callback.\n * @returns Position on which next elements should be inserted after children conversion.\n */\nfunction viewToModelListItemChildrenConverter(\n\tlistItemModel: Element,\n\tviewChildren: Iterable<ViewNode>,\n\tconversionApi: UpcastConversionApi\n) {\n\tconst { writer, schema } = conversionApi;\n\n\t// A position after the last inserted `listItem`.\n\tlet nextPosition = writer.createPositionAfter( listItemModel );\n\n\t// Check all children of the converted `<li>`. At this point we assume there are no \"whitespace\" view text nodes\n\t// in view list, between view list items. This should be handled by `<ul>` and `<ol>` converters.\n\tfor ( const child of viewChildren ) {\n\t\tif ( ( child as any ).name == 'ul' || ( child as any ).name == 'ol' ) {\n\t\t\t// If the children is a list, we will insert its conversion result after currently handled `listItem`.\n\t\t\t// Then, next insertion position will be set after all the new list items (and maybe other elements if\n\t\t\t// something split list item).\n\t\t\t//\n\t\t\t// If this is a list, we expect that some `listItem`s and possibly other blocks will be inserted, however `.modelCursor`\n\t\t\t// should be set after last `listItem` (or block). This is why it feels safe to use it as `nextPosition`\n\t\t\tnextPosition = conversionApi.convertItem( child, nextPosition ).modelCursor;\n\t\t} else {\n\t\t\t// If this is not a list, try inserting content at the end of the currently handled `listItem`.\n\t\t\tconst result = conversionApi.convertItem( child, writer.createPositionAt( listItemModel, 'end' ) );\n\n\t\t\t// It may end up that the current `listItem` becomes split (if that content cannot be inside `listItem`). For example:\n\t\t\t//\n\t\t\t// <li><p>Foo</p></li>\n\t\t\t//\n\t\t\t// will be converted to:\n\t\t\t//\n\t\t\t// <listItem></listItem><paragraph>Foo</paragraph><listItem></listItem>\n\t\t\t//\n\t\t\tconst convertedChild = result.modelRange!.start.nodeAfter;\n\t\t\tconst wasSplit = convertedChild && convertedChild.is( 'element' ) && !schema.checkChild( listItemModel, convertedChild.name );\n\n\t\t\tif ( wasSplit ) {\n\t\t\t\t// As `lastListItem` got split, we need to update it to the second part of the split `listItem` element.\n\t\t\t\t//\n\t\t\t\t// `modelCursor` should be set to a position where the conversion should continue. There are multiple possible scenarios\n\t\t\t\t// that may happen. Usually, `modelCursor` (marked as `#` below) would point to the second list item after conversion:\n\t\t\t\t//\n\t\t\t\t//\t\t`<li><p>Foo</p></li>` -> `<listItem></listItem><paragraph>Foo</paragraph><listItem>#</listItem>`\n\t\t\t\t//\n\t\t\t\t// However, in some cases, like auto-paragraphing, the position is placed at the end of the block element:\n\t\t\t\t//\n\t\t\t\t//\t\t`<li><div>Foo</div></li>` -> `<listItem></listItem><paragraph>Foo#</paragraph><listItem></listItem>`\n\t\t\t\t//\n\t\t\t\t// or after an element if another element broken auto-paragraphed element:\n\t\t\t\t//\n\t\t\t\t//\t\t`<li><div><h2>Foo</h2></div></li>` -> `<listItem></listItem><heading1>Foo</heading1>#<listItem></listItem>`\n\t\t\t\t//\n\t\t\t\t// We need to check for such cases and use proper list item and position based on it.\n\t\t\t\t//\n\t\t\t\tif ( result.modelCursor.parent.is( 'element', 'listItem' ) ) {\n\t\t\t\t\t// (1).\n\t\t\t\t\tlistItemModel = result.modelCursor.parent;\n\t\t\t\t} else {\n\t\t\t\t\t// (2), (3).\n\t\t\t\t\tlistItemModel = findNextListItem( result.modelCursor );\n\t\t\t\t}\n\n\t\t\t\tnextPosition = writer.createPositionAfter( listItemModel );\n\t\t\t}\n\t\t}\n\t}\n\n\treturn nextPosition;\n}\n\n/**\n * Helper function that seeks for a next list item starting from given `startPosition`.\n */\nfunction findNextListItem( startPosition: Position ) {\n\tconst treeWalker = new TreeWalker( { startPosition } );\n\n\tlet value;\n\n\tdo {\n\t\tvalue = treeWalker.next();\n\t} while ( !value.value.item.is( 'element', 'listItem' ) );\n\n\treturn value.value.item;\n}\n\n/**\n * Helper function that takes all children of given `viewRemovedItem` and moves them in a correct place, according\n * to other given parameters.\n */\nfunction hoistNestedLists(\n\tnextIndent: number,\n\tmodelRemoveStartPosition: Position,\n\tviewRemoveStartPosition: ViewPosition,\n\tviewRemovedItem: ViewElement,\n\tconversionApi: DowncastConversionApi,\n\tmodel: Model\n) {\n\t// Find correct previous model list item element.\n\t// The element has to have either same or smaller indent than given reference indent.\n\t// This will be the model element which will get nested items (if it has smaller indent) or sibling items (if it has same indent).\n\t// Keep in mind that such element might not be found, if removed item was the first item.\n\tconst prevModelItem = getSiblingListItem( modelRemoveStartPosition.nodeBefore, {\n\t\tsameIndent: true,\n\t\tsmallerIndent: true,\n\t\tlistIndent: nextIndent\n\t} );\n\n\tconst mapper = conversionApi.mapper;\n\tconst viewWriter = conversionApi.writer;\n\n\t// Indent of found element or `null` if the element has not been found.\n\tconst prevIndent = prevModelItem ? prevModelItem.getAttribute( 'listIndent' ) : null;\n\n\tlet insertPosition;\n\n\tif ( !prevModelItem ) {\n\t\t// If element has not been found, simply insert lists at the position where the removed item was:\n\t\t//\n\t\t// Lorem ipsum.\n\t\t// 1 -------- <--- this is removed, no previous list item, put nested items in place of removed item.\n\t\t// 1.1 -------- <--- this is reference indent.\n\t\t// 1.1.1 --------\n\t\t// 1.1.2 --------\n\t\t// 1.2 --------\n\t\t//\n\t\t// Becomes:\n\t\t//\n\t\t// Lorem ipsum.\n\t\t// 1.1 --------\n\t\t// 1.1.1 --------\n\t\t// 1.1.2 --------\n\t\t// 1.2 --------\n\t\tinsertPosition = viewRemoveStartPosition;\n\t} else if ( prevIndent == nextIndent ) {\n\t\t// If element has been found and has same indent as reference indent it means that nested items should\n\t\t// become siblings of found element:\n\t\t//\n\t\t// 1 --------\n\t\t// 1.1 --------\n\t\t// 1.2 -------- <--- this is `prevModelItem`.\n\t\t// 2 -------- <--- this is removed, previous list item has indent same as reference indent.\n\t\t// 2.1 -------- <--- this is reference indent, this and 2.2 should become siblings of 1.2.\n\t\t// 2.2 --------\n\t\t//\n\t\t// Becomes:\n\t\t//\n\t\t// 1 --------\n\t\t// 1.1 --------\n\t\t// 1.2 --------\n\t\t// 2.1 --------\n\t\t// 2.2 --------\n\t\tconst prevViewList = mapper.toViewElement( prevModelItem )!.parent as ViewElement;\n\t\tinsertPosition = viewWriter.createPositionAfter( prevViewList );\n\t} else {\n\t\t// If element has been found and has smaller indent as reference indent it means that nested items\n\t\t// should become nested items of found item:\n\t\t//\n\t\t// 1 -------- <--- this is `prevModelItem`.\n\t\t// 1.1 -------- <--- this is removed, previous list item has indent smaller than reference indent.\n\t\t// 1.1.1 -------- <--- this is reference indent, this and 1.1.1 should become nested items of 1.\n\t\t// 1.1.2 --------\n\t\t// 1.2 --------\n\t\t//\n\t\t// Becomes:\n\t\t//\n\t\t// 1 --------\n\t\t// 1.1.1 --------\n\t\t// 1.1.2 --------\n\t\t// 1.2 --------\n\t\t//\n\t\t// Note: in this case 1.1.1 have indent 2 while 1 have indent 0. In model that should not be possible,\n\t\t// because following item may have indent bigger only by one. But this is fixed by postfixer.\n\t\tconst modelPosition = model.createPositionAt( prevModelItem, 'end' );\n\t\tinsertPosition = mapper.toViewPosition( modelPosition );\n\t}\n\n\tinsertPosition = positionAfterUiElements( insertPosition );\n\n\t// Handle multiple lists. This happens if list item has nested numbered and bulleted lists. Following lists\n\t// are inserted after the first list (no need to recalculate insertion position for them).\n\tfor ( const child of [ ...viewRemovedItem.getChildren() ] ) {\n\t\tif ( isList( child ) ) {\n\t\t\tinsertPosition = viewWriter.move( viewWriter.createRangeOn( child ), insertPosition ).end;\n\n\t\t\tmergeViewLists( viewWriter, child, child.nextSibling! );\n\t\t\tmergeViewLists( viewWriter, child.previousSibling!, child );\n\t\t}\n\t}\n}\n\n/**\n * Checks if view element is a list type (ul or ol).\n */\nfunction isList( viewElement: ViewTypeCheckable ): viewElement is ViewElement & { name: 'ol' | 'ul' } {\n\treturn viewElement.is( 'element', 'ol' ) || viewElement.is( 'element', 'ul' );\n}\n\n/**\n * Calculates the indent value for a list item. Handles HTML compliant and non-compliant lists.\n *\n * Also, fixes non HTML compliant lists indents:\n *\n * ```\n * before: fixed list:\n * OL OL\n * |-> LI (parent LIs: 0) |-> LI (indent: 0)\n * |-> OL |-> OL\n * |-> OL |\n * | |-> OL |\n * | |-> OL |\n * | |-> LI (parent LIs: 1) |-> LI (indent: 1)\n * |-> LI (parent LIs: 1) |-> LI (indent: 1)\n *\n * before: fixed list:\n * OL OL\n * |-> OL |\n * |-> OL |\n * |-> OL |\n * |-> LI (parent LIs: 0) |-> LI (indent: 0)\n *\n * before: fixed list:\n * OL OL\n * |-> LI (parent LIs: 0) |-> LI (indent: 0)\n * |-> OL |-> OL\n * |-> LI (parent LIs: 0) |-> LI (indent: 1)\n * ```\n */\nfunction getIndent( listItem: ViewElement ) {\n\tlet indent = 0;\n\n\tlet parent = listItem.parent as ViewElement;\n\n\twhile ( parent ) {\n\t\t// Each LI in the tree will result in an increased indent for HTML compliant lists.\n\t\tif ( parent.is( 'element', 'li' ) ) {\n\t\t\tindent++;\n\t\t} else {\n\t\t\t// If however the list is nested in other list we should check previous sibling of any of the list elements...\n\t\t\tconst previousSibling = parent.previousSibling;\n\n\t\t\t// ...because the we might need increase its indent:\n\t\t\t//\t\tbefore: fixed list:\n\t\t\t//\t\tOL OL\n\t\t\t//\t\t|-> LI (parent LIs: 0) |-> LI (indent: 0)\n\t\t\t//\t\t|-> OL |-> OL\n\t\t\t//\t\t |-> LI (parent LIs: 0) |-> LI (indent: 1)\n\t\t\tif ( previousSibling && previousSibling.is( 'element', 'li' ) ) {\n\t\t\t\tindent++;\n\t\t\t}\n\t\t}\n\n\t\tparent = parent.parent as ViewElement;\n\t}\n\n\treturn indent;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/list/listediting\n */\n\nimport ListCommand from './listcommand';\nimport IndentCommand from './indentcommand';\nimport ListUtils from './listutils';\n\nimport { Plugin, type MultiCommand } from 'ckeditor5/src/core';\n\nimport { Enter, type ViewDocumentEnterEvent } from 'ckeditor5/src/enter';\nimport { Delete, type ViewDocumentDeleteEvent } from 'ckeditor5/src/typing';\n\nimport type {\n\tDowncastAttributeEvent,\n\tDowncastInsertEvent,\n\tDowncastRemoveEvent,\n\tElement,\n\tMapperModelToViewPositionEvent,\n\tMapperViewToModelPositionEvent,\n\tModelInsertContentEvent,\n\tUpcastElementEvent,\n\tViewDocumentTabEvent,\n\tViewElement\n} from 'ckeditor5/src/engine';\n\nimport {\n\tcleanList,\n\tcleanListItem,\n\tmodelViewInsertion,\n\tmodelViewChangeType,\n\tmodelViewMergeAfterChangeType,\n\tmodelViewMergeAfter,\n\tmodelViewRemove,\n\tmodelViewSplitOnInsert,\n\tmodelViewChangeIndent,\n\tmodelChangePostFixer,\n\tmodelIndentPasteFixer,\n\tviewModelConverter,\n\tmodelToViewPosition,\n\tviewToModelPosition\n} from './converters';\n\nimport '../../theme/list.css';\n\n/**\n * The engine of the list feature. It handles creating, editing and removing lists and list items.\n *\n * It registers the `'numberedList'`, `'bulletedList'`, `'indentList'` and `'outdentList'` commands.\n */\nexport default class ListEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ListEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Enter, Delete, ListUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Schema.\n\t\t// Note: in case `$block` will ever be allowed in `listItem`, keep in mind that this feature\n\t\t// uses `Selection#getSelectedBlocks()` without any additional processing to obtain all selected list items.\n\t\t// If there are blocks allowed inside list item, algorithms using `getSelectedBlocks()` will have to be modified.\n\t\teditor.model.schema.register( 'listItem', {\n\t\t\tinheritAllFrom: '$block',\n\t\t\tallowAttributes: [ 'listType', 'listIndent' ]\n\t\t} );\n\n\t\t// Converters.\n\t\tconst data = editor.data;\n\t\tconst editing = editor.editing;\n\n\t\teditor.model.document.registerPostFixer( writer => modelChangePostFixer( editor.model, writer ) );\n\n\t\tediting.mapper.registerViewToModelLength( 'li', getViewListItemLength );\n\t\tdata.mapper.registerViewToModelLength( 'li', getViewListItemLength );\n\n\t\tediting.mapper.on<MapperModelToViewPositionEvent>( 'modelToViewPosition', modelToViewPosition( editing.view ) );\n\t\tediting.mapper.on<MapperViewToModelPositionEvent>( 'viewToModelPosition', viewToModelPosition( editor.model ) );\n\t\tdata.mapper.on<MapperModelToViewPositionEvent>( 'modelToViewPosition', modelToViewPosition( editing.view ) );\n\n\t\teditor.conversion.for( 'editingDowncast' )\n\t\t\t.add( dispatcher => {\n\t\t\t\tdispatcher.on<DowncastInsertEvent<Element>>( 'insert', modelViewSplitOnInsert, { priority: 'high' } );\n\t\t\t\tdispatcher.on<DowncastInsertEvent<Element>>( 'insert:listItem', modelViewInsertion( editor.model ) );\n\t\t\t\tdispatcher.on<DowncastAttributeEvent<Element>>( 'attribute:listType:listItem', modelViewChangeType, { priority: 'high' } );\n\t\t\t\tdispatcher.on<DowncastAttributeEvent<Element>>(\n\t\t\t\t\t'attribute:listType:listItem', modelViewMergeAfterChangeType, { priority: 'low' } );\n\t\t\t\tdispatcher.on<DowncastAttributeEvent<Element>>( 'attribute:listIndent:listItem', modelViewChangeIndent( editor.model ) );\n\t\t\t\tdispatcher.on<DowncastRemoveEvent>( 'remove:listItem', modelViewRemove( editor.model ) );\n\t\t\t\tdispatcher.on<DowncastRemoveEvent>( 'remove', modelViewMergeAfter, { priority: 'low' } );\n\t\t\t} );\n\n\t\teditor.conversion.for( 'dataDowncast' )\n\t\t\t.add( dispatcher => {\n\t\t\t\tdispatcher.on<DowncastInsertEvent<Element>>( 'insert', modelViewSplitOnInsert, { priority: 'high' } );\n\t\t\t\tdispatcher.on<DowncastInsertEvent<Element>>( 'insert:listItem', modelViewInsertion( editor.model ) );\n\t\t\t} );\n\n\t\teditor.conversion.for( 'upcast' )\n\t\t\t.add( dispatcher => {\n\t\t\t\tdispatcher.on<UpcastElementEvent>( 'element:ul', cleanList, { priority: 'high' } );\n\t\t\t\tdispatcher.on<UpcastElementEvent>( 'element:ol', cleanList, { priority: 'high' } );\n\t\t\t\tdispatcher.on<UpcastElementEvent>( 'element:li', cleanListItem, { priority: 'high' } );\n\t\t\t\tdispatcher.on<UpcastElementEvent>( 'element:li', viewModelConverter );\n\t\t\t} );\n\n\t\t// Fix indentation of pasted items.\n\t\teditor.model.on<ModelInsertContentEvent>( 'insertContent', modelIndentPasteFixer, { priority: 'high' } );\n\n\t\t// Register commands for numbered and bulleted list.\n\t\teditor.commands.add( 'numberedList', new ListCommand( editor, 'numbered' ) );\n\t\teditor.commands.add( 'bulletedList', new ListCommand( editor, 'bulleted' ) );\n\n\t\t// Register commands for indenting.\n\t\teditor.commands.add( 'indentList', new IndentCommand( editor, 'forward' ) );\n\t\teditor.commands.add( 'outdentList', new IndentCommand( editor, 'backward' ) );\n\n\t\tconst viewDocument = editing.view.document;\n\n\t\t// Overwrite default Enter key behavior.\n\t\t// If Enter key is pressed with selection collapsed in empty list item, outdent it instead of breaking it.\n\t\tthis.listenTo<ViewDocumentEnterEvent>( viewDocument, 'enter', ( evt, data ) => {\n\t\t\tconst doc = this.editor.model.document;\n\t\t\tconst positionParent = doc.selection.getLastPosition()!.parent;\n\n\t\t\tif ( doc.selection.isCollapsed && positionParent.name == 'listItem' && positionParent.isEmpty ) {\n\t\t\t\tthis.editor.execute( 'outdentList' );\n\n\t\t\t\tdata.preventDefault();\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { context: 'li' } );\n\n\t\t// Overwrite default Backspace key behavior.\n\t\t// If Backspace key is pressed with selection collapsed on first position in first list item, outdent it. #83\n\t\tthis.listenTo<ViewDocumentDeleteEvent>( viewDocument, 'delete', ( evt, data ) => {\n\t\t\t// Check conditions from those that require less computations like those immediately available.\n\t\t\tif ( data.direction !== 'backward' ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst selection = this.editor.model.document.selection;\n\n\t\t\tif ( !selection.isCollapsed ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst firstPosition = selection.getFirstPosition()!;\n\n\t\t\tif ( !firstPosition.isAtStart ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst positionParent = firstPosition.parent;\n\n\t\t\tif ( positionParent.name !== 'listItem' ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst previousIsAListItem = positionParent.previousSibling && ( positionParent.previousSibling as any ).name === 'listItem';\n\n\t\t\tif ( previousIsAListItem ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tthis.editor.execute( 'outdentList' );\n\n\t\t\tdata.preventDefault();\n\t\t\tevt.stop();\n\t\t}, { context: 'li' } );\n\n\t\tthis.listenTo<ViewDocumentTabEvent>( editor.editing.view.document, 'tab', ( evt, data ) => {\n\t\t\tconst commandName = data.shiftKey ? 'outdentList' : 'indentList';\n\t\t\tconst command = this.editor.commands.get( commandName )!;\n\n\t\t\tif ( command.isEnabled ) {\n\t\t\t\teditor.execute( commandName );\n\n\t\t\t\tdata.stopPropagation();\n\t\t\t\tdata.preventDefault();\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { context: 'li' } );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\tconst commands = this.editor.commands;\n\n\t\tconst indent = commands.get( 'indent' ) as MultiCommand;\n\t\tconst outdent = commands.get( 'outdent' ) as MultiCommand;\n\n\t\tif ( indent ) {\n\t\t\tindent.registerChildCommand( commands.get( 'indentList' )! );\n\t\t}\n\n\t\tif ( outdent ) {\n\t\t\toutdent.registerChildCommand( commands.get( 'outdentList' )! );\n\t\t}\n\t}\n}\n\nfunction getViewListItemLength( element: ViewElement ) {\n\tlet length = 1;\n\n\tfor ( const child of element.getChildren() as Iterable<ViewElement> ) {\n\t\tif ( child.name == 'ul' || child.name == 'ol' ) {\n\t\t\tfor ( const item of child.getChildren() ) {\n\t\t\t\tlength += getViewListItemLength( item as ViewElement );\n\t\t\t}\n\t\t}\n\t}\n\n\treturn length;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/todolist/todolistediting\n */\n\nimport type {\n\tDowncastAttributeEvent,\n\tDowncastInsertEvent,\n\tElement,\n\tItem,\n\tMapperModelToViewPositionEvent,\n\tModel,\n\tModelApplyOperationEvent,\n\tUpcastElementEvent,\n\tViewDocumentArrowKeyEvent,\n\tViewDocumentKeyDownEvent,\n\tAttributeOperation,\n\tRenameOperation\n} from 'ckeditor5/src/engine';\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport {\n\tgetCode,\n\tparseKeystroke,\n\tgetLocalizedArrowKeyCodeDirection,\n\ttype Locale,\n\ttype GetCallback\n} from 'ckeditor5/src/utils';\n\nimport ListCommand from '../list/listcommand';\nimport ListEditing from '../list/listediting';\nimport CheckTodoListCommand from './checktodolistcommand';\nimport {\n\tdataModelViewInsertion,\n\tdataViewModelCheckmarkInsertion,\n\tmapModelToViewPosition,\n\tmodelViewChangeChecked,\n\tmodelViewChangeType,\n\tmodelViewInsertion\n} from './todolistconverters';\n\nconst ITEM_TOGGLE_KEYSTROKE = parseKeystroke( 'Ctrl+Enter' );\n\n/**\n * The engine of the to-do list feature. It handles creating, editing and removing to-do lists and their items.\n *\n * It registers the entire functionality of the {@link module:list/list/listediting~ListEditing list editing plugin} and extends\n * it with the commands:\n *\n * - `'todoList'`,\n * - `'checkTodoList'`,\n * - `'todoListCheck'` as an alias for `checkTodoList` command.\n */\nexport default class TodoListEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TodoListEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ListEditing ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst { editing, data, model } = editor;\n\n\t\t// Extend schema.\n\t\tmodel.schema.extend( 'listItem', {\n\t\t\tallowAttributes: [ 'todoListChecked' ]\n\t\t} );\n\n\t\t// Disallow todoListChecked attribute on other nodes than listItem with to-do listType.\n\t\tmodel.schema.addAttributeCheck( ( context: any, attributeName ) => {\n\t\t\tconst item = context.last;\n\n\t\t\tif ( attributeName == 'todoListChecked' && item.name == 'listItem' && item.getAttribute( 'listType' ) != 'todo' ) {\n\t\t\t\treturn false;\n\t\t\t}\n\t\t} );\n\n\t\t// Register `todoList` command.\n\t\teditor.commands.add( 'todoList', new ListCommand( editor, 'todo' ) );\n\n\t\tconst checkTodoListCommand = new CheckTodoListCommand( editor );\n\n\t\t// Register `checkTodoList` command and add `todoListCheck` command as an alias for backward compatibility.\n\t\teditor.commands.add( 'checkTodoList', checkTodoListCommand );\n\t\teditor.commands.add( 'todoListCheck', checkTodoListCommand );\n\n\t\t// Define converters.\n\t\tdata.downcastDispatcher.on<DowncastInsertEvent<Element>>(\n\t\t\t'insert:listItem',\n\t\t\tdataModelViewInsertion( model ),\n\t\t\t{ priority: 'high' }\n\t\t);\n\t\tdata.upcastDispatcher.on<UpcastElementEvent>(\n\t\t\t'element:input',\n\t\t\tdataViewModelCheckmarkInsertion,\n\t\t\t{ priority: 'high' }\n\t\t);\n\n\t\tediting.downcastDispatcher.on<DowncastInsertEvent<Element>>(\n\t\t\t'insert:listItem',\n\t\t\tmodelViewInsertion( model, listItem => this._handleCheckmarkChange( listItem ) ),\n\t\t\t{ priority: 'high' }\n\t\t);\n\t\tediting.downcastDispatcher.on<DowncastAttributeEvent<Element>>(\n\t\t\t'attribute:listType:listItem',\n\t\t\tmodelViewChangeType( listItem => this._handleCheckmarkChange( listItem ), editing.view )\n\t\t);\n\t\tediting.downcastDispatcher.on<DowncastAttributeEvent<Element>>(\n\t\t\t'attribute:todoListChecked:listItem',\n\t\t\tmodelViewChangeChecked( listItem => this._handleCheckmarkChange( listItem ) )\n\t\t);\n\n\t\tediting.mapper.on<MapperModelToViewPositionEvent>( 'modelToViewPosition', mapModelToViewPosition( editing.view ) );\n\t\tdata.mapper.on<MapperModelToViewPositionEvent>( 'modelToViewPosition', mapModelToViewPosition( editing.view ) );\n\n\t\t// Jump at the end of the previous node on left arrow key press, when selection is after the checkbox.\n\t\t//\n\t\t// <blockquote><p>Foo</p></blockquote>\n\t\t// <ul><li><checkbox/>{}Bar</li></ul>\n\t\t//\n\t\t// press: `<-`\n\t\t//\n\t\t// <blockquote><p>Foo{}</p></blockquote>\n\t\t// <ul><li><checkbox/>Bar</li></ul>\n\t\t//\n\t\tthis.listenTo<ViewDocumentArrowKeyEvent>(\n\t\t\tediting.view.document,\n\t\t\t'arrowKey',\n\t\t\tjumpOverCheckmarkOnSideArrowKeyPress( model, editor.locale ),\n\t\t\t{ context: 'li' }\n\t\t);\n\n\t\t// Toggle check state of selected to-do list items on keystroke.\n\t\tthis.listenTo<ViewDocumentKeyDownEvent>( editing.view.document, 'keydown', ( evt, data ) => {\n\t\t\tif ( getCode( data ) === ITEM_TOGGLE_KEYSTROKE ) {\n\t\t\t\teditor.execute( 'checkTodoList' );\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\n\t\t// Remove `todoListChecked` attribute when a host element is no longer a to-do list item.\n\t\tconst listItemsToFix = new Set<Item>();\n\n\t\tthis.listenTo<ModelApplyOperationEvent>( model, 'applyOperation', ( evt, args ) => {\n\t\t\tconst operation = args[ 0 ] as RenameOperation | AttributeOperation;\n\n\t\t\tif ( operation.type == 'rename' && operation.oldName == 'listItem' ) {\n\t\t\t\tconst item = operation.position.nodeAfter!;\n\n\t\t\t\tif ( item.hasAttribute( 'todoListChecked' ) ) {\n\t\t\t\t\tlistItemsToFix.add( item );\n\t\t\t\t}\n\t\t\t} else if ( operation.type == 'changeAttribute' && operation.key == 'listType' && operation.oldValue === 'todo' ) {\n\t\t\t\tfor ( const item of operation.range.getItems() ) {\n\t\t\t\t\tif ( item.hasAttribute( 'todoListChecked' ) && item.getAttribute( 'listType' ) !== 'todo' ) {\n\t\t\t\t\t\tlistItemsToFix.add( item );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\n\t\tmodel.document.registerPostFixer( writer => {\n\t\t\tlet hasChanged = false;\n\n\t\t\tfor ( const listItem of listItemsToFix ) {\n\t\t\t\twriter.removeAttribute( 'todoListChecked', listItem );\n\t\t\t\thasChanged = true;\n\t\t\t}\n\n\t\t\tlistItemsToFix.clear();\n\n\t\t\treturn hasChanged;\n\t\t} );\n\t}\n\n\t/**\n\t * Handles the checkbox element change, moves the selection to the corresponding model item to make it possible\n\t * to toggle the `todoListChecked` attribute using the command, and restores the selection position.\n\t *\n\t * Some say it's a hack :) Moving the selection only for executing the command on a certain node and restoring it after,\n\t * is not a clear solution. We need to design an API for using commands beyond the selection range.\n\t * See https://github.com/ckeditor/ckeditor5/issues/1954.\n\t */\n\tprivate _handleCheckmarkChange( listItem: Element ) {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst previousSelectionRanges = Array.from( model.document.selection.getRanges() );\n\n\t\tmodel.change( writer => {\n\t\t\twriter.setSelection( listItem, 'end' );\n\t\t\teditor.execute( 'checkTodoList' );\n\t\t\twriter.setSelection( previousSelectionRanges );\n\t\t} );\n\t}\n}\n\n/**\n * Handles the left/right (LTR/RTL content) arrow key and moves the selection at the end of the previous block element\n * if the selection is just after the checkbox element. In other words, it jumps over the checkbox element when\n * moving the selection to the left/right (LTR/RTL).\n *\n * @returns Callback for 'keydown' events.\n */\nfunction jumpOverCheckmarkOnSideArrowKeyPress( model: Model, locale: Locale ): GetCallback<ViewDocumentArrowKeyEvent> {\n\treturn ( eventInfo, domEventData ) => {\n\t\tconst direction = getLocalizedArrowKeyCodeDirection( domEventData.keyCode, locale.contentLanguageDirection );\n\n\t\tif ( direction != 'left' ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst schema = model.schema;\n\t\tconst selection = model.document.selection;\n\n\t\tif ( !selection.isCollapsed ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst position = selection.getFirstPosition()!;\n\t\tconst parent = position.parent;\n\n\t\tif ( parent.name === 'listItem' && parent.getAttribute( 'listType' ) == 'todo' && position.isAtStart ) {\n\t\t\tconst newRange = schema.getNearestSelectionRange( model.createPositionBefore( parent ), 'backward' );\n\n\t\t\tif ( newRange ) {\n\t\t\t\tmodel.change( writer => writer.setSelection( newRange ) );\n\t\t\t}\n\n\t\t\tdomEventData.preventDefault();\n\t\t\tdomEventData.stopPropagation();\n\t\t\teventInfo.stop();\n\t\t}\n\t};\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./todolist.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/converters\n */\n\nimport type { GetCallback } from 'ckeditor5/src/utils';\nimport type { DowncastAttributeEvent, DowncastDispatcher, Element, ViewElement } from 'ckeditor5/src/engine';\nimport type MediaRegistry from './mediaregistry';\nimport type { MediaOptions } from './utils';\n\n/**\n * Returns a function that converts the model \"url\" attribute to the view representation.\n *\n * Depending on the configuration, the view representation can be \"semantic\" (for the data pipeline):\n *\n * ```html\n * <figure class=\"media\">\n * \t<oembed url=\"foo\"></oembed>\n * </figure>\n * ```\n *\n * or \"non-semantic\" (for the editing view pipeline):\n *\n * ```html\n * <figure class=\"media\">\n * \t<div data-oembed-url=\"foo\">[ non-semantic media preview for \"foo\" ]</div>\n * </figure>\n * ```\n *\n * **Note:** Changing the model \"url\" attribute replaces the entire content of the\n * `<figure>` in the view.\n *\n * @param registry The registry providing\n * the media and their content.\n * @param options options object with following properties:\n * - elementName When set, overrides the default element name for semantic media embeds.\n * - renderMediaPreview When `true`, the converter will create the view in the non-semantic form.\n * - renderForEditingView When `true`, the converter will create a view specific for the\n * editing pipeline (e.g. including CSS classes, content placeholders).\n */\nexport function modelToViewUrlAttributeConverter(\n\tregistry: MediaRegistry,\n\toptions: MediaOptions\n): ( dispatcher: DowncastDispatcher ) => void {\n\tconst converter: GetCallback<DowncastAttributeEvent> = ( evt, data, conversionApi ) => {\n\t\tif ( !conversionApi.consumable.consume( data.item, evt.name ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst url = data.attributeNewValue as string;\n\t\tconst viewWriter = conversionApi.writer;\n\t\tconst figure = conversionApi.mapper.toViewElement( data.item as Element )!;\n\t\tconst mediaContentElement = [ ...figure.getChildren() ]\n\t\t\t.find( child => ( child as ViewElement ).getCustomProperty( 'media-content' ) )!;\n\n\t\t// TODO: removing the wrapper and creating it from scratch is a hack. We can do better than that.\n\t\tviewWriter.remove( mediaContentElement );\n\n\t\tconst mediaViewElement = registry.getMediaViewElement( viewWriter, url, options );\n\n\t\tviewWriter.insert( viewWriter.createPositionAt( figure, 0 ), mediaViewElement );\n\t};\n\n\treturn dispatcher => {\n\t\tdispatcher.on<DowncastAttributeEvent>( 'attribute:url:media', converter );\n\t};\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/utils\n */\n\nimport type {\n\tViewContainerElement,\n\tElement,\n\tModel,\n\tSelectable,\n\tSelection,\n\tDowncastWriter,\n\tViewDocumentSelection,\n\tViewElement,\n\tDocumentSelection\n} from 'ckeditor5/src/engine';\nimport { isWidget, toWidget } from 'ckeditor5/src/widget';\nimport type MediaRegistry from './mediaregistry';\n\n/**\n * Converts a given {@link module:engine/view/element~Element} to a media embed widget:\n * * Adds a {@link module:engine/view/element~Element#_setCustomProperty custom property} allowing to recognize the media widget element.\n * * Calls the {@link module:widget/utils~toWidget} function with the proper element's label creator.\n *\n * @param writer An instance of the view writer.\n * @param label The element's label.\n */\nexport function toMediaWidget( viewElement: ViewElement, writer: DowncastWriter, label: string ): ViewElement {\n\twriter.setCustomProperty( 'media', true, viewElement );\n\n\treturn toWidget( viewElement, writer, { label } );\n}\n\n/**\n * Returns a media widget editing view element if one is selected.\n */\nexport function getSelectedMediaViewWidget( selection: ViewDocumentSelection ): ViewElement | null {\n\tconst viewElement = selection.getSelectedElement();\n\n\tif ( viewElement && isMediaWidget( viewElement ) ) {\n\t\treturn viewElement;\n\t}\n\n\treturn null;\n}\n\n/**\n * Checks if a given view element is a media widget.\n */\nexport function isMediaWidget( viewElement: ViewElement ): boolean {\n\treturn !!viewElement.getCustomProperty( 'media' ) && isWidget( viewElement );\n}\n\n/**\n * Creates a view element representing the media. Either a \"semantic\" one for the data pipeline:\n *\n * ```html\n * <figure class=\"media\">\n * \t<oembed url=\"foo\"></oembed>\n * </figure>\n * ```\n *\n * or a \"non-semantic\" (for the editing view pipeline):\n *\n * ```html\n * <figure class=\"media\">\n * \t<div data-oembed-url=\"foo\">[ non-semantic media preview for \"foo\" ]</div>\n * </figure>\n * ```\n */\nexport function createMediaFigureElement(\n\twriter: DowncastWriter,\n\tregistry: MediaRegistry,\n\turl: string,\n\toptions: MediaOptions\n): ViewContainerElement {\n\treturn writer.createContainerElement( 'figure', { class: 'media' }, [\n\t\tregistry.getMediaViewElement( writer, url, options ),\n\t\twriter.createSlot()\n\t] );\n}\n\n/**\n * Returns a selected media element in the model, if any.\n */\nexport function getSelectedMediaModelWidget( selection: Selection | DocumentSelection ): Element | null {\n\tconst selectedElement = selection.getSelectedElement();\n\n\tif ( selectedElement && selectedElement.is( 'element', 'media' ) ) {\n\t\treturn selectedElement;\n\t}\n\n\treturn null;\n}\n\n/**\n * Creates a media element and inserts it into the model.\n *\n * **Note**: This method will use {@link module:engine/model/model~Model#insertContent `model.insertContent()`} logic of inserting content\n * if no `insertPosition` is passed.\n *\n * @param url An URL of an embeddable media.\n * @param findOptimalPosition If true it will try to find optimal position to insert media without breaking content\n * in which a selection is.\n */\nexport function insertMedia( model: Model, url: string, selectable: Selectable, findOptimalPosition: boolean ): void {\n\tmodel.change( writer => {\n\t\tconst mediaElement = writer.createElement( 'media', { url } );\n\n\t\tmodel.insertObject( mediaElement, selectable, null, {\n\t\t\tsetSelection: 'on',\n\t\t\tfindOptimalPosition: findOptimalPosition ? 'auto' : undefined\n\t\t} );\n\t} );\n}\n\n/**\n * Type for commonly grouped function parameters across this package.\n */\nexport type MediaOptions = {\n\telementName: string;\n\trenderMediaPreview?: boolean;\n\trenderForEditingView?: boolean;\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/mediaembedcommand\n */\n\nimport type { DocumentSelection, Element, Model, Selection } from 'ckeditor5/src/engine';\nimport { Command } from 'ckeditor5/src/core';\nimport { findOptimalInsertionRange } from 'ckeditor5/src/widget';\n\nimport { getSelectedMediaModelWidget, insertMedia } from './utils';\n\n/**\n * The insert media command.\n *\n * The command is registered by the {@link module:media-embed/mediaembedediting~MediaEmbedEditing} as `'mediaEmbed'`.\n *\n * To insert media at the current selection, execute the command and specify the URL:\n *\n * ```ts\n * editor.execute( 'mediaEmbed', 'http://url.to.the/media' );\n * ```\n */\nexport default class MediaEmbedCommand extends Command {\n\t/**\n\t * Media url.\n\t */\n\tdeclare public value: string | undefined;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst selectedMedia = getSelectedMediaModelWidget( selection );\n\n\t\tthis.value = selectedMedia ? selectedMedia.getAttribute( 'url' ) as string : undefined;\n\n\t\tthis.isEnabled = isMediaSelected( selection ) || isAllowedInParent( selection, model );\n\t}\n\n\t/**\n\t * Executes the command, which either:\n\t *\n\t * * updates the URL of the selected media,\n\t * * inserts the new media into the editor and puts the selection around it.\n\t *\n\t * @fires execute\n\t * @param url The URL of the media.\n\t */\n\tpublic override execute( url: string ): void {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst selectedMedia = getSelectedMediaModelWidget( selection );\n\n\t\tif ( selectedMedia ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\twriter.setAttribute( 'url', url, selectedMedia );\n\t\t\t} );\n\t\t} else {\n\t\t\tinsertMedia( model, url, selection, true );\n\t\t}\n\t}\n}\n\n/**\n * Checks if the media embed is allowed in the parent.\n */\nfunction isAllowedInParent( selection: Selection | DocumentSelection, model: Model ): boolean {\n\tconst insertionRange = findOptimalInsertionRange( selection, model );\n\tlet parent = insertionRange.start.parent as Element;\n\n\t// The model.insertContent() will remove empty parent (unless it is a $root or a limit).\n\tif ( parent.isEmpty && !model.schema.isLimit( parent ) ) {\n\t\tparent = parent.parent as Element;\n\t}\n\n\treturn model.schema.checkChild( parent, 'media' );\n}\n\n/**\n * Checks if the media object is selected.\n */\nfunction isMediaSelected( selection: Selection | DocumentSelection ): boolean {\n\tconst element = selection.getSelectedElement();\n\treturn !!element && element.name === 'media';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/mediaregistry\n */\n\nimport type { DowncastWriter, ViewElement } from 'ckeditor5/src/engine';\nimport { IconView, Template } from 'ckeditor5/src/ui';\nimport { type Locale, logWarning, toArray } from 'ckeditor5/src/utils';\n\nimport mediaPlaceholderIcon from '../theme/icons/media-placeholder.svg';\nimport type { MediaEmbedConfig, MediaEmbedProvider } from './mediaembedconfig';\nimport type { MediaOptions } from './utils';\n\nconst mediaPlaceholderIconViewBox = '0 0 64 42';\n\n/**\n * A bridge between the raw media content provider definitions and the editor view content.\n *\n * It helps translating media URLs to corresponding {@link module:engine/view/element~Element view elements}.\n *\n * Mostly used by the {@link module:media-embed/mediaembedediting~MediaEmbedEditing} plugin.\n */\nexport default class MediaRegistry {\n\t/**\n\t * The {@link module:utils/locale~Locale} instance.\n\t */\n\tpublic locale: Locale;\n\n\t/**\n\t * The media provider definitions available for the registry. Usually corresponding with the\n\t * {@link module:media-embed/mediaembedconfig~MediaEmbedConfig media configuration}.\n\t */\n\tpublic providerDefinitions: Array<MediaEmbedProvider>;\n\n\t/**\n\t * Creates an instance of the {@link module:media-embed/mediaregistry~MediaRegistry} class.\n\t *\n\t * @param locale The localization services instance.\n\t * @param config The configuration of the media embed feature.\n\t */\n\tconstructor( locale: Locale, config: MediaEmbedConfig ) {\n\t\tconst providers = config.providers!;\n\t\tconst extraProviders = config.extraProviders || [];\n\t\tconst removedProviders = new Set( config.removeProviders );\n\t\tconst providerDefinitions = providers\n\t\t\t.concat( extraProviders )\n\t\t\t.filter( provider => {\n\t\t\t\tconst name = provider.name;\n\n\t\t\t\tif ( !name ) {\n\t\t\t\t\t/**\n\t\t\t\t\t * One of the providers (or extra providers) specified in the media embed configuration\n\t\t\t\t\t * has no name and will not be used by the editor. In order to get this media\n\t\t\t\t\t * provider working, double check your editor configuration.\n\t\t\t\t\t *\n\t\t\t\t\t * @error media-embed-no-provider-name\n\t\t\t\t\t */\n\t\t\t\t\tlogWarning( 'media-embed-no-provider-name', { provider } );\n\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\treturn !removedProviders.has( name );\n\t\t\t} );\n\n\t\tthis.locale = locale;\n\t\tthis.providerDefinitions = providerDefinitions;\n\t}\n\n\t/**\n\t * Checks whether the passed URL is representing a certain media type allowed in the editor.\n\t *\n\t * @param url The URL to be checked\n\t */\n\tpublic hasMedia( url: string ): boolean {\n\t\treturn !!this._getMedia( url );\n\t}\n\n\t/**\n\t * For the given media URL string and options, it returns the {@link module:engine/view/element~Element view element}\n\t * representing that media.\n\t *\n\t * **Note:** If no URL is specified, an empty view element is returned.\n\t *\n\t * @param writer The view writer used to produce a view element.\n\t * @param url The URL to be translated into a view element.\n\t */\n\tpublic getMediaViewElement(\n\t\twriter: DowncastWriter,\n\t\turl: string,\n\t\toptions: MediaOptions\n\t): ViewElement {\n\t\treturn this._getMedia( url )!.getViewElement( writer, options );\n\t}\n\n\t/**\n\t * Returns a `Media` instance for the given URL.\n\t *\n\t * @param url The URL of the media.\n\t * @returns The `Media` instance or `null` when there is none.\n\t */\n\tprivate _getMedia( url: string ): Media | null {\n\t\tif ( !url ) {\n\t\t\treturn new Media( this.locale );\n\t\t}\n\n\t\turl = url.trim();\n\n\t\tfor ( const definition of this.providerDefinitions ) {\n\t\t\tconst previewRenderer = definition.html!;\n\t\t\tconst pattern = toArray( definition.url );\n\n\t\t\tfor ( const subPattern of pattern ) {\n\t\t\t\tconst match = this._getUrlMatches( url, subPattern );\n\n\t\t\t\tif ( match ) {\n\t\t\t\t\treturn new Media( this.locale, url, match, previewRenderer );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Tries to match `url` to `pattern`.\n\t *\n\t * @param url The URL of the media.\n\t * @param pattern The pattern that should accept the media URL.\n\t */\n\tprivate _getUrlMatches( url: string, pattern: RegExp ): RegExpMatchArray | null {\n\t\t// 1. Try to match without stripping the protocol and \"www\" subdomain.\n\t\tlet match = url.match( pattern );\n\n\t\tif ( match ) {\n\t\t\treturn match;\n\t\t}\n\n\t\t// 2. Try to match after stripping the protocol.\n\t\tlet rawUrl = url.replace( /^https?:\\/\\//, '' );\n\t\tmatch = rawUrl.match( pattern );\n\n\t\tif ( match ) {\n\t\t\treturn match;\n\t\t}\n\n\t\t// 3. Try to match after stripping the \"www\" subdomain.\n\t\trawUrl = rawUrl.replace( /^www\\./, '' );\n\t\tmatch = rawUrl.match( pattern );\n\n\t\tif ( match ) {\n\t\t\treturn match;\n\t\t}\n\n\t\treturn null;\n\t}\n}\n\n/**\n * Represents media defined by the provider configuration.\n *\n * It can be rendered to the {@link module:engine/view/element~Element view element} and used in the editing or data pipeline.\n */\nclass Media {\n\t/**\n\t * The URL this Media instance represents.\n\t */\n\tpublic url: string | null;\n\n\t/**\n\t * Shorthand for {@link module:utils/locale~Locale#t}.\n\t *\n\t * @see module:utils/locale~Locale#t\n\t */\n\tprivate _locale: Locale;\n\n\t/**\n\t * The output of the `RegExp.match` which validated the {@link #url} of this media.\n\t */\n\tprivate _match?: RegExpMatchArray;\n\n\t/**\n\t * The function returning the HTML string preview of this media.\n\t */\n\tprivate _previewRenderer?: ( match: RegExpMatchArray ) => string;\n\n\tconstructor( locale: Locale, url?: string, match?: RegExpMatchArray, previewRenderer?: ( match: RegExpMatchArray ) => string ) {\n\t\tthis.url = this._getValidUrl( url );\n\t\tthis._locale = locale;\n\t\tthis._match = match;\n\t\tthis._previewRenderer = previewRenderer;\n\t}\n\n\t/**\n\t * Returns the view element representation of the media.\n\t *\n\t * @param writer The view writer used to produce a view element.\n\t */\n\tpublic getViewElement(\n\t\twriter: DowncastWriter,\n\t\toptions: MediaOptions\n\t): ViewElement {\n\t\tconst attributes: Record<string, unknown> = {};\n\t\tlet viewElement;\n\n\t\tif ( options.renderForEditingView || ( options.renderMediaPreview && this.url && this._previewRenderer ) ) {\n\t\t\tif ( this.url ) {\n\t\t\t\tattributes[ 'data-oembed-url' ] = this.url;\n\t\t\t}\n\n\t\t\tif ( options.renderForEditingView ) {\n\t\t\t\tattributes.class = 'ck-media__wrapper';\n\t\t\t}\n\n\t\t\tconst mediaHtml = this._getPreviewHtml( options );\n\n\t\t\tviewElement = writer.createRawElement( 'div', attributes, ( domElement, domConverter ) => {\n\t\t\t\tdomConverter!.setContentOf( domElement, mediaHtml );\n\t\t\t} );\n\t\t} else {\n\t\t\tif ( this.url ) {\n\t\t\t\tattributes.url = this.url;\n\t\t\t}\n\n\t\t\tviewElement = writer.createEmptyElement( options.elementName, attributes );\n\t\t}\n\n\t\twriter.setCustomProperty( 'media-content', true, viewElement );\n\n\t\treturn viewElement;\n\t}\n\n\t/**\n\t * Returns the HTML string of the media content preview.\n\t */\n\tprivate _getPreviewHtml( options: { renderForEditingView?: boolean } ): string {\n\t\tif ( this._previewRenderer ) {\n\t\t\treturn this._previewRenderer( this._match! );\n\t\t} else {\n\t\t\t// The placeholder only makes sense for editing view and media which have URLs.\n\t\t\t// Placeholder is never displayed in data and URL-less media have no content.\n\t\t\tif ( this.url && options.renderForEditingView ) {\n\t\t\t\treturn this._getPlaceholderHtml();\n\t\t\t}\n\n\t\t\treturn '';\n\t\t}\n\t}\n\n\t/**\n\t * Returns the placeholder HTML when the media has no content preview.\n\t */\n\tprivate _getPlaceholderHtml(): string {\n\t\tconst icon = new IconView();\n\t\tconst t = this._locale.t;\n\n\t\ticon.content = mediaPlaceholderIcon;\n\t\ticon.viewBox = mediaPlaceholderIconViewBox;\n\n\t\tconst placeholder = new Template( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: 'ck ck-reset_all ck-media__placeholder'\n\t\t\t},\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttag: 'div',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: 'ck-media__placeholder__icon'\n\t\t\t\t\t},\n\t\t\t\t\tchildren: [ icon ]\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttag: 'a',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: 'ck-media__placeholder__url',\n\t\t\t\t\t\ttarget: '_blank',\n\t\t\t\t\t\trel: 'noopener noreferrer',\n\t\t\t\t\t\thref: this.url,\n\t\t\t\t\t\t'data-cke-tooltip-text': t( 'Open media in new tab' )\n\t\t\t\t\t},\n\t\t\t\t\tchildren: [\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\ttag: 'span',\n\t\t\t\t\t\t\tattributes: {\n\t\t\t\t\t\t\t\tclass: 'ck-media__placeholder__url__text'\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t\tchildren: [ this.url ]\n\t\t\t\t\t\t}\n\t\t\t\t\t]\n\t\t\t\t}\n\t\t\t]\n\t\t} ).render() as HTMLElement;\n\n\t\treturn placeholder.outerHTML;\n\t}\n\n\t/**\n\t * Returns the full URL to the specified media.\n\t *\n\t * @param url The URL of the media.\n\t */\n\tprivate _getValidUrl( url: string | undefined ): string | null {\n\t\tif ( !url ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( url.match( /^https?/ ) ) {\n\t\t\treturn url;\n\t\t}\n\n\t\treturn 'https://' + url;\n\t}\n}\n","export default \"<svg viewBox=\\\"0 0 64 42\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M47.426 17V3.713L63.102 0v19.389h-.001l.001.272c0 1.595-2.032 3.43-4.538 4.098-2.506.668-4.538-.083-4.538-1.678 0-1.594 2.032-3.43 4.538-4.098.914-.244 2.032-.565 2.888-.603V4.516L49.076 7.447v9.556A1.014 1.014 0 0 0 49 17h-1.574zM29.5 17h-8.343a7.073 7.073 0 1 0-4.657 4.06v3.781H3.3a2.803 2.803 0 0 1-2.8-2.804V8.63a2.803 2.803 0 0 1 2.8-2.805h4.082L8.58 2.768A1.994 1.994 0 0 1 10.435 1.5h8.985c.773 0 1.477.448 1.805 1.149l1.488 3.177H26.7c1.546 0 2.8 1.256 2.8 2.805V17zm-11.637 0H17.5a1 1 0 0 0-1 1v.05A4.244 4.244 0 1 1 17.863 17zm29.684 2c.97 0 .953-.048.953.889v20.743c0 .953.016.905-.953.905H19.453c-.97 0-.953.048-.953-.905V19.89c0-.937-.016-.889.97-.889h28.077zm-4.701 19.338V22.183H24.154v16.155h18.692zM20.6 21.375v1.616h1.616v-1.616H20.6zm0 3.231v1.616h1.616v-1.616H20.6zm0 3.231v1.616h1.616v-1.616H20.6zm0 3.231v1.616h1.616v-1.616H20.6zm0 3.231v1.616h1.616v-1.616H20.6zm0 3.231v1.616h1.616V37.53H20.6zm24.233-16.155v1.616h1.615v-1.616h-1.615zm0 3.231v1.616h1.615v-1.616h-1.615zm0 3.231v1.616h1.615v-1.616h-1.615zm0 3.231v1.616h1.615v-1.616h-1.615zm0 3.231v1.616h1.615v-1.616h-1.615zm0 3.231v1.616h1.615V37.53h-1.615zM29.485 25.283a.4.4 0 0 1 .593-.35l9.05 4.977a.4.4 0 0 1 0 .701l-9.05 4.978a.4.4 0 0 1-.593-.35v-9.956z\\\"/></svg>\";","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./mediaembedediting.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/mediaembedediting\n */\n\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\nimport type { UpcastElementEvent } from 'ckeditor5/src/engine';\nimport { first, type GetCallback } from 'ckeditor5/src/utils';\n\nimport { modelToViewUrlAttributeConverter } from './converters';\nimport type { MediaEmbedConfig } from './mediaembedconfig';\nimport MediaEmbedCommand from './mediaembedcommand';\nimport MediaRegistry from './mediaregistry';\nimport { toMediaWidget, createMediaFigureElement } from './utils';\n\nimport '../theme/mediaembedediting.css';\n\n/**\n * The media embed editing feature.\n */\nexport default class MediaEmbedEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'MediaEmbedEditing' as const;\n\t}\n\n\t/**\n\t * The media registry managing the media providers in the editor.\n\t */\n\tpublic registry: MediaRegistry;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\t\teditor.config.define( 'mediaEmbed', {\n\t\t\telementName: 'oembed',\n\t\t\tproviders: [\n\t\t\t\t{\n\t\t\t\t\tname: 'dailymotion',\n\t\t\t\t\turl: /^dailymotion\\.com\\/video\\/(\\w+)/,\n\t\t\t\t\thtml: match => {\n\t\t\t\t\t\tconst id = match[ 1 ];\n\n\t\t\t\t\t\treturn (\n\t\t\t\t\t\t\t'<div style=\"position: relative; padding-bottom: 100%; height: 0; \">' +\n\t\t\t\t\t\t\t\t`<iframe src=\"https://www.dailymotion.com/embed/video/${ id }\" ` +\n\t\t\t\t\t\t\t\t\t'style=\"position: absolute; width: 100%; height: 100%; top: 0; left: 0;\" ' +\n\t\t\t\t\t\t\t\t\t'frameborder=\"0\" width=\"480\" height=\"270\" allowfullscreen allow=\"autoplay\">' +\n\t\t\t\t\t\t\t\t'</iframe>' +\n\t\t\t\t\t\t\t'</div>'\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t},\n\n\t\t\t\t{\n\t\t\t\t\tname: 'spotify',\n\t\t\t\t\turl: [\n\t\t\t\t\t\t/^open\\.spotify\\.com\\/(artist\\/\\w+)/,\n\t\t\t\t\t\t/^open\\.spotify\\.com\\/(album\\/\\w+)/,\n\t\t\t\t\t\t/^open\\.spotify\\.com\\/(track\\/\\w+)/\n\t\t\t\t\t],\n\t\t\t\t\thtml: match => {\n\t\t\t\t\t\tconst id = match[ 1 ];\n\n\t\t\t\t\t\treturn (\n\t\t\t\t\t\t\t'<div style=\"position: relative; padding-bottom: 100%; height: 0; padding-bottom: 126%;\">' +\n\t\t\t\t\t\t\t\t`<iframe src=\"https://open.spotify.com/embed/${ id }\" ` +\n\t\t\t\t\t\t\t\t\t'style=\"position: absolute; width: 100%; height: 100%; top: 0; left: 0;\" ' +\n\t\t\t\t\t\t\t\t\t'frameborder=\"0\" allowtransparency=\"true\" allow=\"encrypted-media\">' +\n\t\t\t\t\t\t\t\t'</iframe>' +\n\t\t\t\t\t\t\t'</div>'\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t},\n\n\t\t\t\t{\n\t\t\t\t\tname: 'youtube',\n\t\t\t\t\turl: [\n\t\t\t\t\t\t/^(?:m\\.)?youtube\\.com\\/watch\\?v=([\\w-]+)(?:&t=(\\d+))?/,\n\t\t\t\t\t\t/^(?:m\\.)?youtube\\.com\\/v\\/([\\w-]+)(?:\\?t=(\\d+))?/,\n\t\t\t\t\t\t/^youtube\\.com\\/embed\\/([\\w-]+)(?:\\?start=(\\d+))?/,\n\t\t\t\t\t\t/^youtu\\.be\\/([\\w-]+)(?:\\?t=(\\d+))?/\n\t\t\t\t\t],\n\t\t\t\t\thtml: match => {\n\t\t\t\t\t\tconst id = match[ 1 ];\n\t\t\t\t\t\tconst time = match[ 2 ];\n\n\t\t\t\t\t\treturn (\n\t\t\t\t\t\t\t'<div style=\"position: relative; padding-bottom: 100%; height: 0; padding-bottom: 56.2493%;\">' +\n\t\t\t\t\t\t\t\t`<iframe src=\"https://www.youtube.com/embed/${ id }${ time ? `?start=${ time }` : '' }\" ` +\n\t\t\t\t\t\t\t\t\t'style=\"position: absolute; width: 100%; height: 100%; top: 0; left: 0;\" ' +\n\t\t\t\t\t\t\t\t\t'frameborder=\"0\" allow=\"autoplay; encrypted-media\" allowfullscreen>' +\n\t\t\t\t\t\t\t\t'</iframe>' +\n\t\t\t\t\t\t\t'</div>'\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t},\n\n\t\t\t\t{\n\t\t\t\t\tname: 'vimeo',\n\t\t\t\t\turl: [\n\t\t\t\t\t\t/^vimeo\\.com\\/(\\d+)/,\n\t\t\t\t\t\t/^vimeo\\.com\\/[^/]+\\/[^/]+\\/video\\/(\\d+)/,\n\t\t\t\t\t\t/^vimeo\\.com\\/album\\/[^/]+\\/video\\/(\\d+)/,\n\t\t\t\t\t\t/^vimeo\\.com\\/channels\\/[^/]+\\/(\\d+)/,\n\t\t\t\t\t\t/^vimeo\\.com\\/groups\\/[^/]+\\/videos\\/(\\d+)/,\n\t\t\t\t\t\t/^vimeo\\.com\\/ondemand\\/[^/]+\\/(\\d+)/,\n\t\t\t\t\t\t/^player\\.vimeo\\.com\\/video\\/(\\d+)/\n\t\t\t\t\t],\n\t\t\t\t\thtml: match => {\n\t\t\t\t\t\tconst id = match[ 1 ];\n\n\t\t\t\t\t\treturn (\n\t\t\t\t\t\t\t'<div style=\"position: relative; padding-bottom: 100%; height: 0; padding-bottom: 56.2493%;\">' +\n\t\t\t\t\t\t\t\t`<iframe src=\"https://player.vimeo.com/video/${ id }\" ` +\n\t\t\t\t\t\t\t\t\t'style=\"position: absolute; width: 100%; height: 100%; top: 0; left: 0;\" ' +\n\t\t\t\t\t\t\t\t\t'frameborder=\"0\" webkitallowfullscreen mozallowfullscreen allowfullscreen>' +\n\t\t\t\t\t\t\t\t'</iframe>' +\n\t\t\t\t\t\t\t'</div>'\n\t\t\t\t\t\t);\n\t\t\t\t\t}\n\t\t\t\t},\n\n\t\t\t\t{\n\t\t\t\t\tname: 'instagram',\n\t\t\t\t\turl: /^instagram\\.com\\/p\\/(\\w+)/\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tname: 'twitter',\n\t\t\t\t\turl: /^twitter\\.com/\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tname: 'googleMaps',\n\t\t\t\t\turl: [\n\t\t\t\t\t\t/^google\\.com\\/maps/,\n\t\t\t\t\t\t/^goo\\.gl\\/maps/,\n\t\t\t\t\t\t/^maps\\.google\\.com/,\n\t\t\t\t\t\t/^maps\\.app\\.goo\\.gl/\n\t\t\t\t\t]\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tname: 'flickr',\n\t\t\t\t\turl: /^flickr\\.com/\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\tname: 'facebook',\n\t\t\t\t\turl: /^facebook\\.com/\n\t\t\t\t}\n\t\t\t]\n\t\t} as MediaEmbedConfig );\n\n\t\tthis.registry = new MediaRegistry( editor.locale, editor.config.get( 'mediaEmbed' )! );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst schema = editor.model.schema;\n\t\tconst t = editor.t;\n\t\tconst conversion = editor.conversion;\n\t\tconst renderMediaPreview = editor.config.get( 'mediaEmbed.previewsInData' );\n\t\tconst elementName = editor.config.get( 'mediaEmbed.elementName' )!;\n\n\t\tconst registry = this.registry;\n\n\t\teditor.commands.add( 'mediaEmbed', new MediaEmbedCommand( editor ) );\n\n\t\t// Configure the schema.\n\t\tschema.register( 'media', {\n\t\t\tinheritAllFrom: '$blockObject',\n\t\t\tallowAttributes: [ 'url' ]\n\t\t} );\n\n\t\t// Model -> Data\n\t\tconversion.for( 'dataDowncast' ).elementToStructure( {\n\t\t\tmodel: 'media',\n\t\t\tview: ( modelElement, { writer } ) => {\n\t\t\t\tconst url = modelElement.getAttribute( 'url' ) as string;\n\n\t\t\t\treturn createMediaFigureElement( writer, registry, url, {\n\t\t\t\t\telementName,\n\t\t\t\t\trenderMediaPreview: !!url && renderMediaPreview\n\t\t\t\t} );\n\t\t\t}\n\t\t} );\n\n\t\t// Model -> Data (url -> data-oembed-url)\n\t\tconversion.for( 'dataDowncast' ).add(\n\t\t\tmodelToViewUrlAttributeConverter( registry, {\n\t\t\t\telementName,\n\t\t\t\trenderMediaPreview\n\t\t\t} ) );\n\n\t\t// Model -> View (element)\n\t\tconversion.for( 'editingDowncast' ).elementToStructure( {\n\t\t\tmodel: 'media',\n\t\t\tview: ( modelElement, { writer } ) => {\n\t\t\t\tconst url = modelElement.getAttribute( 'url' ) as string;\n\t\t\t\tconst figure = createMediaFigureElement( writer, registry, url, {\n\t\t\t\t\telementName,\n\t\t\t\t\trenderForEditingView: true\n\t\t\t\t} );\n\n\t\t\t\treturn toMediaWidget( figure, writer, t( 'media widget' ) );\n\t\t\t}\n\t\t} );\n\n\t\t// Model -> View (url -> data-oembed-url)\n\t\tconversion.for( 'editingDowncast' ).add(\n\t\t\tmodelToViewUrlAttributeConverter( registry, {\n\t\t\t\telementName,\n\t\t\t\trenderForEditingView: true\n\t\t\t} ) );\n\n\t\t// View -> Model (data-oembed-url -> url)\n\t\tconversion.for( 'upcast' )\n\t\t\t// Upcast semantic media.\n\t\t\t.elementToElement( {\n\t\t\t\tview: element => [ 'oembed', elementName ].includes( element.name ) && element.getAttribute( 'url' ) ?\n\t\t\t\t\t{ name: true } :\n\t\t\t\t\tnull,\n\t\t\t\tmodel: ( viewMedia, { writer } ) => {\n\t\t\t\t\tconst url = viewMedia.getAttribute( 'url' ) as string;\n\n\t\t\t\t\tif ( registry.hasMedia( url ) ) {\n\t\t\t\t\t\treturn writer.createElement( 'media', { url } );\n\t\t\t\t\t}\n\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\t\t\t} )\n\t\t\t// Upcast non-semantic media.\n\t\t\t.elementToElement( {\n\t\t\t\tview: {\n\t\t\t\t\tname: 'div',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\t'data-oembed-url': true\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\tmodel: ( viewMedia, { writer } ) => {\n\t\t\t\t\tconst url = viewMedia.getAttribute( 'data-oembed-url' ) as string;\n\n\t\t\t\t\tif ( registry.hasMedia( url ) ) {\n\t\t\t\t\t\treturn writer.createElement( 'media', { url } );\n\t\t\t\t\t}\n\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\t\t\t} )\n\t\t\t// Consume `<figure class=\"media\">` elements, that were left after upcast.\n\t\t\t.add( dispatcher => {\n\t\t\t\tconst converter: GetCallback<UpcastElementEvent> = ( evt, data, conversionApi ) => {\n\t\t\t\t\tif ( !conversionApi.consumable.consume( data.viewItem, { name: true, classes: 'media' } ) ) {\n\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\n\t\t\t\t\tconst { modelRange, modelCursor } = conversionApi.convertChildren( data.viewItem, data.modelCursor );\n\n\t\t\t\t\tdata.modelRange = modelRange;\n\t\t\t\t\tdata.modelCursor = modelCursor;\n\n\t\t\t\t\tconst modelElement = first( modelRange!.getItems() );\n\n\t\t\t\t\tif ( !modelElement ) {\n\t\t\t\t\t\t// Revert consumed figure so other features can convert it.\n\t\t\t\t\t\tconversionApi.consumable.revert( data.viewItem, { name: true, classes: 'media' } );\n\t\t\t\t\t}\n\t\t\t\t};\n\n\t\t\t\tdispatcher.on<UpcastElementEvent>( 'element:figure', converter );\n\t\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/automediaembed\n */\n\nimport { type Editor, Plugin } from 'ckeditor5/src/core';\nimport { LiveRange, LivePosition } from 'ckeditor5/src/engine';\nimport { Clipboard, type ClipboardPipeline } from 'ckeditor5/src/clipboard';\nimport { Delete } from 'ckeditor5/src/typing';\nimport { Undo, type UndoCommand } from 'ckeditor5/src/undo';\nimport { global } from 'ckeditor5/src/utils';\n\nimport MediaEmbedEditing from './mediaembedediting';\nimport { insertMedia } from './utils';\nimport type MediaEmbedCommand from './mediaembedcommand';\n\nconst URL_REGEXP = /^(?:http(s)?:\\/\\/)?[\\w-]+\\.[\\w-.~:/?#[\\]@!$&'()*+,;=%]+$/;\n\n/**\n * The auto-media embed plugin. It recognizes media links in the pasted content and embeds\n * them shortly after they are injected into the document.\n */\nexport default class AutoMediaEmbed extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Clipboard, Delete, Undo ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'AutoMediaEmbed' as const;\n\t}\n\n\t/**\n\t * The pastetoembed `setTimeout` ID. Stored as a property to allow\n\t * cleaning of the timeout.\n\t */\n\tprivate _timeoutId: number | null;\n\n\t/**\n\t * The position where the `<media>` element will be inserted after the timeout,\n\t * determined each time the new content is pasted into the document.\n\t */\n\tprivate _positionToInsert: LivePosition | null;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis._timeoutId = null;\n\t\tthis._positionToInsert = null;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst modelDocument = editor.model.document;\n\n\t\t// We need to listen on `Clipboard#inputTransformation` because we need to save positions of selection.\n\t\t// After pasting, the content between those positions will be checked for a URL that could be transformed\n\t\t// into media.\n\t\tconst clipboardPipeline: ClipboardPipeline = editor.plugins.get( 'ClipboardPipeline' );\n\t\tthis.listenTo( clipboardPipeline, 'inputTransformation', () => {\n\t\t\tconst firstRange = modelDocument.selection.getFirstRange()!;\n\n\t\t\tconst leftLivePosition = LivePosition.fromPosition( firstRange.start );\n\t\t\tleftLivePosition.stickiness = 'toPrevious';\n\n\t\t\tconst rightLivePosition = LivePosition.fromPosition( firstRange.end );\n\t\t\trightLivePosition.stickiness = 'toNext';\n\n\t\t\tmodelDocument.once( 'change:data', () => {\n\t\t\t\tthis._embedMediaBetweenPositions( leftLivePosition, rightLivePosition );\n\n\t\t\t\tleftLivePosition.detach();\n\t\t\t\trightLivePosition.detach();\n\t\t\t}, { priority: 'high' } );\n\t\t} );\n\n\t\tconst undoCommand: UndoCommand = editor.commands.get( 'undo' )!;\n\t\tundoCommand.on( 'execute', () => {\n\t\t\tif ( this._timeoutId ) {\n\t\t\t\tglobal.window.clearTimeout( this._timeoutId );\n\t\t\t\tthis._positionToInsert!.detach();\n\n\t\t\t\tthis._timeoutId = null;\n\t\t\t\tthis._positionToInsert = null;\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\t}\n\n\t/**\n\t * Analyzes the part of the document between provided positions in search for a URL representing media.\n\t * When the URL is found, it is automatically converted into media.\n\t *\n\t * @param leftPosition Left position of the selection.\n\t * @param rightPosition Right position of the selection.\n\t */\n\tprivate _embedMediaBetweenPositions( leftPosition: LivePosition, rightPosition: LivePosition ): void {\n\t\tconst editor = this.editor;\n\t\tconst mediaRegistry = editor.plugins.get( MediaEmbedEditing ).registry;\n\t\t// TODO: Use marker instead of LiveRange & LivePositions.\n\t\tconst urlRange = new LiveRange( leftPosition, rightPosition );\n\t\tconst walker = urlRange.getWalker( { ignoreElementEnd: true } );\n\n\t\tlet url = '';\n\n\t\tfor ( const node of walker ) {\n\t\t\tif ( node.item.is( '$textProxy' ) ) {\n\t\t\t\turl += node.item.data;\n\t\t\t}\n\t\t}\n\n\t\turl = url.trim();\n\n\t\t// If the URL does not match to universal URL regexp, let's skip that.\n\t\tif ( !url.match( URL_REGEXP ) ) {\n\t\t\turlRange.detach();\n\n\t\t\treturn;\n\t\t}\n\n\t\t// If the URL represents a media, let's use it.\n\t\tif ( !mediaRegistry.hasMedia( url ) ) {\n\t\t\turlRange.detach();\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst mediaEmbedCommand: MediaEmbedCommand = editor.commands.get( 'mediaEmbed' )!;\n\n\t\t// Do not anything if media element cannot be inserted at the current position (#47).\n\t\tif ( !mediaEmbedCommand.isEnabled ) {\n\t\t\turlRange.detach();\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Position won't be available in the `setTimeout` function so let's clone it.\n\t\tthis._positionToInsert = LivePosition.fromPosition( leftPosition );\n\n\t\t// This action mustn't be executed if undo was called between pasting and auto-embedding.\n\t\tthis._timeoutId = global.window.setTimeout( () => {\n\t\t\teditor.model.change( writer => {\n\t\t\t\tthis._timeoutId = null;\n\n\t\t\t\twriter.remove( urlRange );\n\t\t\t\turlRange.detach();\n\n\t\t\t\tlet insertionPosition: LivePosition | null = null;\n\n\t\t\t\t// Check if position where the media element should be inserted is still valid.\n\t\t\t\t// Otherwise leave it as undefined to use document.selection - default behavior of model.insertContent().\n\t\t\t\tif ( this._positionToInsert!.root.rootName !== '$graveyard' ) {\n\t\t\t\t\tinsertionPosition = this._positionToInsert;\n\t\t\t\t}\n\n\t\t\t\tinsertMedia( editor.model, url, insertionPosition, false );\n\n\t\t\t\tthis._positionToInsert!.detach();\n\t\t\t\tthis._positionToInsert = null;\n\t\t\t} );\n\n\t\t\teditor.plugins.get( Delete ).requestUndoOnBackspace();\n\t\t}, 100 );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./mediaform.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/ui/mediaformview\n */\n\nimport {\n\ttype InputTextView,\n\tButtonView,\n\tFocusCycler,\n\tLabeledFieldView,\n\tView,\n\tViewCollection,\n\tcreateLabeledInputText,\n\tsubmitHandler\n} from 'ckeditor5/src/ui';\nimport { FocusTracker, KeystrokeHandler, type Locale } from 'ckeditor5/src/utils';\nimport { icons } from 'ckeditor5/src/core';\n\n// See: #8833.\n// eslint-disable-next-line ckeditor5-rules/ckeditor-imports\nimport '@ckeditor/ckeditor5-ui/theme/components/responsive-form/responsiveform.css';\nimport '../../theme/mediaform.css';\n\n/**\n * The media form view controller class.\n *\n * See {@link module:media-embed/ui/mediaformview~MediaFormView}.\n */\nexport default class MediaFormView extends View {\n\t/**\n\t * Tracks information about the DOM focus in the form.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * An instance of the {@link module:utils/keystrokehandler~KeystrokeHandler}.\n\t */\n\tpublic readonly keystrokes: KeystrokeHandler;\n\n\t/**\n\t * The value of the URL input.\n\t */\n\tdeclare public mediaURLInputValue: string;\n\n\t/**\n\t * The URL input view.\n\t */\n\tpublic urlInputView: LabeledFieldView<InputTextView>;\n\n\t/**\n\t * The Save button view.\n\t */\n\tpublic saveButtonView: ButtonView;\n\n\t/**\n\t * The Cancel button view.\n\t */\n\tpublic cancelButtonView: ButtonView;\n\n\t/**\n\t * A collection of views that can be focused in the form.\n\t */\n\tprivate readonly _focusables: ViewCollection;\n\n\t/**\n\t * Helps cycling over {@link #_focusables} in the form.\n\t */\n\tprivate readonly _focusCycler: FocusCycler;\n\n\t/**\n\t * An array of form validators used by {@link #isValid}.\n\t */\n\tprivate readonly _validators: Array<( v: MediaFormView ) => string | undefined>;\n\n\t/**\n\t * The default info text for the {@link #urlInputView}.\n\t */\n\tprivate _urlInputViewInfoDefault?: string;\n\n\t/**\n\t * The info text with an additional tip for the {@link #urlInputView},\n\t * displayed when the input has some value.\n\t */\n\tprivate _urlInputViewInfoTip?: string;\n\n\t/**\n\t * @param validators Form validators used by {@link #isValid}.\n\t * @param locale The localization services instance.\n\t */\n\tconstructor( validators: Array<( v: MediaFormView ) => string | undefined>, locale: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst t = locale.t;\n\n\t\tthis.focusTracker = new FocusTracker();\n\t\tthis.keystrokes = new KeystrokeHandler();\n\t\tthis.set( 'mediaURLInputValue', '' );\n\t\tthis.urlInputView = this._createUrlInput();\n\n\t\tthis.saveButtonView = this._createButton( t( 'Save' ), icons.check, 'ck-button-save' );\n\t\tthis.saveButtonView.type = 'submit';\n\t\tthis.saveButtonView.bind( 'isEnabled' ).to( this, 'mediaURLInputValue', value => !!value );\n\n\t\tthis.cancelButtonView = this._createButton( t( 'Cancel' ), icons.cancel, 'ck-button-cancel', 'cancel' );\n\n\t\tthis._focusables = new ViewCollection();\n\n\t\tthis._focusCycler = new FocusCycler( {\n\t\t\tfocusables: this._focusables,\n\t\t\tfocusTracker: this.focusTracker,\n\t\t\tkeystrokeHandler: this.keystrokes,\n\t\t\tactions: {\n\t\t\t\t// Navigate form fields backwards using the <kbd>Shift</kbd> + <kbd>Tab</kbd> keystroke.\n\t\t\t\tfocusPrevious: 'shift + tab',\n\n\t\t\t\t// Navigate form fields forwards using the <kbd>Tab</kbd> key.\n\t\t\t\tfocusNext: 'tab'\n\t\t\t}\n\t\t} );\n\n\t\tthis._validators = validators;\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'form',\n\n\t\t\tattributes: {\n\t\t\t\tclass: [\n\t\t\t\t\t'ck',\n\t\t\t\t\t'ck-media-form',\n\t\t\t\t\t'ck-responsive-form'\n\t\t\t\t],\n\n\t\t\t\ttabindex: '-1'\n\t\t\t},\n\n\t\t\tchildren: [\n\t\t\t\tthis.urlInputView,\n\t\t\t\tthis.saveButtonView,\n\t\t\t\tthis.cancelButtonView\n\t\t\t]\n\t\t} );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\tsubmitHandler( {\n\t\t\tview: this\n\t\t} );\n\n\t\tconst childViews = [\n\t\t\tthis.urlInputView,\n\t\t\tthis.saveButtonView,\n\t\t\tthis.cancelButtonView\n\t\t];\n\n\t\tchildViews.forEach( v => {\n\t\t\t// Register the view as focusable.\n\t\t\tthis._focusables.add( v );\n\n\t\t\t// Register the view in the focus tracker.\n\t\t\tthis.focusTracker.add( v.element! );\n\t\t} );\n\n\t\t// Start listening for the keystrokes coming from #element.\n\t\tthis.keystrokes.listenTo( this.element! );\n\n\t\tconst stopPropagation = ( data: KeyboardEvent ) => data.stopPropagation();\n\n\t\t// Since the form is in the dropdown panel which is a child of the toolbar, the toolbar's\n\t\t// keystroke handler would take over the key management in the URL input. We need to prevent\n\t\t// this ASAP. Otherwise, the basic caret movement using the arrow keys will be impossible.\n\t\tthis.keystrokes.set( 'arrowright', stopPropagation );\n\t\tthis.keystrokes.set( 'arrowleft', stopPropagation );\n\t\tthis.keystrokes.set( 'arrowup', stopPropagation );\n\t\tthis.keystrokes.set( 'arrowdown', stopPropagation );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tthis.focusTracker.destroy();\n\t\tthis.keystrokes.destroy();\n\t}\n\n\t/**\n\t * Focuses the fist {@link #_focusables} in the form.\n\t */\n\tpublic focus(): void {\n\t\tthis._focusCycler.focusFirst();\n\t}\n\n\t/**\n\t * The native DOM `value` of the {@link #urlInputView} element.\n\t *\n\t * **Note**: Do not confuse it with the {@link module:ui/inputtext/inputtextview~InputTextView#value}\n\t * which works one way only and may not represent the actual state of the component in the DOM.\n\t */\n\tpublic get url(): string {\n\t\treturn this.urlInputView.fieldView.element!.value.trim();\n\t}\n\n\tpublic set url( url: string ) {\n\t\tthis.urlInputView.fieldView.element!.value = url.trim();\n\t}\n\n\t/**\n\t * Validates the form and returns `false` when some fields are invalid.\n\t */\n\tpublic isValid(): boolean {\n\t\tthis.resetFormStatus();\n\n\t\tfor ( const validator of this._validators ) {\n\t\t\tconst errorText = validator( this );\n\n\t\t\t// One error per field is enough.\n\t\t\tif ( errorText ) {\n\t\t\t\t// Apply updated error.\n\t\t\t\tthis.urlInputView.errorText = errorText;\n\n\t\t\t\treturn false;\n\t\t\t}\n\t\t}\n\n\t\treturn true;\n\t}\n\n\t/**\n\t * Cleans up the supplementary error and information text of the {@link #urlInputView}\n\t * bringing them back to the state when the form has been displayed for the first time.\n\t *\n\t * See {@link #isValid}.\n\t */\n\tpublic resetFormStatus(): void {\n\t\tthis.urlInputView.errorText = null;\n\t\tthis.urlInputView.infoText = this._urlInputViewInfoDefault!;\n\t}\n\n\t/**\n\t * Creates a labeled input view.\n\t *\n\t * @returns Labeled input view instance.\n\t */\n\tprivate _createUrlInput(): LabeledFieldView<InputTextView> {\n\t\tconst t = this.locale!.t;\n\n\t\tconst labeledInput = new LabeledFieldView( this.locale, createLabeledInputText );\n\t\tconst inputField = labeledInput.fieldView;\n\n\t\tthis._urlInputViewInfoDefault = t( 'Paste the media URL in the input.' );\n\t\tthis._urlInputViewInfoTip = t( 'Tip: Paste the URL into the content to embed faster.' );\n\n\t\tlabeledInput.label = t( 'Media URL' );\n\t\tlabeledInput.infoText = this._urlInputViewInfoDefault;\n\n\t\tinputField.on( 'input', () => {\n\t\t\t// Display the tip text only when there is some value. Otherwise fall back to the default info text.\n\t\t\tlabeledInput.infoText = inputField.element!.value ? this._urlInputViewInfoTip! : this._urlInputViewInfoDefault!;\n\t\t\tthis.mediaURLInputValue = inputField.element!.value.trim();\n\t\t} );\n\n\t\treturn labeledInput;\n\t}\n\n\t/**\n\t * Creates a button view.\n\t *\n\t * @param label The button label.\n\t * @param icon The button icon.\n\t * @param className The additional button CSS class name.\n\t * @param eventName An event name that the `ButtonView#execute` event will be delegated to.\n\t * @returns The button view instance.\n\t */\n\tprivate _createButton( label: string, icon: string, className: string, eventName?: string ): ButtonView {\n\t\tconst button = new ButtonView( this.locale );\n\n\t\tbutton.set( {\n\t\t\tlabel,\n\t\t\ticon,\n\t\t\ttooltip: true\n\t\t} );\n\n\t\tbutton.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\tclass: className\n\t\t\t}\n\t\t} );\n\n\t\tif ( eventName ) {\n\t\t\tbutton.delegate( 'execute' ).to( this, eventName );\n\t\t}\n\n\t\treturn button;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/mediaembedui\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { createDropdown, CssTransitionDisablerMixin, type DropdownView } from 'ckeditor5/src/ui';\n\nimport MediaFormView from './ui/mediaformview';\nimport MediaEmbedEditing from './mediaembedediting';\nimport mediaIcon from '../theme/icons/media.svg';\nimport type MediaEmbedCommand from './mediaembedcommand';\nimport type { LocaleTranslate } from 'ckeditor5/src/utils';\nimport type MediaRegistry from './mediaregistry';\n\n/**\n * The media embed UI plugin.\n */\nexport default class MediaEmbedUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ MediaEmbedEditing ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'MediaEmbedUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst command: MediaEmbedCommand = editor.commands.get( 'mediaEmbed' )!;\n\n\t\teditor.ui.componentFactory.add( 'mediaEmbed', locale => {\n\t\t\tconst dropdown = createDropdown( locale );\n\n\t\t\tthis._setUpDropdown( dropdown, command );\n\n\t\t\treturn dropdown;\n\t\t} );\n\t}\n\n\tprivate _setUpDropdown( dropdown: DropdownView, command: MediaEmbedCommand ): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst button = dropdown.buttonView;\n\t\tconst registry = editor.plugins.get( MediaEmbedEditing ).registry;\n\n\t\tdropdown.once( 'change:isOpen', () => {\n\t\t\tconst form = new ( CssTransitionDisablerMixin( MediaFormView ) )( getFormValidators( editor.t, registry ), editor.locale );\n\n\t\t\tdropdown.panelView.children.add( form );\n\n\t\t\t// Note: Use the low priority to make sure the following listener starts working after the\n\t\t\t// default action of the drop-down is executed (i.e. the panel showed up). Otherwise, the\n\t\t\t// invisible form/input cannot be focused/selected.\n\t\t\tbutton.on( 'open', () => {\n\t\t\t\tform.disableCssTransitions();\n\n\t\t\t\t// Make sure that each time the panel shows up, the URL field remains in sync with the value of\n\t\t\t\t// the command. If the user typed in the input, then canceled (`urlInputView#fieldView#value` stays\n\t\t\t\t// unaltered) and re-opened it without changing the value of the media command (e.g. because they\n\t\t\t\t// didn't change the selection), they would see the old value instead of the actual value of the\n\t\t\t\t// command.\n\t\t\t\tform.url = command.value || '';\n\t\t\t\tform.urlInputView.fieldView.select();\n\t\t\t\tform.enableCssTransitions();\n\t\t\t}, { priority: 'low' } );\n\n\t\t\tdropdown.on( 'submit', () => {\n\t\t\t\tif ( form.isValid() ) {\n\t\t\t\t\teditor.execute( 'mediaEmbed', form.url );\n\t\t\t\t\teditor.editing.view.focus();\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tdropdown.on( 'change:isOpen', () => form.resetFormStatus() );\n\t\t\tdropdown.on( 'cancel', () => {\n\t\t\t\teditor.editing.view.focus();\n\t\t\t} );\n\n\t\t\tform.delegate( 'submit', 'cancel' ).to( dropdown );\n\t\t\tform.urlInputView.fieldView.bind( 'value' ).to( command, 'value' );\n\n\t\t\t// Form elements should be read-only when corresponding commands are disabled.\n\t\t\tform.urlInputView.bind( 'isEnabled' ).to( command, 'isEnabled' );\n\t\t} );\n\n\t\tdropdown.bind( 'isEnabled' ).to( command );\n\n\t\tbutton.set( {\n\t\t\tlabel: t( 'Insert media' ),\n\t\t\ticon: mediaIcon,\n\t\t\ttooltip: true\n\t\t} );\n\t}\n}\n\nfunction getFormValidators( t: LocaleTranslate, registry: MediaRegistry ): Array<( v: MediaFormView ) => string | undefined> {\n\treturn [\n\t\tform => {\n\t\t\tif ( !form.url.length ) {\n\t\t\t\treturn t( 'The URL must not be empty.' );\n\t\t\t}\n\t\t},\n\t\tform => {\n\t\t\tif ( !registry.hasMedia( form.url ) ) {\n\t\t\t\treturn t( 'This media URL is not supported.' );\n\t\t\t}\n\t\t}\n\t];\n}\n","export default \"<svg xmlns=\\\"http://www.w3.org/2000/svg\\\" viewBox=\\\"0 0 20 20\\\"><path d=\\\"M18.68 3.03c.6 0 .59-.03.59.55v12.84c0 .59.01.56-.59.56H1.29c-.6 0-.59.03-.59-.56V3.58c0-.58-.01-.55.6-.55h17.38zM15.77 15V5H4.2v10h11.57zM2 4v1h1V4H2zm0 2v1h1V6H2zm0 2v1h1V8H2zm0 2v1h1v-1H2zm0 2v1h1v-1H2zm0 2v1h1v-1H2zM17 4v1h1V4h-1zm0 2v1h1V6h-1zm0 2v1h1V8h-1zm0 2v1h1v-1h-1zm0 2v1h1v-1h-1zm0 2v1h1v-1h-1zM7.5 7.177a.4.4 0 0 1 .593-.351l5.133 2.824a.4.4 0 0 1 0 .7l-5.133 2.824a.4.4 0 0 1-.593-.35V7.176v.001z\\\"/></svg>\";","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./mediaembed.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/list\n */\n\nimport {\n\tMatcher,\n\tUpcastWriter,\n\ttype ViewDocumentFragment,\n\ttype ViewElement,\n\ttype ViewNode,\n\ttype ViewText\n} from 'ckeditor5/src/engine';\n\n/**\n * Transforms Word specific list-like elements to the semantic HTML lists.\n *\n * Lists in Word are represented by block elements with special attributes like:\n *\n * ```xml\n * <p class=MsoListParagraphCxSpFirst style='mso-list:l1 level1 lfo1'>...</p> // Paragraph based list.\n * <h1 style='mso-list:l0 level1 lfo1'>...</h1> // Heading 1 based list.\n * ```\n *\n * @param documentFragment The view structure to be transformed.\n * @param stylesString Styles from which list-like elements styling will be extracted.\n */\nexport function transformListItemLikeElementsIntoLists(\n\tdocumentFragment: ViewDocumentFragment,\n\tstylesString: string\n): void {\n\tif ( !documentFragment.childCount ) {\n\t\treturn;\n\t}\n\n\tconst writer = new UpcastWriter( documentFragment.document );\n\tconst itemLikeElements = findAllItemLikeElements( documentFragment, writer );\n\n\tif ( !itemLikeElements.length ) {\n\t\treturn;\n\t}\n\n\tlet currentList: ViewElement | null = null;\n\tlet currentIndentation = 1;\n\n\titemLikeElements.forEach( ( itemLikeElement, i ) => {\n\t\tconst isDifferentList = isNewListNeeded( itemLikeElements[ i - 1 ], itemLikeElement );\n\t\tconst previousItemLikeElement = isDifferentList ? null : itemLikeElements[ i - 1 ];\n\t\tconst indentationDifference = getIndentationDifference( previousItemLikeElement, itemLikeElement );\n\n\t\tif ( isDifferentList ) {\n\t\t\tcurrentList = null;\n\t\t\tcurrentIndentation = 1;\n\t\t}\n\n\t\tif ( !currentList || indentationDifference !== 0 ) {\n\t\t\tconst listStyle = detectListStyle( itemLikeElement, stylesString );\n\n\t\t\tif ( !currentList ) {\n\t\t\t\tcurrentList = insertNewEmptyList( listStyle, itemLikeElement.element, writer );\n\t\t\t} else if ( itemLikeElement.indent > currentIndentation ) {\n\t\t\t\tconst lastListItem = currentList.getChild( currentList.childCount - 1 ) as ViewElement;\n\t\t\t\tconst lastListItemChild = lastListItem!.getChild( lastListItem.childCount - 1 ) as ViewElement;\n\n\t\t\t\tcurrentList = insertNewEmptyList( listStyle, lastListItemChild, writer );\n\t\t\t\tcurrentIndentation += 1;\n\t\t\t} else if ( itemLikeElement.indent < currentIndentation ) {\n\t\t\t\tconst differentIndentation = currentIndentation - itemLikeElement.indent;\n\n\t\t\t\tcurrentList = findParentListAtLevel( currentList, differentIndentation );\n\t\t\t\tcurrentIndentation = itemLikeElement.indent;\n\t\t\t}\n\n\t\t\tif ( itemLikeElement.indent <= currentIndentation ) {\n\t\t\t\tif ( !currentList.is( 'element', listStyle.type ) ) {\n\t\t\t\t\tcurrentList = writer.rename( listStyle.type, currentList );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tconst listItem = transformElementIntoListItem( itemLikeElement.element, writer );\n\n\t\twriter.appendChild( listItem, currentList! );\n\t} );\n}\n\n/**\n * Removes paragraph wrapping content inside a list item.\n */\nexport function unwrapParagraphInListItem(\n\tdocumentFragment: ViewDocumentFragment,\n\twriter: UpcastWriter\n): void {\n\tfor ( const value of writer.createRangeIn( documentFragment ) ) {\n\t\tconst element = value.item;\n\n\t\tif ( element.is( 'element', 'li' ) ) {\n\t\t\t// Google Docs allows for single paragraph inside LI.\n\t\t\tconst firstChild = element.getChild( 0 );\n\n\t\t\tif ( firstChild && firstChild.is( 'element', 'p' ) ) {\n\t\t\t\twriter.unwrapElement( firstChild );\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Finds all list-like elements in a given document fragment.\n *\n * @param documentFragment Document fragment in which to look for list-like nodes.\n * @returns Array of found list-like items. Each item is an object containing:\n */\nfunction findAllItemLikeElements(\n\tdocumentFragment: ViewDocumentFragment,\n\twriter: UpcastWriter\n): Array<ListLikeElement> {\n\tconst range = writer.createRangeIn( documentFragment );\n\n\t// Matcher for finding list-like elements.\n\tconst itemLikeElementsMatcher = new Matcher( {\n\t\tname: /^p|h\\d+$/,\n\t\tstyles: {\n\t\t\t'mso-list': /.*/\n\t\t}\n\t} );\n\n\tconst itemLikeElements: Array<ListLikeElement> = [];\n\n\tfor ( const value of range ) {\n\t\tif ( value.type === 'elementStart' && itemLikeElementsMatcher.match( value.item as ViewElement ) ) {\n\t\t\tconst itemData = getListItemData( value.item as ViewElement );\n\n\t\t\titemLikeElements.push( {\n\t\t\t\telement: value.item as ViewElement,\n\t\t\t\tid: itemData.id,\n\t\t\t\torder: itemData.order,\n\t\t\t\tindent: itemData.indent\n\t\t\t} );\n\t\t}\n\t}\n\n\treturn itemLikeElements;\n}\n\n/**\n * Extracts list item style from the provided CSS.\n *\n * List item style is extracted from the CSS stylesheet. Each list with its specific style attribute\n * value (`mso-list:l1 level1 lfo1`) has its dedicated properties in a CSS stylesheet defined with a selector like:\n *\n * ```css\n * @list l1:level1 { ... }\n * ```\n *\n * It contains `mso-level-number-format` property which defines list numbering/bullet style. If this property\n * is not defined it means default `decimal` numbering.\n *\n * Here CSS string representation is used as `mso-level-number-format` property is an invalid CSS property\n * and will be removed during CSS parsing.\n *\n * @param listLikeItem List-like item for which list style will be searched for. Usually\n * a result of `findAllItemLikeElements()` function.\n * @param stylesString CSS stylesheet.\n * @returns An object with properties:\n *\n * * type - List type, could be `ul` or `ol`.\n * * startIndex - List start index, valid only for ordered lists.\n * * style - List style, for example: `decimal`, `lower-roman`, etc. It is extracted\n * directly from Word stylesheet and adjusted to represent proper values for the CSS `list-style-type` property.\n * If it cannot be adjusted, the `null` value is returned.\n */\nfunction detectListStyle( listLikeItem: ListLikeElement, stylesString: string ) {\n\tconst listStyleRegexp = new RegExp( `@list l${ listLikeItem.id }:level${ listLikeItem.indent }\\\\s*({[^}]*)`, 'gi' );\n\tconst listStyleTypeRegex = /mso-level-number-format:([^;]{0,100});/gi;\n\tconst listStartIndexRegex = /mso-level-start-at:\\s{0,100}([0-9]{0,10})\\s{0,100};/gi;\n\n\tconst listStyleMatch = listStyleRegexp.exec( stylesString );\n\n\tlet listStyleType = 'decimal'; // Decimal is default one.\n\tlet type = 'ol'; // <ol> is default list.\n\tlet startIndex = null;\n\n\tif ( listStyleMatch && listStyleMatch[ 1 ] ) {\n\t\tconst listStyleTypeMatch = listStyleTypeRegex.exec( listStyleMatch[ 1 ] );\n\n\t\tif ( listStyleTypeMatch && listStyleTypeMatch[ 1 ] ) {\n\t\t\tlistStyleType = listStyleTypeMatch[ 1 ].trim();\n\t\t\ttype = listStyleType !== 'bullet' && listStyleType !== 'image' ? 'ol' : 'ul';\n\t\t}\n\n\t\t// Styles for the numbered lists are always defined in the Word CSS stylesheet.\n\t\t// Unordered lists MAY contain a value for the Word CSS definition `mso-level-text` but sometimes\n\t\t// this tag is missing. And because of that, we cannot depend on that. We need to predict the list style value\n\t\t// based on the list style marker element.\n\t\tif ( listStyleType === 'bullet' ) {\n\t\t\tconst bulletedStyle = findBulletedListStyle( listLikeItem.element );\n\n\t\t\tif ( bulletedStyle ) {\n\t\t\t\tlistStyleType = bulletedStyle;\n\t\t\t}\n\t\t} else {\n\t\t\tconst listStartIndexMatch = listStartIndexRegex.exec( listStyleMatch[ 1 ] );\n\n\t\t\tif ( listStartIndexMatch && listStartIndexMatch[ 1 ] ) {\n\t\t\t\tstartIndex = parseInt( listStartIndexMatch[ 1 ] );\n\t\t\t}\n\t\t}\n\t}\n\n\treturn {\n\t\ttype,\n\t\tstartIndex,\n\t\tstyle: mapListStyleDefinition( listStyleType )\n\t};\n}\n\n/**\n * Tries to extract the `list-style-type` value based on the marker element for bulleted list.\n */\nfunction findBulletedListStyle( element: ViewElement ) {\n\tconst listMarkerElement = findListMarkerNode( element );\n\n\tif ( !listMarkerElement ) {\n\t\treturn null;\n\t}\n\n\tconst listMarker = listMarkerElement._data;\n\n\tif ( listMarker === 'o' ) {\n\t\treturn 'circle';\n\t} else if ( listMarker === '·' ) {\n\t\treturn 'disc';\n\t}\n\t// Word returns '§' instead of '■' for the square list style.\n\telse if ( listMarker === '§' ) {\n\t\treturn 'square';\n\t}\n\n\treturn null;\n}\n\n/**\n * Tries to find a text node that represents the marker element (list-style-type).\n */\nfunction findListMarkerNode( element: ViewElement ): ViewText | null {\n\t// If the first child is a text node, it is the data for the element.\n\t// The list-style marker is not present here.\n\tif ( element.getChild( 0 )!.is( '$text' ) ) {\n\t\treturn null;\n\t}\n\n\tfor ( const childNode of element.getChildren() ) {\n\t\t// The list-style marker will be inside the `<span>` element. Let's ignore all non-span elements.\n\t\t// It may happen that the `<a>` element is added as the first child. Most probably, it's an anchor element.\n\t\tif ( !childNode.is( 'element', 'span' ) ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst textNodeOrElement = childNode.getChild( 0 );\n\n\t\tif ( !textNodeOrElement ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\t// If already found the marker element, use it.\n\t\tif ( textNodeOrElement.is( '$text' ) ) {\n\t\t\treturn textNodeOrElement;\n\t\t}\n\n\t\treturn ( textNodeOrElement as any ).getChild( 0 );\n\t}\n\n\t/* istanbul ignore next -- @preserve */\n\treturn null;\n}\n\n/**\n * Parses the `list-style-type` value extracted directly from the Word CSS stylesheet and returns proper CSS definition.\n */\nfunction mapListStyleDefinition( value: string ) {\n\tif ( value.startsWith( 'arabic-leading-zero' ) ) {\n\t\treturn 'decimal-leading-zero';\n\t}\n\n\tswitch ( value ) {\n\t\tcase 'alpha-upper':\n\t\t\treturn 'upper-alpha';\n\t\tcase 'alpha-lower':\n\t\t\treturn 'lower-alpha';\n\t\tcase 'roman-upper':\n\t\t\treturn 'upper-roman';\n\t\tcase 'roman-lower':\n\t\t\treturn 'lower-roman';\n\t\tcase 'circle':\n\t\tcase 'disc':\n\t\tcase 'square':\n\t\t\treturn value;\n\t\tdefault:\n\t\t\treturn null;\n\t}\n}\n\n/**\n * Creates an empty list of a given type and inserts it after a specified element.\n *\n * @param listStyle List style object which determines the type of newly created list.\n * Usually a result of `detectListStyle()` function.\n * @param element Element after which list is inserted.\n * @returns Newly created list element.\n */\nfunction insertNewEmptyList(\n\tlistStyle: ReturnType<typeof detectListStyle>,\n\telement: ViewElement,\n\twriter: UpcastWriter\n) {\n\tconst parent = element.parent!;\n\tconst list = writer.createElement( listStyle.type );\n\tconst position = parent.getChildIndex( element ) + 1;\n\n\twriter.insertChild( position, list, parent );\n\n\t// We do not support modifying the marker for a particular list item.\n\t// Set the value for the `list-style-type` property directly to the list container.\n\tif ( listStyle.style ) {\n\t\twriter.setStyle( 'list-style-type', listStyle.style, list );\n\t}\n\n\tif ( listStyle.startIndex && listStyle.startIndex > 1 ) {\n\t\twriter.setAttribute( 'start', listStyle.startIndex, list );\n\t}\n\n\treturn list;\n}\n\n/**\n * Transforms a given element into a semantic list item. As the function operates on a provided\n * {module:engine/src/view/element~Element element} it will modify the view structure to which this element belongs.\n *\n * @param element Element which will be transformed into a list item.\n * @returns New element to which the given one was transformed. It is\n * inserted in place of the old element (the reference to the old element is lost due to renaming).\n */\nfunction transformElementIntoListItem( element: ViewElement, writer: UpcastWriter ) {\n\tremoveBulletElement( element, writer );\n\twriter.removeStyle( 'text-indent', element ); // #12361\n\n\treturn writer.rename( 'li', element )!;\n}\n\n/**\n * Extracts list item information from Word specific list-like element style:\n *\n * ```\n * `style=\"mso-list:l1 level1 lfo1\"`\n * ```\n *\n * where:\n *\n * ```\n * * `l1` is a list id (however it does not mean this is a continuous list - see #43),\n * * `level1` is a list item indentation level,\n * * `lfo1` is a list insertion order in a document.\n * ```\n *\n * @param element Element from which style data is extracted.\n */\nfunction getListItemData( element: ViewElement ): ListItemData {\n\tconst data: ListItemData = {} as any;\n\tconst listStyle = element.getStyle( 'mso-list' );\n\n\tif ( listStyle ) {\n\t\tconst idMatch = listStyle.match( /(^|\\s{1,100})l(\\d+)/i );\n\t\tconst orderMatch = listStyle.match( /\\s{0,100}lfo(\\d+)/i );\n\t\tconst indentMatch = listStyle.match( /\\s{0,100}level(\\d+)/i );\n\n\t\tif ( idMatch && orderMatch && indentMatch ) {\n\t\t\tdata.id = idMatch[ 2 ];\n\t\t\tdata.order = orderMatch[ 1 ];\n\t\t\tdata.indent = parseInt( indentMatch[ 1 ] );\n\t\t}\n\t}\n\n\treturn data;\n}\n\n/**\n * Removes span with a numbering/bullet from a given element.\n */\nfunction removeBulletElement( element: ViewElement, writer: UpcastWriter ) {\n\t// Matcher for finding `span` elements holding lists numbering/bullets.\n\tconst bulletMatcher = new Matcher( {\n\t\tname: 'span',\n\t\tstyles: {\n\t\t\t'mso-list': 'Ignore'\n\t\t}\n\t} );\n\n\tconst range = writer.createRangeIn( element );\n\n\tfor ( const value of range ) {\n\t\tif ( value.type === 'elementStart' && bulletMatcher.match( value.item as ViewElement ) ) {\n\t\t\twriter.remove( value.item as ViewElement );\n\t\t}\n\t}\n}\n\n/**\n * Whether the previous and current items belong to the same list. It is determined based on `item.id`\n * (extracted from `mso-list` style, see #getListItemData) and a previous sibling of the current item.\n *\n * However, it's quite easy to change the `id` attribute for nested lists in Word. It will break the list feature while pasting.\n * Let's check also the `indent` attribute. If the difference between those two elements is equal to 1, we can assume that\n * the `currentItem` is a beginning of the nested list because lists in CKEditor 5 always start with the `indent=0` attribute.\n * See: https://github.com/ckeditor/ckeditor5/issues/7805.\n */\nfunction isNewListNeeded( previousItem: ListLikeElement, currentItem: ListLikeElement ) {\n\tif ( !previousItem ) {\n\t\treturn true;\n\t}\n\n\tif ( previousItem.id !== currentItem.id ) {\n\t\t// See: https://github.com/ckeditor/ckeditor5/issues/7805.\n\t\t//\n\t\t// * List item 1.\n\t\t// - Nested list item 1.\n\t\tif ( currentItem.indent - previousItem.indent === 1 ) {\n\t\t\treturn false;\n\t\t}\n\n\t\treturn true;\n\t}\n\n\tconst previousSibling = currentItem.element.previousSibling;\n\n\tif ( !previousSibling ) {\n\t\treturn true;\n\t}\n\n\t// Even with the same id the list does not have to be continuous (#43).\n\treturn !isList( previousSibling );\n}\n\nfunction isList( element: ViewNode ) {\n\treturn element.is( 'element', 'ol' ) || element.is( 'element', 'ul' );\n}\n\n/**\n * Calculates the indentation difference between two given list items (based on the indent attribute\n * extracted from the `mso-list` style, see #getListItemData).\n */\nfunction getIndentationDifference( previousItem: ListLikeElement | null, currentItem: ListLikeElement ) {\n\treturn previousItem ? currentItem.indent - previousItem.indent : currentItem.indent - 1;\n}\n\n/**\n * Finds the parent list element (ul/ol) of a given list element with indentation level lower by a given value.\n *\n * @param listElement List element from which to start looking for a parent list.\n * @param indentationDifference Indentation difference between lists.\n * @returns Found list element with indentation level lower by a given value.\n */\nfunction findParentListAtLevel( listElement: ViewElement, indentationDifference: number ) {\n\tconst ancestors = listElement.getAncestors( { parentFirst: true } );\n\n\tlet parentList = null;\n\tlet levelChange = 0;\n\n\tfor ( const ancestor of ancestors ) {\n\t\tif ( ancestor.is( 'element', 'ul' ) || ancestor.is( 'element', 'ol' ) ) {\n\t\t\tlevelChange++;\n\t\t}\n\n\t\tif ( levelChange === indentationDifference ) {\n\t\t\tparentList = ancestor;\n\t\t\tbreak;\n\t\t}\n\t}\n\n\treturn parentList as ViewElement;\n}\n\ninterface ListItemData {\n\n\t/**\n\t * Parent list id.\n\t */\n\tid: string;\n\n\t/**\n\t * List item creation order.\n\t */\n\torder: string;\n\n\t/**\n\t * List item indentation level.\n\t */\n\tindent: number;\n}\n\ninterface ListLikeElement extends ListItemData {\n\n\t/**\n\t * List-like element.\n\t */\n\telement: ViewElement;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/image\n */\n\n/* globals btoa */\n\nimport {\n\tMatcher,\n\tUpcastWriter,\n\ttype ViewDocumentFragment,\n\ttype ViewElement,\n\ttype ViewNode\n} from 'ckeditor5/src/engine';\n\n/**\n * Replaces source attribute of all `<img>` elements representing regular\n * images (not the Word shapes) with inlined base64 image representation extracted from RTF or Blob data.\n *\n * @param documentFragment Document fragment on which transform images.\n * @param rtfData The RTF data from which images representation will be used.\n */\nexport function replaceImagesSourceWithBase64( documentFragment: ViewDocumentFragment, rtfData: string ): void {\n\tif ( !documentFragment.childCount ) {\n\t\treturn;\n\t}\n\n\tconst upcastWriter = new UpcastWriter( documentFragment.document );\n\tconst shapesIds = findAllShapesIds( documentFragment, upcastWriter );\n\n\tremoveAllImgElementsRepresentingShapes( shapesIds, documentFragment, upcastWriter );\n\tinsertMissingImgs( shapesIds, documentFragment, upcastWriter );\n\tremoveAllShapeElements( documentFragment, upcastWriter );\n\n\tconst images = findAllImageElementsWithLocalSource( documentFragment, upcastWriter );\n\n\tif ( images.length ) {\n\t\treplaceImagesFileSourceWithInlineRepresentation( images, extractImageDataFromRtf( rtfData ), upcastWriter );\n\t}\n}\n\n/**\n * Converts given HEX string to base64 representation.\n *\n * @internal\n * @param hexString The HEX string to be converted.\n * @returns Base64 representation of a given HEX string.\n */\nexport function _convertHexToBase64( hexString: string ): string {\n\treturn btoa( hexString.match( /\\w{2}/g )!.map( char => {\n\t\treturn String.fromCharCode( parseInt( char, 16 ) );\n\t} ).join( '' ) );\n}\n\n/**\n * Finds all shapes (`<v:*>...</v:*>`) ids. Shapes can represent images (canvas)\n * or Word shapes (which does not have RTF or Blob representation).\n *\n * @param documentFragment Document fragment from which to extract shape ids.\n * @returns Array of shape ids.\n */\nfunction findAllShapesIds( documentFragment: ViewDocumentFragment, writer: UpcastWriter ): Array<string> {\n\tconst range = writer.createRangeIn( documentFragment );\n\n\tconst shapeElementsMatcher = new Matcher( {\n\t\tname: /v:(.+)/\n\t} );\n\n\tconst shapesIds = [];\n\n\tfor ( const value of range ) {\n\t\tif ( value.type != 'elementStart' ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst el = value.item as ViewElement;\n\t\tconst previousSibling = el.previousSibling;\n\t\tconst prevSiblingName = previousSibling && previousSibling.is( 'element' ) ? previousSibling.name : null;\n\n\t\t// If shape element have 'o:gfxdata' attribute and is not directly before `<v:shapetype>` element it means it represent Word shape.\n\t\tif ( shapeElementsMatcher.match( el ) && el.getAttribute( 'o:gfxdata' ) && prevSiblingName !== 'v:shapetype' ) {\n\t\t\tshapesIds.push( ( value.item as ViewElement ).getAttribute( 'id' )! );\n\t\t}\n\t}\n\n\treturn shapesIds;\n}\n\n/**\n * Removes all `<img>` elements which represents Word shapes and not regular images.\n *\n * @param shapesIds Shape ids which will be checked against `<img>` elements.\n * @param documentFragment Document fragment from which to remove `<img>` elements.\n */\nfunction removeAllImgElementsRepresentingShapes(\n\tshapesIds: Array<string>,\n\tdocumentFragment: ViewDocumentFragment,\n\twriter: UpcastWriter\n): void {\n\tconst range = writer.createRangeIn( documentFragment );\n\n\tconst imageElementsMatcher = new Matcher( {\n\t\tname: 'img'\n\t} );\n\n\tconst imgs = [];\n\n\tfor ( const value of range ) {\n\t\tif ( value.item.is( 'element' ) && imageElementsMatcher.match( value.item ) ) {\n\t\t\tconst el = value.item;\n\t\t\tconst shapes = el.getAttribute( 'v:shapes' ) ? el.getAttribute( 'v:shapes' )!.split( ' ' ) : [];\n\n\t\t\tif ( shapes.length && shapes.every( shape => shapesIds.indexOf( shape ) > -1 ) ) {\n\t\t\t\timgs.push( el );\n\t\t\t// Shapes may also have empty source while content is paste in some browsers (Safari).\n\t\t\t} else if ( !el.getAttribute( 'src' ) ) {\n\t\t\t\timgs.push( el );\n\t\t\t}\n\t\t}\n\t}\n\n\tfor ( const img of imgs ) {\n\t\twriter.remove( img );\n\t}\n}\n\n/**\n * Removes all shape elements (`<v:*>...</v:*>`) so they do not pollute the output structure.\n *\n * @param documentFragment Document fragment from which to remove shape elements.\n */\nfunction removeAllShapeElements( documentFragment: ViewDocumentFragment, writer: UpcastWriter ) {\n\tconst range = writer.createRangeIn( documentFragment );\n\n\tconst shapeElementsMatcher = new Matcher( {\n\t\tname: /v:(.+)/\n\t} );\n\n\tconst shapes = [];\n\n\tfor ( const value of range ) {\n\t\tif ( value.type == 'elementStart' && shapeElementsMatcher.match( value.item as ViewElement ) ) {\n\t\t\tshapes.push( value.item as ViewElement );\n\t\t}\n\t}\n\n\tfor ( const shape of shapes ) {\n\t\twriter.remove( shape );\n\t}\n}\n\n/**\n * Inserts `img` tags if there is none after a shape.\n */\nfunction insertMissingImgs( shapeIds: Array<string>, documentFragment: ViewDocumentFragment, writer: UpcastWriter ) {\n\tconst range = writer.createRangeIn( documentFragment );\n\n\tconst shapes: Array<ViewElement> = [];\n\n\tfor ( const value of range ) {\n\t\tif ( value.type == 'elementStart' && value.item.is( 'element', 'v:shape' ) ) {\n\t\t\tconst id = value.item.getAttribute( 'id' )!;\n\n\t\t\tif ( shapeIds.includes( id ) ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tif ( !containsMatchingImg( value.item.parent!.getChildren(), id ) ) {\n\t\t\t\tshapes.push( value.item );\n\t\t\t}\n\t\t}\n\t}\n\n\tfor ( const shape of shapes ) {\n\t\tconst attrs: Record<string, unknown> = {\n\t\t\tsrc: findSrc( shape )\n\t\t};\n\n\t\tif ( shape.hasAttribute( 'alt' ) ) {\n\t\t\tattrs.alt = shape.getAttribute( 'alt' );\n\t\t}\n\n\t\tconst img = writer.createElement( 'img', attrs );\n\n\t\twriter.insertChild( shape.index! + 1, img, shape.parent! );\n\t}\n\n\tfunction containsMatchingImg( nodes: Iterable<ViewNode>, id: string ): boolean {\n\t\tfor ( const node of nodes ) {\n\t\t\t/* istanbul ignore else -- @preserve */\n\t\t\tif ( node.is( 'element' ) ) {\n\t\t\t\tif ( node.name == 'img' && node.getAttribute( 'v:shapes' ) == id ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\n\t\t\t\tif ( containsMatchingImg( node.getChildren(), id ) ) {\n\t\t\t\t\treturn true;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn false;\n\t}\n\n\tfunction findSrc( shape: ViewElement ) {\n\t\tfor ( const child of shape.getChildren() ) {\n\t\t\t/* istanbul ignore else -- @preserve */\n\t\t\tif ( child.is( 'element' ) && child.getAttribute( 'src' ) ) {\n\t\t\t\treturn child.getAttribute( 'src' );\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Finds all `<img>` elements in a given document fragment which have source pointing to local `file://` resource.\n *\n * @param documentFragment Document fragment in which to look for `<img>` elements.\n * @returns result All found images grouped by source type.\n */\nfunction findAllImageElementsWithLocalSource(\n\tdocumentFragment: ViewDocumentFragment,\n\twriter: UpcastWriter\n): Array<ViewElement> {\n\tconst range = writer.createRangeIn( documentFragment );\n\n\tconst imageElementsMatcher = new Matcher( {\n\t\tname: 'img'\n\t} );\n\n\tconst imgs = [];\n\n\tfor ( const value of range ) {\n\t\tif ( value.item.is( 'element' ) && imageElementsMatcher.match( value.item ) ) {\n\t\t\tif ( value.item.getAttribute( 'src' )!.startsWith( 'file://' ) ) {\n\t\t\t\timgs.push( value.item );\n\t\t\t}\n\t\t}\n\t}\n\n\treturn imgs;\n}\n\n/**\n * Extracts all images HEX representations from a given RTF data.\n *\n * @param rtfData The RTF data from which to extract images HEX representation.\n * @returns Array of found HEX representations. Each array item is an object containing:\n *\n * * hex Image representation in HEX format.\n * * type Type of image, `image/png` or `image/jpeg`.\n */\nfunction extractImageDataFromRtf( rtfData: string ): Array<{ hex: string; type: string }> {\n\tif ( !rtfData ) {\n\t\treturn [];\n\t}\n\n\tconst regexPictureHeader = /{\\\\pict[\\s\\S]+?\\\\bliptag-?\\d+(\\\\blipupi-?\\d+)?({\\\\\\*\\\\blipuid\\s?[\\da-fA-F]+)?[\\s}]*?/;\n\tconst regexPicture = new RegExp( '(?:(' + regexPictureHeader.source + '))([\\\\da-fA-F\\\\s]+)\\\\}', 'g' );\n\tconst images = rtfData.match( regexPicture );\n\tconst result = [];\n\n\tif ( images ) {\n\t\tfor ( const image of images ) {\n\t\t\tlet imageType: string | false = false;\n\n\t\t\tif ( image.includes( '\\\\pngblip' ) ) {\n\t\t\t\timageType = 'image/png';\n\t\t\t} else if ( image.includes( '\\\\jpegblip' ) ) {\n\t\t\t\timageType = 'image/jpeg';\n\t\t\t}\n\n\t\t\tif ( imageType ) {\n\t\t\t\tresult.push( {\n\t\t\t\t\thex: image.replace( regexPictureHeader, '' ).replace( /[^\\da-fA-F]/g, '' ),\n\t\t\t\t\ttype: imageType\n\t\t\t\t} );\n\t\t\t}\n\t\t}\n\t}\n\n\treturn result;\n}\n\n/**\n * Replaces `src` attribute value of all given images with the corresponding base64 image representation.\n *\n * @param imageElements Array of image elements which will have its source replaced.\n * @param imagesHexSources Array of images hex sources (usually the result of `extractImageDataFromRtf()` function).\n * The array should be the same length as `imageElements` parameter.\n */\nfunction replaceImagesFileSourceWithInlineRepresentation(\n\timageElements: Array<ViewElement>,\n\timagesHexSources: ReturnType<typeof extractImageDataFromRtf>,\n\twriter: UpcastWriter\n) {\n\t// Assume there is an equal amount of image elements and images HEX sources so they can be matched accordingly based on existing order.\n\tif ( imageElements.length === imagesHexSources.length ) {\n\t\tfor ( let i = 0; i < imageElements.length; i++ ) {\n\t\t\tconst newSrc = `data:${ imagesHexSources[ i ].type };base64,${ _convertHexToBase64( imagesHexSources[ i ].hex ) }`;\n\t\t\twriter.setAttribute( 'src', newSrc, imageElements[ i ] );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/normalizers/mswordnormalizer\n */\n\nimport { transformListItemLikeElementsIntoLists } from '../filters/list';\nimport { replaceImagesSourceWithBase64 } from '../filters/image';\nimport { setTableAlignment } from '../filters/table';\nimport type { ViewDocument } from 'ckeditor5/src/engine';\nimport type { Normalizer, NormalizerData } from '../normalizer';\n\nconst msWordMatch1 = /<meta\\s*name=\"?generator\"?\\s*content=\"?microsoft\\s*word\\s*\\d+\"?\\/?>/i;\nconst msWordMatch2 = /xmlns:o=\"urn:schemas-microsoft-com/i;\n\n/**\n * Normalizer for the content pasted from Microsoft Word.\n */\nexport default class MSWordNormalizer implements Normalizer {\n\tpublic readonly document: ViewDocument;\n\n\t/**\n\t * Creates a new `MSWordNormalizer` instance.\n\t *\n\t * @param document View document.\n\t */\n\tconstructor( document: ViewDocument ) {\n\t\tthis.document = document;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic isActive( htmlString: string ): boolean {\n\t\treturn msWordMatch1.test( htmlString ) || msWordMatch2.test( htmlString );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic execute( data: NormalizerData ): void {\n\t\tconst { body: documentFragment, stylesString } = data._parsedData;\n\n\t\ttransformListItemLikeElementsIntoLists( documentFragment, stylesString );\n\t\treplaceImagesSourceWithBase64( documentFragment, data.dataTransfer.getData( 'text/rtf' ) );\n\t\tsetTableAlignment( documentFragment );\n\t\tdata.content = documentFragment;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/table\n */\n\nimport { UpcastWriter, type ViewDocumentFragment } from 'ckeditor5/src/engine';\n\n/**\n * Set alignment for table pasted from MS Word.\n *\n * @param documentFragment The view structure to be transformed.\n */\nexport function setTableAlignment( documentFragment: ViewDocumentFragment ): void {\n\tconst upcastWriter = new UpcastWriter( documentFragment.document );\n\n\tfor ( const item of documentFragment.getChildren() ) {\n\t\tif ( !item.is( 'element' ) ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\t// If table is not wrapped into div[align], it should be aligned left.\n\t\t// More details: https://github.com/ckeditor/ckeditor5/issues/8752#issuecomment-1623507171.\n\t\t// RTL tables have the `align` attribute set explicitly -\n\t\t// see https://github.com/ckeditor/ckeditor5/issues/8752#issuecomment-1628876074.\n\t\tif ( item.is( 'element', 'table' ) ) {\n\t\t\tupcastWriter.setAttribute( 'align', 'left', item );\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst align = item.getAttribute( 'align' );\n\t\tconst child = item.getChild( 0 );\n\n\t\t// We're looking for the `<div>` elements with `align` attribute and a child.\n\t\tif ( item.name !== 'div' || !align || !child ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\t// If table is wrapped in div[align], the defined alignment value should be preserved.\n\t\t// More details: https://github.com/ckeditor/ckeditor5/issues/8752#issuecomment-1629065676.\n\t\tif ( child.is( 'element', 'table' ) ) {\n\t\t\tupcastWriter.setAttribute( 'align', align === 'center' ? 'none' : align, child );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/br\n */\n\nimport {\n\tDomConverter,\n\tViewDocument,\n\ttype UpcastWriter,\n\ttype ViewDocumentFragment,\n\ttype ViewElement,\n\ttype ViewNode\n} from 'ckeditor5/src/engine';\n\n/**\n * Transforms `<br>` elements that are siblings to some block element into a paragraphs.\n *\n * @param documentFragment The view structure to be transformed.\n */\nexport default function transformBlockBrsToParagraphs(\n\tdocumentFragment: ViewDocumentFragment,\n\twriter: UpcastWriter\n): void {\n\tconst viewDocument = new ViewDocument( writer.document.stylesProcessor );\n\tconst domConverter = new DomConverter( viewDocument, { renderingMode: 'data' } );\n\n\tconst blockElements = domConverter.blockElements;\n\tconst inlineObjectElements = domConverter.inlineObjectElements;\n\n\tconst elementsToReplace = [];\n\n\tfor ( const value of writer.createRangeIn( documentFragment ) ) {\n\t\tconst element = value.item;\n\n\t\tif ( element.is( 'element', 'br' ) ) {\n\t\t\tconst nextSibling = findSibling( element, 'forward', writer, { blockElements, inlineObjectElements } );\n\t\t\tconst previousSibling = findSibling( element, 'backward', writer, { blockElements, inlineObjectElements } );\n\n\t\t\tconst nextSiblingIsBlock = isBlockViewElement( nextSibling, blockElements );\n\t\t\tconst previousSiblingIsBlock = isBlockViewElement( previousSibling, blockElements );\n\n\t\t\t// If the <br> is surrounded by blocks then convert it to a paragraph:\n\t\t\t// * <p>foo</p>[<br>]<p>bar</p> -> <p>foo</p>[<p></p>]<p>bar</p>\n\t\t\t// * <p>foo</p>[<br>] -> <p>foo</p>[<p></p>]\n\t\t\t// * [<br>]<p>foo</p> -> [<p></p>]<p>foo</p>\n\t\t\tif ( previousSiblingIsBlock || nextSiblingIsBlock ) {\n\t\t\t\telementsToReplace.push( element );\n\t\t\t}\n\t\t}\n\t}\n\n\tfor ( const element of elementsToReplace ) {\n\t\tif ( element.hasClass( 'Apple-interchange-newline' ) ) {\n\t\t\twriter.remove( element );\n\t\t} else {\n\t\t\twriter.replace( element, writer.createElement( 'p' ) );\n\t\t}\n\t}\n}\n\n/**\n * Returns sibling node, threats inline elements as transparent (but should stop on an inline objects).\n */\nfunction findSibling(\n\tviewElement: ViewElement,\n\tdirection: 'forward' | 'backward',\n\twriter: UpcastWriter,\n\t{ blockElements, inlineObjectElements }: { blockElements: Array<string>; inlineObjectElements: Array<string> }\n) {\n\tlet position = writer.createPositionAt( viewElement, direction == 'forward' ? 'after' : 'before' );\n\n\t// Find first position that is just before a first:\n\t// * text node,\n\t// * block element,\n\t// * inline object element.\n\t// It's ignoring any inline (non-object) elements like span, strong, etc.\n\tposition = position.getLastMatchingPosition( ( { item } ) => (\n\t\titem.is( 'element' ) &&\n\t\t!blockElements.includes( item.name ) &&\n\t\t!inlineObjectElements.includes( item.name )\n\t), { direction } );\n\n\treturn direction == 'forward' ? position.nodeAfter : position.nodeBefore;\n}\n\n/**\n * Returns true for view elements that are listed as block view elements.\n */\nfunction isBlockViewElement( node: ViewNode | null, blockElements: Array<string> ) {\n\treturn !!node && node.is( 'element' ) && blockElements.includes( node.name );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/normalizers/googledocsnormalizer\n */\n\nimport { UpcastWriter, type ViewDocument } from 'ckeditor5/src/engine';\n\nimport removeBoldWrapper from '../filters/removeboldwrapper';\nimport transformBlockBrsToParagraphs from '../filters/br';\nimport { unwrapParagraphInListItem } from '../filters/list';\nimport type { Normalizer, NormalizerData } from '../normalizer';\n\nconst googleDocsMatch = /id=(\"|')docs-internal-guid-[-0-9a-f]+(\"|')/i;\n\n/**\n * Normalizer for the content pasted from Google Docs.\n */\nexport default class GoogleDocsNormalizer implements Normalizer {\n\tpublic readonly document: ViewDocument;\n\n\t/**\n\t * Creates a new `GoogleDocsNormalizer` instance.\n\t *\n\t * @param document View document.\n\t */\n\tconstructor( document: ViewDocument ) {\n\t\tthis.document = document;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic isActive( htmlString: string ): boolean {\n\t\treturn googleDocsMatch.test( htmlString );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic execute( data: NormalizerData ): void {\n\t\tconst writer = new UpcastWriter( this.document );\n\t\tconst { body: documentFragment } = data._parsedData;\n\n\t\tremoveBoldWrapper( documentFragment, writer );\n\t\tunwrapParagraphInListItem( documentFragment, writer );\n\t\ttransformBlockBrsToParagraphs( documentFragment, writer );\n\n\t\tdata.content = documentFragment;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/removeboldwrapper\n */\n\nimport type { UpcastWriter, ViewDocumentFragment } from 'ckeditor5/src/engine';\n\n/**\n * Removes the `<b>` tag wrapper added by Google Docs to a copied content.\n *\n * @param documentFragment element `data.content` obtained from clipboard\n */\nexport default function removeBoldWrapper( documentFragment: ViewDocumentFragment, writer: UpcastWriter ): void {\n\tfor ( const child of documentFragment.getChildren() ) {\n\t\tif ( child.is( 'element', 'b' ) && child.getStyle( 'font-weight' ) === 'normal' ) {\n\t\t\tconst childIndex = documentFragment.getChildIndex( child );\n\n\t\t\twriter.remove( child );\n\t\t\twriter.insertChild( childIndex, child.getChildren(), documentFragment );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/normalizers/googlesheetsnormalizer\n */\n\nimport { UpcastWriter, type ViewDocument } from 'ckeditor5/src/engine';\n\nimport removeXmlns from '../filters/removexmlns';\nimport removeGoogleSheetsTag from '../filters/removegooglesheetstag';\nimport removeInvalidTableWidth from '../filters/removeinvalidtablewidth';\nimport removeStyleBlock from '../filters/removestyleblock';\nimport type { Normalizer, NormalizerData } from '../normalizer';\n\nconst googleSheetsMatch = /<google-sheets-html-origin/i;\n\n/**\n * Normalizer for the content pasted from Google Sheets.\n */\nexport default class GoogleSheetsNormalizer implements Normalizer {\n\tpublic readonly document: ViewDocument;\n\n\t/**\n\t * Creates a new `GoogleSheetsNormalizer` instance.\n\t *\n\t * @param document View document.\n\t */\n\tconstructor( document: ViewDocument ) {\n\t\tthis.document = document;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic isActive( htmlString: string ): boolean {\n\t\treturn googleSheetsMatch.test( htmlString );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic execute( data: NormalizerData ): void {\n\t\tconst writer = new UpcastWriter( this.document );\n\t\tconst { body: documentFragment } = data._parsedData;\n\n\t\tremoveGoogleSheetsTag( documentFragment, writer );\n\t\tremoveXmlns( documentFragment, writer );\n\t\tremoveInvalidTableWidth( documentFragment, writer );\n\t\tremoveStyleBlock( documentFragment, writer );\n\n\t\tdata.content = documentFragment;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/removegooglesheetstag\n */\n\nimport type { UpcastWriter, ViewDocumentFragment } from 'ckeditor5/src/engine';\n\n/**\n * Removes the `<google-sheets-html-origin>` tag wrapper added by Google Sheets to a copied content.\n *\n * @param documentFragment element `data.content` obtained from clipboard\n */\nexport default function removeGoogleSheetsTag( documentFragment: ViewDocumentFragment, writer: UpcastWriter ): void {\n\tfor ( const child of documentFragment.getChildren() ) {\n\t\tif ( child.is( 'element', 'google-sheets-html-origin' ) ) {\n\t\t\tconst childIndex = documentFragment.getChildIndex( child );\n\n\t\t\twriter.remove( child );\n\t\t\twriter.insertChild( childIndex, child.getChildren(), documentFragment );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/removexmlns\n */\n\nimport type { UpcastWriter, ViewDocumentFragment } from 'ckeditor5/src/engine';\n\n/**\n * Removes the `xmlns` attribute from table pasted from Google Sheets.\n *\n * @param documentFragment element `data.content` obtained from clipboard\n */\nexport default function removeXmlns( documentFragment: ViewDocumentFragment, writer: UpcastWriter ): void {\n\tfor ( const child of documentFragment.getChildren() ) {\n\t\tif ( child.is( 'element', 'table' ) && child.hasAttribute( 'xmlns' ) ) {\n\t\t\twriter.removeAttribute( 'xmlns', child );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/removeinvalidtablewidth\n */\n\nimport type { UpcastWriter, ViewDocumentFragment } from 'ckeditor5/src/engine';\n\n/**\n * Removes the `width:0px` style from table pasted from Google Sheets.\n *\n * @param documentFragment element `data.content` obtained from clipboard\n */\nexport default function removeInvalidTableWidth( documentFragment: ViewDocumentFragment, writer: UpcastWriter ): void {\n\tfor ( const child of documentFragment.getChildren() ) {\n\t\tif ( child.is( 'element', 'table' ) && child.getStyle( 'width' ) === '0px' ) {\n\t\t\twriter.removeStyle( 'width', child );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/removestyleblock\n */\n\nimport type { UpcastWriter, ViewDocumentFragment } from 'ckeditor5/src/engine';\n\n/**\n * Removes `<style>` block added by Google Sheets to a copied content.\n *\n * @param documentFragment element `data.content` obtained from clipboard\n */\nexport default function removeStyleBlock( documentFragment: ViewDocumentFragment, writer: UpcastWriter ): void {\n\tfor ( const child of Array.from( documentFragment.getChildren() ) ) {\n\t\tif ( child.is( 'element', 'style' ) ) {\n\t\t\twriter.remove( child );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/space\n */\n\n/**\n * Replaces last space preceding elements closing tag with `&nbsp;`. Such operation prevents spaces from being removed\n * during further DOM/View processing (see especially {@link module:engine/view/domconverter~DomConverter#_processDomInlineNodes}).\n * This method also takes into account Word specific `<o:p></o:p>` empty tags.\n * Additionally multiline sequences of spaces and new lines between tags are removed (see #39 and #40).\n *\n * @param htmlString HTML string in which spacing should be normalized.\n * @returns Input HTML with spaces normalized.\n */\nexport function normalizeSpacing( htmlString: string ): string {\n\t// Run normalizeSafariSpaceSpans() two times to cover nested spans.\n\treturn normalizeSafariSpaceSpans( normalizeSafariSpaceSpans( htmlString ) )\n\t\t// Remove all \\r\\n from \"spacerun spans\" so the last replace line doesn't strip all whitespaces.\n\t\t.replace( /(<span\\s+style=['\"]mso-spacerun:yes['\"]>[^\\S\\r\\n]*?)[\\r\\n]+([^\\S\\r\\n]*<\\/span>)/g, '$1$2' )\n\t\t.replace( /<span\\s+style=['\"]mso-spacerun:yes['\"]><\\/span>/g, '' )\n\t\t.replace( /(<span\\s+style=['\"]letter-spacing:[^'\"]+?['\"]>)[\\r\\n]+(<\\/span>)/g, '$1 $2' )\n\t\t.replace( / <\\//g, '\\u00A0</' )\n\t\t.replace( / <o:p><\\/o:p>/g, '\\u00A0<o:p></o:p>' )\n\t\t// Remove <o:p> block filler from empty paragraph. Safari uses \\u00A0 instead of &nbsp;.\n\t\t.replace( /<o:p>(&nbsp;|\\u00A0)<\\/o:p>/g, '' )\n\t\t// Remove all whitespaces when they contain any \\r or \\n.\n\t\t.replace( />([^\\S\\r\\n]*[\\r\\n]\\s*)</g, '><' );\n}\n\n/**\n * Normalizes spacing in special Word `spacerun spans` (`<span style='mso-spacerun:yes'>\\s+</span>`) by replacing\n * all spaces with `&nbsp; ` pairs. This prevents spaces from being removed during further DOM/View processing\n * (see especially {@link module:engine/view/domconverter~DomConverter#_processDomInlineNodes}).\n *\n * @param htmlDocument Native `Document` object in which spacing should be normalized.\n */\nexport function normalizeSpacerunSpans( htmlDocument: Document ): void {\n\thtmlDocument.querySelectorAll( 'span[style*=spacerun]' ).forEach( el => {\n\t\tconst htmlElement = el as HTMLElement;\n\t\tconst innerTextLength = htmlElement.innerText.length || 0;\n\n\t\thtmlElement.innerText = Array( innerTextLength + 1 ).join( '\\u00A0 ' ).substr( 0, innerTextLength );\n\t} );\n}\n\n/**\n * Normalizes specific spacing generated by Safari when content pasted from Word (`<span class=\"Apple-converted-space\"> </span>`)\n * by replacing all spaces sequences longer than 1 space with `&nbsp; ` pairs. This prevents spaces from being removed during\n * further DOM/View processing (see especially {@link module:engine/view/domconverter~DomConverter#_processDataFromDomText}).\n *\n * This function is similar to {@link module:clipboard/utils/normalizeclipboarddata normalizeClipboardData util} but uses\n * regular spaces / &nbsp; sequence for replacement.\n *\n * @param htmlString HTML string in which spacing should be normalized\n * @returns Input HTML with spaces normalized.\n */\nfunction normalizeSafariSpaceSpans( htmlString: string ) {\n\treturn htmlString.replace( /<span(?: class=\"Apple-converted-space\"|)>(\\s+)<\\/span>/g, ( fullMatch, spaces ) => {\n\t\treturn spaces.length === 1 ? ' ' : Array( spaces.length + 1 ).join( '\\u00A0 ' ).substr( 0, spaces.length );\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/filters/parse\n */\n\n/* globals DOMParser */\n\nimport {\n\tDomConverter,\n\tViewDocument,\n\ttype StylesProcessor,\n\ttype ViewDocumentFragment\n} from 'ckeditor5/src/engine';\n\nimport { normalizeSpacing, normalizeSpacerunSpans } from './space';\n\n/**\n * Parses the provided HTML extracting contents of `<body>` and `<style>` tags.\n *\n * @param htmlString HTML string to be parsed.\n */\nexport function parseHtml( htmlString: string, stylesProcessor: StylesProcessor ): ParseHtmlResult {\n\tconst domParser = new DOMParser();\n\n\t// Remove Word specific \"if comments\" so content inside is not omitted by the parser.\n\thtmlString = htmlString.replace( /<!--\\[if gte vml 1]>/g, '' );\n\n\tconst normalizedHtml = normalizeSpacing( cleanContentAfterBody( htmlString ) );\n\n\t// Parse htmlString as native Document object.\n\tconst htmlDocument = domParser.parseFromString( normalizedHtml, 'text/html' );\n\n\tnormalizeSpacerunSpans( htmlDocument );\n\n\t// Get `innerHTML` first as transforming to View modifies the source document.\n\tconst bodyString = htmlDocument.body.innerHTML;\n\n\t// Transform document.body to View.\n\tconst bodyView = documentToView( htmlDocument, stylesProcessor );\n\n\t// Extract stylesheets.\n\tconst stylesObject = extractStyles( htmlDocument );\n\n\treturn {\n\t\tbody: bodyView,\n\t\tbodyString,\n\t\tstyles: stylesObject.styles,\n\t\tstylesString: stylesObject.stylesString\n\t};\n}\n\n/**\n * The result of {@link ~parseHtml}.\n */\nexport interface ParseHtmlResult {\n\n\t/**\n\t * Parsed body content as a traversable structure.\n\t */\n\tbody: ViewDocumentFragment;\n\n\t/**\n\t * Entire body content as a string.\n\t */\n\tbodyString: string;\n\n\t/**\n\t * Array of native `CSSStyleSheet` objects, each representing separate `style` tag from the source HTML.\n\t */\n\tstyles: Array<CSSStyleSheet>;\n\n\t/**\n\t * All `style` tags contents combined in the order of occurrence into one string.\n\t */\n\tstylesString: string;\n}\n\n/**\n * Transforms native `Document` object into {@link module:engine/view/documentfragment~DocumentFragment}. Comments are skipped.\n *\n * @param htmlDocument Native `Document` object to be transformed.\n */\nfunction documentToView( htmlDocument: Document, stylesProcessor: StylesProcessor ) {\n\tconst viewDocument = new ViewDocument( stylesProcessor );\n\tconst domConverter = new DomConverter( viewDocument, { renderingMode: 'data' } );\n\tconst fragment = htmlDocument.createDocumentFragment();\n\tconst nodes = htmlDocument.body.childNodes;\n\n\twhile ( nodes.length > 0 ) {\n\t\tfragment.appendChild( nodes[ 0 ] );\n\t}\n\n\treturn domConverter.domToView( fragment, { skipComments: true } ) as ViewDocumentFragment;\n}\n\n/**\n * Extracts both `CSSStyleSheet` and string representation from all `style` elements available in a provided `htmlDocument`.\n *\n * @param htmlDocument Native `Document` object from which styles will be extracted.\n */\nfunction extractStyles( htmlDocument: Document ): { styles: Array<CSSStyleSheet>; stylesString: string } {\n\tconst styles = [];\n\tconst stylesString = [];\n\tconst styleTags = Array.from( htmlDocument.getElementsByTagName( 'style' ) );\n\n\tfor ( const style of styleTags ) {\n\t\tif ( style.sheet && style.sheet.cssRules && style.sheet.cssRules.length ) {\n\t\t\tstyles.push( style.sheet );\n\t\t\tstylesString.push( style.innerHTML );\n\t\t}\n\t}\n\n\treturn {\n\t\tstyles,\n\t\tstylesString: stylesString.join( ' ' )\n\t};\n}\n\n/**\n * Removes leftover content from between closing </body> and closing </html> tag:\n *\n * ```html\n * <html><body><p>Foo Bar</p></body><span>Fo</span></html> -> <html><body><p>Foo Bar</p></body></html>\n * ```\n *\n * This function is used as specific browsers (Edge) add some random content after `body` tag when pasting from Word.\n * @param htmlString The HTML string to be cleaned.\n * @returns The HTML string with leftover content removed.\n */\nfunction cleanContentAfterBody( htmlString: string ) {\n\tconst bodyCloseTag = '</body>';\n\tconst htmlCloseTag = '</html>';\n\n\tconst bodyCloseIndex = htmlString.indexOf( bodyCloseTag );\n\n\tif ( bodyCloseIndex < 0 ) {\n\t\treturn htmlString;\n\t}\n\n\tconst htmlCloseIndex = htmlString.indexOf( htmlCloseTag, bodyCloseIndex + bodyCloseTag.length );\n\n\treturn htmlString.substring( 0, bodyCloseIndex + bodyCloseTag.length ) +\n\t\t( htmlCloseIndex >= 0 ? htmlString.substring( htmlCloseIndex ) : '' );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/utils/common\n */\n\nimport type {\n\tConversion,\n\tElement,\n\tItem,\n\tPosition,\n\tSchema,\n\tWriter\n} from 'ckeditor5/src/engine';\n\nimport { downcastAttributeToStyle, upcastStyleToAttribute } from './../converters/tableproperties';\nimport type TableUtils from '../tableutils';\n\n/**\n * A common method to update the numeric value. If a value is the default one, it will be unset.\n *\n * @param key An attribute key.\n * @param value The new attribute value.\n * @param item A model item on which the attribute will be set.\n * @param defaultValue The default attribute value. If a value is lower or equal, it will be unset.\n */\nexport function updateNumericAttribute( key: string, value: unknown, item: Item, writer: Writer, defaultValue: unknown = 1 ): void {\n\tif ( value !== undefined && value !== null && defaultValue !== undefined && defaultValue !== null && value > defaultValue ) {\n\t\twriter.setAttribute( key, value, item );\n\t} else {\n\t\twriter.removeAttribute( key, item );\n\t}\n}\n\n/**\n * A common method to create an empty table cell. It creates a proper model structure as a table cell must have at least one block inside.\n *\n * @param writer The model writer.\n * @param insertPosition The position at which the table cell should be inserted.\n * @param attributes The element attributes.\n * @returns Created table cell.\n */\nexport function createEmptyTableCell( writer: Writer, insertPosition: Position, attributes: Record<string, unknown> = {} ): Element {\n\tconst tableCell = writer.createElement( 'tableCell', attributes );\n\n\twriter.insertElement( 'paragraph', tableCell );\n\twriter.insert( tableCell, insertPosition );\n\n\treturn tableCell;\n}\n\n/**\n * Checks if a table cell belongs to the heading column section.\n */\nexport function isHeadingColumnCell( tableUtils: TableUtils, tableCell: Element ): boolean {\n\tconst table = tableCell.parent!.parent as Element;\n\tconst headingColumns = parseInt( table.getAttribute( 'headingColumns' ) as string || '0' );\n\tconst { column } = tableUtils.getCellLocation( tableCell );\n\n\treturn !!headingColumns && column < headingColumns;\n}\n\n/**\n * Enables conversion for an attribute for simple view-model mappings.\n *\n * @param options.defaultValue The default value for the specified `modelAttribute`.\n */\nexport function enableProperty(\n\tschema: Schema,\n\tconversion: Conversion,\n\toptions: {\n\t\tmodelAttribute: string;\n\t\tstyleName: string;\n\t\tdefaultValue: string;\n\t\treduceBoxSides?: boolean;\n\t}\n): void {\n\tconst { modelAttribute } = options;\n\n\tschema.extend( 'tableCell', {\n\t\tallowAttributes: [ modelAttribute ]\n\t} );\n\n\tupcastStyleToAttribute( conversion, { viewElement: /^(td|th)$/, ...options } );\n\tdowncastAttributeToStyle( conversion, { modelElement: 'tableCell', ...options } );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/converters/upcasttable\n */\n\nimport type { Element, UpcastDispatcher, UpcastElementEvent, ViewElement, ViewNode } from 'ckeditor5/src/engine';\n\nimport { createEmptyTableCell } from '../utils/common';\nimport { first } from 'ckeditor5/src/utils';\n\n/**\n * Returns a function that converts the table view representation:\n *\n * ```xml\n * <figure class=\"table\"><table>...</table></figure>\n * ```\n *\n * to the model representation:\n *\n * ```xml\n * <table></table>\n * ```\n */\nexport function upcastTableFigure() {\n\treturn ( dispatcher: UpcastDispatcher ): void => {\n\t\tdispatcher.on<UpcastElementEvent>( 'element:figure', ( evt, data, conversionApi ) => {\n\t\t\t// Do not convert if this is not a \"table figure\".\n\t\t\tif ( !conversionApi.consumable.test( data.viewItem, { name: true, classes: 'table' } ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Find a table element inside the figure element.\n\t\t\tconst viewTable = getViewTableFromFigure( data.viewItem );\n\n\t\t\t// Do not convert if table element is absent or was already converted.\n\t\t\tif ( !viewTable || !conversionApi.consumable.test( viewTable, { name: true } ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Consume the figure to prevent other converters from processing it again.\n\t\t\tconversionApi.consumable.consume( data.viewItem, { name: true, classes: 'table' } );\n\n\t\t\t// Convert view table to model table.\n\t\t\tconst conversionResult = conversionApi.convertItem( viewTable, data.modelCursor );\n\n\t\t\t// Get table element from conversion result.\n\t\t\tconst modelTable = first( conversionResult.modelRange!.getItems() as Iterator<Element> );\n\n\t\t\t// When table wasn't successfully converted then finish conversion.\n\t\t\tif ( !modelTable ) {\n\t\t\t\t// Revert consumed figure so other features can convert it.\n\t\t\t\tconversionApi.consumable.revert( data.viewItem, { name: true, classes: 'table' } );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconversionApi.convertChildren( data.viewItem, conversionApi.writer.createPositionAt( modelTable, 'end' ) );\n\t\t\tconversionApi.updateConversionResult( modelTable, data );\n\t\t} );\n\t};\n}\n\n/**\n * View table element to model table element conversion helper.\n *\n * This conversion helper converts the table element as well as table rows.\n *\n * @returns Conversion helper.\n */\nexport default function upcastTable() {\n\treturn ( dispatcher: UpcastDispatcher ): void => {\n\t\tdispatcher.on<UpcastElementEvent>( 'element:table', ( evt, data, conversionApi ) => {\n\t\t\tconst viewTable = data.viewItem;\n\n\t\t\t// When element was already consumed then skip it.\n\t\t\tif ( !conversionApi.consumable.test( viewTable, { name: true } ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst { rows, headingRows, headingColumns } = scanTable( viewTable );\n\n\t\t\t// Only set attributes if values is greater then 0.\n\t\t\tconst attributes: { headingColumns?: number; headingRows?: number } = {};\n\n\t\t\tif ( headingColumns ) {\n\t\t\t\tattributes.headingColumns = headingColumns;\n\t\t\t}\n\n\t\t\tif ( headingRows ) {\n\t\t\t\tattributes.headingRows = headingRows;\n\t\t\t}\n\n\t\t\tconst table = conversionApi.writer.createElement( 'table', attributes );\n\n\t\t\tif ( !conversionApi.safeInsert( table, data.modelCursor ) ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconversionApi.consumable.consume( viewTable, { name: true } );\n\n\t\t\t// Upcast table rows in proper order (heading rows first).\n\t\t\trows.forEach( row => conversionApi.convertItem( row, conversionApi.writer.createPositionAt( table, 'end' ) ) );\n\n\t\t\t// Convert everything else.\n\t\t\tconversionApi.convertChildren( viewTable, conversionApi.writer.createPositionAt( table, 'end' ) );\n\n\t\t\t// Create one row and one table cell for empty table.\n\t\t\tif ( table.isEmpty ) {\n\t\t\t\tconst row = conversionApi.writer.createElement( 'tableRow' );\n\t\t\t\tconversionApi.writer.insert( row, conversionApi.writer.createPositionAt( table, 'end' ) );\n\n\t\t\t\tcreateEmptyTableCell( conversionApi.writer, conversionApi.writer.createPositionAt( row, 'end' ) );\n\t\t\t}\n\n\t\t\tconversionApi.updateConversionResult( table, data );\n\t\t} );\n\t};\n}\n\n/**\n * A conversion helper that skips empty <tr> elements from upcasting at the beginning of the table.\n *\n * An empty row is considered a table model error but when handling clipboard data there could be rows that contain only row-spanned cells\n * and empty TR-s are used to maintain the table structure (also {@link module:table/tablewalker~TableWalker} assumes that there are only\n * rows that have related `tableRow` elements).\n *\n * *Note:* Only the first empty rows are removed because they have no meaning and it solves the issue\n * of an improper table with all empty rows.\n *\n * @returns Conversion helper.\n */\nexport function skipEmptyTableRow() {\n\treturn ( dispatcher: UpcastDispatcher ): void => {\n\t\tdispatcher.on<UpcastElementEvent>( 'element:tr', ( evt, data ) => {\n\t\t\tif ( data.viewItem.isEmpty && data.modelCursor.index == 0 ) {\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { priority: 'high' } );\n\t};\n}\n\n/**\n * A converter that ensures an empty paragraph is inserted in a table cell if no other content was converted.\n *\n * @returns Conversion helper.\n */\nexport function ensureParagraphInTableCell( elementName: string ) {\n\treturn ( dispatcher: UpcastDispatcher ): void => {\n\t\tdispatcher.on<UpcastElementEvent>( `element:${ elementName }`, ( evt, data, { writer } ) => {\n\t\t\t// The default converter will create a model range on converted table cell.\n\t\t\tif ( !data.modelRange ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst tableCell = data.modelRange.start.nodeAfter as Element;\n\t\t\tconst modelCursor = writer.createPositionAt( tableCell, 0 );\n\n\t\t\t// Ensure a paragraph in the model for empty table cells for converted table cells.\n\t\t\tif ( data.viewItem.isEmpty ) {\n\t\t\t\twriter.insertElement( 'paragraph', modelCursor );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst childNodes = Array.from( tableCell.getChildren() );\n\n\t\t\t// In case there are only markers inside the table cell then move them to the paragraph.\n\t\t\tif ( childNodes.every( node => node.is( 'element', '$marker' ) ) ) {\n\t\t\t\tconst paragraph = writer.createElement( 'paragraph' );\n\n\t\t\t\twriter.insert( paragraph, writer.createPositionAt( tableCell, 0 ) );\n\n\t\t\t\tfor ( const node of childNodes ) {\n\t\t\t\t\twriter.move( writer.createRangeOn( node ), writer.createPositionAt( paragraph, 'end' ) );\n\t\t\t\t}\n\t\t\t}\n\t\t}, { priority: 'low' } );\n\t};\n}\n\n/**\n * Get view `<table>` element from the view widget (`<figure>`).\n */\nfunction getViewTableFromFigure( figureView: ViewElement ) {\n\tfor ( const figureChild of figureView.getChildren() ) {\n\t\tif ( figureChild.is( 'element', 'table' ) ) {\n\t\t\treturn figureChild;\n\t\t}\n\t}\n}\n\n/**\n * Scans table rows and extracts required metadata from the table:\n *\n * headingRows - The number of rows that go as table headers.\n * headingColumns - The maximum number of row headings.\n * rows - Sorted `<tr>` elements as they should go into the model - ie. if `<thead>` is inserted after `<tbody>` in the view.\n */\nfunction scanTable( viewTable: ViewElement ) {\n\tlet headingRows = 0;\n\tlet headingColumns: number | undefined = undefined;\n\n\t// The `<tbody>` and `<thead>` sections in the DOM do not have to be in order `<thead>` -> `<tbody>` and there might be more than one\n\t// of them.\n\t// As the model does not have these sections, rows from different sections must be sorted.\n\t// For example, below is a valid HTML table:\n\t//\n\t// <table>\n\t// <tbody><tr><td>2</td></tr></tbody>\n\t// <thead><tr><td>1</td></tr></thead>\n\t// <tbody><tr><td>3</td></tr></tbody>\n\t// </table>\n\t//\n\t// But browsers will render rows in order as: 1 as the heading and 2 and 3 as the body.\n\tconst headRows = [];\n\tconst bodyRows = [];\n\n\t// Currently the editor does not support more then one <thead> section.\n\t// Only the first <thead> from the view will be used as a heading row and the others will be converted to body rows.\n\tlet firstTheadElement;\n\n\tfor ( const tableChild of Array.from( viewTable.getChildren() as IterableIterator<ViewElement> ) ) {\n\t\t// Only `<thead>`, `<tbody>` & `<tfoot>` from allowed table children can have `<tr>`s.\n\t\t// The else is for future purposes (mainly `<caption>`).\n\t\tif ( tableChild.name !== 'tbody' && tableChild.name !== 'thead' && tableChild.name !== 'tfoot' ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\t// Save the first `<thead>` in the table as table header - all other ones will be converted to table body rows.\n\t\tif ( tableChild.name === 'thead' && !firstTheadElement ) {\n\t\t\tfirstTheadElement = tableChild;\n\t\t}\n\n\t\t// There might be some extra empty text nodes between the `<tr>`s.\n\t\t// Make sure further code operates on `tr`s only. (#145)\n\t\tconst trs = Array.from( tableChild.getChildren() ).filter(\n\t\t\t( el: ViewNode ): el is ViewElement & { name: 'tr' } => el.is( 'element', 'tr' )\n\t\t);\n\n\t\tfor ( const tr of trs ) {\n\t\t\t// This <tr> is a child of a first <thead> element.\n\t\t\tif (\n\t\t\t\t( firstTheadElement && tableChild === firstTheadElement ) ||\n\t\t\t\t(\n\t\t\t\t\ttableChild.name === 'tbody' &&\n\t\t\t\t\tArray.from( tr.getChildren() ).length &&\n\t\t\t\t\tArray.from( tr.getChildren() ).every( e => e.is( 'element', 'th' ) )\n\t\t\t\t)\n\t\t\t) {\n\t\t\t\theadingRows++;\n\t\t\t\theadRows.push( tr );\n\t\t\t} else {\n\t\t\t\tbodyRows.push( tr );\n\t\t\t\t// For other rows check how many column headings this row has.\n\n\t\t\t\tconst headingCols = scanRowForHeadingColumns( tr );\n\n\t\t\t\tif ( !headingColumns || headingCols < headingColumns ) {\n\t\t\t\t\theadingColumns = headingCols;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\treturn {\n\t\theadingRows,\n\t\theadingColumns: headingColumns || 0,\n\t\trows: [ ...headRows, ...bodyRows ]\n\t};\n}\n\n/**\n * Scans a `<tr>` element and its children for metadata:\n * - For heading row:\n * - Adds this row to either the heading or the body rows.\n * - Updates the number of heading rows.\n * - For body rows:\n * - Calculates the number of column headings.\n */\nfunction scanRowForHeadingColumns( tr: ViewElement ) {\n\tlet headingColumns = 0;\n\tlet index = 0;\n\n\t// Filter out empty text nodes from tr children.\n\tconst children = Array.from( tr.getChildren() as IterableIterator<ViewElement> )\n\t\t.filter( child => child.name === 'th' || child.name === 'td' );\n\n\t// Count starting adjacent <th> elements of a <tr>.\n\twhile ( index < children.length && children[ index ].name === 'th' ) {\n\t\tconst th = children[ index ];\n\n\t\t// Adjust columns calculation by the number of spanned columns.\n\t\tconst colspan = parseInt( th.getAttribute( 'colspan' ) as string || '1' );\n\n\t\theadingColumns = headingColumns + colspan;\n\t\tindex++;\n\t}\n\n\treturn headingColumns;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tablewalker\n */\nimport type { Element, Position } from 'ckeditor5/src/engine';\n\n// @if CK_DEBUG // const CKEditorError = require( '@ckeditor/ckeditor5-utils/src/ckeditorerror' ).default;\n\ninterface CellData {\n\tcell: Element;\n\trow: number;\n\tcolumn: number;\n}\n\n/**\n * An object with configuration for `TableWalker`.\n */\nexport interface TableWalkerOptions {\n\n\t/**\n\t * A row index for which this iterator will output cells. Can't be used together with `startRow` and `endRow`.\n\t */\n\trow?: number | null;\n\n\t/**\n\t * A row index from which this iterator should start. Can't be used together with `row`. Default value is 0.\n\t */\n\tstartRow?: number;\n\n\t/**\n\t * A row index at which this iterator should end. Can't be used together with `row`.\n\t */\n\tendRow?: number;\n\n\t/**\n\t * A column index for which this iterator will output cells. Can't be used together with `startColumn` and `endColumn`.\n\t */\n\tcolumn?: number;\n\n\t/**\n\t * A column index from which this iterator should start. Can't be used together with `column`. Default value is 0.\n\t */\n\tstartColumn?: number;\n\n\t/**\n\t * A column index at which this iterator should end. Can't be used together with `column`.\n\t */\n\tendColumn?: number;\n\n\t/**\n\t * Also return values for spanned cells. Default value is false.\n\t */\n\tincludeAllSlots?: boolean;\n}\n\n/**\n * The table iterator class. It allows to iterate over table cells. For each cell the iterator yields\n * {@link module:table/tablewalker~TableSlot} with proper table cell attributes.\n */\nexport default class TableWalker implements IterableIterator<TableSlot> {\n\t/**\n\t * The walker's table element.\n\t *\n\t * @internal\n\t */\n\tpublic readonly _table: Element;\n\n\t/**\n\t * A row index from which this iterator will start.\n\t */\n\tprivate readonly _startRow: number | null;\n\n\t/**\n\t * A row index at which this iterator will end.\n\t */\n\tprivate readonly _endRow?: number | null;\n\n\t/**\n\t * If set, the table walker will only output cells from a given column and following ones or cells that overlap them.\n\t */\n\tprivate readonly _startColumn: number;\n\n\t/**\n\t * If set, the table walker will only output cells up to a given column.\n\t */\n\tprivate readonly _endColumn?: number;\n\n\t/**\n\t * Enables output of spanned cells that are normally not yielded.\n\t */\n\tprivate readonly _includeAllSlots: boolean;\n\n\t/**\n\t * Row indexes to skip from the iteration.\n\t */\n\tprivate readonly _skipRows: Set<number>;\n\n\t/**\n\t * The current row index.\n\t *\n\t * @internal\n\t */\n\tpublic _row: number;\n\n\t/**\n\t * The index of the current row element in the table.\n\t *\n\t * @internal\n\t */\n\tpublic _rowIndex: number;\n\n\t/**\n\t * The current column index.\n\t *\n\t * @internal\n\t */\n\tpublic _column: number;\n\n\t/**\n\t * The cell index in a parent row. For spanned cells when {@link #_includeAllSlots} is set to `true`,\n\t * this represents the index of the next table cell.\n\t *\n\t * @internal\n\t */\n\tpublic _cellIndex: number;\n\n\t/**\n\t * Holds a map of spanned cells in a table.\n\t */\n\tprivate readonly _spannedCells: Map<number, Map<number, CellData>>;\n\n\t/**\n\t * Index of the next column where a cell is anchored.\n\t */\n\tprivate _nextCellAtColumn: number;\n\n\t/**\n\t * Creates an instance of the table walker.\n\t *\n\t * The table walker iterates internally by traversing the table from row index = 0 and column index = 0.\n\t * It walks row by row and column by column in order to output values defined in the constructor.\n\t * By default it will output only the locations that are occupied by a cell. To include also spanned rows and columns,\n\t * pass the `includeAllSlots` option to the constructor.\n\t *\n\t * The most important values of the iterator are column and row indexes of a cell.\n\t *\n\t * See {@link module:table/tablewalker~TableSlot} what values are returned by the table walker.\n\t *\n\t * To iterate over a given row:\n\t *\n\t * ```ts\n\t * const tableWalker = new TableWalker( table, { startRow: 1, endRow: 2 } );\n\t *\n\t * for ( const tableSlot of tableWalker ) {\n\t * console.log( 'A cell at row', tableSlot.row, 'and column', tableSlot.column );\n\t * }\n\t * ```\n\t *\n\t * For instance the code above for the following table:\n\t *\n\t * +----+----+----+----+----+----+\n\t * | 00 | 02 | 03 | 04 | 05 |\n\t * | +----+----+----+----+\n\t * | | 12 | 14 | 15 |\n\t * | +----+----+----+ +\n\t * | | 22 | |\n\t * |----+----+----+----+----+ +\n\t * | 30 | 31 | 32 | 33 | 34 | |\n\t * +----+----+----+----+----+----+\n\t *\n\t * will log in the console:\n\t *\n\t * 'A cell at row 1 and column 2'\n\t * 'A cell at row 1 and column 4'\n\t * 'A cell at row 1 and column 5'\n\t * 'A cell at row 2 and column 2'\n\t *\n\t * To also iterate over spanned cells:\n\t *\n\t * ```ts\n\t * const tableWalker = new TableWalker( table, { row: 1, includeAllSlots: true } );\n\t *\n\t * for ( const tableSlot of tableWalker ) {\n\t * console.log( 'Slot at', tableSlot.row, 'x', tableSlot.column, ':', tableSlot.isAnchor ? 'is anchored' : 'is spanned' );\n\t * }\n\t * ```\n\t *\n\t * will log in the console for the table from the previous example:\n\t *\n\t * 'Cell at 1 x 0 : is spanned'\n\t * 'Cell at 1 x 1 : is spanned'\n\t * 'Cell at 1 x 2 : is anchored'\n\t * 'Cell at 1 x 3 : is spanned'\n\t * 'Cell at 1 x 4 : is anchored'\n\t * 'Cell at 1 x 5 : is anchored'\n\t *\n\t * **Note**: Option `row` is a shortcut that sets both `startRow` and `endRow` to the same row.\n\t * (Use either `row` or `startRow` and `endRow` but never together). Similarly the `column` option sets both `startColumn`\n\t * and `endColumn` to the same column (Use either `column` or `startColumn` and `endColumn` but never together).\n\t *\n\t * @param table A table over which the walker iterates.\n\t * @param options An object with configuration.\n\t * @param options.row A row index for which this iterator will output cells. Can't be used together with `startRow` and `endRow`.\n\t * @param options.startRow A row index from which this iterator should start. Can't be used together with `row`. Default value is 0.\n\t * @param options.endRow A row index at which this iterator should end. Can't be used together with `row`.\n\t * @param options.column A column index for which this iterator will output cells.\n\t * Can't be used together with `startColumn` and `endColumn`.\n\t * @param options.startColumn A column index from which this iterator should start.\n\t * Can't be used together with `column`. Default value is 0.\n\t * @param options.endColumn A column index at which this iterator should end. Can't be used together with `column`.\n\t * @param options.includeAllSlots Also return values for spanned cells. Default value is \"false\".\n\t */\n\tconstructor( table: Element, options: TableWalkerOptions = {} ) {\n\t\tthis._table = table;\n\t\tthis._startRow = options.row !== undefined ? options.row : options.startRow || 0;\n\t\tthis._endRow = options.row !== undefined ? options.row : options.endRow;\n\t\tthis._startColumn = options.column !== undefined ? options.column : options.startColumn || 0;\n\t\tthis._endColumn = options.column !== undefined ? options.column : options.endColumn;\n\t\tthis._includeAllSlots = !!options.includeAllSlots;\n\t\tthis._skipRows = new Set();\n\t\tthis._row = 0;\n\t\tthis._rowIndex = 0;\n\t\tthis._column = 0;\n\t\tthis._cellIndex = 0;\n\t\tthis._spannedCells = new Map();\n\t\tthis._nextCellAtColumn = -1;\n\t}\n\n\t/**\n\t * Iterable interface.\n\t */\n\tpublic [ Symbol.iterator ](): IterableIterator<TableSlot> {\n\t\treturn this;\n\t}\n\n\t/**\n\t * Gets the next table walker's value.\n\t *\n\t * @returns The next table walker's value.\n\t */\n\tpublic next(): IteratorResult<TableSlot, undefined> {\n\t\tconst row = this._table.getChild( this._rowIndex );\n\n\t\t// Iterator is done when there's no row (table ended) or the row is after `endRow` limit.\n\t\tif ( !row || this._isOverEndRow() ) {\n\t\t\treturn { done: true, value: undefined };\n\t\t}\n\n\t\t// We step over current element when it is not a tableRow instance.\n\t\tif ( !row.is( 'element', 'tableRow' ) ) {\n\t\t\tthis._rowIndex++;\n\n\t\t\treturn this.next();\n\t\t}\n\n\t\tif ( this._isOverEndColumn() ) {\n\t\t\treturn this._advanceToNextRow();\n\t\t}\n\n\t\tlet outValue: IteratorYieldResult<TableSlot> | null = null;\n\n\t\tconst spanData = this._getSpanned();\n\n\t\tif ( spanData ) {\n\t\t\tif ( this._includeAllSlots && !this._shouldSkipSlot() ) {\n\t\t\t\toutValue = this._formatOutValue( spanData.cell, spanData.row, spanData.column );\n\t\t\t}\n\t\t} else {\n\t\t\tconst cell = row.getChild( this._cellIndex ) as Element;\n\n\t\t\tif ( !cell ) {\n\t\t\t\t// If there are no more cells left in row advance to the next row.\n\t\t\t\treturn this._advanceToNextRow();\n\t\t\t}\n\n\t\t\tconst colspan = parseInt( cell.getAttribute( 'colspan' ) as string || '1' );\n\t\t\tconst rowspan = parseInt( cell.getAttribute( 'rowspan' ) as string || '1' );\n\n\t\t\t// Record this cell spans if it's not 1x1 cell.\n\t\t\tif ( colspan > 1 || rowspan > 1 ) {\n\t\t\t\tthis._recordSpans( cell, rowspan, colspan );\n\t\t\t}\n\n\t\t\tif ( !this._shouldSkipSlot() ) {\n\t\t\t\toutValue = this._formatOutValue( cell );\n\t\t\t}\n\n\t\t\tthis._nextCellAtColumn = this._column + colspan;\n\t\t}\n\n\t\t// Advance to the next column before returning value.\n\t\tthis._column++;\n\n\t\tif ( this._column == this._nextCellAtColumn ) {\n\t\t\tthis._cellIndex++;\n\t\t}\n\n\t\t// The current value will be returned only if current row and column are not skipped.\n\t\treturn outValue || this.next();\n\t}\n\n\t/**\n\t * Marks a row to skip in the next iteration. It will also skip cells from the current row if there are any cells from the current row\n\t * to output.\n\t *\n\t * @param row The row index to skip.\n\t */\n\tpublic skipRow( row: number ): void {\n\t\tthis._skipRows.add( row );\n\t}\n\n\t/**\n\t * Advances internal cursor to the next row.\n\t */\n\tprivate _advanceToNextRow() {\n\t\tthis._row++;\n\t\tthis._rowIndex++;\n\t\tthis._column = 0;\n\t\tthis._cellIndex = 0;\n\t\tthis._nextCellAtColumn = -1;\n\n\t\treturn this.next();\n\t}\n\n\t/**\n\t * Checks if the current row is over {@link #_endRow}.\n\t */\n\tprivate _isOverEndRow() {\n\t\t// If #_endRow is defined skip all rows after it.\n\t\treturn this._endRow !== undefined && this._row > this._endRow!;\n\t}\n\n\t/**\n\t * Checks if the current cell is over {@link #_endColumn}\n\t */\n\tprivate _isOverEndColumn() {\n\t\t// If #_endColumn is defined skip all cells after it.\n\t\treturn this._endColumn !== undefined && this._column > this._endColumn;\n\t}\n\n\t/**\n\t * A common method for formatting the iterator's output value.\n\t *\n\t * @param cell The table cell to output.\n\t * @param anchorRow The row index of a cell anchor slot.\n\t * @param anchorColumn The column index of a cell anchor slot.\n\t */\n\tprivate _formatOutValue( cell: Element, anchorRow = this._row, anchorColumn = this._column ): IteratorYieldResult<TableSlot> {\n\t\treturn {\n\t\t\tdone: false,\n\t\t\tvalue: new TableSlot( this, cell, anchorRow, anchorColumn )\n\t\t};\n\t}\n\n\t/**\n\t * Checks if the current slot should be skipped.\n\t */\n\tprivate _shouldSkipSlot(): boolean {\n\t\tconst rowIsMarkedAsSkipped = this._skipRows.has( this._row );\n\t\tconst rowIsBeforeStartRow = this._row < this._startRow!;\n\n\t\tconst columnIsBeforeStartColumn = this._column < this._startColumn;\n\t\tconst columnIsAfterEndColumn = this._endColumn !== undefined && this._column > this._endColumn;\n\n\t\treturn rowIsMarkedAsSkipped || rowIsBeforeStartRow || columnIsBeforeStartColumn || columnIsAfterEndColumn;\n\t}\n\n\t/**\n\t * Returns the cell element that is spanned over the current cell location.\n\t */\n\tprivate _getSpanned(): CellData | null {\n\t\tconst rowMap = this._spannedCells.get( this._row );\n\n\t\t// No spans for given row.\n\t\tif ( !rowMap ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// If spans for given rows has entry for column it means that this location if spanned by other cell.\n\t\treturn rowMap.get( this._column ) || null;\n\t}\n\n\t/**\n\t * Updates spanned cells map relative to the current cell location and its span dimensions.\n\t *\n\t * @param cell A cell that is spanned.\n\t * @param rowspan Cell height.\n\t * @param colspan Cell width.\n\t */\n\tprivate _recordSpans( cell: Element, rowspan: number, colspan: number ) {\n\t\tconst data = {\n\t\t\tcell,\n\t\t\trow: this._row,\n\t\t\tcolumn: this._column\n\t\t};\n\n\t\tfor ( let rowToUpdate = this._row; rowToUpdate < this._row + rowspan; rowToUpdate++ ) {\n\t\t\tfor ( let columnToUpdate = this._column; columnToUpdate < this._column + colspan; columnToUpdate++ ) {\n\t\t\t\tif ( rowToUpdate != this._row || columnToUpdate != this._column ) {\n\t\t\t\t\tthis._markSpannedCell( rowToUpdate, columnToUpdate, data );\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Marks the cell location as spanned by another cell.\n\t *\n\t * @param row The row index of the cell location.\n\t * @param column The column index of the cell location.\n\t * @param data A spanned cell details (cell element, anchor row and column).\n\t */\n\tprivate _markSpannedCell( row: number, column: number, data: CellData ) {\n\t\tif ( !this._spannedCells.has( row ) ) {\n\t\t\tthis._spannedCells.set( row, new Map() );\n\t\t}\n\n\t\tconst rowSpans = this._spannedCells.get( row )!;\n\n\t\trowSpans.set( column, data );\n\t}\n}\n\n/**\n * An object returned by {@link module:table/tablewalker~TableWalker} when traversing table cells.\n */\nclass TableSlot {\n\t/**\n\t * The current table cell.\n\t */\n\tpublic readonly cell: Element;\n\n\t/**\n\t * The row index of a table slot.\n\t */\n\tpublic readonly row: number;\n\n\t/**\n\t * The column index of a table slot.\n\t */\n\tpublic readonly column: number;\n\n\t/**\n\t * The row index of a cell anchor slot.\n\t */\n\tpublic readonly cellAnchorRow: number;\n\n\t/**\n\t * The column index of a cell anchor slot.\n\t */\n\tpublic readonly cellAnchorColumn: number;\n\n\t/**\n\t * The index of the current cell in the parent row.\n\t */\n\tprivate readonly _cellIndex: number;\n\n\t/**\n\t * The index of the current row element in the table.\n\t */\n\tprivate readonly _rowIndex: number;\n\n\t/**\n\t * The table element.\n\t */\n\tprivate readonly _table: Element;\n\n\t/**\n\t * Creates an instance of the table walker value.\n\t *\n\t * @param tableWalker The table walker instance.\n\t * @param cell The current table cell.\n\t * @param anchorRow The row index of a cell anchor slot.\n\t * @param anchorColumn The column index of a cell anchor slot.\n\t */\n\tconstructor( tableWalker: TableWalker, cell: Element, anchorRow: number, anchorColumn: number ) {\n\t\tthis.cell = cell;\n\t\tthis.row = tableWalker._row;\n\t\tthis.column = tableWalker._column;\n\t\tthis.cellAnchorRow = anchorRow;\n\t\tthis.cellAnchorColumn = anchorColumn;\n\t\tthis._cellIndex = tableWalker._cellIndex;\n\t\tthis._rowIndex = tableWalker._rowIndex;\n\t\tthis._table = tableWalker._table;\n\t}\n\n\t// @if CK_DEBUG // public get isSpanned(): unknown { return throwMissingGetterError( 'isSpanned' ); }\n\t// @if CK_DEBUG // public get colspan(): unknown { return throwMissingGetterError( 'colspan' ); }\n\t// @if CK_DEBUG // public get rowspan(): unknown { return throwMissingGetterError( 'rowspan' ); }\n\t// @if CK_DEBUG // public get cellIndex(): unknown { return throwMissingGetterError( 'cellIndex' ); }\n\n\t/**\n\t * Whether the cell is anchored in the current slot.\n\t */\n\tpublic get isAnchor(): boolean {\n\t\treturn this.row === this.cellAnchorRow && this.column === this.cellAnchorColumn;\n\t}\n\n\t/**\n\t * The width of a cell defined by a `colspan` attribute. If the model attribute is not present, it is set to `1`.\n\t */\n\tpublic get cellWidth(): number {\n\t\treturn parseInt( this.cell.getAttribute( 'colspan' ) as string || '1' );\n\t}\n\n\t/**\n\t * The height of a cell defined by a `rowspan` attribute. If the model attribute is not present, it is set to `1`.\n\t */\n\tpublic get cellHeight(): number {\n\t\treturn parseInt( this.cell.getAttribute( 'rowspan' ) as string || '1' );\n\t}\n\n\t/**\n\t * The index of the current row element in the table.\n\t */\n\tpublic get rowIndex(): number {\n\t\treturn this._rowIndex;\n\t}\n\n\t/**\n\t * Returns the {@link module:engine/model/position~Position} before the table slot.\n\t */\n\tpublic getPositionBefore(): Position {\n\t\tconst model = this._table.root.document!.model;\n\n\t\treturn model.createPositionAt( this._table.getChild( this.row ) as Element, this._cellIndex );\n\t}\n}\n\nexport type { TableSlot };\n\n/**\n * This `TableSlot`'s getter (property) was removed in CKEditor 5 v20.0.0.\n *\n * Check out the new `TableWalker`'s API in the documentation.\n *\n * @error tableslot-getter-removed\n * @param getterName\n */\n// @if CK_DEBUG // function throwMissingGetterError( getterName: string ): void {\n// @if CK_DEBUG //\t\tthrow new CKEditorError( 'tableslot-getter-removed', null, {\n// @if CK_DEBUG //\t\t\tgetterName\n// @if CK_DEBUG //\t\t} );\n// @if CK_DEBUG // }\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/converters/downcast\n */\n\nimport { toWidget, toWidgetEditable } from 'ckeditor5/src/widget';\nimport type { Node, ViewElement, Element, DowncastWriter, ElementCreatorFunction } from 'ckeditor5/src/engine';\n\nimport TableWalker from './../tablewalker';\nimport type TableUtils from '../tableutils';\nimport type { AdditionalSlot } from '../tableediting';\n\n/**\n * Model table element to view table element conversion helper.\n */\nexport function downcastTable( tableUtils: TableUtils, options: DowncastTableOptions ): ElementCreatorFunction {\n\treturn ( table, { writer } ) => {\n\t\tconst headingRows = table.getAttribute( 'headingRows' ) || 0;\n\t\tconst tableElement = writer.createContainerElement( 'table', null, [] );\n\t\tconst figureElement = writer.createContainerElement( 'figure', { class: 'table' }, tableElement );\n\n\t\t// Table head slot.\n\t\tif ( headingRows > 0 ) {\n\t\t\twriter.insert(\n\t\t\t\twriter.createPositionAt( tableElement, 'end' ),\n\t\t\t\twriter.createContainerElement(\n\t\t\t\t\t'thead',\n\t\t\t\t\tnull,\n\t\t\t\t\twriter.createSlot( element => element.is( 'element', 'tableRow' ) && element.index! < headingRows )\n\t\t\t\t)\n\t\t\t);\n\t\t}\n\n\t\t// Table body slot.\n\t\tif ( headingRows < tableUtils.getRows( table ) ) {\n\t\t\twriter.insert(\n\t\t\t\twriter.createPositionAt( tableElement, 'end' ),\n\t\t\t\twriter.createContainerElement(\n\t\t\t\t\t'tbody',\n\t\t\t\t\tnull,\n\t\t\t\t\twriter.createSlot( element => element.is( 'element', 'tableRow' ) && element.index! >= headingRows )\n\t\t\t\t)\n\t\t\t);\n\t\t}\n\n\t\t// Dynamic slots.\n\t\tfor ( const { positionOffset, filter } of options.additionalSlots ) {\n\t\t\twriter.insert(\n\t\t\t\twriter.createPositionAt( tableElement, positionOffset ),\n\t\t\t\twriter.createSlot( filter )\n\t\t\t);\n\t\t}\n\n\t\t// Create a slot with items that don't fit into the table.\n\t\twriter.insert(\n\t\t\twriter.createPositionAt( tableElement, 'after' ),\n\t\t\twriter.createSlot( element => {\n\t\t\t\tif ( element.is( 'element', 'tableRow' ) ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\treturn !options.additionalSlots.some( ( { filter } ) => filter( element ) );\n\t\t\t} )\n\t\t);\n\n\t\treturn options.asWidget ? toTableWidget( figureElement, writer ) : figureElement;\n\t};\n}\n\n/**\n * Model table row element to view `<tr>` element conversion helper.\n *\n * @returns Element creator.\n */\nexport function downcastRow(): ElementCreatorFunction {\n\treturn ( tableRow, { writer } ) => {\n\t\treturn tableRow.isEmpty ?\n\t\t\twriter.createEmptyElement( 'tr' ) :\n\t\t\twriter.createContainerElement( 'tr' );\n\t};\n}\n\n/**\n * Model table cell element to view `<td>` or `<th>` element conversion helper.\n *\n * This conversion helper will create proper `<th>` elements for table cells that are in the heading section (heading row or column)\n * and `<td>` otherwise.\n *\n * @param options.asWidget If set to `true`, the downcast conversion will produce a widget.\n * @returns Element creator.\n */\nexport function downcastCell( options: { asWidget?: boolean } = {} ): ElementCreatorFunction {\n\treturn ( tableCell, { writer } ) => {\n\t\tconst tableRow = tableCell.parent as Element;\n\t\tconst table = tableRow.parent as Element;\n\t\tconst rowIndex = table.getChildIndex( tableRow )!;\n\n\t\tconst tableWalker = new TableWalker( table, { row: rowIndex } );\n\t\tconst headingRows = table.getAttribute( 'headingRows' ) as number || 0;\n\t\tconst headingColumns = table.getAttribute( 'headingColumns' ) as number || 0;\n\n\t\tlet result: ViewElement | null = null;\n\n\t\t// We need to iterate over a table in order to get proper row & column values from a walker.\n\t\tfor ( const tableSlot of tableWalker ) {\n\t\t\tif ( tableSlot.cell == tableCell ) {\n\t\t\t\tconst isHeading = tableSlot.row < headingRows || tableSlot.column < headingColumns;\n\t\t\t\tconst cellElementName = isHeading ? 'th' : 'td';\n\n\t\t\t\tresult = options.asWidget ?\n\t\t\t\t\ttoWidgetEditable( writer.createEditableElement( cellElementName ), writer ) :\n\t\t\t\t\twriter.createContainerElement( cellElementName );\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn result;\n\t};\n}\n\n/**\n * Overrides paragraph inside table cell conversion.\n *\n * This converter:\n * * should be used to override default paragraph conversion.\n * * It will only convert `<paragraph>` placed directly inside `<tableCell>`.\n * * For a single paragraph without attributes it returns `<span>` to simulate data table.\n * * For all other cases it returns `<p>` element.\n *\n * @param options.asWidget If set to `true`, the downcast conversion will produce a widget.\n * @returns Element creator.\n */\nexport function convertParagraphInTableCell( options: { asWidget?: boolean } = {} ): ElementCreatorFunction {\n\treturn ( modelElement, { writer } ) => {\n\t\tif ( !modelElement.parent!.is( 'element', 'tableCell' ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( !isSingleParagraphWithoutAttributes( modelElement ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\tif ( options.asWidget ) {\n\t\t\treturn writer.createContainerElement( 'span', { class: 'ck-table-bogus-paragraph' } );\n\t\t} else {\n\t\t\t// Using `<p>` in case there are some markers on it and transparentRendering will render it anyway.\n\t\t\tconst viewElement = writer.createContainerElement( 'p' );\n\n\t\t\twriter.setCustomProperty( 'dataPipeline:transparentRendering', true, viewElement );\n\n\t\t\treturn viewElement;\n\t\t}\n\t};\n}\n\n/**\n * Checks if given model `<paragraph>` is an only child of a parent (`<tableCell>`) and if it has any attribute set.\n *\n * The paragraph should be converted in the editing view to:\n *\n * * If returned `true` - to a `<span class=\"ck-table-bogus-paragraph\">`\n * * If returned `false` - to a `<p>`\n */\nexport function isSingleParagraphWithoutAttributes( modelElement: Element ): boolean {\n\tconst tableCell = modelElement.parent!;\n\n\tconst isSingleParagraph = tableCell.childCount == 1;\n\n\treturn isSingleParagraph && !hasAnyAttribute( modelElement );\n}\n\n/**\n * Converts a given {@link module:engine/view/element~Element} to a table widget:\n * * Adds a {@link module:engine/view/element~Element#_setCustomProperty custom property} allowing to recognize the table widget element.\n * * Calls the {@link module:widget/utils~toWidget} function with the proper element's label creator.\n *\n * @param writer An instance of the view writer.\n * @param label The element's label. It will be concatenated with the table `alt` attribute if one is present.\n */\nfunction toTableWidget( viewElement: ViewElement, writer: DowncastWriter ): ViewElement {\n\twriter.setCustomProperty( 'table', true, viewElement );\n\n\treturn toWidget( viewElement, writer, { hasSelectionHandle: true } );\n}\n\n/**\n * Checks if an element has any attributes set.\n */\nfunction hasAnyAttribute( element: Node ): boolean {\n\tconst iteratorItem = element.getAttributeKeys().next();\n\n\treturn !iteratorItem.done;\n}\n\nexport interface DowncastTableOptions {\n\n\t/**\n\t * If set to `true`, the downcast conversion will produce a widget.\n\t */\n\tasWidget?: boolean;\n\n\t/**\n\t * Array of additional slot handlers.\n\t */\n\tadditionalSlots: Array<AdditionalSlot>;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/inserttablecommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\n\nimport type {\n\tDocumentSelection,\n\tSchema,\n\tSelection,\n\tElement\n} from 'ckeditor5/src/engine';\nimport type TableUtils from '../tableutils';\n\n/**\n * The insert table command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'insertTable'` editor command.\n *\n * To insert a table at the current selection, execute the command and specify the dimensions:\n *\n * ```ts\n * editor.execute( 'insertTable', { rows: 20, columns: 5 } );\n * ```\n */\nexport default class InsertTableCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst schema = model.schema;\n\n\t\tthis.isEnabled = isAllowedInParent( selection, schema );\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * Inserts a table with the given number of rows and columns into the editor.\n\t *\n\t * @param options.rows The number of rows to create in the inserted table. Default value is 2.\n\t * @param options.columns The number of columns to create in the inserted table. Default value is 2.\n\t * @param options.headingRows The number of heading rows. If not provided it will default to\n\t * {@link module:table/tableconfig~TableConfig#defaultHeadings `config.table.defaultHeadings.rows`} table config.\n\t * @param options.headingColumns The number of heading columns. If not provided it will default to\n\t * {@link module:table/tableconfig~TableConfig#defaultHeadings `config.table.defaultHeadings.columns`} table config.\n\t * @fires execute\n\t */\n\tpublic override execute(\n\t\toptions: {\n\t\t\trows?: number;\n\t\t\tcolumns?: number;\n\t\t\theadingRows?: number;\n\t\t\theadingColumns?: number;\n\t\t} = {}\n\t): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst tableUtils: TableUtils = editor.plugins.get( 'TableUtils' );\n\n\t\tconst defaultRows = editor.config.get( 'table.defaultHeadings.rows' );\n\t\tconst defaultColumns = editor.config.get( 'table.defaultHeadings.columns' );\n\n\t\tif ( options.headingRows === undefined && defaultRows ) {\n\t\t\toptions.headingRows = defaultRows;\n\t\t}\n\n\t\tif ( options.headingColumns === undefined && defaultColumns ) {\n\t\t\toptions.headingColumns = defaultColumns;\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\tconst table = tableUtils.createTable( writer, options );\n\n\t\t\tmodel.insertObject( table, null, null, { findOptimalPosition: 'auto' } );\n\n\t\t\twriter.setSelection( writer.createPositionAt( table.getNodeByPath( [ 0, 0, 0 ] ), 0 ) );\n\t\t} );\n\t}\n}\n\n/**\n * Checks if the table is allowed in the parent.\n */\nfunction isAllowedInParent( selection: Selection | DocumentSelection, schema: Schema ) {\n\tconst positionParent = selection.getFirstPosition()!.parent;\n\tconst validParent = positionParent === positionParent.root ? positionParent : positionParent.parent;\n\n\treturn schema.checkChild( validParent as Element, 'table' );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/insertrowcommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport type TableUtils from '../tableutils';\n\n/**\n * The insert row command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'insertTableRowBelow'` and\n * `'insertTableRowAbove'` editor commands.\n *\n * To insert a row below the selected cell, execute the following command:\n *\n * ```ts\n * editor.execute( 'insertTableRowBelow' );\n * ```\n *\n * To insert a row above the selected cell, execute the following command:\n *\n * ```ts\n * editor.execute( 'insertTableRowAbove' );\n * ```\n */\nexport default class InsertRowCommand extends Command {\n\t/**\n\t * The order of insertion relative to the row in which the caret is located.\n\t */\n\tpublic readonly order: 'above' | 'below';\n\n\t/**\n\t * Creates a new `InsertRowCommand` instance.\n\t *\n\t * @param editor The editor on which this command will be used.\n\t * @param options.order The order of insertion relative to the row in which the caret is located.\n\t * Possible values: `\"above\"` and `\"below\"`. Default value is \"below\"\n\t */\n\tconstructor( editor: Editor, options: { order?: 'above' | 'below' } = {} ) {\n\t\tsuper( editor );\n\n\t\tthis.order = options.order || 'below';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst selection = this.editor.model.document.selection;\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst isAnyCellSelected = !!tableUtils.getSelectionAffectedTableCells( selection ).length;\n\n\t\tthis.isEnabled = isAnyCellSelected;\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * Depending on the command's {@link #order} value, it inserts a row `'below'` or `'above'` the row in which selection is set.\n\t *\n\t * @fires execute\n\t */\n\tpublic override execute(): void {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\t\tconst tableUtils: TableUtils = editor.plugins.get( 'TableUtils' );\n\t\tconst insertAbove = this.order === 'above';\n\n\t\tconst affectedTableCells = tableUtils.getSelectionAffectedTableCells( selection );\n\t\tconst rowIndexes = tableUtils.getRowIndexes( affectedTableCells );\n\n\t\tconst row = insertAbove ? rowIndexes.first : rowIndexes.last;\n\t\tconst table = affectedTableCells[ 0 ].findAncestor( 'table' )!;\n\n\t\ttableUtils.insertRows( table, { at: insertAbove ? row : row + 1, copyStructureFromAbove: !insertAbove } );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/insertcolumncommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport type TableUtils from '../tableutils';\n\n/**\n * The insert column command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'insertTableColumnLeft'` and\n * `'insertTableColumnRight'` editor commands.\n *\n * To insert a column to the left of the selected cell, execute the following command:\n *\n * ```ts\n * editor.execute( 'insertTableColumnLeft' );\n * ```\n *\n * To insert a column to the right of the selected cell, execute the following command:\n *\n * ```ts\n * editor.execute( 'insertTableColumnRight' );\n * ```\n */\nexport default class InsertColumnCommand extends Command {\n\t/**\n\t * The order of insertion relative to the column in which the caret is located.\n\t */\n\tpublic readonly order: 'left' | 'right';\n\n\t/**\n\t * Creates a new `InsertColumnCommand` instance.\n\t *\n\t * @param editor An editor on which this command will be used.\n\t * @param options.order The order of insertion relative to the column in which the caret is located.\n\t * Possible values: `\"left\"` and `\"right\"`. Default value is \"right\".\n\t */\n\tconstructor( editor: Editor, options: { order?: 'left' | 'right' } = {} ) {\n\t\tsuper( editor );\n\n\t\tthis.order = options.order || 'right';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst selection = this.editor.model.document.selection;\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst isAnyCellSelected = !!tableUtils.getSelectionAffectedTableCells( selection ).length;\n\n\t\tthis.isEnabled = isAnyCellSelected;\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * Depending on the command's {@link #order} value, it inserts a column to the `'left'` or `'right'` of the column\n\t * in which the selection is set.\n\t *\n\t * @fires execute\n\t */\n\tpublic override execute(): void {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\t\tconst tableUtils: TableUtils = editor.plugins.get( 'TableUtils' );\n\t\tconst insertBefore = this.order === 'left';\n\n\t\tconst affectedTableCells = tableUtils.getSelectionAffectedTableCells( selection );\n\t\tconst columnIndexes = tableUtils.getColumnIndexes( affectedTableCells );\n\n\t\tconst column = insertBefore ? columnIndexes.first : columnIndexes.last;\n\t\tconst table = affectedTableCells[ 0 ].findAncestor( 'table' )!;\n\n\t\ttableUtils.insertColumns( table, { columns: 1, at: insertBefore ? column : column + 1 } );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/splitcellcommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport type TableUtils from '../tableutils';\n\n/**\n * The split cell command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'splitTableCellVertically'`\n * and `'splitTableCellHorizontally'` editor commands.\n *\n * You can split any cell vertically or horizontally by executing this command. For example, to split the selected table cell vertically:\n *\n * ```ts\n * editor.execute( 'splitTableCellVertically' );\n * ```\n */\nexport default class SplitCellCommand extends Command {\n\t/**\n\t * The direction that indicates which cell will be split.\n\t */\n\tpublic readonly direction: 'horizontally' | 'vertically';\n\n\t/**\n\t * Creates a new `SplitCellCommand` instance.\n\t *\n\t * @param editor The editor on which this command will be used.\n\t * @param options.direction Indicates whether the command should split cells `'horizontally'` or `'vertically'`.\n\t */\n\tconstructor( editor: Editor, options: { direction?: 'horizontally' | 'vertically' } = {} ) {\n\t\tsuper( editor );\n\n\t\tthis.direction = options.direction || 'horizontally';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( this.editor.model.document.selection );\n\n\t\tthis.isEnabled = selectedCells.length === 1;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst tableCell = tableUtils.getSelectionAffectedTableCells( this.editor.model.document.selection )[ 0 ];\n\t\tconst isHorizontal = this.direction === 'horizontally';\n\n\t\tif ( isHorizontal ) {\n\t\t\ttableUtils.splitCellHorizontally( tableCell, 2 );\n\t\t} else {\n\t\t\ttableUtils.splitCellVertically( tableCell, 2 );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/utils/structure\n */\n\nimport type { Element, Node, Writer } from 'ckeditor5/src/engine';\n\nimport { default as TableWalker, type TableSlot } from '../tablewalker';\nimport { createEmptyTableCell, updateNumericAttribute } from './common';\nimport type TableUtils from '../tableutils';\n\ntype CellAttributes = {\n\trowspan?: number;\n\tcolspan?: number;\n};\n\n/**\n * Returns a cropped table according to given dimensions.\n\n * To return a cropped table that starts at first row and first column and end in third row and column:\n *\n * ```ts\n * const croppedTable = cropTableToDimensions( table, {\n * startRow: 1,\n * endRow: 3,\n * startColumn: 1,\n * endColumn: 3\n * }, writer );\n * ```\n *\n * Calling the code above for the table below:\n *\n * 0 1 2 3 4 0 1 2\n * ┌───┬───┬───┬───┬───┐\n * 0 │ a │ b │ c │ d │ e │\n * ├───┴───┤ ├───┴───┤ ┌───┬───┬───┐\n * 1 │ f │ │ g │ │ │ │ g │ 0\n * ├───┬───┴───┼───┬───┤ will return: ├───┴───┼───┤\n * 2 │ h │ i │ j │ k │ │ i │ j │ 1\n * ├───┤ ├───┤ │ │ ├───┤\n * 3 │ l │ │ m │ │ │ │ m │ 2\n * ├───┼───┬───┤ ├───┤ └───────┴───┘\n * 4 │ n │ o │ p │ │ q │\n * └───┴───┴───┴───┴───┘\n */\nexport function cropTableToDimensions(\n\tsourceTable: Element,\n\tcropDimensions: {\n\t\tstartRow: number;\n\t\tstartColumn: number;\n\t\tendRow: number;\n\t\tendColumn: number;\n\t},\n\twriter: Writer\n): Element {\n\tconst { startRow, startColumn, endRow, endColumn } = cropDimensions;\n\n\t// Create empty table with empty rows equal to crop height.\n\tconst croppedTable = writer.createElement( 'table' );\n\tconst cropHeight = endRow - startRow + 1;\n\n\tfor ( let i = 0; i < cropHeight; i++ ) {\n\t\twriter.insertElement( 'tableRow', croppedTable, 'end' );\n\t}\n\n\tconst tableMap = [ ...new TableWalker( sourceTable, { startRow, endRow, startColumn, endColumn, includeAllSlots: true } ) ];\n\n\t// Iterate over source table slots (including empty - spanned - ones).\n\tfor ( const { row: sourceRow, column: sourceColumn, cell: tableCell, isAnchor, cellAnchorRow, cellAnchorColumn } of tableMap ) {\n\t\t// Row index in cropped table.\n\t\tconst rowInCroppedTable = sourceRow - startRow;\n\t\tconst row = croppedTable.getChild( rowInCroppedTable ) as Element;\n\n\t\t// For empty slots: fill the gap with empty table cell.\n\t\tif ( !isAnchor ) {\n\t\t\t// But fill the gap only if the spanning cell is anchored outside cropped area.\n\t\t\t// In the table from method jsdoc those cells are: \"c\" & \"f\".\n\t\t\tif ( cellAnchorRow < startRow || cellAnchorColumn < startColumn ) {\n\t\t\t\tcreateEmptyTableCell( writer, writer.createPositionAt( row, 'end' ) );\n\t\t\t}\n\t\t}\n\t\t// Otherwise clone the cell with all children and trim if it exceeds cropped area.\n\t\telse {\n\t\t\tconst tableCellCopy = writer.cloneElement( tableCell );\n\n\t\t\twriter.append( tableCellCopy, row );\n\n\t\t\t// Trim table if it exceeds cropped area.\n\t\t\t// In the table from method jsdoc those cells are: \"g\" & \"m\".\n\t\t\ttrimTableCellIfNeeded( tableCellCopy, sourceRow, sourceColumn, endRow, endColumn, writer );\n\t\t}\n\t}\n\n\t// Adjust heading rows & columns in cropped table if crop selection includes headings parts.\n\taddHeadingsToCroppedTable( croppedTable, sourceTable, startRow, startColumn, writer );\n\n\treturn croppedTable;\n}\n\n/**\n * Returns slot info of cells that starts above and overlaps a given row.\n *\n * In a table below, passing `overlapRow = 3`\n *\n * ┌───┬───┬───┬───┬───┐\n * 0 │ a │ b │ c │ d │ e │\n * │ ├───┼───┼───┼───┤\n * 1 │ │ f │ g │ h │ i │\n * ├───┤ ├───┼───┤ │\n * 2 │ j │ │ k │ l │ │\n * │ │ │ ├───┼───┤\n * 3 │ │ │ │ m │ n │ <- overlap row to check\n * ├───┼───┤ │ ├───│\n * 4 │ o │ p │ │ │ q │\n * └───┴───┴───┴───┴───┘\n *\n * will return slot info for cells: \"j\", \"f\", \"k\".\n *\n * @param table The table to check.\n * @param overlapRow The index of the row to check.\n * @param startRow row to start analysis. Use it when it is known that the cells above that row will not overlap. Default value is 0.\n */\nexport function getVerticallyOverlappingCells( table: Element, overlapRow: number, startRow: number = 0 ): Array<TableSlot> {\n\tconst cells: Array<TableSlot> = [];\n\n\tconst tableWalker = new TableWalker( table, { startRow, endRow: overlapRow - 1 } );\n\n\tfor ( const slotInfo of tableWalker ) {\n\t\tconst { row, cellHeight } = slotInfo;\n\t\tconst cellEndRow = row + cellHeight - 1;\n\n\t\tif ( row < overlapRow && overlapRow <= cellEndRow ) {\n\t\t\tcells.push( slotInfo );\n\t\t}\n\t}\n\n\treturn cells;\n}\n\n/**\n * Splits the table cell horizontally.\n *\n * @returns Created table cell, if any were created.\n */\nexport function splitHorizontally( tableCell: Element, splitRow: number, writer: Writer ): Element | null {\n\tconst tableRow = tableCell.parent as Node;\n\tconst table = tableRow.parent as Element;\n\tconst rowIndex = tableRow.index!;\n\n\tconst rowspan = parseInt( tableCell.getAttribute( 'rowspan' ) as string );\n\tconst newRowspan = splitRow - rowIndex;\n\n\tconst newCellAttributes: CellAttributes = {};\n\tconst newCellRowSpan = rowspan - newRowspan;\n\n\tif ( newCellRowSpan > 1 ) {\n\t\tnewCellAttributes.rowspan = newCellRowSpan;\n\t}\n\n\tconst colspan = parseInt( tableCell.getAttribute( 'colspan' ) as string || '1' );\n\n\tif ( colspan > 1 ) {\n\t\tnewCellAttributes.colspan = colspan;\n\t}\n\n\tconst startRow = rowIndex;\n\tconst endRow = startRow + newRowspan;\n\tconst tableMap = [ ...new TableWalker( table, { startRow, endRow, includeAllSlots: true } ) ];\n\n\tlet newCell = null;\n\tlet columnIndex;\n\n\tfor ( const tableSlot of tableMap ) {\n\t\tconst { row, column, cell } = tableSlot;\n\n\t\tif ( cell === tableCell && columnIndex === undefined ) {\n\t\t\tcolumnIndex = column;\n\t\t}\n\n\t\tif ( columnIndex !== undefined && columnIndex === column && row === endRow ) {\n\t\t\tnewCell = createEmptyTableCell( writer, tableSlot.getPositionBefore(), newCellAttributes );\n\t\t}\n\t}\n\n\t// Update the rowspan attribute after updating table.\n\tupdateNumericAttribute( 'rowspan', newRowspan, tableCell, writer );\n\n\treturn newCell;\n}\n\n/**\n * Returns slot info of cells that starts before and overlaps a given column.\n *\n * In a table below, passing `overlapColumn = 3`\n *\n * 0 1 2 3 4\n * ┌───────┬───────┬───┐\n * │ a │ b │ c │\n * │───┬───┴───────┼───┤\n * │ d │ e │ f │\n * ├───┼───┬───────┴───┤\n * │ g │ h │ i │\n * ├───┼───┼───┬───────┤\n * │ j │ k │ l │ m │\n * ├───┼───┴───┼───┬───┤\n * │ n │ o │ p │ q │\n * └───┴───────┴───┴───┘\n * ^\n * Overlap column to check\n *\n * will return slot info for cells: \"b\", \"e\", \"i\".\n *\n * @param table The table to check.\n * @param overlapColumn The index of the column to check.\n */\nexport function getHorizontallyOverlappingCells( table: Element, overlapColumn: number ): Array<TableSlot> {\n\tconst cellsToSplit = [];\n\n\tconst tableWalker = new TableWalker( table );\n\n\tfor ( const slotInfo of tableWalker ) {\n\t\tconst { column, cellWidth } = slotInfo;\n\t\tconst cellEndColumn = column + cellWidth - 1;\n\n\t\tif ( column < overlapColumn && overlapColumn <= cellEndColumn ) {\n\t\t\tcellsToSplit.push( slotInfo );\n\t\t}\n\t}\n\n\treturn cellsToSplit;\n}\n\n/**\n * Splits the table cell vertically.\n *\n * @param columnIndex The table cell column index.\n * @param splitColumn The index of column to split cell on.\n * @returns Created table cell.\n */\nexport function splitVertically( tableCell: Element, columnIndex: number, splitColumn: number, writer: Writer ): Element {\n\tconst colspan = parseInt( tableCell.getAttribute( 'colspan' ) as string );\n\tconst newColspan = splitColumn - columnIndex;\n\n\tconst newCellAttributes: CellAttributes = {};\n\tconst newCellColSpan = colspan - newColspan;\n\n\tif ( newCellColSpan > 1 ) {\n\t\tnewCellAttributes.colspan = newCellColSpan;\n\t}\n\n\tconst rowspan = parseInt( tableCell.getAttribute( 'rowspan' ) as string || '1' );\n\n\tif ( rowspan > 1 ) {\n\t\tnewCellAttributes.rowspan = rowspan;\n\t}\n\n\tconst newCell = createEmptyTableCell( writer, writer.createPositionAfter( tableCell ), newCellAttributes );\n\n\t// Update the colspan attribute after updating table.\n\tupdateNumericAttribute( 'colspan', newColspan, tableCell, writer );\n\n\treturn newCell;\n}\n\n/**\n * Adjusts table cell dimensions to not exceed limit row and column.\n *\n * If table cell width (or height) covers a column (or row) that is after a limit column (or row)\n * this method will trim \"colspan\" (or \"rowspan\") attribute so the table cell will fit in a defined limits.\n */\nexport function trimTableCellIfNeeded(\n\ttableCell: Element,\n\tcellRow: number,\n\tcellColumn: number,\n\tlimitRow: number,\n\tlimitColumn: number,\n\twriter: Writer\n): void {\n\tconst colspan = parseInt( tableCell.getAttribute( 'colspan' ) as string || '1' );\n\tconst rowspan = parseInt( tableCell.getAttribute( 'rowspan' ) as string || '1' );\n\n\tconst endColumn = cellColumn + colspan - 1;\n\n\tif ( endColumn > limitColumn ) {\n\t\tconst trimmedSpan = limitColumn - cellColumn + 1;\n\n\t\tupdateNumericAttribute( 'colspan', trimmedSpan, tableCell, writer, 1 );\n\t}\n\n\tconst endRow = cellRow + rowspan - 1;\n\n\tif ( endRow > limitRow ) {\n\t\tconst trimmedSpan = limitRow - cellRow + 1;\n\n\t\tupdateNumericAttribute( 'rowspan', trimmedSpan, tableCell, writer, 1 );\n\t}\n}\n\n/**\n * Sets proper heading attributes to a cropped table.\n */\nfunction addHeadingsToCroppedTable( croppedTable: Element, sourceTable: Element, startRow: number, startColumn: number, writer: Writer ) {\n\tconst headingRows = parseInt( sourceTable.getAttribute( 'headingRows' ) as string || '0' );\n\n\tif ( headingRows > 0 ) {\n\t\tconst headingRowsInCrop = headingRows - startRow;\n\t\tupdateNumericAttribute( 'headingRows', headingRowsInCrop, croppedTable, writer, 0 );\n\t}\n\n\tconst headingColumns = parseInt( sourceTable.getAttribute( 'headingColumns' ) as string || '0' );\n\n\tif ( headingColumns > 0 ) {\n\t\tconst headingColumnsInCrop = headingColumns - startColumn;\n\t\tupdateNumericAttribute( 'headingColumns', headingColumnsInCrop, croppedTable, writer, 0 );\n\t}\n}\n\n/**\n * Removes columns that have no cells anchored.\n *\n * In table below:\n *\n * +----+----+----+----+----+----+----+\n * | 00 | 01 | 03 | 04 | 06 |\n * +----+----+----+----+ +----+\n * | 10 | 11 | 13 | | 16 |\n * +----+----+----+----+----+----+----+\n * | 20 | 21 | 23 | 24 | 26 |\n * +----+----+----+----+----+----+----+\n * ^--- empty ---^\n *\n * Will remove columns 2 and 5.\n *\n * **Note:** This is a low-level helper method for clearing invalid model state when doing table modifications.\n * To remove a column from a table use {@link module:table/tableutils~TableUtils#removeColumns `TableUtils.removeColumns()`}.\n *\n * @internal\n * @returns True if removed some columns.\n */\nexport function removeEmptyColumns( table: Element, tableUtils: TableUtils ): boolean {\n\tconst width = tableUtils.getColumns( table );\n\tconst columnsMap = new Array( width ).fill( 0 );\n\n\tfor ( const { column } of new TableWalker( table ) ) {\n\t\tcolumnsMap[ column ]++;\n\t}\n\n\tconst emptyColumns = columnsMap.reduce( ( result, cellsCount, column ) => {\n\t\treturn cellsCount ? result : [ ...result, column ];\n\t}, [] );\n\n\tif ( emptyColumns.length > 0 ) {\n\t\t// Remove only last empty column because it will recurrently trigger removing empty rows.\n\t\tconst emptyColumn = emptyColumns[ emptyColumns.length - 1 ];\n\n\t\t// @if CK_DEBUG_TABLE // console.log( `Removing empty column: ${ emptyColumn }.` );\n\t\ttableUtils.removeColumns( table, { at: emptyColumn } );\n\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n\n/**\n * Removes rows that have no cells anchored.\n *\n * In table below:\n *\n * +----+----+----+\n * | 00 | 01 | 02 |\n * +----+----+----+\n * | 10 | 11 | 12 |\n * + + + +\n * | | | | <-- empty\n * +----+----+----+\n * | 30 | 31 | 32 |\n * +----+----+----+\n * | 40 | 42 |\n * + + +\n * | | | <-- empty\n * +----+----+----+\n * | 60 | 61 | 62 |\n * +----+----+----+\n *\n * Will remove rows 2 and 5.\n *\n * **Note:** This is a low-level helper method for clearing invalid model state when doing table modifications.\n * To remove a row from a table use {@link module:table/tableutils~TableUtils#removeRows `TableUtils.removeRows()`}.\n *\n * @internal\n * @returns True if removed some rows.\n */\nexport function removeEmptyRows( table: Element, tableUtils: TableUtils ): boolean {\n\tconst emptyRows = [];\n\tconst tableRowCount = tableUtils.getRows( table );\n\n\tfor ( let rowIndex = 0; rowIndex < tableRowCount; rowIndex++ ) {\n\t\tconst tableRow = table.getChild( rowIndex ) as Element;\n\n\t\tif ( tableRow.isEmpty ) {\n\t\t\temptyRows.push( rowIndex );\n\t\t}\n\t}\n\n\tif ( emptyRows.length > 0 ) {\n\t\t// Remove only last empty row because it will recurrently trigger removing empty columns.\n\t\tconst emptyRow = emptyRows[ emptyRows.length - 1 ];\n\n\t\t// @if CK_DEBUG_TABLE // console.log( `Removing empty row: ${ emptyRow }.` );\n\t\ttableUtils.removeRows( table, { at: emptyRow } );\n\n\t\treturn true;\n\t}\n\n\treturn false;\n}\n\n/**\n * Removes rows and columns that have no cells anchored.\n *\n * In table below:\n *\n * +----+----+----+----+\n * | 00 | 02 |\n * +----+----+ +\n * | 10 | |\n * +----+----+----+----+\n * | 20 | 22 | 23 |\n * + + + +\n * | | | | <-- empty row\n * +----+----+----+----+\n * ^--- empty column\n *\n * Will remove row 3 and column 1.\n *\n * **Note:** This is a low-level helper method for clearing invalid model state when doing table modifications.\n * To remove a rows from a table use {@link module:table/tableutils~TableUtils#removeRows `TableUtils.removeRows()`} and\n * {@link module:table/tableutils~TableUtils#removeColumns `TableUtils.removeColumns()`} to remove a column.\n *\n * @internal\n */\nexport function removeEmptyRowsColumns( table: Element, tableUtils: TableUtils ): void {\n\tconst removedColumns = removeEmptyColumns( table, tableUtils );\n\n\t// If there was some columns removed then cleaning empty rows was already triggered.\n\tif ( !removedColumns ) {\n\t\tremoveEmptyRows( table, tableUtils );\n\t}\n}\n\n/**\n * Returns adjusted last row index if selection covers part of a row with empty slots (spanned by other cells).\n * The `dimensions.lastRow` is equal to last row index but selection might be bigger.\n *\n * This happens *only* on rectangular selection so we analyze a case like this:\n *\n * +---+---+---+---+\n * 0 | a | b | c | d |\n * + + +---+---+\n * 1 | | e | f | g |\n * + +---+ +---+\n * 2 | | h | | i | <- last row, each cell has rowspan = 2,\n * + + + + + so we need to return 3, not 2\n * 3 | | | | |\n * +---+---+---+---+\n *\n * @returns Adjusted last row index.\n */\nexport function adjustLastRowIndex(\n\ttable: Element,\n\tdimensions: {\n\t\tfirstRow: number;\n\t\tfirstColumn: number;\n\t\tlastRow: number;\n\t\tlastColumn: number;\n\t}\n): number {\n\tconst lastRowMap = Array.from( new TableWalker( table, {\n\t\tstartColumn: dimensions.firstColumn,\n\t\tendColumn: dimensions.lastColumn,\n\t\trow: dimensions.lastRow\n\t} ) );\n\n\tconst everyCellHasSingleRowspan = lastRowMap.every( ( { cellHeight } ) => cellHeight === 1 );\n\n\t// It is a \"flat\" row, so the last row index is OK.\n\tif ( everyCellHasSingleRowspan ) {\n\t\treturn dimensions.lastRow;\n\t}\n\n\t// Otherwise get any cell's rowspan and adjust the last row index.\n\tconst rowspanAdjustment = lastRowMap[ 0 ].cellHeight - 1;\n\treturn dimensions.lastRow + rowspanAdjustment;\n}\n\n/**\n * Returns adjusted last column index if selection covers part of a column with empty slots (spanned by other cells).\n * The `dimensions.lastColumn` is equal to last column index but selection might be bigger.\n *\n * This happens *only* on rectangular selection so we analyze a case like this:\n *\n * 0 1 2 3\n * +---+---+---+---+\n * | a |\n * +---+---+---+---+\n * | b | c | d |\n * +---+---+---+---+\n * | e | f |\n * +---+---+---+---+\n * | g | h |\n * +---+---+---+---+\n * ^\n * last column, each cell has colspan = 2, so we need to return 3, not 2\n *\n * @returns Adjusted last column index.\n */\nexport function adjustLastColumnIndex(\n\ttable: Element,\n\tdimensions: {\n\t\tfirstRow: number;\n\t\tfirstColumn: number;\n\t\tlastRow: number;\n\t\tlastColumn: number;\n\t}\n): number {\n\tconst lastColumnMap = Array.from( new TableWalker( table, {\n\t\tstartRow: dimensions.firstRow,\n\t\tendRow: dimensions.lastRow,\n\t\tcolumn: dimensions.lastColumn\n\t} ) );\n\n\tconst everyCellHasSingleColspan = lastColumnMap.every( ( { cellWidth } ) => cellWidth === 1 );\n\n\t// It is a \"flat\" column, so the last column index is OK.\n\tif ( everyCellHasSingleColspan ) {\n\t\treturn dimensions.lastColumn;\n\t}\n\n\t// Otherwise get any cell's colspan and adjust the last column index.\n\tconst colspanAdjustment = lastColumnMap[ 0 ].cellWidth - 1;\n\treturn dimensions.lastColumn + colspanAdjustment;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/mergecellcommand\n */\n\nimport type {\n\tElement,\n\tNode,\n\tWriter\n} from 'ckeditor5/src/engine';\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport TableWalker from '../tablewalker';\nimport { isHeadingColumnCell } from '../utils/common';\nimport { removeEmptyRowsColumns } from '../utils/structure';\nimport type { ArrowKeyCodeDirection } from 'ckeditor5/src/utils';\n\nimport type TableUtils from '../tableutils';\n\n/**\n * The merge cell command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'mergeTableCellRight'`, `'mergeTableCellLeft'`,\n * `'mergeTableCellUp'` and `'mergeTableCellDown'` editor commands.\n *\n * To merge a table cell at the current selection with another cell, execute the command corresponding with the preferred direction.\n *\n * For example, to merge with a cell to the right:\n *\n * ```ts\n * editor.execute( 'mergeTableCellRight' );\n * ```\n *\n * **Note**: If a table cell has a different [`rowspan`](https://www.w3.org/TR/html50/tabular-data.html#attr-tdth-rowspan)\n * (for `'mergeTableCellRight'` and `'mergeTableCellLeft'`) or [`colspan`](https://www.w3.org/TR/html50/tabular-data.html#attr-tdth-colspan)\n * (for `'mergeTableCellUp'` and `'mergeTableCellDown'`), the command will be disabled.\n */\nexport default class MergeCellCommand extends Command {\n\t/**\n\t * The direction that indicates which cell will be merged with the currently selected one.\n\t */\n\tpublic readonly direction: ArrowKeyCodeDirection;\n\n\t/**\n\t * Whether the merge is horizontal (left/right) or vertical (up/down).\n\t */\n\tpublic readonly isHorizontal: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic declare value: Node | undefined;\n\n\t/**\n\t * Creates a new `MergeCellCommand` instance.\n\t *\n\t * @param editor The editor on which this command will be used.\n\t * @param options.direction Indicates which cell to merge with the currently selected one.\n\t * Possible values are: `'left'`, `'right'`, `'up'` and `'down'`.\n\t */\n\tconstructor( editor: Editor, options: { direction: ArrowKeyCodeDirection } ) {\n\t\tsuper( editor );\n\n\t\tthis.direction = options.direction;\n\t\tthis.isHorizontal = this.direction == 'right' || this.direction == 'left';\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst cellToMerge = this._getMergeableCell();\n\n\t\tthis.value = cellToMerge;\n\t\tthis.isEnabled = !!cellToMerge;\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * Depending on the command's {@link #direction} value, it will merge the cell that is to the `'left'`, `'right'`, `'up'` or `'down'`.\n\t *\n\t * @fires execute\n\t */\n\tpublic override execute(): void {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst tableCell = tableUtils.getTableCellsContainingSelection( doc.selection )[ 0 ];\n\n\t\tconst cellToMerge = this.value!;\n\t\tconst direction = this.direction;\n\n\t\tmodel.change( writer => {\n\t\t\tconst isMergeNext = direction == 'right' || direction == 'down';\n\n\t\t\t// The merge mechanism is always the same so sort cells to be merged.\n\t\t\tconst cellToExpand = ( isMergeNext ? tableCell : cellToMerge ) as Element;\n\t\t\tconst cellToRemove = ( isMergeNext ? cellToMerge : tableCell ) as Element;\n\n\t\t\t// Cache the parent of cell to remove for later check.\n\t\t\tconst removedTableCellRow = cellToRemove.parent as Element;\n\n\t\t\tmergeTableCells( cellToRemove, cellToExpand, writer );\n\n\t\t\tconst spanAttribute = this.isHorizontal ? 'colspan' : 'rowspan';\n\t\t\tconst cellSpan = parseInt( tableCell.getAttribute( spanAttribute ) as string || '1' );\n\t\t\tconst cellToMergeSpan = parseInt( cellToMerge.getAttribute( spanAttribute ) as string || '1' );\n\n\t\t\t// Update table cell span attribute and merge set selection on merged contents.\n\t\t\twriter.setAttribute( spanAttribute, cellSpan + cellToMergeSpan, cellToExpand );\n\t\t\twriter.setSelection( writer.createRangeIn( cellToExpand ) );\n\n\t\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\t\tconst table = removedTableCellRow.findAncestor( 'table' )!;\n\n\t\t\t// Remove empty rows and columns after merging.\n\t\t\tremoveEmptyRowsColumns( table, tableUtils );\n\t\t} );\n\t}\n\n\t/**\n\t * Returns a cell that can be merged with the current cell depending on the command's direction.\n\t */\n\tprivate _getMergeableCell(): Node | undefined {\n\t\tconst model = this.editor.model;\n\t\tconst doc = model.document;\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst tableCell = tableUtils.getTableCellsContainingSelection( doc.selection )[ 0 ];\n\n\t\tif ( !tableCell ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// First get the cell on proper direction.\n\t\tconst cellToMerge = this.isHorizontal ?\n\t\t\tgetHorizontalCell( tableCell, this.direction, tableUtils ) :\n\t\t\tgetVerticalCell( tableCell, this.direction, tableUtils );\n\n\t\tif ( !cellToMerge ) {\n\t\t\treturn;\n\t\t}\n\n\t\t// If found check if the span perpendicular to merge direction is equal on both cells.\n\t\tconst spanAttribute = this.isHorizontal ? 'rowspan' : 'colspan';\n\t\tconst span = parseInt( tableCell.getAttribute( spanAttribute ) as string || '1' );\n\n\t\tconst cellToMergeSpan = parseInt( cellToMerge.getAttribute( spanAttribute ) as string || '1' );\n\n\t\tif ( cellToMergeSpan === span ) {\n\t\t\treturn cellToMerge;\n\t\t}\n\t}\n}\n\n/**\n * Returns the cell that can be merged horizontally.\n */\nfunction getHorizontalCell( tableCell: Element, direction: ArrowKeyCodeDirection, tableUtils: TableUtils ) {\n\tconst tableRow = tableCell.parent!;\n\tconst table = tableRow.parent as Element;\n\tconst horizontalCell = direction == 'right' ? tableCell.nextSibling : tableCell.previousSibling;\n\tconst hasHeadingColumns = ( table.getAttribute( 'headingColumns' ) || 0 ) > 0;\n\n\tif ( !horizontalCell ) {\n\t\treturn;\n\t}\n\n\t// Sort cells:\n\tconst cellOnLeft = ( direction == 'right' ? tableCell : horizontalCell ) as Element;\n\tconst cellOnRight = ( direction == 'right' ? horizontalCell : tableCell ) as Element;\n\n\t// Get their column indexes:\n\tconst { column: leftCellColumn } = tableUtils.getCellLocation( cellOnLeft );\n\tconst { column: rightCellColumn } = tableUtils.getCellLocation( cellOnRight );\n\n\tconst leftCellSpan = parseInt( cellOnLeft.getAttribute( 'colspan' ) as string || '1' );\n\n\tconst isCellOnLeftInHeadingColumn = isHeadingColumnCell( tableUtils, cellOnLeft );\n\tconst isCellOnRightInHeadingColumn = isHeadingColumnCell( tableUtils, cellOnRight );\n\n\t// We cannot merge heading columns cells with regular cells.\n\tif ( hasHeadingColumns && isCellOnLeftInHeadingColumn != isCellOnRightInHeadingColumn ) {\n\t\treturn;\n\t}\n\n\t// The cell on the right must have index that is distant to the cell on the left by the left cell's width (colspan).\n\tconst cellsAreTouching = leftCellColumn + leftCellSpan === rightCellColumn;\n\n\t// If the right cell's column index is different it means that there are rowspanned cells between them.\n\treturn cellsAreTouching ? horizontalCell : undefined;\n}\n\n/**\n * Returns the cell that can be merged vertically.\n */\nfunction getVerticalCell( tableCell: Element, direction: ArrowKeyCodeDirection, tableUtils: TableUtils ): Node | null {\n\tconst tableRow = tableCell.parent as Element;\n\tconst table = tableRow.parent as Element;\n\n\tconst rowIndex = table.getChildIndex( tableRow )!;\n\n\t// Don't search for mergeable cell if direction points out of the table.\n\tif ( ( direction == 'down' && rowIndex === tableUtils.getRows( table ) - 1 ) || ( direction == 'up' && rowIndex === 0 ) ) {\n\t\treturn null;\n\t}\n\n\tconst rowspan = parseInt( tableCell.getAttribute( 'rowspan' ) as string || '1' );\n\tconst headingRows = table.getAttribute( 'headingRows' ) || 0;\n\n\tconst isMergeWithBodyCell = direction == 'down' && ( rowIndex + rowspan ) === headingRows;\n\tconst isMergeWithHeadCell = direction == 'up' && rowIndex === headingRows;\n\n\t// Don't search for mergeable cell if direction points out of the current table section.\n\tif ( headingRows && ( isMergeWithBodyCell || isMergeWithHeadCell ) ) {\n\t\treturn null;\n\t}\n\n\tconst currentCellRowSpan = parseInt( tableCell.getAttribute( 'rowspan' ) as string || '1' );\n\tconst rowOfCellToMerge = direction == 'down' ? rowIndex + currentCellRowSpan : rowIndex;\n\n\tconst tableMap = [ ...new TableWalker( table, { endRow: rowOfCellToMerge } ) ];\n\n\tconst currentCellData = tableMap.find( value => value.cell === tableCell )!;\n\tconst mergeColumn = currentCellData.column;\n\n\tconst cellToMergeData = tableMap.find( ( { row, cellHeight, column } ) => {\n\t\tif ( column !== mergeColumn ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tif ( direction == 'down' ) {\n\t\t\t// If merging a cell below the mergeRow is already calculated.\n\t\t\treturn row === rowOfCellToMerge;\n\t\t} else {\n\t\t\t// If merging a cell above calculate if it spans to mergeRow.\n\t\t\treturn rowOfCellToMerge === row + cellHeight;\n\t\t}\n\t} );\n\n\treturn cellToMergeData && cellToMergeData.cell ? cellToMergeData.cell : null;\n}\n\n/**\n * Merges two table cells. It will ensure that after merging cells with an empty paragraph, the resulting table cell will only have one\n * paragraph. If one of the merged table cells is empty, the merged table cell will have the contents of the non-empty table cell.\n * If both are empty, the merged table cell will have only one empty paragraph.\n */\nfunction mergeTableCells( cellToRemove: Element, cellToExpand: Element, writer: Writer ) {\n\tif ( !isEmpty( cellToRemove ) ) {\n\t\tif ( isEmpty( cellToExpand ) ) {\n\t\t\twriter.remove( writer.createRangeIn( cellToExpand ) );\n\t\t}\n\n\t\twriter.move( writer.createRangeIn( cellToRemove ), writer.createPositionAt( cellToExpand, 'end' ) );\n\t}\n\n\t// Remove merged table cell.\n\twriter.remove( cellToRemove );\n}\n\n/**\n * Checks if the passed table cell contains an empty paragraph.\n */\nfunction isEmpty( tableCell: Element ): boolean {\n\tconst firstTableChild = tableCell.getChild( 0 ) as Element;\n\n\treturn tableCell.childCount == 1 && firstTableChild.is( 'element', 'paragraph' ) && firstTableChild.isEmpty;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/removerowcommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport type { Element, Node } from 'ckeditor5/src/engine';\nimport type TableUtils from '../tableutils';\n\n/**\n * The remove row command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'removeTableRow'` editor command.\n *\n * To remove the row containing the selected cell, execute the command:\n *\n * ```ts\n * editor.execute( 'removeTableRow' );\n * ```\n */\nexport default class RemoveRowCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( this.editor.model.document.selection );\n\t\tconst firstCell = selectedCells[ 0 ];\n\n\t\tif ( firstCell ) {\n\t\t\tconst table = firstCell.findAncestor( 'table' )!;\n\t\t\tconst tableRowCount = tableUtils.getRows( table );\n\t\t\tconst lastRowIndex = tableRowCount - 1;\n\n\t\t\tconst selectedRowIndexes = tableUtils.getRowIndexes( selectedCells );\n\n\t\t\tconst areAllRowsSelected = selectedRowIndexes.first === 0 && selectedRowIndexes.last === lastRowIndex;\n\n\t\t\t// Disallow selecting whole table -> delete whole table should be used instead.\n\t\t\tthis.isEnabled = !areAllRowsSelected;\n\t\t} else {\n\t\t\tthis.isEnabled = false;\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tconst model = this.editor.model;\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\n\t\tconst referenceCells = tableUtils.getSelectionAffectedTableCells( model.document.selection );\n\t\tconst removedRowIndexes = tableUtils.getRowIndexes( referenceCells );\n\n\t\tconst firstCell = referenceCells[ 0 ];\n\t\tconst table = firstCell.findAncestor( 'table' )!;\n\n\t\tconst columnIndexToFocus = tableUtils.getCellLocation( firstCell ).column;\n\n\t\tmodel.change( writer => {\n\t\t\tconst rowsToRemove = removedRowIndexes.last - removedRowIndexes.first + 1;\n\n\t\t\ttableUtils.removeRows( table, {\n\t\t\t\tat: removedRowIndexes.first,\n\t\t\t\trows: rowsToRemove\n\t\t\t} );\n\n\t\t\tconst cellToFocus = getCellToFocus( table, removedRowIndexes.first, columnIndexToFocus, tableUtils.getRows( table ) );\n\n\t\t\twriter.setSelection( writer.createPositionAt( cellToFocus, 0 ) );\n\t\t} );\n\t}\n}\n\n/**\n * Returns a cell that should be focused before removing the row, belonging to the same column as the currently focused cell.\n * - If the row was not the last one, the cell to focus will be in the row that followed it (before removal).\n * - If the row was the last one, the cell to focus will be in the row that preceded it (before removal).\n */\nfunction getCellToFocus( table: Element, removedRowIndex: number, columnToFocus: number, tableRowCount: number ): Node {\n\t// Don't go beyond last row's index.\n\tconst row = table.getChild( Math.min( removedRowIndex, tableRowCount - 1 ) ) as Element;\n\n\t// Default to first table cell.\n\tlet cellToFocus = row.getChild( 0 )!;\n\tlet column = 0;\n\n\tfor ( const tableCell of row.getChildren() ) {\n\t\tif ( column > columnToFocus ) {\n\t\t\treturn cellToFocus;\n\t\t}\n\n\t\tcellToFocus = tableCell;\n\t\tcolumn += parseInt( tableCell.getAttribute( 'colspan' ) as string || '1' );\n\t}\n\n\treturn cellToFocus;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/removecolumncommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport type {\n\tDocumentSelection,\n\tElement,\n\tSelection\n} from 'ckeditor5/src/engine';\n\nimport TableWalker, { type TableSlot } from '../tablewalker';\nimport type TableUtils from '../tableutils';\n\n/**\n * The remove column command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'removeTableColumn'` editor command.\n *\n * To remove the column containing the selected cell, execute the command:\n *\n * ```ts\n * editor.execute( 'removeTableColumn' );\n * ```\n */\nexport default class RemoveColumnCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( this.editor.model.document.selection );\n\t\tconst firstCell = selectedCells[ 0 ];\n\n\t\tif ( firstCell ) {\n\t\t\tconst table = firstCell.findAncestor( 'table' )!;\n\t\t\tconst tableColumnCount = tableUtils.getColumns( table );\n\n\t\t\tconst { first, last } = tableUtils.getColumnIndexes( selectedCells );\n\n\t\t\tthis.isEnabled = last - first < ( tableColumnCount - 1 );\n\t\t} else {\n\t\t\tthis.isEnabled = false;\n\t\t}\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst [ firstCell, lastCell ] = getBoundaryCells( this.editor.model.document.selection, tableUtils );\n\t\tconst table = firstCell.parent!.parent as Element;\n\n\t\t// Cache the table before removing or updating colspans.\n\t\tconst tableMap = [ ...new TableWalker( table ) ];\n\n\t\t// Store column indexes of removed columns.\n\t\tconst removedColumnIndexes = {\n\t\t\tfirst: tableMap.find( value => value.cell === firstCell )!.column,\n\t\t\tlast: tableMap.find( value => value.cell === lastCell )!.column\n\t\t};\n\n\t\tconst cellToFocus = getCellToFocus( tableMap, firstCell, lastCell, removedColumnIndexes )!;\n\n\t\tthis.editor.model.change( writer => {\n\t\t\tconst columnsToRemove = removedColumnIndexes.last - removedColumnIndexes.first + 1;\n\n\t\t\ttableUtils.removeColumns( table, {\n\t\t\t\tat: removedColumnIndexes.first,\n\t\t\t\tcolumns: columnsToRemove\n\t\t\t} );\n\n\t\t\twriter.setSelection( writer.createPositionAt( cellToFocus, 0 ) );\n\t\t} );\n\t}\n}\n\n/**\n * Returns a proper table cell to focus after removing a column.\n * - selection is on last table cell it will return previous cell.\n */\nfunction getCellToFocus(\n\ttableMap: Array<TableSlot>,\n\tfirstCell: Element,\n\tlastCell: Element,\n\tremovedColumnIndexes: {\n\t\tfirst: number;\n\t\tlast: number;\n\t}\n) {\n\tconst colspan = parseInt( lastCell.getAttribute( 'colspan' ) as string || '1' );\n\n\t// If the table cell is spanned over 2+ columns - it will be truncated so the selection should\n\t// stay in that cell.\n\tif ( colspan > 1 ) {\n\t\treturn lastCell;\n\t}\n\t// Normally, look for the cell in the same row that precedes the first cell to put selection there (\"column on the left\").\n\t// If the deleted column is the first column of the table, there will be no predecessor: use the cell\n\t// from the column that follows then (also in the same row).\n\telse if ( firstCell.previousSibling || lastCell.nextSibling ) {\n\t\treturn lastCell.nextSibling || firstCell.previousSibling;\n\t}\n\t// It can happen that table cells have no siblings in a row, for instance, when there are row spans\n\t// in the table (in the previous row). Then just look for the closest cell that is in a column\n\t// that will not be removed to put the selection there.\n\telse {\n\t\t// Look for any cell in a column that precedes the first removed column.\n\t\tif ( removedColumnIndexes.first ) {\n\t\t\treturn tableMap.reverse().find( ( { column } ) => {\n\t\t\t\treturn column < removedColumnIndexes.first;\n\t\t\t} )!.cell;\n\t\t}\n\t\t// If the first removed column is the first column of the table, then\n\t\t// look for any cell that is in a column that follows the last removed column.\n\t\telse {\n\t\t\treturn tableMap.reverse().find( ( { column } ) => {\n\t\t\t\treturn column > removedColumnIndexes.last;\n\t\t\t} )!.cell;\n\t\t}\n\t}\n}\n\n/**\n * Returns helper object returning the first and the last cell contained in given selection, based on DOM order.\n */\nfunction getBoundaryCells( selection: Selection | DocumentSelection, tableUtils: TableUtils ) {\n\tconst referenceCells = tableUtils.getSelectionAffectedTableCells( selection );\n\tconst firstCell = referenceCells[ 0 ];\n\tconst lastCell = referenceCells.pop()!;\n\n\tconst returnValue = [ firstCell, lastCell ];\n\n\treturn firstCell.isBefore( lastCell ) ? returnValue : returnValue.reverse();\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/setheaderrowcommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport type { Element } from 'ckeditor5/src/engine';\nimport type TableUtils from '../tableutils';\n\nimport { updateNumericAttribute } from '../utils/common';\nimport { getVerticallyOverlappingCells, splitHorizontally } from '../utils/structure';\n\n/**\n * The header row command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'setTableColumnHeader'` editor command.\n *\n * You can make the row containing the selected cell a [header](https://www.w3.org/TR/html50/tabular-data.html#the-th-element) by executing:\n *\n * ```ts\n * editor.execute( 'setTableRowHeader' );\n * ```\n *\n * **Note:** All preceding rows will also become headers. If the current row is already a header, executing this command\n * will make it a regular row back again (including the following rows).\n */\nexport default class SetHeaderRowCommand extends Command {\n\t/**\n\t * Flag indicating whether the command is active. The command is active when the\n\t * {@link module:engine/model/selection~Selection} is in a header row.\n\t *\n\t * @observable\n\t */\n\tpublic declare value: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst model = this.editor.model;\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( model.document.selection );\n\t\tconst isInTable = selectedCells.length > 0;\n\n\t\tthis.isEnabled = isInTable;\n\t\tthis.value = isInTable && selectedCells.every( cell => this._isInHeading( cell, cell.parent!.parent as Element ) );\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * When the selection is in a non-header row, the command will set the `headingRows` table attribute to cover that row.\n\t *\n\t * When the selection is already in a header row, it will set `headingRows` so the heading section will end before that row.\n\t *\n\t * @fires execute\n\t * @param options.forceValue If set, the command will set (`true`) or unset (`false`) the header rows according to\n\t * the `forceValue` parameter instead of the current model state.\n\t */\n\tpublic override execute( options: { forceValue?: boolean } = {} ): void {\n\t\tif ( options.forceValue === this.value ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst model = this.editor.model;\n\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( model.document.selection );\n\t\tconst table = selectedCells[ 0 ].findAncestor( 'table' )!;\n\n\t\tconst { first, last } = tableUtils.getRowIndexes( selectedCells );\n\t\tconst headingRowsToSet = this.value ? first : last + 1;\n\t\tconst currentHeadingRows = table.getAttribute( 'headingRows' ) as number || 0;\n\n\t\tmodel.change( writer => {\n\t\t\tif ( headingRowsToSet ) {\n\t\t\t\t// Changing heading rows requires to check if any of a heading cell is overlapping vertically the table head.\n\t\t\t\t// Any table cell that has a rowspan attribute > 1 will not exceed the table head so we need to fix it in rows below.\n\t\t\t\tconst startRow = headingRowsToSet > currentHeadingRows ? currentHeadingRows : 0;\n\t\t\t\tconst overlappingCells = getVerticallyOverlappingCells( table, headingRowsToSet, startRow );\n\n\t\t\t\tfor ( const { cell } of overlappingCells ) {\n\t\t\t\t\tsplitHorizontally( cell, headingRowsToSet, writer );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tupdateNumericAttribute( 'headingRows', headingRowsToSet, table, writer, 0 );\n\t\t} );\n\t}\n\n\t/**\n\t * Checks if a table cell is in the heading section.\n\t */\n\tprivate _isInHeading( tableCell: Element, table: Element ): boolean {\n\t\tconst headingRows = parseInt( table.getAttribute( 'headingRows' ) as string || '0' );\n\n\t\treturn !!headingRows && ( tableCell.parent as Element ).index! < headingRows;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/setheadercolumncommand\n */\n\nimport { Command } from 'ckeditor5/src/core';\nimport type TableUtils from '../tableutils';\n\nimport {\n\tisHeadingColumnCell,\n\tupdateNumericAttribute\n} from '../utils/common';\nimport { getHorizontallyOverlappingCells, splitVertically } from '../utils/structure';\n\n/**\n * The header column command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'setTableColumnHeader'` editor command.\n *\n * You can make the column containing the selected cell a [header](https://www.w3.org/TR/html50/tabular-data.html#the-th-element)\n * by executing:\n *\n * ```ts\n * editor.execute( 'setTableColumnHeader' );\n * ```\n *\n * **Note:** All preceding columns will also become headers. If the current column is already a header, executing this command\n * will make it a regular column back again (including the following columns).\n */\nexport default class SetHeaderColumnCommand extends Command {\n\t/**\n\t * Flag indicating whether the command is active. The command is active when the\n\t * {@link module:engine/model/selection~Selection} is in a header column.\n\t *\n\t * @observable\n\t */\n\tpublic declare value: boolean;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst model = this.editor.model;\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( model.document.selection );\n\t\tconst isInTable = selectedCells.length > 0;\n\n\t\tthis.isEnabled = isInTable;\n\t\tthis.value = isInTable && selectedCells.every( cell => isHeadingColumnCell( tableUtils, cell ) );\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * When the selection is in a non-header column, the command will set the `headingColumns` table attribute to cover that column.\n\t *\n\t * When the selection is already in a header column, it will set `headingColumns` so the heading section will end before that column.\n\t *\n\t * @fires execute\n\t * @param options.forceValue If set, the command will set (`true`) or unset (`false`) the header columns according to\n\t * the `forceValue` parameter instead of the current model state.\n\t */\n\tpublic override execute( options: { forceValue?: boolean } = {} ): void {\n\t\tif ( options.forceValue === this.value ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst model = this.editor.model;\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( model.document.selection );\n\t\tconst table = selectedCells[ 0 ].findAncestor( 'table' )!;\n\n\t\tconst { first, last } = tableUtils.getColumnIndexes( selectedCells );\n\t\tconst headingColumnsToSet = this.value ? first : last + 1;\n\n\t\tmodel.change( writer => {\n\t\t\tif ( headingColumnsToSet ) {\n\t\t\t\t// Changing heading columns requires to check if any of a heading cell is overlapping horizontally the table head.\n\t\t\t\t// Any table cell that has a colspan attribute > 1 will not exceed the table head so we need to fix it in columns before.\n\t\t\t\tconst overlappingCells = getHorizontallyOverlappingCells( table, headingColumnsToSet );\n\n\t\t\t\tfor ( const { cell, column } of overlappingCells ) {\n\t\t\t\t\tsplitVertically( cell, column, headingColumnsToSet, writer );\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tupdateNumericAttribute( 'headingColumns', headingColumnsToSet, table, writer, 0 );\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tableutils\n */\n\nimport { CKEditorError } from 'ckeditor5/src/utils';\nimport { Plugin } from 'ckeditor5/src/core';\nimport type {\n\tDocumentSelection,\n\tElement,\n\tNode,\n\tPosition,\n\tRange,\n\tSelection,\n\tWriter\n} from 'ckeditor5/src/engine';\n\nimport TableWalker, { type TableWalkerOptions } from './tablewalker';\nimport { createEmptyTableCell, updateNumericAttribute } from './utils/common';\nimport { removeEmptyColumns, removeEmptyRows } from './utils/structure';\n\ntype Cell = { cell: Element; rowspan: number };\ntype CellsToMove = Map<number, Cell>;\ntype CellsToTrim = Array<Cell>;\n\ntype IndexesObject = { first: number; last: number };\n\n/**\n * The table utilities plugin.\n */\nexport default class TableUtils extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TableUtils' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tthis.decorate( 'insertColumns' );\n\t\tthis.decorate( 'insertRows' );\n\t}\n\n\t/**\n\t * Returns the table cell location as an object with table row and table column indexes.\n\t *\n\t * For instance, in the table below:\n\t *\n\t * 0 1 2 3\n\t * +---+---+---+---+\n\t * 0 | a | b | c |\n\t * + + +---+\n\t * 1 | | | d |\n\t * +---+---+ +---+\n\t * 2 | e | | f |\n\t * +---+---+---+---+\n\t *\n\t * the method will return:\n\t *\n\t * ```ts\n\t * const cellA = table.getNodeByPath( [ 0, 0 ] );\n\t * editor.plugins.get( 'TableUtils' ).getCellLocation( cellA );\n\t * // will return { row: 0, column: 0 }\n\t *\n\t * const cellD = table.getNodeByPath( [ 1, 0 ] );\n\t * editor.plugins.get( 'TableUtils' ).getCellLocation( cellD );\n\t * // will return { row: 1, column: 3 }\n\t * ```\n\t *\n\t * @returns Returns a `{row, column}` object.\n\t */\n\tpublic getCellLocation( tableCell: Element ): { row: number; column: number } {\n\t\tconst tableRow = tableCell.parent!;\n\t\tconst table = tableRow.parent as Element;\n\n\t\tconst rowIndex = table.getChildIndex( tableRow as Node );\n\n\t\tconst tableWalker = new TableWalker( table, { row: rowIndex } );\n\n\t\tfor ( const { cell, row, column } of tableWalker ) {\n\t\t\tif ( cell === tableCell ) {\n\t\t\t\treturn { row, column };\n\t\t\t}\n\t\t}\n\n\t\t// Should be unreachable code.\n\t\t/* istanbul ignore next -- @preserve */\n\t\treturn undefined as any;\n\t}\n\n\t/**\n\t * Creates an empty table with a proper structure. The table needs to be inserted into the model,\n\t * for example, by using the {@link module:engine/model/model~Model#insertContent} function.\n\t *\n\t * ```ts\n\t * model.change( ( writer ) => {\n\t * // Create a table of 2 rows and 7 columns:\n\t * const table = tableUtils.createTable( writer, { rows: 2, columns: 7 } );\n\t *\n\t * // Insert a table to the model at the best position taking the current selection:\n\t * model.insertContent( table );\n\t * }\n\t * ```\n\t *\n\t * @param writer The model writer.\n\t * @param options.rows The number of rows to create. Default value is 2.\n\t * @param options.columns The number of columns to create. Default value is 2.\n\t * @param options.headingRows The number of heading rows. Default value is 0.\n\t * @param options.headingColumns The number of heading columns. Default value is 0.\n\t * @returns The created table element.\n\t */\n\tpublic createTable(\n\t\twriter: Writer,\n\t\toptions: {\n\t\t\trows?: number;\n\t\t\tcolumns?: number;\n\t\t\theadingRows?: number;\n\t\t\theadingColumns?: number;\n\t\t}\n\t): Element {\n\t\tconst table = writer.createElement( 'table' );\n\n\t\tconst rows = options.rows || 2;\n\t\tconst columns = options.columns || 2;\n\n\t\tcreateEmptyRows( writer, table, 0, rows, columns );\n\n\t\tif ( options.headingRows ) {\n\t\t\tupdateNumericAttribute( 'headingRows', Math.min( options.headingRows, rows ), table, writer, 0 );\n\t\t}\n\n\t\tif ( options.headingColumns ) {\n\t\t\tupdateNumericAttribute( 'headingColumns', Math.min( options.headingColumns, columns ), table, writer, 0 );\n\t\t}\n\n\t\treturn table;\n\t}\n\n\t/**\n\t * Inserts rows into a table.\n\t *\n\t * ```ts\n\t * editor.plugins.get( 'TableUtils' ).insertRows( table, { at: 1, rows: 2 } );\n\t * ```\n\t *\n\t * Assuming the table on the left, the above code will transform it to the table on the right:\n\t *\n\t * row index\n\t * 0 +---+---+---+ `at` = 1, +---+---+---+ 0\n\t * | a | b | c | `rows` = 2, | a | b | c |\n\t * 1 + +---+---+ <-- insert here + +---+---+ 1\n\t * | | d | e | | | | |\n\t * 2 + +---+---+ will give: + +---+---+ 2\n\t * | | f | g | | | | |\n\t * 3 +---+---+---+ + +---+---+ 3\n\t * | | d | e |\n\t * + +---+---+ 4\n\t * + + f | g |\n\t * +---+---+---+ 5\n\t *\n\t * @param table The table model element where the rows will be inserted.\n\t * @param options.at The row index at which the rows will be inserted. Default value is 0.\n\t * @param options.rows The number of rows to insert. Default value is 1.\n\t * @param options.copyStructureFromAbove The flag for copying row structure. Note that\n\t * the row structure will not be copied if this option is not provided.\n\t */\n\tpublic insertRows( table: Element, options: { at?: number; rows?: number; copyStructureFromAbove?: boolean } = {} ): void {\n\t\tconst model = this.editor.model;\n\n\t\tconst insertAt = options.at || 0;\n\t\tconst rowsToInsert = options.rows || 1;\n\t\tconst isCopyStructure = options.copyStructureFromAbove !== undefined;\n\t\tconst copyStructureFrom = options.copyStructureFromAbove ? insertAt - 1 : insertAt;\n\n\t\tconst rows = this.getRows( table );\n\t\tconst columns = this.getColumns( table );\n\n\t\tif ( insertAt > rows ) {\n\t\t\t/**\n\t\t\t * The `options.at` points at a row position that does not exist.\n\t\t\t *\n\t\t\t * @error tableutils-insertrows-insert-out-of-range\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'tableutils-insertrows-insert-out-of-range',\n\t\t\t\tthis,\n\t\t\t\t{ options }\n\t\t\t);\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\tconst headingRows = table.getAttribute( 'headingRows' ) as number || 0;\n\n\t\t\t// Inserting rows inside heading section requires to update `headingRows` attribute as the heading section will grow.\n\t\t\tif ( headingRows > insertAt ) {\n\t\t\t\tupdateNumericAttribute( 'headingRows', headingRows + rowsToInsert, table, writer, 0 );\n\t\t\t}\n\n\t\t\t// Inserting at the end or at the beginning of a table doesn't require to calculate anything special.\n\t\t\tif ( !isCopyStructure && ( insertAt === 0 || insertAt === rows ) ) {\n\t\t\t\tcreateEmptyRows( writer, table, insertAt, rowsToInsert, columns );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Iterate over all the rows above the inserted rows in order to check for the row-spanned cells.\n\t\t\tconst walkerEndRow = isCopyStructure ? Math.max( insertAt, copyStructureFrom ) : insertAt;\n\t\t\tconst tableIterator = new TableWalker( table, { endRow: walkerEndRow } );\n\n\t\t\t// Store spans of the reference row to reproduce it's structure. This array is column number indexed.\n\t\t\tconst rowColSpansMap = new Array( columns ).fill( 1 );\n\n\t\t\tfor ( const { row, column, cellHeight, cellWidth, cell } of tableIterator ) {\n\t\t\t\tconst lastCellRow = row + cellHeight - 1;\n\n\t\t\t\tconst isOverlappingInsertedRow = row < insertAt && insertAt <= lastCellRow;\n\t\t\t\tconst isReferenceRow = row <= copyStructureFrom && copyStructureFrom <= lastCellRow;\n\n\t\t\t\t// If the cell is row-spanned and overlaps the inserted row, then reserve space for it in the row map.\n\t\t\t\tif ( isOverlappingInsertedRow ) {\n\t\t\t\t\t// This cell overlaps the inserted rows so we need to expand it further.\n\t\t\t\t\twriter.setAttribute( 'rowspan', cellHeight + rowsToInsert, cell );\n\n\t\t\t\t\t// Mark this cell with negative number to indicate how many cells should be skipped when adding the new cells.\n\t\t\t\t\trowColSpansMap[ column ] = -cellWidth;\n\t\t\t\t}\n\t\t\t\t// Store the colspan from reference row.\n\t\t\t\telse if ( isCopyStructure && isReferenceRow ) {\n\t\t\t\t\trowColSpansMap[ column ] = cellWidth;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfor ( let rowIndex = 0; rowIndex < rowsToInsert; rowIndex++ ) {\n\t\t\t\tconst tableRow = writer.createElement( 'tableRow' );\n\n\t\t\t\twriter.insert( tableRow, table, insertAt );\n\n\t\t\t\tfor ( let cellIndex = 0; cellIndex < rowColSpansMap.length; cellIndex++ ) {\n\t\t\t\t\tconst colspan = rowColSpansMap[ cellIndex ];\n\t\t\t\t\tconst insertPosition = writer.createPositionAt( tableRow, 'end' );\n\n\t\t\t\t\t// Insert the empty cell only if this slot is not row-spanned from any other cell.\n\t\t\t\t\tif ( colspan > 0 ) {\n\t\t\t\t\t\tcreateEmptyTableCell( writer, insertPosition, colspan > 1 ? { colspan } : undefined );\n\t\t\t\t\t}\n\n\t\t\t\t\t// Skip the col-spanned slots, there won't be any cells.\n\t\t\t\t\tcellIndex += Math.abs( colspan ) - 1;\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Inserts columns into a table.\n\t *\n\t * ```ts\n\t * editor.plugins.get( 'TableUtils' ).insertColumns( table, { at: 1, columns: 2 } );\n\t * ```\n\t *\n\t * Assuming the table on the left, the above code will transform it to the table on the right:\n\t *\n\t * 0 1 2 3 0 1 2 3 4 5\n\t * +---+---+---+ +---+---+---+---+---+\n\t * | a | b | | a | b |\n\t * + +---+ + +---+\n\t * | | c | | | c |\n\t * +---+---+---+ will give: +---+---+---+---+---+\n\t * | d | e | f | | d | | | e | f |\n\t * +---+ +---+ +---+---+---+ +---+\n\t * | g | | h | | g | | | | h |\n\t * +---+---+---+ +---+---+---+---+---+\n\t * | i | | i |\n\t * +---+---+---+ +---+---+---+---+---+\n\t * ^---- insert here, `at` = 1, `columns` = 2\n\t *\n\t * @param table The table model element where the columns will be inserted.\n\t * @param options.at The column index at which the columns will be inserted. Default value is 0.\n\t * @param options.columns The number of columns to insert. Default value is 1.\n\t */\n\tpublic insertColumns( table: Element, options: { at?: number; columns?: number } = {} ): void {\n\t\tconst model = this.editor.model;\n\n\t\tconst insertAt = options.at || 0;\n\t\tconst columnsToInsert = options.columns || 1;\n\n\t\tmodel.change( writer => {\n\t\t\tconst headingColumns = table.getAttribute( 'headingColumns' ) as number;\n\n\t\t\t// Inserting columns inside heading section requires to update `headingColumns` attribute as the heading section will grow.\n\t\t\tif ( insertAt < headingColumns ) {\n\t\t\t\twriter.setAttribute( 'headingColumns', headingColumns + columnsToInsert, table );\n\t\t\t}\n\n\t\t\tconst tableColumns = this.getColumns( table );\n\n\t\t\t// Inserting at the end and at the beginning of a table doesn't require to calculate anything special.\n\t\t\tif ( insertAt === 0 || tableColumns === insertAt ) {\n\t\t\t\tfor ( const tableRow of table.getChildren() ) {\n\t\t\t\t\t// Ignore non-row elements inside the table (e.g. caption).\n\t\t\t\t\tif ( !tableRow.is( 'element', 'tableRow' ) ) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\tcreateCells( columnsToInsert, writer, writer.createPositionAt( tableRow, insertAt ? 'end' : 0 ) );\n\t\t\t\t}\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst tableWalker = new TableWalker( table, { column: insertAt, includeAllSlots: true } );\n\n\t\t\tfor ( const tableSlot of tableWalker ) {\n\t\t\t\tconst { row, cell, cellAnchorColumn, cellAnchorRow, cellWidth, cellHeight } = tableSlot;\n\n\t\t\t\t// When iterating over column the table walker outputs either:\n\t\t\t\t// - cells at given column index (cell \"e\" from method docs),\n\t\t\t\t// - spanned columns (spanned cell from row between cells \"g\" and \"h\" - spanned by \"e\", only if `includeAllSlots: true`),\n\t\t\t\t// - or a cell from the same row which spans over this column (cell \"a\").\n\n\t\t\t\tif ( cellAnchorColumn < insertAt ) {\n\t\t\t\t\t// If cell is anchored in previous column, it is a cell that spans over an inserted column (cell \"a\" & \"i\").\n\t\t\t\t\t// For such cells expand them by a number of columns inserted.\n\t\t\t\t\twriter.setAttribute( 'colspan', cellWidth + columnsToInsert, cell );\n\n\t\t\t\t\t// This cell will overlap cells in rows below so skip them (because of `includeAllSlots` option) - (cell \"a\")\n\t\t\t\t\tconst lastCellRow = cellAnchorRow + cellHeight - 1;\n\n\t\t\t\t\tfor ( let i = row; i <= lastCellRow; i++ ) {\n\t\t\t\t\t\ttableWalker.skipRow( i );\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\t// It's either cell at this column index or spanned cell by a row-spanned cell from row above.\n\t\t\t\t\t// In table above it's cell \"e\" and a spanned position from row below (empty cell between cells \"g\" and \"h\")\n\t\t\t\t\tcreateCells( columnsToInsert, writer, tableSlot.getPositionBefore() );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Removes rows from the given `table`.\n\t *\n\t * This method re-calculates the table geometry including `rowspan` attribute of table cells overlapping removed rows\n\t * and table headings values.\n\t *\n\t * ```ts\n\t * editor.plugins.get( 'TableUtils' ).removeRows( table, { at: 1, rows: 2 } );\n\t * ```\n\t *\n\t * Executing the above code in the context of the table on the left will transform its structure as presented on the right:\n\t *\n\t * row index\n\t * ┌───┬───┬───┐ `at` = 1 ┌───┬───┬───┐\n\t * 0 │ a │ b │ c │ `rows` = 2 │ a │ b │ c │ 0\n\t * │ ├───┼───┤ │ ├───┼───┤\n\t * 1 │ │ d │ e │ <-- remove from here │ │ d │ g │ 1\n\t * │ │ ├───┤ will give: ├───┼───┼───┤\n\t * 2 │ │ │ f │ │ h │ i │ j │ 2\n\t * │ │ ├───┤ └───┴───┴───┘\n\t * 3 │ │ │ g │\n\t * ├───┼───┼───┤\n\t * 4 │ h │ i │ j │\n\t * └───┴───┴───┘\n\t *\n\t * @param options.at The row index at which the removing rows will start.\n\t * @param options.rows The number of rows to remove. Default value is 1.\n\t */\n\tpublic removeRows( table: Element, options: { at: number; rows?: number } ): void {\n\t\tconst model = this.editor.model;\n\n\t\tconst rowsToRemove = options.rows || 1;\n\t\tconst rowCount = this.getRows( table );\n\t\tconst first = options.at;\n\t\tconst last = first + rowsToRemove - 1;\n\n\t\tif ( last > rowCount - 1 ) {\n\t\t\t/**\n\t\t\t * The `options.at` param must point at existing row and `options.rows` must not exceed the rows in the table.\n\t\t\t *\n\t\t\t * @error tableutils-removerows-row-index-out-of-range\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'tableutils-removerows-row-index-out-of-range',\n\t\t\t\tthis,\n\t\t\t\t{ table, options }\n\t\t\t);\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\tconst indexesObject = { first, last };\n\n\t\t\t// Removing rows from the table require that most calculations to be done prior to changing table structure.\n\t\t\t// Preparations must be done in the same enqueueChange callback to use the current table structure.\n\n\t\t\t// 1. Preparation - get row-spanned cells that have to be modified after removing rows.\n\t\t\tconst { cellsToMove, cellsToTrim } = getCellsToMoveAndTrimOnRemoveRow( table, indexesObject );\n\n\t\t\t// 2. Execution\n\n\t\t\t// 2a. Move cells from removed rows that extends over a removed section - must be done before removing rows.\n\t\t\t// This will fill any gaps in a rows below that previously were empty because of row-spanned cells.\n\t\t\tif ( cellsToMove.size ) {\n\t\t\t\tconst rowAfterRemovedSection = last + 1;\n\t\t\t\tmoveCellsToRow( table, rowAfterRemovedSection, cellsToMove, writer );\n\t\t\t}\n\n\t\t\t// 2b. Remove all required rows.\n\t\t\tfor ( let i = last; i >= first; i-- ) {\n\t\t\t\twriter.remove( table.getChild( i )! );\n\t\t\t}\n\n\t\t\t// 2c. Update cells from rows above that overlap removed section. Similar to step 2 but does not involve moving cells.\n\t\t\tfor ( const { rowspan, cell } of cellsToTrim ) {\n\t\t\t\tupdateNumericAttribute( 'rowspan', rowspan, cell, writer );\n\t\t\t}\n\n\t\t\t// 2d. Adjust heading rows if removed rows were in a heading section.\n\t\t\tupdateHeadingRows( table, indexesObject, writer );\n\n\t\t\t// 2e. Remove empty columns (without anchored cells) if there are any.\n\t\t\tif ( !removeEmptyColumns( table, this ) ) {\n\t\t\t\t// If there wasn't any empty columns then we still need to check if this wasn't called\n\t\t\t\t// because of cleaning empty rows and we only removed one of them.\n\t\t\t\tremoveEmptyRows( table, this );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Removes columns from the given `table`.\n\t *\n\t * This method re-calculates the table geometry including the `colspan` attribute of table cells overlapping removed columns\n\t * and table headings values.\n\t *\n\t * ```ts\n\t * editor.plugins.get( 'TableUtils' ).removeColumns( table, { at: 1, columns: 2 } );\n\t * ```\n\t *\n\t * Executing the above code in the context of the table on the left will transform its structure as presented on the right:\n\t *\n\t * 0 1 2 3 4 0 1 2\n\t * ┌───────────────┬───┐ ┌───────┬───┐\n\t * │ a │ b │ │ a │ b │\n\t * │ ├───┤ │ ├───┤\n\t * │ │ c │ │ │ c │\n\t * ├───┬───┬───┬───┼───┤ will give: ├───┬───┼───┤\n\t * │ d │ e │ f │ g │ h │ │ d │ g │ h │\n\t * ├───┼───┼───┤ ├───┤ ├───┤ ├───┤\n\t * │ i │ j │ k │ │ l │ │ i │ │ l │\n\t * ├───┴───┴───┴───┴───┤ ├───┴───┴───┤\n\t * │ m │ │ m │\n\t * └───────────────────┘ └───────────┘\n\t * ^---- remove from here, `at` = 1, `columns` = 2\n\t *\n\t * @param options.at The row index at which the removing columns will start.\n\t * @param options.columns The number of columns to remove.\n\t */\n\tpublic removeColumns( table: Element, options: { at: number; columns?: number } ): void {\n\t\tconst model = this.editor.model;\n\t\tconst first = options.at;\n\t\tconst columnsToRemove = options.columns || 1;\n\t\tconst last = options.at + columnsToRemove - 1;\n\n\t\tmodel.change( writer => {\n\t\t\tadjustHeadingColumns( table, { first, last }, writer );\n\n\t\t\tfor ( let removedColumnIndex = last; removedColumnIndex >= first; removedColumnIndex-- ) {\n\t\t\t\tfor ( const { cell, column, cellWidth } of [ ...new TableWalker( table ) ] ) {\n\t\t\t\t\t// If colspaned cell overlaps removed column decrease its span.\n\t\t\t\t\tif ( column <= removedColumnIndex && cellWidth > 1 && column + cellWidth > removedColumnIndex ) {\n\t\t\t\t\t\tupdateNumericAttribute( 'colspan', cellWidth - 1, cell, writer );\n\t\t\t\t\t} else if ( column === removedColumnIndex ) {\n\t\t\t\t\t\t// The cell in removed column has colspan of 1.\n\t\t\t\t\t\twriter.remove( cell );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Remove empty rows that could appear after removing columns.\n\t\t\tif ( !removeEmptyRows( table, this ) ) {\n\t\t\t\t// If there wasn't any empty rows then we still need to check if this wasn't called\n\t\t\t\t// because of cleaning empty columns and we only removed one of them.\n\t\t\t\tremoveEmptyColumns( table, this );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Divides a table cell vertically into several ones.\n\t *\n\t * The cell will be visually split into more cells by updating colspans of other cells in a column\n\t * and inserting cells (columns) after that cell.\n\t *\n\t * In the table below, if cell \"a\" is split into 3 cells:\n\t *\n\t * +---+---+---+\n\t * | a | b | c |\n\t * +---+---+---+\n\t * | d | e | f |\n\t * +---+---+---+\n\t *\n\t * it will result in the table below:\n\t *\n\t * +---+---+---+---+---+\n\t * | a | | | b | c |\n\t * +---+---+---+---+---+\n\t * | d | e | f |\n\t * +---+---+---+---+---+\n\t *\n\t * So cell \"d\" will get its `colspan` updated to `3` and 2 cells will be added (2 columns will be created).\n\t *\n\t * Splitting a cell that already has a `colspan` attribute set will distribute the cell `colspan` evenly and the remainder\n\t * will be left to the original cell:\n\t *\n\t * +---+---+---+\n\t * | a |\n\t * +---+---+---+\n\t * | b | c | d |\n\t * +---+---+---+\n\t *\n\t * Splitting cell \"a\" with `colspan=3` into 2 cells will create 1 cell with a `colspan=a` and cell \"a\" that will have `colspan=2`:\n\t *\n\t * +---+---+---+\n\t * | a | |\n\t * +---+---+---+\n\t * | b | c | d |\n\t * +---+---+---+\n\t */\n\tpublic splitCellVertically( tableCell: Element, numberOfCells = 2 ): void {\n\t\tconst model = this.editor.model;\n\t\tconst tableRow = tableCell.parent!;\n\t\tconst table = tableRow.parent as Element;\n\n\t\tconst rowspan = parseInt( tableCell.getAttribute( 'rowspan' ) as string || '1' );\n\t\tconst colspan = parseInt( tableCell.getAttribute( 'colspan' ) as string || '1' );\n\n\t\tmodel.change( writer => {\n\t\t\t// First check - the cell spans over multiple rows so before doing anything else just split this cell.\n\t\t\tif ( colspan > 1 ) {\n\t\t\t\t// Get spans of new (inserted) cells and span to update of split cell.\n\t\t\t\tconst { newCellsSpan, updatedSpan } = breakSpanEvenly( colspan, numberOfCells );\n\n\t\t\t\tupdateNumericAttribute( 'colspan', updatedSpan, tableCell, writer );\n\n\t\t\t\t// Each inserted cell will have the same attributes:\n\t\t\t\tconst newCellsAttributes: { colspan?: number; rowspan?: number } = {};\n\n\t\t\t\t// Do not store default value in the model.\n\t\t\t\tif ( newCellsSpan > 1 ) {\n\t\t\t\t\tnewCellsAttributes.colspan = newCellsSpan;\n\t\t\t\t}\n\n\t\t\t\t// Copy rowspan of split cell.\n\t\t\t\tif ( rowspan > 1 ) {\n\t\t\t\t\tnewCellsAttributes.rowspan = rowspan;\n\t\t\t\t}\n\n\t\t\t\tconst cellsToInsert = colspan > numberOfCells ? numberOfCells - 1 : colspan - 1;\n\t\t\t\tcreateCells( cellsToInsert, writer, writer.createPositionAfter( tableCell ), newCellsAttributes );\n\t\t\t}\n\n\t\t\t// Second check - the cell has colspan of 1 or we need to create more cells then the currently one spans over.\n\t\t\tif ( colspan < numberOfCells ) {\n\t\t\t\tconst cellsToInsert = numberOfCells - colspan;\n\n\t\t\t\t// First step: expand cells on the same column as split cell.\n\t\t\t\tconst tableMap = [ ...new TableWalker( table ) ];\n\n\t\t\t\t// Get the column index of split cell.\n\t\t\t\tconst { column: splitCellColumn } = tableMap.find( ( { cell } ) => cell === tableCell )!;\n\n\t\t\t\t// Find cells which needs to be expanded vertically - those on the same column or those that spans over split cell's column.\n\t\t\t\tconst cellsToUpdate = tableMap.filter( ( { cell, cellWidth, column } ) => {\n\t\t\t\t\tconst isOnSameColumn = cell !== tableCell && column === splitCellColumn;\n\t\t\t\t\tconst spansOverColumn = ( column < splitCellColumn && column + cellWidth > splitCellColumn );\n\n\t\t\t\t\treturn isOnSameColumn || spansOverColumn;\n\t\t\t\t} );\n\n\t\t\t\t// Expand cells vertically.\n\t\t\t\tfor ( const { cell, cellWidth } of cellsToUpdate ) {\n\t\t\t\t\twriter.setAttribute( 'colspan', cellWidth + cellsToInsert, cell );\n\t\t\t\t}\n\n\t\t\t\t// Second step: create columns after split cell.\n\n\t\t\t\t// Each inserted cell will have the same attributes:\n\t\t\t\tconst newCellsAttributes: { rowspan?: number } = {};\n\n\t\t\t\t// Do not store default value in the model.\n\n\t\t\t\t// Copy rowspan of split cell.\n\t\t\t\tif ( rowspan > 1 ) {\n\t\t\t\t\tnewCellsAttributes.rowspan = rowspan;\n\t\t\t\t}\n\n\t\t\t\tcreateCells( cellsToInsert, writer, writer.createPositionAfter( tableCell ), newCellsAttributes );\n\n\t\t\t\tconst headingColumns = table.getAttribute( 'headingColumns' ) as number || 0;\n\n\t\t\t\t// Update heading section if split cell is in heading section.\n\t\t\t\tif ( headingColumns > splitCellColumn ) {\n\t\t\t\t\tupdateNumericAttribute( 'headingColumns', headingColumns + cellsToInsert, table, writer );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Divides a table cell horizontally into several ones.\n\t *\n\t * The cell will be visually split into more cells by updating rowspans of other cells in the row and inserting rows with a single cell\n\t * below.\n\t *\n\t * If in the table below cell \"b\" is split into 3 cells:\n\t *\n\t * +---+---+---+\n\t * | a | b | c |\n\t * +---+---+---+\n\t * | d | e | f |\n\t * +---+---+---+\n\t *\n\t * It will result in the table below:\n\t *\n\t * +---+---+---+\n\t * | a | b | c |\n\t * + +---+ +\n\t * | | | |\n\t * + +---+ +\n\t * | | | |\n\t * +---+---+---+\n\t * | d | e | f |\n\t * +---+---+---+\n\t *\n\t * So cells \"a\" and \"b\" will get their `rowspan` updated to `3` and 2 rows with a single cell will be added.\n\t *\n\t * Splitting a cell that already has a `rowspan` attribute set will distribute the cell `rowspan` evenly and the remainder\n\t * will be left to the original cell:\n\t *\n\t * +---+---+---+\n\t * | a | b | c |\n\t * + +---+---+\n\t * | | d | e |\n\t * + +---+---+\n\t * | | f | g |\n\t * + +---+---+\n\t * | | h | i |\n\t * +---+---+---+\n\t *\n\t * Splitting cell \"a\" with `rowspan=4` into 3 cells will create 2 cells with a `rowspan=1` and cell \"a\" will have `rowspan=2`:\n\t *\n\t * +---+---+---+\n\t * | a | b | c |\n\t * + +---+---+\n\t * | | d | e |\n\t * +---+---+---+\n\t * | | f | g |\n\t * +---+---+---+\n\t * | | h | i |\n\t * +---+---+---+\n\t */\n\tpublic splitCellHorizontally( tableCell: Element, numberOfCells = 2 ): void {\n\t\tconst model = this.editor.model;\n\n\t\tconst tableRow = tableCell.parent as Node;\n\t\tconst table = tableRow.parent! as Element;\n\t\tconst splitCellRow = table.getChildIndex( tableRow )!;\n\n\t\tconst rowspan = parseInt( tableCell.getAttribute( 'rowspan' ) as string || '1' );\n\t\tconst colspan = parseInt( tableCell.getAttribute( 'colspan' ) as string || '1' );\n\n\t\tmodel.change( writer => {\n\t\t\t// First check - the cell spans over multiple rows so before doing anything else just split this cell.\n\t\t\tif ( rowspan > 1 ) {\n\t\t\t\t// Cache table map before updating table.\n\t\t\t\tconst tableMap = [ ...new TableWalker( table, {\n\t\t\t\t\tstartRow: splitCellRow,\n\t\t\t\t\tendRow: splitCellRow + rowspan - 1,\n\t\t\t\t\tincludeAllSlots: true\n\t\t\t\t} ) ];\n\n\t\t\t\t// Get spans of new (inserted) cells and span to update of split cell.\n\t\t\t\tconst { newCellsSpan, updatedSpan } = breakSpanEvenly( rowspan, numberOfCells );\n\n\t\t\t\tupdateNumericAttribute( 'rowspan', updatedSpan, tableCell, writer );\n\n\t\t\t\tconst { column: cellColumn } = tableMap.find( ( { cell } ) => cell === tableCell )!;\n\n\t\t\t\t// Each inserted cell will have the same attributes:\n\t\t\t\tconst newCellsAttributes: { rowspan?: number; colspan?: number } = {};\n\n\t\t\t\t// Do not store default value in the model.\n\t\t\t\tif ( newCellsSpan > 1 ) {\n\t\t\t\t\tnewCellsAttributes.rowspan = newCellsSpan;\n\t\t\t\t}\n\n\t\t\t\t// Copy colspan of split cell.\n\t\t\t\tif ( colspan > 1 ) {\n\t\t\t\t\tnewCellsAttributes.colspan = colspan;\n\t\t\t\t}\n\n\t\t\t\tfor ( const tableSlot of tableMap ) {\n\t\t\t\t\tconst { column, row } = tableSlot;\n\n\t\t\t\t\t// As both newly created cells and the split cell might have rowspan,\n\t\t\t\t\t// the insertion of new cells must go to appropriate rows:\n\t\t\t\t\t//\n\t\t\t\t\t// 1. It's a row after split cell + it's height.\n\t\t\t\t\tconst isAfterSplitCell = row >= splitCellRow + updatedSpan;\n\t\t\t\t\t// 2. Is on the same column.\n\t\t\t\t\tconst isOnSameColumn = column === cellColumn;\n\t\t\t\t\t// 3. And it's row index is after previous cell height.\n\t\t\t\t\tconst isInEvenlySplitRow = ( row + splitCellRow + updatedSpan ) % newCellsSpan === 0;\n\n\t\t\t\t\tif ( isAfterSplitCell && isOnSameColumn && isInEvenlySplitRow ) {\n\t\t\t\t\t\tcreateCells( 1, writer, tableSlot.getPositionBefore(), newCellsAttributes );\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Second check - the cell has rowspan of 1 or we need to create more cells than the current cell spans over.\n\t\t\tif ( rowspan < numberOfCells ) {\n\t\t\t\t// We already split the cell in check one so here we split to the remaining number of cells only.\n\t\t\t\tconst cellsToInsert = numberOfCells - rowspan;\n\n\t\t\t\t// This check is needed since we need to check if there are any cells from previous rows than spans over this cell's row.\n\t\t\t\tconst tableMap = [ ...new TableWalker( table, { startRow: 0, endRow: splitCellRow } ) ];\n\n\t\t\t\t// First step: expand cells.\n\t\t\t\tfor ( const { cell, cellHeight, row } of tableMap ) {\n\t\t\t\t\t// Expand rowspan of cells that are either:\n\t\t\t\t\t// - on the same row as current cell,\n\t\t\t\t\t// - or are below split cell row and overlaps that row.\n\t\t\t\t\tif ( cell !== tableCell && row + cellHeight > splitCellRow ) {\n\t\t\t\t\t\tconst rowspanToSet = cellHeight + cellsToInsert;\n\n\t\t\t\t\t\twriter.setAttribute( 'rowspan', rowspanToSet, cell );\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\t// Second step: create rows with single cell below split cell.\n\t\t\t\tconst newCellsAttributes: { colspan?: number } = {};\n\n\t\t\t\t// Copy colspan of split cell.\n\t\t\t\tif ( colspan > 1 ) {\n\t\t\t\t\tnewCellsAttributes.colspan = colspan;\n\t\t\t\t}\n\n\t\t\t\tcreateEmptyRows( writer, table, splitCellRow + 1, cellsToInsert, 1, newCellsAttributes );\n\n\t\t\t\t// Update heading section if split cell is in heading section.\n\t\t\t\tconst headingRows = table.getAttribute( 'headingRows' ) as number || 0;\n\n\t\t\t\tif ( headingRows > splitCellRow ) {\n\t\t\t\t\tupdateNumericAttribute( 'headingRows', headingRows + cellsToInsert, table, writer );\n\t\t\t\t}\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Returns the number of columns for a given table.\n\t *\n\t * ```ts\n\t * editor.plugins.get( 'TableUtils' ).getColumns( table );\n\t * ```\n\t *\n\t * @param table The table to analyze.\n\t */\n\tpublic getColumns( table: Element ): number {\n\t\t// Analyze first row only as all the rows should have the same width.\n\t\t// Using the first row without checking if it's a tableRow because we expect\n\t\t// that table will have only tableRow model elements at the beginning.\n\t\tconst row = table.getChild( 0 ) as Element;\n\n\t\treturn [ ...row.getChildren() ].reduce( ( columns, row ) => {\n\t\t\tconst columnWidth = parseInt( row.getAttribute( 'colspan' ) as string || '1' );\n\n\t\t\treturn columns + columnWidth;\n\t\t}, 0 );\n\t}\n\n\t/**\n\t * Returns the number of rows for a given table. Any other element present in the table model is omitted.\n\t *\n\t * ```ts\n\t * editor.plugins.get( 'TableUtils' ).getRows( table );\n\t * ```\n\t *\n\t * @param table The table to analyze.\n\t */\n\tpublic getRows( table: Element ): number {\n\t\t// Rowspan not included due to #6427.\n\t\treturn Array.from( table.getChildren() )\n\t\t\t.reduce( ( rowCount, child ) => child.is( 'element', 'tableRow' ) ? rowCount + 1 : rowCount, 0 );\n\t}\n\n\t/**\n\t * Creates an instance of the table walker.\n\t *\n\t * The table walker iterates internally by traversing the table from row index = 0 and column index = 0.\n\t * It walks row by row and column by column in order to output values defined in the options.\n\t * By default it will output only the locations that are occupied by a cell. To include also spanned rows and columns,\n\t * pass the `includeAllSlots` option.\n\t *\n\t * @internal\n\t * @param table A table over which the walker iterates.\n\t * @param options An object with configuration.\n\t */\n\tpublic createTableWalker( table: Element, options: TableWalkerOptions = {} ): TableWalker {\n\t\treturn new TableWalker( table, options );\n\t}\n\n\t/**\n\t * Returns all model table cells that are fully selected (from the outside)\n\t * within the provided model selection's ranges.\n\t *\n\t * To obtain the cells selected from the inside, use\n\t * {@link #getTableCellsContainingSelection}.\n\t */\n\tpublic getSelectedTableCells( selection: Selection | DocumentSelection ): Array<Element> {\n\t\tconst cells = [];\n\n\t\tfor ( const range of this.sortRanges( selection.getRanges() ) ) {\n\t\t\tconst element = range.getContainedElement();\n\n\t\t\tif ( element && element.is( 'element', 'tableCell' ) ) {\n\t\t\t\tcells.push( element );\n\t\t\t}\n\t\t}\n\n\t\treturn cells;\n\t}\n\n\t/**\n\t * Returns all model table cells that the provided model selection's ranges\n\t * {@link module:engine/model/range~Range#start} inside.\n\t *\n\t * To obtain the cells selected from the outside, use\n\t * {@link #getSelectedTableCells}.\n\t */\n\tpublic getTableCellsContainingSelection( selection: Selection | DocumentSelection ): Array<Element> {\n\t\tconst cells = [];\n\n\t\tfor ( const range of selection.getRanges() ) {\n\t\t\tconst cellWithSelection = range.start.findAncestor( 'tableCell' );\n\n\t\t\tif ( cellWithSelection ) {\n\t\t\t\tcells.push( cellWithSelection );\n\t\t\t}\n\t\t}\n\n\t\treturn cells;\n\t}\n\n\t/**\n\t * Returns all model table cells that are either completely selected\n\t * by selection ranges or host selection range\n\t * {@link module:engine/model/range~Range#start start positions} inside them.\n\t *\n\t * Combines {@link #getTableCellsContainingSelection} and\n\t * {@link #getSelectedTableCells}.\n\t */\n\tpublic getSelectionAffectedTableCells( selection: Selection | DocumentSelection ): Array<Element> {\n\t\tconst selectedCells = this.getSelectedTableCells( selection );\n\n\t\tif ( selectedCells.length ) {\n\t\t\treturn selectedCells;\n\t\t}\n\n\t\treturn this.getTableCellsContainingSelection( selection );\n\t}\n\n\t/**\n\t * Returns an object with the `first` and `last` row index contained in the given `tableCells`.\n\t *\n\t * ```ts\n\t * const selectedTableCells = getSelectedTableCells( editor.model.document.selection );\n\t *\n\t * const { first, last } = getRowIndexes( selectedTableCells );\n\t *\n\t * console.log( `Selected rows: ${ first } to ${ last }` );\n\t * ```\n\t *\n\t * @returns Returns an object with the `first` and `last` table row indexes.\n\t */\n\tpublic getRowIndexes( tableCells: Array<Element> ): IndexesObject {\n\t\tconst indexes = tableCells.map( cell => ( cell.parent as Element ).index! );\n\n\t\treturn this._getFirstLastIndexesObject( indexes );\n\t}\n\n\t/**\n\t * Returns an object with the `first` and `last` column index contained in the given `tableCells`.\n\t *\n\t * ```ts\n\t * const selectedTableCells = getSelectedTableCells( editor.model.document.selection );\n\t *\n\t * const { first, last } = getColumnIndexes( selectedTableCells );\n\t *\n\t * console.log( `Selected columns: ${ first } to ${ last }` );\n\t * ```\n\t *\n\t * @returns Returns an object with the `first` and `last` table column indexes.\n\t */\n\tpublic getColumnIndexes( tableCells: Array<Element> ): IndexesObject {\n\t\tconst table = tableCells[ 0 ].findAncestor( 'table' )!;\n\t\tconst tableMap = [ ...new TableWalker( table ) ];\n\n\t\tconst indexes = tableMap\n\t\t\t.filter( entry => tableCells.includes( entry.cell ) )\n\t\t\t.map( entry => entry.column );\n\n\t\treturn this._getFirstLastIndexesObject( indexes );\n\t}\n\n\t/**\n\t * Checks if the selection contains cells that do not exceed rectangular selection.\n\t *\n\t * In a table below:\n\t *\n\t * ┌───┬───┬───┬───┐\n\t * │ a │ b │ c │ d │\n\t * ├───┴───┼───┤ │\n\t * │ e │ f │ │\n\t * │ ├───┼───┤\n\t * │ │ g │ h │\n\t * └───────┴───┴───┘\n\t *\n\t * Valid selections are these which create a solid rectangle (without gaps), such as:\n\t * - a, b (two horizontal cells)\n\t * - c, f (two vertical cells)\n\t * - a, b, e (cell \"e\" spans over four cells)\n\t * - c, d, f (cell d spans over a cell in the row below)\n\t *\n\t * While an invalid selection would be:\n\t * - a, c (the unselected cell \"b\" creates a gap)\n\t * - f, g, h (cell \"d\" spans over a cell from the row of \"f\" cell - thus creates a gap)\n\t */\n\tpublic isSelectionRectangular( selectedTableCells: Array<Element> ): boolean {\n\t\tif ( selectedTableCells.length < 2 || !this._areCellInTheSameTableSection( selectedTableCells ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// A valid selection is a fully occupied rectangle composed of table cells.\n\t\t// Below we will calculate the area of a selected table cells and the area of valid selection.\n\t\t// The area of a valid selection is defined by top-left and bottom-right cells.\n\t\tconst rows: Set<number> = new Set();\n\t\tconst columns: Set<number> = new Set();\n\n\t\tlet areaOfSelectedCells = 0;\n\n\t\tfor ( const tableCell of selectedTableCells ) {\n\t\t\tconst { row, column } = this.getCellLocation( tableCell );\n\t\t\tconst rowspan = parseInt( tableCell.getAttribute( 'rowspan' ) as string ) || 1;\n\t\t\tconst colspan = parseInt( tableCell.getAttribute( 'colspan' ) as string ) || 1;\n\n\t\t\t// Record row & column indexes of current cell.\n\t\t\trows.add( row );\n\t\t\tcolumns.add( column );\n\n\t\t\t// For cells that spans over multiple rows add also the last row that this cell spans over.\n\t\t\tif ( rowspan > 1 ) {\n\t\t\t\trows.add( row + rowspan - 1 );\n\t\t\t}\n\n\t\t\t// For cells that spans over multiple columns add also the last column that this cell spans over.\n\t\t\tif ( colspan > 1 ) {\n\t\t\t\tcolumns.add( column + colspan - 1 );\n\t\t\t}\n\n\t\t\tareaOfSelectedCells += ( rowspan * colspan );\n\t\t}\n\n\t\t// We can only merge table cells that are in adjacent rows...\n\t\tconst areaOfValidSelection = getBiggestRectangleArea( rows, columns );\n\n\t\treturn areaOfValidSelection == areaOfSelectedCells;\n\t}\n\n\t/**\n\t * Returns array of sorted ranges.\n\t */\n\tpublic sortRanges( ranges: Iterable<Range> ): Array<Range> {\n\t\treturn Array.from( ranges ).sort( compareRangeOrder );\n\t}\n\n\t/**\n\t * Helper method to get an object with `first` and `last` indexes from an unsorted array of indexes.\n\t */\n\tprivate _getFirstLastIndexesObject( indexes: Array<number> ): IndexesObject {\n\t\tconst allIndexesSorted = indexes.sort( ( indexA, indexB ) => indexA - indexB );\n\n\t\tconst first = allIndexesSorted[ 0 ];\n\t\tconst last = allIndexesSorted[ allIndexesSorted.length - 1 ];\n\n\t\treturn { first, last };\n\t}\n\n\t/**\n\t * Checks if the selection does not mix a header (column or row) with other cells.\n\t *\n\t * For instance, in the table below valid selections consist of cells with the same letter only.\n\t * So, a-a (same heading row and column) or d-d (body cells) are valid while c-d or a-b are not.\n\t *\n\t * header columns\n\t * ↓ ↓\n\t * ┌───┬───┬───┬───┐\n\t * │ a │ a │ b │ b │ ← header row\n\t * ├───┼───┼───┼───┤\n\t * │ c │ c │ d │ d │\n\t * ├───┼───┼───┼───┤\n\t * │ c │ c │ d │ d │\n\t * └───┴───┴───┴───┘\n\t */\n\tprivate _areCellInTheSameTableSection( tableCells: Array<Element> ): boolean {\n\t\tconst table = tableCells[ 0 ].findAncestor( 'table' )!;\n\n\t\tconst rowIndexes = this.getRowIndexes( tableCells );\n\t\tconst headingRows = parseInt( table.getAttribute( 'headingRows' ) as string ) || 0;\n\n\t\t// Calculating row indexes is a bit cheaper so if this check fails we can't merge.\n\t\tif ( !this._areIndexesInSameSection( rowIndexes, headingRows ) ) {\n\t\t\treturn false;\n\t\t}\n\n\t\tconst columnIndexes = this.getColumnIndexes( tableCells );\n\t\tconst headingColumns = parseInt( table.getAttribute( 'headingColumns' ) as string ) || 0;\n\n\t\t// Similarly cells must be in same column section.\n\t\treturn this._areIndexesInSameSection( columnIndexes, headingColumns );\n\t}\n\n\t/**\n\t * Unified check if table rows/columns indexes are in the same heading/body section.\n\t */\n\tprivate _areIndexesInSameSection( { first, last }: IndexesObject, headingSectionSize: number ): boolean {\n\t\tconst firstCellIsInHeading = first < headingSectionSize;\n\t\tconst lastCellIsInHeading = last < headingSectionSize;\n\n\t\treturn firstCellIsInHeading === lastCellIsInHeading;\n\t}\n}\n\n/**\n * Creates empty rows at the given index in an existing table.\n *\n * @param insertAt The row index of row insertion.\n * @param rows The number of rows to create.\n * @param tableCellToInsert The number of cells to insert in each row.\n */\nfunction createEmptyRows( writer: Writer, table: Element, insertAt: number, rows: number, tableCellToInsert: number, attributes = {} ) {\n\tfor ( let i = 0; i < rows; i++ ) {\n\t\tconst tableRow = writer.createElement( 'tableRow' );\n\n\t\twriter.insert( tableRow, table, insertAt );\n\n\t\tcreateCells( tableCellToInsert, writer, writer.createPositionAt( tableRow, 'end' ), attributes );\n\t}\n}\n\n/**\n * Creates cells at a given position.\n *\n * @param cells The number of cells to create\n */\nfunction createCells( cells: number, writer: Writer, insertPosition: Position, attributes = {} ) {\n\tfor ( let i = 0; i < cells; i++ ) {\n\t\tcreateEmptyTableCell( writer, insertPosition, attributes );\n\t}\n}\n\n/**\n * Evenly distributes the span of a cell to a number of provided cells.\n * The resulting spans will always be integer values.\n *\n * For instance breaking a span of 7 into 3 cells will return:\n *\n * ```ts\n * { newCellsSpan: 2, updatedSpan: 3 }\n * ```\n *\n * as two cells will have a span of 2 and the remainder will go the first cell so its span will change to 3.\n *\n * @param span The span value do break.\n * @param numberOfCells The number of resulting spans.\n */\nfunction breakSpanEvenly( span: number, numberOfCells: number ): { newCellsSpan: number; updatedSpan: number } {\n\tif ( span < numberOfCells ) {\n\t\treturn { newCellsSpan: 1, updatedSpan: 1 };\n\t}\n\n\tconst newCellsSpan = Math.floor( span / numberOfCells );\n\tconst updatedSpan = ( span - newCellsSpan * numberOfCells ) + newCellsSpan;\n\n\treturn { newCellsSpan, updatedSpan };\n}\n\n/**\n * Updates heading columns attribute if removing a row from head section.\n */\nfunction adjustHeadingColumns( table: Element, removedColumnIndexes: IndexesObject, writer: Writer ) {\n\tconst headingColumns = table.getAttribute( 'headingColumns' ) as number || 0;\n\n\tif ( headingColumns && removedColumnIndexes.first < headingColumns ) {\n\t\tconst headingsRemoved = Math.min( headingColumns - 1 /* Other numbers are 0-based */, removedColumnIndexes.last ) -\n\t\t\tremovedColumnIndexes.first + 1;\n\n\t\twriter.setAttribute( 'headingColumns', headingColumns - headingsRemoved, table );\n\t}\n}\n\n/**\n * Calculates a new heading rows value for removing rows from heading section.\n */\nfunction updateHeadingRows( table: Element, { first, last }: IndexesObject, writer: Writer ) {\n\tconst headingRows = table.getAttribute( 'headingRows' ) as number || 0;\n\n\tif ( first < headingRows ) {\n\t\tconst newRows = last < headingRows ? headingRows - ( last - first + 1 ) : first;\n\n\t\tupdateNumericAttribute( 'headingRows', newRows, table, writer, 0 );\n\t}\n}\n\n/**\n * Finds cells that will be:\n * - trimmed - Cells that are \"above\" removed rows sections and overlap the removed section - their rowspan must be trimmed.\n * - moved - Cells from removed rows section might stick out of. These cells are moved to the next row after a removed section.\n *\n * Sample table with overlapping & sticking out cells:\n *\n * +----+----+----+----+----+\n * | 00 | 01 | 02 | 03 | 04 |\n * +----+ + + + +\n * | 10 | | | | |\n * +----+----+ + + +\n * | 20 | 21 | | | | <-- removed row\n * + + +----+ + +\n * | | | 32 | | | <-- removed row\n * +----+ + +----+ +\n * | 40 | | | 43 | |\n * +----+----+----+----+----+\n *\n * In a table above:\n * - cells to trim: '02', '03' & '04'.\n * - cells to move: '21' & '32'.\n */\nfunction getCellsToMoveAndTrimOnRemoveRow( table: Element, { first, last }: IndexesObject ) {\n\tconst cellsToMove: CellsToMove = new Map();\n\tconst cellsToTrim: CellsToTrim = [];\n\n\tfor ( const { row, column, cellHeight, cell } of new TableWalker( table, { endRow: last } ) ) {\n\t\tconst lastRowOfCell = row + cellHeight - 1;\n\n\t\tconst isCellStickingOutFromRemovedRows = row >= first && row <= last && lastRowOfCell > last;\n\n\t\tif ( isCellStickingOutFromRemovedRows ) {\n\t\t\tconst rowspanInRemovedSection = last - row + 1;\n\t\t\tconst rowSpanToSet = cellHeight - rowspanInRemovedSection;\n\n\t\t\tcellsToMove.set( column, {\n\t\t\t\tcell,\n\t\t\t\trowspan: rowSpanToSet\n\t\t\t} );\n\t\t}\n\n\t\tconst isCellOverlappingRemovedRows = row < first && lastRowOfCell >= first;\n\n\t\tif ( isCellOverlappingRemovedRows ) {\n\t\t\tlet rowspanAdjustment;\n\n\t\t\t// Cell fully covers removed section - trim it by removed rows count.\n\t\t\tif ( lastRowOfCell >= last ) {\n\t\t\t\trowspanAdjustment = last - first + 1;\n\t\t\t}\n\t\t\t// Cell partially overlaps removed section - calculate cell's span that is in removed section.\n\t\t\telse {\n\t\t\t\trowspanAdjustment = lastRowOfCell - first + 1;\n\t\t\t}\n\n\t\t\tcellsToTrim.push( {\n\t\t\t\tcell,\n\t\t\t\trowspan: cellHeight - rowspanAdjustment\n\t\t\t} );\n\t\t}\n\t}\n\n\treturn { cellsToMove, cellsToTrim };\n}\n\nfunction moveCellsToRow( table: Element, targetRowIndex: number, cellsToMove: CellsToMove, writer: Writer ) {\n\tconst tableWalker = new TableWalker( table, {\n\t\tincludeAllSlots: true,\n\t\trow: targetRowIndex\n\t} );\n\n\tconst tableRowMap = [ ...tableWalker ];\n\tconst row = table.getChild( targetRowIndex )!;\n\n\tlet previousCell;\n\n\tfor ( const { column, cell, isAnchor } of tableRowMap ) {\n\t\tif ( cellsToMove.has( column ) ) {\n\t\t\tconst { cell: cellToMove, rowspan } = cellsToMove.get( column )!;\n\n\t\t\tconst targetPosition = previousCell ?\n\t\t\t\twriter.createPositionAfter( previousCell ) :\n\t\t\t\twriter.createPositionAt( row, 0 );\n\n\t\t\twriter.move( writer.createRangeOn( cellToMove ), targetPosition );\n\t\t\tupdateNumericAttribute( 'rowspan', rowspan, cellToMove, writer );\n\n\t\t\tpreviousCell = cellToMove;\n\t\t} else if ( isAnchor ) {\n\t\t\t// If cell is spanned then `cell` holds reference to overlapping cell. See ckeditor/ckeditor5#6502.\n\t\t\tpreviousCell = cell;\n\t\t}\n\t}\n}\n\nfunction compareRangeOrder( rangeA: Range, rangeB: Range ) {\n\t// Since table cell ranges are disjoint, it's enough to check their start positions.\n\tconst posA = rangeA.start;\n\tconst posB = rangeB.start;\n\n\t// Checking for equal position (returning 0) is not needed because this would be either:\n\t// a. Intersecting range (not allowed by model)\n\t// b. Collapsed range on the same position (allowed by model but should not happen).\n\treturn posA.isBefore( posB ) ? -1 : 1;\n}\n\n/**\n * Calculates the area of a maximum rectangle that can span over the provided row & column indexes.\n */\nfunction getBiggestRectangleArea( rows: Set<number>, columns: Set<number> ): number {\n\tconst rowsIndexes = Array.from( rows.values() );\n\tconst columnIndexes = Array.from( columns.values() );\n\n\tconst lastRow = Math.max( ...rowsIndexes );\n\tconst firstRow = Math.min( ...rowsIndexes );\n\tconst lastColumn = Math.max( ...columnIndexes );\n\tconst firstColumn = Math.min( ...columnIndexes );\n\n\treturn ( lastRow - firstRow + 1 ) * ( lastColumn - firstColumn + 1 );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/mergecellscommand\n */\n\nimport type {\n\tElement,\n\tWriter\n} from 'ckeditor5/src/engine';\n\nimport { Command } from 'ckeditor5/src/core';\nimport TableUtils from '../tableutils';\nimport { updateNumericAttribute } from '../utils/common';\nimport { removeEmptyRowsColumns } from '../utils/structure';\n\n/**\n * The merge cells command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'mergeTableCells'` editor command.\n *\n * For example, to merge selected table cells:\n *\n * ```ts\n * editor.execute( 'mergeTableCells' );\n * ```\n */\nexport default class MergeCellsCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst tableUtils = this.editor.plugins.get( TableUtils );\n\n\t\tconst selectedTableCells = tableUtils.getSelectedTableCells( this.editor.model.document.selection );\n\t\tthis.isEnabled = tableUtils.isSelectionRectangular( selectedTableCells );\n\t}\n\n\t/**\n\t * Executes the command.\n\t *\n\t * @fires execute\n\t */\n\tpublic override execute(): void {\n\t\tconst model = this.editor.model;\n\t\tconst tableUtils = this.editor.plugins.get( TableUtils );\n\n\t\tmodel.change( writer => {\n\t\t\tconst selectedTableCells = tableUtils.getSelectedTableCells( model.document.selection );\n\n\t\t\t// All cells will be merged into the first one.\n\t\t\tconst firstTableCell = selectedTableCells.shift()!;\n\n\t\t\t// Update target cell dimensions.\n\t\t\tconst { mergeWidth, mergeHeight } = getMergeDimensions( firstTableCell, selectedTableCells, tableUtils );\n\t\t\tupdateNumericAttribute( 'colspan', mergeWidth, firstTableCell, writer );\n\t\t\tupdateNumericAttribute( 'rowspan', mergeHeight, firstTableCell, writer );\n\n\t\t\tfor ( const tableCell of selectedTableCells ) {\n\t\t\t\tmergeTableCells( tableCell, firstTableCell, writer );\n\t\t\t}\n\n\t\t\tconst table = firstTableCell.findAncestor( 'table' )!;\n\n\t\t\t// Remove rows and columns that become empty (have no anchored cells).\n\t\t\tremoveEmptyRowsColumns( table, tableUtils );\n\n\t\t\twriter.setSelection( firstTableCell, 'in' );\n\t\t} );\n\t}\n}\n\n/**\n * Merges two table cells. It will ensure that after merging cells with empty paragraphs the resulting table cell will only have one\n * paragraph. If one of the merged table cells is empty, the merged table cell will have contents of the non-empty table cell.\n * If both are empty, the merged table cell will have only one empty paragraph.\n */\nfunction mergeTableCells( cellBeingMerged: Element, targetCell: Element, writer: Writer ) {\n\tif ( !isEmpty( cellBeingMerged ) ) {\n\t\tif ( isEmpty( targetCell ) ) {\n\t\t\twriter.remove( writer.createRangeIn( targetCell ) );\n\t\t}\n\n\t\twriter.move( writer.createRangeIn( cellBeingMerged ), writer.createPositionAt( targetCell, 'end' ) );\n\t}\n\n\t// Remove merged table cell.\n\twriter.remove( cellBeingMerged );\n}\n\n/**\n * Checks if the passed table cell contains an empty paragraph.\n */\nfunction isEmpty( tableCell: Element ): boolean {\n\tconst firstTableChild = tableCell.getChild( 0 );\n\n\treturn tableCell.childCount == 1 && firstTableChild!.is( 'element', 'paragraph' ) && firstTableChild.isEmpty;\n}\n\nfunction getMergeDimensions( firstTableCell: Element, selectedTableCells: Array<Element>, tableUtils: TableUtils ) {\n\tlet maxWidthOffset = 0;\n\tlet maxHeightOffset = 0;\n\n\tfor ( const tableCell of selectedTableCells ) {\n\t\tconst { row, column } = tableUtils.getCellLocation( tableCell );\n\n\t\tmaxWidthOffset = getMaxOffset( tableCell, column, maxWidthOffset, 'colspan' );\n\t\tmaxHeightOffset = getMaxOffset( tableCell, row, maxHeightOffset, 'rowspan' );\n\t}\n\n\t// Update table cell span attribute and merge set selection on a merged contents.\n\tconst { row: firstCellRow, column: firstCellColumn } = tableUtils.getCellLocation( firstTableCell );\n\n\tconst mergeWidth = maxWidthOffset - firstCellColumn;\n\tconst mergeHeight = maxHeightOffset - firstCellRow;\n\n\treturn { mergeWidth, mergeHeight };\n}\n\nfunction getMaxOffset( tableCell: Element, start: number, currentMaxOffset: number, which: string ) {\n\tconst dimensionValue = parseInt( tableCell.getAttribute( which ) as string || '1' );\n\n\treturn Math.max( currentMaxOffset, start + dimensionValue );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/selectrowcommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport type { Range, Element } from 'ckeditor5/src/engine';\nimport type TableUtils from '../tableutils';\n\n/**\n * The select row command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'selectTableRow'` editor command.\n *\n * To select the rows containing the selected cells, execute the command:\n *\n * ```ts\n * editor.execute( 'selectTableRow' );\n * ```\n */\nexport default class SelectRowCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\t// It does not affect data so should be enabled in read-only mode.\n\t\tthis.affectsData = false;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( this.editor.model.document.selection );\n\n\t\tthis.isEnabled = selectedCells.length > 0;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tconst model = this.editor.model;\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst referenceCells = tableUtils.getSelectionAffectedTableCells( model.document.selection );\n\t\tconst rowIndexes = tableUtils.getRowIndexes( referenceCells );\n\n\t\tconst table = referenceCells[ 0 ].findAncestor( 'table' )!;\n\t\tconst rangesToSelect: Array<Range> = [];\n\n\t\tfor ( let rowIndex = rowIndexes.first; rowIndex <= rowIndexes.last; rowIndex++ ) {\n\t\t\tfor ( const cell of ( table.getChild( rowIndex ) as Element ).getChildren() ) {\n\t\t\t\trangesToSelect.push( model.createRangeOn( cell ) );\n\t\t\t}\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\twriter.setSelection( rangesToSelect );\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/commands/selectcolumncommand\n */\n\nimport { Command, type Editor } from 'ckeditor5/src/core';\nimport type { Range } from 'ckeditor5/src/engine';\nimport type TableUtils from '../tableutils';\n\nimport TableWalker from '../tablewalker';\n\n/**\n * The select column command.\n *\n * The command is registered by {@link module:table/tableediting~TableEditing} as the `'selectTableColumn'` editor command.\n *\n * To select the columns containing the selected cells, execute the command:\n *\n * ```ts\n * editor.execute( 'selectTableColumn' );\n * ```\n */\nexport default class SelectColumnCommand extends Command {\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\t// It does not affect data so should be enabled in read-only mode.\n\t\tthis.affectsData = false;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override refresh(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst selectedCells = tableUtils.getSelectionAffectedTableCells( this.editor.model.document.selection );\n\n\t\tthis.isEnabled = selectedCells.length > 0;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override execute(): void {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst model = this.editor.model;\n\t\tconst referenceCells = tableUtils.getSelectionAffectedTableCells( model.document.selection );\n\t\tconst firstCell = referenceCells[ 0 ];\n\t\tconst lastCell = referenceCells.pop()!;\n\t\tconst table = firstCell.findAncestor( 'table' )!;\n\n\t\tconst startLocation = tableUtils.getCellLocation( firstCell );\n\t\tconst endLocation = tableUtils.getCellLocation( lastCell );\n\n\t\tconst startColumn = Math.min( startLocation.column, endLocation.column );\n\t\tconst endColumn = Math.max( startLocation.column, endLocation.column );\n\n\t\tconst rangesToSelect: Array<Range> = [];\n\n\t\tfor ( const cellInfo of new TableWalker( table, { startColumn, endColumn } ) ) {\n\t\t\trangesToSelect.push( model.createRangeOn( cellInfo.cell ) );\n\t\t}\n\n\t\tmodel.change( writer => {\n\t\t\twriter.setSelection( rangesToSelect );\n\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/converters/table-layout-post-fixer\n */\n\nimport type {\n\tElement,\n\tModel,\n\tWriter,\n\tDiffItem,\n\tDiffItemAttribute\n} from 'ckeditor5/src/engine';\n\nimport TableWalker from './../tablewalker';\nimport { createEmptyTableCell, updateNumericAttribute } from '../utils/common';\n\n/**\n * Injects a table layout post-fixer into the model.\n *\n * The role of the table layout post-fixer is to ensure that the table rows have the correct structure\n * after a {@link module:engine/model/model~Model#change `change()`} block was executed.\n *\n * The correct structure means that:\n *\n * * All table rows have the same size.\n * * None of the table cells extend vertically beyond their section (either header or body).\n * * A table cell has always at least one element as a child.\n *\n * If the table structure is not correct, the post-fixer will automatically correct it in two steps:\n *\n * 1. It will clip table cells that extend beyond their section.\n * 2. It will add empty table cells to the rows that are narrower than the widest table row.\n *\n * ## Clipping overlapping table cells\n *\n * Such situation may occur when pasting a table (or a part of a table) to the editor from external sources.\n *\n * For example, see the following table which has a cell (FOO) with the rowspan attribute (2):\n *\n * ```xml\n * <table headingRows=\"1\">\n * <tableRow>\n * <tableCell rowspan=\"2\"><paragraph>FOO</paragraph></tableCell>\n * <tableCell colspan=\"2\"><paragraph>BAR</paragraph></tableCell>\n * </tableRow>\n * <tableRow>\n * <tableCell><paragraph>BAZ</paragraph></tableCell>\n * <tableCell><paragraph>XYZ</paragraph></tableCell>\n * </tableRow>\n * </table>\n * ```\n *\n * It will be rendered in the view as:\n *\n * ```xml\n * <table>\n * <thead>\n * <tr>\n * <td rowspan=\"2\">FOO</td>\n * <td colspan=\"2\">BAR</td>\n * </tr>\n * </thead>\n * <tbody>\n * <tr>\n * <td>BAZ</td>\n * <td>XYZ</td>\n * </tr>\n * </tbody>\n * </table>\n * ```\n *\n * In the above example the table will be rendered as a table with two rows: one in the header and second one in the body.\n * The table cell (FOO) cannot span over multiple rows as it would extend from the header to the body section.\n * The `rowspan` attribute must be changed to (1). The value (1) is the default value of the `rowspan` attribute\n * so the `rowspan` attribute will be removed from the model.\n *\n * The table cell with BAZ in the content will be in the first column of the table.\n *\n * ## Adding missing table cells\n *\n * The table post-fixer will insert empty table cells to equalize table row sizes (the number of columns).\n * The size of a table row is calculated by counting column spans of table cells, both horizontal (from the same row) and\n * vertical (from the rows above).\n *\n * In the above example, the table row in the body section of the table is narrower then the row from the header: it has two cells\n * with the default colspan (1). The header row has one cell with colspan (1) and the second with colspan (2).\n * The table cell (FOO) does not extend beyond the head section (and as such will be fixed in the first step of this post-fixer).\n * The post-fixer will add a missing table cell to the row in the body section of the table.\n *\n * The table from the above example will be fixed and rendered to the view as below:\n *\n * ```xml\n * <table>\n * <thead>\n * <tr>\n * <td rowspan=\"2\">FOO</td>\n * <td colspan=\"2\">BAR</td>\n * </tr>\n * </thead>\n * <tbody>\n * <tr>\n * <td>BAZ</td>\n * <td>XYZ</td>\n * </tr>\n * </tbody>\n * </table>\n * ```\n *\n * ## Collaboration and undo - Expectations vs post-fixer results\n *\n * The table post-fixer only ensures proper structure without a deeper analysis of the nature of the change. As such, it might lead\n * to a structure which was not intended by the user. In particular, it will also fix undo steps (in conjunction with collaboration)\n * in which the editor content might not return to the original state.\n *\n * This will usually happen when one or more users change the size of the table.\n *\n * As an example see the table below:\n *\n * ```xml\n * <table>\n * <tbody>\n * <tr>\n * <td>11</td>\n * <td>12</td>\n * </tr>\n * <tr>\n * <td>21</td>\n * <td>22</td>\n * </tr>\n * </tbody>\n * </table>\n * ```\n *\n * and the user actions:\n *\n * 1. Both users have a table with two rows and two columns.\n * 2. User A adds a column at the end of the table. This will insert empty table cells to two rows.\n * 3. User B adds a row at the end of the table. This will insert a row with two empty table cells.\n * 4. Both users will have a table as below:\n *\n * ```xml\n * <table>\n * <tbody>\n * <tr>\n * <td>11</td>\n * <td>12</td>\n * <td>(empty, inserted by A)</td>\n * </tr>\n * <tr>\n * <td>21</td>\n * <td>22</td>\n * <td>(empty, inserted by A)</td>\n * </tr>\n * <tr>\n * <td>(empty, inserted by B)</td>\n * <td>(empty, inserted by B)</td>\n * </tr>\n * </tbody>\n * </table>\n * ```\n *\n * The last row is shorter then others so the table post-fixer will add an empty row to the last row:\n *\n * ```xml\n * <table>\n * <tbody>\n * <tr>\n * <td>11</td>\n * <td>12</td>\n * <td>(empty, inserted by A)</td>\n * </tr>\n * <tr>\n * <td>21</td>\n * <td>22</td>\n * <td>(empty, inserted by A)</td>\n * </tr>\n * <tr>\n * <td>(empty, inserted by B)</td>\n * <td>(empty, inserted by B)</td>\n * <td>(empty, inserted by the post-fixer)</td>\n * </tr>\n * </tbody>\n * </table>\n * ```\n *\n * Unfortunately undo does not know the nature of the changes and depending on which user applies the post-fixer changes, undoing them\n * might lead to a broken table. If User B undoes inserting the column to the table, the undo engine will undo only the operations of\n * inserting empty cells to rows from the initial table state (row 1 and 2) but the cell in the post-fixed row will remain:\n *\n * ```xml\n * <table>\n * <tbody>\n * <tr>\n * <td>11</td>\n * <td>12</td>\n * </tr>\n * <tr>\n * <td>21</td>\n * <td>22</td>\n * </tr>\n * <tr>\n * <td>(empty, inserted by B)</td>\n * <td>(empty, inserted by B)</td>\n * <td>(empty, inserted by a post-fixer)</td>\n * </tr>\n * </tbody>\n * </table>\n * ```\n *\n * After undo, the table post-fixer will detect that two rows are shorter than others and will fix the table to:\n *\n * ```xml\n * <table>\n * <tbody>\n * <tr>\n * <td>11</td>\n * <td>12</td>\n * <td>(empty, inserted by a post-fixer after undo)</td>\n * </tr>\n * <tr>\n * <td>21</td>\n * <td>22</td>\n * <td>(empty, inserted by a post-fixer after undo)</td>\n * </tr>\n * <tr>\n * <td>(empty, inserted by B)</td>\n * <td>(empty, inserted by B)</td>\n * <td>(empty, inserted by a post-fixer)</td>\n * </tr>\n * </tbody>\n * </table>\n * ```\n */\nexport default function injectTableLayoutPostFixer( model: Model ): void {\n\tmodel.document.registerPostFixer( writer => tableLayoutPostFixer( writer, model ) );\n}\n\n/**\n * The table layout post-fixer.\n */\nfunction tableLayoutPostFixer( writer: Writer, model: Model ) {\n\tconst changes = model.document.differ.getChanges();\n\n\tlet wasFixed = false;\n\n\t// Do not analyze the same table more then once - may happen for multiple changes in the same table.\n\tconst analyzedTables = new Set();\n\n\tfor ( const entry of changes ) {\n\t\tlet table: Element | null = null;\n\n\t\tif ( entry.type == 'insert' && entry.name == 'table' ) {\n\t\t\ttable = entry.position.nodeAfter as Element;\n\t\t}\n\n\t\t// Fix table on adding/removing table cells and rows.\n\t\tif ( ( entry.type == 'insert' || entry.type == 'remove' ) && ( entry.name == 'tableRow' || entry.name == 'tableCell' ) ) {\n\t\t\ttable = entry.position.findAncestor( 'table' );\n\t\t}\n\n\t\t// Fix table on any table's attribute change - including attributes of table cells.\n\t\tif ( isTableAttributeEntry( entry ) ) {\n\t\t\ttable = entry.range.start.findAncestor( 'table' );\n\t\t}\n\n\t\tif ( table && !analyzedTables.has( table ) ) {\n\t\t\t// Step 1: correct rowspans of table cells if necessary.\n\t\t\t// The wasFixed flag should be true if any of tables in batch was fixed - might be more then one.\n\t\t\twasFixed = fixTableCellsRowspan( table, writer ) || wasFixed;\n\t\t\t// Step 2: fix table rows sizes.\n\t\t\twasFixed = fixTableRowsSizes( table, writer ) || wasFixed;\n\n\t\t\tanalyzedTables.add( table );\n\t\t}\n\t}\n\n\treturn wasFixed;\n}\n\n/**\n * Fixes the invalid value of the `rowspan` attribute because a table cell cannot vertically extend beyond the table section it belongs to.\n *\n * @returns Returns `true` if the table was fixed.\n */\nfunction fixTableCellsRowspan( table: Element, writer: Writer ) {\n\tlet wasFixed = false;\n\n\tconst cellsToTrim = findCellsToTrim( table );\n\n\tif ( cellsToTrim.length ) {\n\t\t// @if CK_DEBUG_TABLE // console.log( `Post-fixing table: trimming cells row-spans (${ cellsToTrim.length }).` );\n\n\t\twasFixed = true;\n\n\t\tfor ( const data of cellsToTrim ) {\n\t\t\tupdateNumericAttribute( 'rowspan', data.rowspan, data.cell, writer, 1 );\n\t\t}\n\t}\n\n\treturn wasFixed;\n}\n\n/**\n * Makes all table rows in a table the same size.\n *\n * @returns Returns `true` if the table was fixed.\n */\nfunction fixTableRowsSizes( table: Element, writer: Writer ) {\n\tlet wasFixed = false;\n\n\tconst childrenLengths = getChildrenLengths( table );\n\tconst rowsToRemove = [];\n\n\t// Find empty rows.\n\tfor ( const [ rowIndex, size ] of childrenLengths.entries() ) {\n\t\t// Ignore all non-row models.\n\t\tif ( !size && table.getChild( rowIndex )!.is( 'element', 'tableRow' ) ) {\n\t\t\trowsToRemove.push( rowIndex );\n\t\t}\n\t}\n\n\t// Remove empty rows.\n\tif ( rowsToRemove.length ) {\n\t\t// @if CK_DEBUG_TABLE // console.log( `Post-fixing table: remove empty rows (${ rowsToRemove.length }).` );\n\n\t\twasFixed = true;\n\n\t\tfor ( const rowIndex of rowsToRemove.reverse() ) {\n\t\t\twriter.remove( table.getChild( rowIndex )! );\n\t\t\tchildrenLengths.splice( rowIndex, 1 );\n\t\t}\n\t}\n\n\t// Filter out everything that's not a table row.\n\tconst rowsLengths = childrenLengths.filter( ( row, rowIndex ) => table.getChild( rowIndex )!.is( 'element', 'tableRow' ) );\n\n\t// Verify if all the rows have the same number of columns.\n\tconst tableSize = rowsLengths[ 0 ];\n\tconst isValid = rowsLengths.every( length => length === tableSize );\n\n\tif ( !isValid ) {\n\t\t// @if CK_DEBUG_TABLE // console.log( 'Post-fixing table: adding missing cells.' );\n\n\t\t// Find the maximum number of columns.\n\t\tconst maxColumns = rowsLengths.reduce( ( prev, current ) => current > prev ? current : prev, 0 );\n\n\t\tfor ( const [ rowIndex, size ] of rowsLengths.entries() ) {\n\t\t\tconst columnsToInsert = maxColumns - size;\n\n\t\t\tif ( columnsToInsert ) {\n\t\t\t\tfor ( let i = 0; i < columnsToInsert; i++ ) {\n\t\t\t\t\tcreateEmptyTableCell( writer, writer.createPositionAt( table.getChild( rowIndex )!, 'end' ) );\n\t\t\t\t}\n\n\t\t\t\twasFixed = true;\n\t\t\t}\n\t\t}\n\t}\n\n\treturn wasFixed;\n}\n\n/**\n * Searches for table cells that extend beyond the table section to which they belong to. It will return an array of objects\n * that stores table cells to be trimmed and the correct value of the `rowspan` attribute to set.\n */\nfunction findCellsToTrim( table: Element ) {\n\tconst headingRows = parseInt( table.getAttribute( 'headingRows' ) as string || '0' );\n\tconst maxRows = Array.from( table.getChildren() )\n\t\t.reduce( ( count, row ) => row.is( 'element', 'tableRow' ) ? count + 1 : count, 0 );\n\n\tconst cellsToTrim = [];\n\n\tfor ( const { row, cell, cellHeight } of new TableWalker( table ) ) {\n\t\t// Skip cells that do not expand over its row.\n\t\tif ( cellHeight < 2 ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst isInHeader = row < headingRows;\n\n\t\t// Row limit is either end of header section or whole table as table body is after the header.\n\t\tconst rowLimit = isInHeader ? headingRows : maxRows;\n\n\t\t// If table cell expands over its limit reduce it height to proper value.\n\t\tif ( row + cellHeight > rowLimit ) {\n\t\t\tconst newRowspan = rowLimit - row;\n\n\t\t\tcellsToTrim.push( { cell, rowspan: newRowspan } );\n\t\t}\n\t}\n\n\treturn cellsToTrim;\n}\n\n/**\n * Returns an array with lengths of rows assigned to the corresponding row index.\n */\nfunction getChildrenLengths( table: Element ) {\n\t// TableWalker will not provide items for the empty rows, we need to pre-fill this array.\n\tconst lengths = new Array( table.childCount ).fill( 0 );\n\n\tfor ( const { rowIndex } of new TableWalker( table, { includeAllSlots: true } ) ) {\n\t\tlengths[ rowIndex ]++;\n\t}\n\n\treturn lengths;\n}\n\n/**\n * Checks if the differ entry for an attribute change is one of the table's attributes.\n */\nfunction isTableAttributeEntry( entry: DiffItem ): entry is DiffItemAttribute {\n\tif ( entry.type !== 'attribute' ) {\n\t\treturn false;\n\t}\n\n\tconst key = entry.attributeKey;\n\n\treturn key === 'headingRows' || key === 'colspan' || key === 'rowspan';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/converters/table-cell-paragraph-post-fixer\n */\n\nimport type { Model, Writer, Element, DiffItemInsert, DiffItemRemove } from 'ckeditor5/src/engine';\n\n/**\n * Injects a table cell post-fixer into the model which inserts a `paragraph` element into empty table cells.\n *\n * A table cell must contain at least one block element as a child. An empty table cell will have an empty `paragraph` as a child.\n *\n * ```xml\n * <table>\n * <tableRow>\n * <tableCell></tableCell>\n * </tableRow>\n * </table>\n * ```\n *\n * Will be fixed to:\n *\n * ```xml\n * <table>\n * <tableRow>\n * <tableCell><paragraph></paragraph></tableCell>\n * </tableRow>\n * </table>\n * ```\n */\nexport default function injectTableCellParagraphPostFixer( model: Model ): void {\n\tmodel.document.registerPostFixer( writer => tableCellContentsPostFixer( writer, model ) );\n}\n\n/**\n * The table cell contents post-fixer.\n */\nfunction tableCellContentsPostFixer( writer: Writer, model: Model ) {\n\tconst changes = model.document.differ.getChanges();\n\n\tlet wasFixed = false;\n\n\tfor ( const entry of changes ) {\n\t\tif ( entry.type == 'insert' && entry.name == 'table' ) {\n\t\t\twasFixed = fixTable( entry.position.nodeAfter as Element, writer ) || wasFixed;\n\t\t}\n\n\t\tif ( entry.type == 'insert' && entry.name == 'tableRow' ) {\n\t\t\twasFixed = fixTableRow( entry.position.nodeAfter as Element, writer ) || wasFixed;\n\t\t}\n\n\t\tif ( entry.type == 'insert' && entry.name == 'tableCell' ) {\n\t\t\twasFixed = fixTableCellContent( entry.position.nodeAfter as Element, writer ) || wasFixed;\n\t\t}\n\n\t\tif ( ( entry.type == 'remove' || entry.type == 'insert' ) && checkTableCellChange( entry ) ) {\n\t\t\twasFixed = fixTableCellContent( entry.position.parent as Element, writer ) || wasFixed;\n\t\t}\n\t}\n\n\treturn wasFixed;\n}\n\n/**\n * Fixes all table cells in a table.\n */\nfunction fixTable( table: Element, writer: Writer ) {\n\tlet wasFixed = false;\n\n\tfor ( const row of table.getChildren() ) {\n\t\tif ( row.is( 'element', 'tableRow' ) ) {\n\t\t\twasFixed = fixTableRow( row, writer ) || wasFixed;\n\t\t}\n\t}\n\n\treturn wasFixed;\n}\n\n/**\n * Fixes all table cells in a table row.\n */\nfunction fixTableRow( tableRow: Element, writer: Writer ) {\n\tlet wasFixed = false;\n\n\tfor ( const tableCell of tableRow.getChildren() as IterableIterator<Element> ) {\n\t\twasFixed = fixTableCellContent( tableCell, writer ) || wasFixed;\n\t}\n\n\treturn wasFixed;\n}\n\n/**\n * Fixes all table cell content by:\n * - Adding a paragraph to a table cell without any child.\n * - Wrapping direct $text in a `<paragraph>`.\n */\nfunction fixTableCellContent( tableCell: Element, writer: Writer ) {\n\t// Insert paragraph to an empty table cell.\n\tif ( tableCell.childCount == 0 ) {\n\t\t// @if CK_DEBUG_TABLE // console.log( 'Post-fixing table: insert paragraph in empty cell.' );\n\n\t\twriter.insertElement( 'paragraph', tableCell );\n\n\t\treturn true;\n\t}\n\n\t// Check table cell children for directly placed text nodes.\n\t// Temporary solution. See https://github.com/ckeditor/ckeditor5/issues/1464.\n\tconst textNodes = Array.from( tableCell.getChildren() ).filter( child => child.is( '$text' ) );\n\n\t// @if CK_DEBUG_TABLE // textNodes.length && console.log( 'Post-fixing table: wrap cell content with paragraph.' );\n\n\tfor ( const child of textNodes ) {\n\t\twriter.wrap( writer.createRangeOn( child ), 'paragraph' );\n\t}\n\n\t// Return true when there were text nodes to fix.\n\treturn !!textNodes.length;\n}\n\n/**\n * Checks if a differ change should fix the table cell. This happens on:\n * - Removing content from the table cell (i.e. `tableCell` can be left empty).\n * - Adding a text node directly into a table cell.\n */\nfunction checkTableCellChange( entry: DiffItemInsert | DiffItemRemove ) {\n\tif ( !entry.position.parent.is( 'element', 'tableCell' ) ) {\n\t\treturn false;\n\t}\n\n\treturn entry.type == 'insert' && entry.name == '$text' || entry.type == 'remove';\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/converters/table-cell-refresh-handler\n */\n\nimport type {\n\tEditingController,\n\tElement,\n\tMapper,\n\tModel\n} from 'ckeditor5/src/engine';\n\nimport { isSingleParagraphWithoutAttributes } from './downcast';\n\n/**\n * A table cell refresh handler which marks the table cell in the differ to have it re-rendered.\n *\n * Model `paragraph` inside a table cell can be rendered as `<span>` or `<p>`. It is rendered as `<span>` if this is the only block\n * element in that table cell and it does not have any attributes. It is rendered as `<p>` otherwise.\n *\n * When table cell content changes, for example a second `paragraph` element is added, we need to ensure that the first `paragraph` is\n * re-rendered so it changes from `<span>` to `<p>`. The easiest way to do it is to re-render the entire table cell.\n */\nexport default function tableCellRefreshHandler( model: Model, editing: EditingController ): void {\n\tconst differ = model.document.differ;\n\n\t// Stores cells to be refreshed, so the table cell will be refreshed once for multiple changes.\n\tconst cellsToCheck = new Set();\n\n\tfor ( const change of differ.getChanges() ) {\n\t\tconst parent = change.type == 'attribute' ? change.range.start.parent : change.position.parent;\n\n\t\tif ( parent.is( 'element', 'tableCell' ) ) {\n\t\t\tcellsToCheck.add( parent );\n\t\t}\n\t}\n\n\tfor ( const tableCell of cellsToCheck.values() as Iterable<Element> ) {\n\t\tconst paragraphsToRefresh = Array.from( tableCell.getChildren() )\n\t\t\t.filter( child => shouldRefresh( child as Element, editing.mapper ) );\n\n\t\tfor ( const paragraph of paragraphsToRefresh ) {\n\t\t\tediting.reconvertItem( paragraph );\n\t\t}\n\t}\n}\n\n/**\n * Check if given model element needs refreshing.\n */\nfunction shouldRefresh( child: Element, mapper: Mapper ) {\n\tif ( !child.is( 'element', 'paragraph' ) ) {\n\t\treturn false;\n\t}\n\n\tconst viewElement = mapper.toViewElement( child );\n\n\tif ( !viewElement ) {\n\t\treturn false;\n\t}\n\n\treturn isSingleParagraphWithoutAttributes( child ) !== viewElement.is( 'element', 'span' );\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./tableediting.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tableediting\n */\n\nimport { Plugin, type Editor } from 'ckeditor5/src/core';\nimport type { PositionOffset, ViewElement, SlotFilter } from 'ckeditor5/src/engine';\n\nimport upcastTable, { ensureParagraphInTableCell, skipEmptyTableRow, upcastTableFigure } from './converters/upcasttable';\nimport { convertParagraphInTableCell, downcastCell, downcastRow, downcastTable } from './converters/downcast';\n\nimport InsertTableCommand from './commands/inserttablecommand';\nimport InsertRowCommand from './commands/insertrowcommand';\nimport InsertColumnCommand from './commands/insertcolumncommand';\nimport SplitCellCommand from './commands/splitcellcommand';\nimport MergeCellCommand from './commands/mergecellcommand';\nimport RemoveRowCommand from './commands/removerowcommand';\nimport RemoveColumnCommand from './commands/removecolumncommand';\nimport SetHeaderRowCommand from './commands/setheaderrowcommand';\nimport SetHeaderColumnCommand from './commands/setheadercolumncommand';\nimport MergeCellsCommand from './commands/mergecellscommand';\nimport SelectRowCommand from './commands/selectrowcommand';\nimport SelectColumnCommand from './commands/selectcolumncommand';\nimport TableUtils from '../src/tableutils';\n\nimport injectTableLayoutPostFixer from './converters/table-layout-post-fixer';\nimport injectTableCellParagraphPostFixer from './converters/table-cell-paragraph-post-fixer';\n\nimport tableHeadingsRefreshHandler from './converters/table-headings-refresh-handler';\nimport tableCellRefreshHandler from './converters/table-cell-refresh-handler';\n\nimport '../theme/tableediting.css';\n\n/**\n * The table editing feature.\n */\nexport default class TableEditing extends Plugin {\n\t/**\n\t * Handlers for creating additional slots in the table.\n\t */\n\tprivate _additionalSlots: Array<AdditionalSlot>;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TableEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ TableUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( editor: Editor ) {\n\t\tsuper( editor );\n\n\t\tthis._additionalSlots = [];\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst schema = model.schema;\n\t\tconst conversion = editor.conversion;\n\t\tconst tableUtils = editor.plugins.get( TableUtils );\n\n\t\tschema.register( 'table', {\n\t\t\tinheritAllFrom: '$blockObject',\n\t\t\tallowAttributes: [ 'headingRows', 'headingColumns' ]\n\t\t} );\n\n\t\tschema.register( 'tableRow', {\n\t\t\tallowIn: 'table',\n\t\t\tisLimit: true\n\t\t} );\n\n\t\tschema.register( 'tableCell', {\n\t\t\tallowContentOf: '$container',\n\t\t\tallowIn: 'tableRow',\n\t\t\tallowAttributes: [ 'colspan', 'rowspan' ],\n\t\t\tisLimit: true,\n\t\t\tisSelectable: true\n\t\t} );\n\n\t\t// Figure conversion.\n\t\tconversion.for( 'upcast' ).add( upcastTableFigure() );\n\n\t\t// Table conversion.\n\t\tconversion.for( 'upcast' ).add( upcastTable() );\n\n\t\tconversion.for( 'editingDowncast' ).elementToStructure( {\n\t\t\tmodel: {\n\t\t\t\tname: 'table',\n\t\t\t\tattributes: [ 'headingRows' ]\n\t\t\t},\n\t\t\tview: downcastTable( tableUtils, {\n\t\t\t\tasWidget: true,\n\t\t\t\tadditionalSlots: this._additionalSlots\n\t\t\t} )\n\t\t} );\n\t\tconversion.for( 'dataDowncast' ).elementToStructure( {\n\t\t\tmodel: {\n\t\t\t\tname: 'table',\n\t\t\t\tattributes: [ 'headingRows' ]\n\t\t\t},\n\t\t\tview: downcastTable( tableUtils, {\n\t\t\t\tadditionalSlots: this._additionalSlots\n\t\t\t} )\n\t\t} );\n\n\t\t// Table row conversion.\n\t\tconversion.for( 'upcast' ).elementToElement( { model: 'tableRow', view: 'tr' } );\n\t\tconversion.for( 'upcast' ).add( skipEmptyTableRow() );\n\n\t\tconversion.for( 'downcast' ).elementToElement( {\n\t\t\tmodel: 'tableRow',\n\t\t\tview: downcastRow()\n\t\t} );\n\n\t\t// Table cell conversion.\n\t\tconversion.for( 'upcast' ).elementToElement( { model: 'tableCell', view: 'td' } );\n\t\tconversion.for( 'upcast' ).elementToElement( { model: 'tableCell', view: 'th' } );\n\t\tconversion.for( 'upcast' ).add( ensureParagraphInTableCell( 'td' ) );\n\t\tconversion.for( 'upcast' ).add( ensureParagraphInTableCell( 'th' ) );\n\n\t\tconversion.for( 'editingDowncast' ).elementToElement( {\n\t\t\tmodel: 'tableCell',\n\t\t\tview: downcastCell( { asWidget: true } )\n\t\t} );\n\t\tconversion.for( 'dataDowncast' ).elementToElement( {\n\t\t\tmodel: 'tableCell',\n\t\t\tview: downcastCell()\n\t\t} );\n\n\t\t// Duplicates code - needed to properly refresh paragraph inside a table cell.\n\t\tconversion.for( 'editingDowncast' ).elementToElement( {\n\t\t\tmodel: 'paragraph',\n\t\t\tview: convertParagraphInTableCell( { asWidget: true } ),\n\t\t\tconverterPriority: 'high'\n\t\t} );\n\t\tconversion.for( 'dataDowncast' ).elementToElement( {\n\t\t\tmodel: 'paragraph',\n\t\t\tview: convertParagraphInTableCell(),\n\t\t\tconverterPriority: 'high'\n\t\t} );\n\n\t\t// Table attributes conversion.\n\t\tconversion.for( 'downcast' ).attributeToAttribute( { model: 'colspan', view: 'colspan' } );\n\t\tconversion.for( 'upcast' ).attributeToAttribute( {\n\t\t\tmodel: { key: 'colspan', value: upcastCellSpan( 'colspan' ) },\n\t\t\tview: 'colspan'\n\t\t} );\n\n\t\tconversion.for( 'downcast' ).attributeToAttribute( { model: 'rowspan', view: 'rowspan' } );\n\t\tconversion.for( 'upcast' ).attributeToAttribute( {\n\t\t\tmodel: { key: 'rowspan', value: upcastCellSpan( 'rowspan' ) },\n\t\t\tview: 'rowspan'\n\t\t} );\n\n\t\t// Define the config.\n\t\teditor.config.define( 'table.defaultHeadings.rows', 0 );\n\t\teditor.config.define( 'table.defaultHeadings.columns', 0 );\n\n\t\t// Define all the commands.\n\t\teditor.commands.add( 'insertTable', new InsertTableCommand( editor ) );\n\t\teditor.commands.add( 'insertTableRowAbove', new InsertRowCommand( editor, { order: 'above' } ) );\n\t\teditor.commands.add( 'insertTableRowBelow', new InsertRowCommand( editor, { order: 'below' } ) );\n\t\teditor.commands.add( 'insertTableColumnLeft', new InsertColumnCommand( editor, { order: 'left' } ) );\n\t\teditor.commands.add( 'insertTableColumnRight', new InsertColumnCommand( editor, { order: 'right' } ) );\n\n\t\teditor.commands.add( 'removeTableRow', new RemoveRowCommand( editor ) );\n\t\teditor.commands.add( 'removeTableColumn', new RemoveColumnCommand( editor ) );\n\n\t\teditor.commands.add( 'splitTableCellVertically', new SplitCellCommand( editor, { direction: 'vertically' } ) );\n\t\teditor.commands.add( 'splitTableCellHorizontally', new SplitCellCommand( editor, { direction: 'horizontally' } ) );\n\n\t\teditor.commands.add( 'mergeTableCells', new MergeCellsCommand( editor ) );\n\n\t\teditor.commands.add( 'mergeTableCellRight', new MergeCellCommand( editor, { direction: 'right' } ) );\n\t\teditor.commands.add( 'mergeTableCellLeft', new MergeCellCommand( editor, { direction: 'left' } ) );\n\t\teditor.commands.add( 'mergeTableCellDown', new MergeCellCommand( editor, { direction: 'down' } ) );\n\t\teditor.commands.add( 'mergeTableCellUp', new MergeCellCommand( editor, { direction: 'up' } ) );\n\n\t\teditor.commands.add( 'setTableColumnHeader', new SetHeaderColumnCommand( editor ) );\n\t\teditor.commands.add( 'setTableRowHeader', new SetHeaderRowCommand( editor ) );\n\n\t\teditor.commands.add( 'selectTableRow', new SelectRowCommand( editor ) );\n\t\teditor.commands.add( 'selectTableColumn', new SelectColumnCommand( editor ) );\n\n\t\tinjectTableLayoutPostFixer( model );\n\t\tinjectTableCellParagraphPostFixer( model );\n\n\t\tthis.listenTo( model.document, 'change:data', () => {\n\t\t\ttableHeadingsRefreshHandler( model, editor.editing );\n\t\t\ttableCellRefreshHandler( model, editor.editing );\n\t\t} );\n\t}\n\n\t/**\n\t * Registers downcast handler for the additional table slot.\n\t */\n\tpublic registerAdditionalSlot( slotHandler: AdditionalSlot ): void {\n\t\tthis._additionalSlots.push( slotHandler );\n\t}\n}\n\n/**\n * Returns fixed colspan and rowspan attrbutes values.\n *\n * @param type colspan or rowspan.\n * @returns conversion value function.\n */\nfunction upcastCellSpan( type: string ) {\n\treturn ( cell: ViewElement ) => {\n\t\tconst span = parseInt( cell.getAttribute( type )! );\n\n\t\tif ( Number.isNaN( span ) || span <= 0 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn span;\n\t};\n}\n\n/**\n * By default, only the `tableRow` elements from the `table` model are downcast inside the `<table>` and\n * all other elements are pushed outside the table. This handler allows creating additional slots inside\n * the table for other elements.\n *\n * Take this model as an example:\n *\n * ```xml\n * <table>\n * <tableRow>...</tableRow>\n * <tableRow>...</tableRow>\n * <tableColumnGroup>...</tableColumnGroup>\n * </table>\n * ```\n *\n * By default, downcasting result will be:\n *\n * ```xml\n * <table>\n * <tbody>\n * <tr>...</tr>\n * <tr>...</tr>\n * </tbody>\n * </table>\n * <colgroup>...</colgroup>\n * ```\n *\n * To allow the `tableColumnGroup` element at the end of the table, use the following configuration:\n *\n * ```ts\n * const additionalSlot = {\n * filter: element => element.is( 'element', 'tableColumnGroup' ),\n * positionOffset: 'end'\n * }\n * ```\n *\n * Now, the downcast result will be:\n *\n * ```xml\n * <table>\n * <tbody>\n * <tr>...</tr>\n * <tr>...</tr>\n * </tbody>\n * <colgroup>...</colgroup>\n * </table>\n * ```\n */\nexport interface AdditionalSlot {\n\n\t/**\n\t * Filter for elements that should be placed inside given slot.\n\t */\n\tfilter: SlotFilter;\n\n\t/**\n\t * Position of the slot within the table.\n\t */\n\tpositionOffset: PositionOffset;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/converters/table-headings-refresh-handler\n */\n\nimport type {\n\tEditingController,\n\tElement,\n\tModel\n} from 'ckeditor5/src/engine';\n\nimport TableWalker from '../tablewalker';\n\n/**\n * A table headings refresh handler which marks the table cells or rows in the differ to have it re-rendered\n * if the headings attribute changed.\n *\n * Table heading rows and heading columns are represented in the model by a `headingRows` and `headingColumns` attributes.\n *\n * When table headings attribute changes, all the cells/rows are marked to re-render to change between `<td>` and `<th>`.\n */\nexport default function tableHeadingsRefreshHandler( model: Model, editing: EditingController ): void {\n\tconst differ = model.document.differ;\n\n\tfor ( const change of differ.getChanges() ) {\n\t\tlet table;\n\t\tlet isRowChange = false;\n\n\t\tif ( change.type == 'attribute' ) {\n\t\t\tconst element = change.range.start.nodeAfter;\n\n\t\t\tif ( !element || !element.is( 'element', 'table' ) ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\tif ( change.attributeKey != 'headingRows' && change.attributeKey != 'headingColumns' ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\ttable = element;\n\t\t\tisRowChange = change.attributeKey == 'headingRows';\n\t\t} else if ( change.name == 'tableRow' || change.name == 'tableCell' ) {\n\t\t\ttable = change.position.findAncestor( 'table' );\n\t\t\tisRowChange = change.name == 'tableRow';\n\t\t}\n\n\t\tif ( !table ) {\n\t\t\tcontinue;\n\t\t}\n\n\t\tconst headingRows = table.getAttribute( 'headingRows' ) || 0;\n\t\tconst headingColumns = table.getAttribute( 'headingColumns' ) || 0;\n\n\t\tconst tableWalker = new TableWalker( table );\n\n\t\tfor ( const tableSlot of tableWalker ) {\n\t\t\tconst isHeading = tableSlot.row < headingRows || tableSlot.column < headingColumns;\n\t\t\tconst expectedElementName = isHeading ? 'th' : 'td';\n\n\t\t\tconst viewElement = editing.mapper.toViewElement( tableSlot.cell );\n\n\t\t\tif ( viewElement && viewElement.is( 'element' ) && viewElement.name != expectedElementName ) {\n\t\t\t\tediting.reconvertItem( ( isRowChange ? tableSlot.cell.parent : tableSlot.cell ) as Element );\n\t\t\t}\n\t\t}\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./inserttable.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/ui/inserttableview\n */\n\nimport { View, ButtonView, addKeyboardHandlingForGrid, type ViewCollection } from 'ckeditor5/src/ui';\n\nimport { KeystrokeHandler, FocusTracker, type Locale } from 'ckeditor5/src/utils';\n\nimport './../../theme/inserttable.css';\n\n/**\n * The table size view.\n *\n * It renders a 10x10 grid to choose the inserted table size.\n */\nexport default class InsertTableView extends View {\n\t/**\n\t * A collection of table size box items.\n\t */\n\tpublic readonly items: ViewCollection<ButtonView>;\n\n\t/**\n\t * Listen to `keydown` events fired in this view's main element.\n\t */\n\tpublic readonly keystrokes: KeystrokeHandler;\n\n\t/**\n\t * Tracks information about the DOM focus in the grid.\n\t */\n\tpublic readonly focusTracker: FocusTracker;\n\n\t/**\n\t * The currently selected number of rows of the new table.\n\t *\n\t * @observable\n\t */\n\tdeclare public rows: number;\n\n\t/**\n\t * The currently selected number of columns of the new table.\n\t *\n\t * @observable\n\t */\n\tdeclare public columns: number;\n\n\t/**\n\t * The label text displayed under the boxes.\n\t *\n\t * @observable\n\t */\n\tdeclare public label: string;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tconstructor( locale: Locale ) {\n\t\tsuper( locale );\n\n\t\tconst bind = this.bindTemplate;\n\t\tthis.items = this._createGridCollection();\n\t\tthis.keystrokes = new KeystrokeHandler();\n\t\tthis.focusTracker = new FocusTracker();\n\t\tthis.set( 'rows', 0 );\n\t\tthis.set( 'columns', 0 );\n\t\tthis.bind( 'label' ).to( this, 'columns', this, 'rows', ( columns, rows ) => `${ rows } × ${ columns }` );\n\n\t\tthis.setTemplate( {\n\t\t\ttag: 'div',\n\t\t\tattributes: {\n\t\t\t\tclass: [ 'ck' ]\n\t\t\t},\n\n\t\t\tchildren: [\n\t\t\t\t{\n\t\t\t\t\ttag: 'div',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: [ 'ck-insert-table-dropdown__grid' ]\n\t\t\t\t\t},\n\t\t\t\t\ton: {\n\t\t\t\t\t\t'mouseover@.ck-insert-table-dropdown-grid-box': bind.to( 'boxover' )\n\t\t\t\t\t},\n\t\t\t\t\tchildren: this.items\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttag: 'div',\n\t\t\t\t\tattributes: {\n\t\t\t\t\t\tclass: [\n\t\t\t\t\t\t\t'ck',\n\t\t\t\t\t\t\t'ck-insert-table-dropdown__label'\n\t\t\t\t\t\t],\n\t\t\t\t\t\t'aria-hidden': true\n\t\t\t\t\t},\n\t\t\t\t\tchildren: [\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\ttext: bind.to( 'label' )\n\t\t\t\t\t\t}\n\t\t\t\t\t]\n\t\t\t\t}\n\t\t\t],\n\n\t\t\ton: {\n\t\t\t\tmousedown: bind.to( evt => {\n\t\t\t\t\tevt.preventDefault();\n\t\t\t\t} ),\n\n\t\t\t\tclick: bind.to( () => {\n\t\t\t\t\tthis.fire( 'execute' );\n\t\t\t\t} )\n\t\t\t}\n\t\t} );\n\n\t\t// #rows and #columns are set via changes to #focusTracker on mouse over.\n\t\tthis.on( 'boxover', ( evt, domEvt ) => {\n\t\t\tconst { row, column } = domEvt.target.dataset;\n\t\t\tthis.items.get( ( parseInt( row, 10 ) - 1 ) * 10 + ( parseInt( column, 10 ) - 1 ) )!.focus();\n\t\t} );\n\n\t\t// This allows the #rows and #columns to be updated when:\n\t\t// * the user navigates the grid using the keyboard,\n\t\t// * the user moves the mouse over grid items.\n\t\tthis.focusTracker.on( 'change:focusedElement', ( evt, name, focusedElement ) => {\n\t\t\tif ( !focusedElement ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst { row, column } = focusedElement.dataset;\n\n\t\t\t// As row & column indexes are zero-based transform it to number of selected rows & columns.\n\t\t\tthis.set( {\n\t\t\t\trows: parseInt( row ),\n\t\t\t\tcolumns: parseInt( column )\n\t\t\t} );\n\t\t} );\n\n\t\tthis.on( 'change:columns', () => this._highlightGridBoxes() );\n\t\tthis.on( 'change:rows', () => this._highlightGridBoxes() );\n\t}\n\n\tpublic override render(): void {\n\t\tsuper.render();\n\n\t\taddKeyboardHandlingForGrid( {\n\t\t\tkeystrokeHandler: this.keystrokes,\n\t\t\tfocusTracker: this.focusTracker,\n\t\t\tgridItems: this.items,\n\t\t\tnumberOfColumns: 10,\n\t\t\tuiLanguageDirection: this.locale && this.locale.uiLanguageDirection\n\t\t} );\n\n\t\tfor ( const item of this.items ) {\n\t\t\tthis.focusTracker.add( item.element! );\n\t\t}\n\n\t\tthis.keystrokes.listenTo( this.element! );\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic focus(): void {\n\t\tthis.items.get( 0 )!.focus();\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic focusLast(): void {\n\t\tthis.items.get( 0 )!.focus();\n\t}\n\n\t/**\n\t * Highlights grid boxes depending on rows and columns selected.\n\t */\n\tprivate _highlightGridBoxes() {\n\t\tconst rows = this.rows;\n\t\tconst columns = this.columns;\n\n\t\tthis.items.map( ( boxView, index ) => {\n\t\t\t// Translate box index to the row & column index.\n\t\t\tconst itemRow = Math.floor( index / 10 );\n\t\t\tconst itemColumn = index % 10;\n\n\t\t\t// Grid box is highlighted when its row & column index belongs to selected number of rows & columns.\n\t\t\tconst isOn = itemRow < rows && itemColumn < columns;\n\n\t\t\tboxView.set( 'isOn', isOn );\n\t\t} );\n\t}\n\n\t/**\n\t * Creates a new Button for the grid.\n\t *\n\t * @param locale The locale instance.\n\t * @param row Row number.\n\t * @param column Column number.\n\t * @param label The grid button label.\n\t */\n\tprivate _createGridButton( locale: Locale, row: number, column: number, label: string ): ButtonView {\n\t\tconst button = new ButtonView( locale );\n\n\t\tbutton.set( {\n\t\t\tlabel,\n\t\t\tclass: 'ck-insert-table-dropdown-grid-box'\n\t\t} );\n\n\t\tbutton.extendTemplate( {\n\t\t\tattributes: {\n\t\t\t\t'data-row': row,\n\t\t\t\t'data-column': column\n\t\t\t}\n\t\t} );\n\n\t\treturn button;\n\t}\n\n\t/**\n\t * @returns A view collection containing boxes to be placed in a table grid.\n\t */\n\tprivate _createGridCollection() {\n\t\tconst boxes = [];\n\n\t\t// Add grid boxes to table selection view.\n\t\tfor ( let index = 0; index < 100; index++ ) {\n\t\t\tconst row = Math.floor( index / 10 );\n\t\t\tconst column = index % 10;\n\t\t\tconst label = `${ row + 1 } × ${ column + 1 }`;\n\n\t\t\tboxes.push( this._createGridButton( this.locale!, row + 1, column + 1, label ) );\n\t\t}\n\n\t\treturn this.createCollection( boxes );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ui/bindings/addkeyboardhandlingforgrid\n */\n\nimport type { FocusTracker, KeystrokeHandler } from '@ckeditor/ckeditor5-utils';\nimport type { FocusableView } from '../focuscycler';\nimport type ViewCollection from '../viewcollection';\n\n/**\n * A helper that adds a keyboard navigation support (arrow up/down/left/right) for grids.\n *\n * @param options Configuration options.\n * @param options.keystrokeHandler Keystroke handler to register navigation with arrow keys.\n * @param options.focusTracker A focus tracker for grid elements.\n * @param options.gridItems A collection of grid items.\n * @param options.numberOfColumns Number of columns in the grid. Can be specified as a function that returns\n * the number (e.g. for responsive grids).\n * @param options.uiLanguageDirection String of ui language direction.\n */\nexport default function addKeyboardHandlingForGrid(\n\t{ keystrokeHandler, focusTracker, gridItems, numberOfColumns, uiLanguageDirection }: {\n\t\tkeystrokeHandler: KeystrokeHandler;\n\t\tfocusTracker: FocusTracker;\n\t\tgridItems: ViewCollection;\n\t\tnumberOfColumns: number | ( () => number );\n\t\tuiLanguageDirection?: string;\n\t}\n): void {\n\tconst getNumberOfColumns = typeof numberOfColumns === 'number' ? () => numberOfColumns : numberOfColumns;\n\n\tkeystrokeHandler.set( 'arrowright', getGridItemFocuser( ( focusedElementIndex, gridItems ) => {\n\t\treturn uiLanguageDirection === 'rtl' ?\n\t\t\tgetLeftElementIndex( focusedElementIndex, gridItems.length ) :\n\t\t\tgetRightElementIndex( focusedElementIndex, gridItems.length );\n\t} ) );\n\n\tkeystrokeHandler.set( 'arrowleft', getGridItemFocuser( ( focusedElementIndex, gridItems ) => {\n\t\treturn uiLanguageDirection === 'rtl' ?\n\t\t\tgetRightElementIndex( focusedElementIndex, gridItems.length ) :\n\t\t\tgetLeftElementIndex( focusedElementIndex, gridItems.length );\n\t} ) );\n\n\tkeystrokeHandler.set( 'arrowup', getGridItemFocuser( ( focusedElementIndex, gridItems ) => {\n\t\tlet nextIndex = focusedElementIndex - getNumberOfColumns();\n\n\t\tif ( nextIndex < 0 ) {\n\t\t\tnextIndex = focusedElementIndex + getNumberOfColumns() * Math.floor( gridItems.length / getNumberOfColumns() );\n\n\t\t\tif ( nextIndex > gridItems.length - 1 ) {\n\t\t\t\tnextIndex -= getNumberOfColumns();\n\t\t\t}\n\t\t}\n\n\t\treturn nextIndex;\n\t} ) );\n\n\tkeystrokeHandler.set( 'arrowdown', getGridItemFocuser( ( focusedElementIndex, gridItems ) => {\n\t\tlet nextIndex = focusedElementIndex + getNumberOfColumns();\n\n\t\tif ( nextIndex > gridItems.length - 1 ) {\n\t\t\tnextIndex = focusedElementIndex % getNumberOfColumns();\n\t\t}\n\n\t\treturn nextIndex;\n\t} ) );\n\n\tfunction getGridItemFocuser( getIndexToFocus: ( focusedElementIndex: number, gridItems: ViewCollection ) => number ) {\n\t\treturn ( evt: KeyboardEvent ) => {\n\t\t\tconst focusedElement = gridItems.find( item => item.element === focusTracker.focusedElement );\n\t\t\tconst focusedElementIndex = gridItems.getIndex( focusedElement! );\n\t\t\tconst nextIndexToFocus = getIndexToFocus( focusedElementIndex, gridItems );\n\n\t\t\t( gridItems.get( nextIndexToFocus ) as FocusableView ).focus();\n\n\t\t\tevt.stopPropagation();\n\t\t\tevt.preventDefault();\n\t\t};\n\t}\n\n\t/**\n\t * Function returning the next index.\n\t *\n\t * ```\n\t * before: [ ][x][ ]\tafter: [ ][ ][x]\n\t * index = 1 index = 2\n\t * ```\n\t *\n\t * If current index is last, function returns first index.\n\t *\n\t * ```\n\t * before: [ ][ ][x]\tafter: [x][ ][ ]\n\t * index = 2 index = 0\n\t * ```\n\t *\n\t * @param elementIndex Number of current index.\n\t * @param collectionLength A count of collection items.\n\t */\n\tfunction getRightElementIndex( elementIndex: number, collectionLength: number ) {\n\t\tif ( elementIndex === collectionLength - 1 ) {\n\t\t\treturn 0;\n\t\t} else {\n\t\t\treturn elementIndex + 1;\n\t\t}\n\t}\n\n\t/**\n\t * Function returning the previous index.\n\t *\n\t * ```\n\t * before: [ ][x][ ]\tafter: [x][ ][ ]\n\t * index = 1 index = 0\n\t * ```\n\t *\n\t * If current index is first, function returns last index.\n\t *\n\t * ```\n\t * before: [x][ ][ ]\tafter: [ ][ ][x]\n\t * index = 0 index = 2\n\t * ```\n\t *\n\t * @param elementIndex Number of current index.\n\t * @param collectionLength A count of collection items.\n\t */\n\tfunction getLeftElementIndex( elementIndex: number, collectionLength: number ) {\n\t\tif ( elementIndex === 0 ) {\n\t\t\treturn collectionLength - 1;\n\t\t} else {\n\t\t\treturn elementIndex - 1;\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tableui\n */\n\nimport { Plugin, type Command, type Editor } from 'ckeditor5/src/core';\nimport {\n\taddListToDropdown,\n\tcreateDropdown,\n\tModel,\n\tSplitButtonView,\n\tSwitchButtonView,\n\ttype DropdownView,\n\ttype ListDropdownItemDefinition\n} from 'ckeditor5/src/ui';\nimport { Collection, type Locale } from 'ckeditor5/src/utils';\n\nimport InsertTableView from './ui/inserttableview';\n\nimport tableIcon from './../theme/icons/table.svg';\nimport tableColumnIcon from './../theme/icons/table-column.svg';\nimport tableRowIcon from './../theme/icons/table-row.svg';\nimport tableMergeCellIcon from './../theme/icons/table-merge-cell.svg';\nimport type InsertTableCommand from './commands/inserttablecommand';\nimport type MergeCellsCommand from './commands/mergecellscommand';\n\n/**\n * The table UI plugin. It introduces:\n *\n * * The `'insertTable'` dropdown,\n * * The `'tableColumn'` dropdown,\n * * The `'tableRow'` dropdown,\n * * The `'mergeTableCells'` split button.\n *\n * The `'tableColumn'`, `'tableRow'` and `'mergeTableCells'` dropdowns work best with {@link module:table/tabletoolbar~TableToolbar}.\n */\nexport default class TableUI extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TableUI' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = this.editor.t;\n\t\tconst contentLanguageDirection = editor.locale.contentLanguageDirection;\n\t\tconst isContentLtr = contentLanguageDirection === 'ltr';\n\n\t\teditor.ui.componentFactory.add( 'insertTable', locale => {\n\t\t\tconst command: InsertTableCommand = editor.commands.get( 'insertTable' )!;\n\t\t\tconst dropdownView = createDropdown( locale );\n\n\t\t\tdropdownView.bind( 'isEnabled' ).to( command );\n\n\t\t\t// Decorate dropdown's button.\n\t\t\tdropdownView.buttonView.set( {\n\t\t\t\ticon: tableIcon,\n\t\t\t\tlabel: t( 'Insert table' ),\n\t\t\t\ttooltip: true\n\t\t\t} );\n\n\t\t\tlet insertTableView: InsertTableView;\n\n\t\t\tdropdownView.on( 'change:isOpen', () => {\n\t\t\t\tif ( insertTableView ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\t// Prepare custom view for dropdown's panel.\n\t\t\t\tinsertTableView = new InsertTableView( locale );\n\t\t\t\tdropdownView.panelView.children.add( insertTableView );\n\n\t\t\t\tinsertTableView.delegate( 'execute' ).to( dropdownView );\n\n\t\t\t\tdropdownView.on( 'execute', () => {\n\t\t\t\t\teditor.execute( 'insertTable', { rows: insertTableView.rows, columns: insertTableView.columns } );\n\t\t\t\t\teditor.editing.view.focus();\n\t\t\t\t} );\n\t\t\t} );\n\n\t\t\treturn dropdownView;\n\t\t} );\n\n\t\teditor.ui.componentFactory.add( 'tableColumn', locale => {\n\t\t\tconst options = [\n\t\t\t\t{\n\t\t\t\t\ttype: 'switchbutton',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'setTableColumnHeader',\n\t\t\t\t\t\tlabel: t( 'Header column' ),\n\t\t\t\t\t\tbindIsOn: true\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{ type: 'separator' },\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: isContentLtr ? 'insertTableColumnLeft' : 'insertTableColumnRight',\n\t\t\t\t\t\tlabel: t( 'Insert column left' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: isContentLtr ? 'insertTableColumnRight' : 'insertTableColumnLeft',\n\t\t\t\t\t\tlabel: t( 'Insert column right' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'removeTableColumn',\n\t\t\t\t\t\tlabel: t( 'Delete column' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'selectTableColumn',\n\t\t\t\t\t\tlabel: t( 'Select column' )\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t] as Array<ListDropdownItemDefinition>;\n\n\t\t\treturn this._prepareDropdown( t( 'Column' ), tableColumnIcon, options, locale );\n\t\t} );\n\n\t\teditor.ui.componentFactory.add( 'tableRow', locale => {\n\t\t\tconst options = [\n\t\t\t\t{\n\t\t\t\t\ttype: 'switchbutton',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'setTableRowHeader',\n\t\t\t\t\t\tlabel: t( 'Header row' ),\n\t\t\t\t\t\tbindIsOn: true\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{ type: 'separator' },\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'insertTableRowAbove',\n\t\t\t\t\t\tlabel: t( 'Insert row above' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'insertTableRowBelow',\n\t\t\t\t\t\tlabel: t( 'Insert row below' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'removeTableRow',\n\t\t\t\t\t\tlabel: t( 'Delete row' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'selectTableRow',\n\t\t\t\t\t\tlabel: t( 'Select row' )\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t] as Array<ListDropdownItemDefinition>;\n\n\t\t\treturn this._prepareDropdown( t( 'Row' ), tableRowIcon, options, locale );\n\t\t} );\n\n\t\teditor.ui.componentFactory.add( 'mergeTableCells', locale => {\n\t\t\tconst options = [\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'mergeTableCellUp',\n\t\t\t\t\t\tlabel: t( 'Merge cell up' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: isContentLtr ? 'mergeTableCellRight' : 'mergeTableCellLeft',\n\t\t\t\t\t\tlabel: t( 'Merge cell right' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'mergeTableCellDown',\n\t\t\t\t\t\tlabel: t( 'Merge cell down' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: isContentLtr ? 'mergeTableCellLeft' : 'mergeTableCellRight',\n\t\t\t\t\t\tlabel: t( 'Merge cell left' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{ type: 'separator' },\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'splitTableCellVertically',\n\t\t\t\t\t\tlabel: t( 'Split cell vertically' )\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t{\n\t\t\t\t\ttype: 'button',\n\t\t\t\t\tmodel: {\n\t\t\t\t\t\tcommandName: 'splitTableCellHorizontally',\n\t\t\t\t\t\tlabel: t( 'Split cell horizontally' )\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t] as Array<ListDropdownItemDefinition>;\n\n\t\t\treturn this._prepareMergeSplitButtonDropdown( t( 'Merge cells' ), tableMergeCellIcon, options, locale );\n\t\t} );\n\t}\n\n\t/**\n\t * Creates a dropdown view from a set of options.\n\t *\n\t * @param label The dropdown button label.\n\t * @param icon An icon for the dropdown button.\n\t * @param options The list of options for the dropdown.\n\t */\n\tprivate _prepareDropdown( label: string, icon: string, options: Array<ListDropdownItemDefinition>, locale: Locale ) {\n\t\tconst editor = this.editor;\n\t\tconst dropdownView = createDropdown( locale );\n\t\tconst commands = this._fillDropdownWithListOptions( dropdownView, options );\n\n\t\t// Decorate dropdown's button.\n\t\tdropdownView.buttonView.set( {\n\t\t\tlabel,\n\t\t\ticon,\n\t\t\ttooltip: true\n\t\t} );\n\n\t\t// Make dropdown button disabled when all options are disabled.\n\t\tdropdownView.bind( 'isEnabled' ).toMany( commands, 'isEnabled', ( ...areEnabled ) => {\n\t\t\treturn areEnabled.some( isEnabled => isEnabled );\n\t\t} );\n\n\t\tthis.listenTo( dropdownView, 'execute', evt => {\n\t\t\teditor.execute( ( evt.source as any ).commandName );\n\n\t\t\t// Toggling a switch button view should not move the focus to the editable.\n\t\t\tif ( !( evt.source instanceof SwitchButtonView ) ) {\n\t\t\t\teditor.editing.view.focus();\n\t\t\t}\n\t\t} );\n\n\t\treturn dropdownView;\n\t}\n\n\t/**\n\t * Creates a dropdown view with a {@link module:ui/dropdown/button/splitbuttonview~SplitButtonView} for\n\t * merge (and split)related commands.\n\t *\n\t * @param label The dropdown button label.\n\t * @param icon An icon for the dropdown button.\n\t * @param options The list of options for the dropdown.\n\t */\n\tprivate _prepareMergeSplitButtonDropdown( label: string, icon: string, options: Array<ListDropdownItemDefinition>, locale: Locale ) {\n\t\tconst editor = this.editor;\n\t\tconst dropdownView = createDropdown( locale, SplitButtonView );\n\t\tconst mergeCommandName = 'mergeTableCells';\n\n\t\t// Main command.\n\t\tconst mergeCommand: MergeCellsCommand = editor.commands.get( mergeCommandName )!;\n\n\t\t// Subcommands in the dropdown.\n\t\tconst commands = this._fillDropdownWithListOptions( dropdownView, options );\n\n\t\tdropdownView.buttonView.set( {\n\t\t\tlabel,\n\t\t\ticon,\n\t\t\ttooltip: true,\n\t\t\tisEnabled: true\n\t\t} );\n\n\t\t// Make dropdown button disabled when all options are disabled together with the main command.\n\t\tdropdownView.bind( 'isEnabled' ).toMany( [ mergeCommand, ...commands ], 'isEnabled', ( ...areEnabled ) => {\n\t\t\treturn areEnabled.some( isEnabled => isEnabled );\n\t\t} );\n\n\t\t// Merge selected table cells when the main part of the split button is clicked.\n\t\tthis.listenTo( dropdownView.buttonView, 'execute', () => {\n\t\t\teditor.execute( mergeCommandName );\n\t\t\teditor.editing.view.focus();\n\t\t} );\n\n\t\t// Execute commands for events coming from the list in the dropdown panel.\n\t\tthis.listenTo( dropdownView, 'execute', evt => {\n\t\t\teditor.execute( ( evt.source as any ).commandName );\n\t\t\teditor.editing.view.focus();\n\t\t} );\n\n\t\treturn dropdownView;\n\t}\n\n\t/**\n\t * Injects a {@link module:ui/list/listview~ListView} into the passed dropdown with buttons\n\t * which execute editor commands as configured in passed options.\n\t *\n\t * @param options The list of options for the dropdown.\n\t * @returns Commands the list options are interacting with.\n\t */\n\tprivate _fillDropdownWithListOptions( dropdownView: DropdownView, options: Array<ListDropdownItemDefinition> ) {\n\t\tconst editor = this.editor;\n\t\tconst commands: Array<Command> = [];\n\t\tconst itemDefinitions = new Collection<ListDropdownItemDefinition>();\n\n\t\tfor ( const option of options ) {\n\t\t\taddListOption( option, editor, commands, itemDefinitions );\n\t\t}\n\n\t\taddListToDropdown( dropdownView, itemDefinitions );\n\n\t\treturn commands;\n\t}\n}\n\n/**\n * Adds an option to a list view.\n *\n * @param option A configuration option.\n * @param commands The list of commands to update.\n * @param itemDefinitions A collection of dropdown items to update with the given option.\n */\nfunction addListOption(\n\toption: ListDropdownItemDefinition,\n\teditor: Editor,\n\tcommands: Array<Command>,\n\titemDefinitions: Collection<ListDropdownItemDefinition>\n) {\n\tif ( option.type === 'button' || option.type === 'switchbutton' ) {\n\t\tconst model = option.model = new Model( option.model );\n\t\tconst { commandName, bindIsOn } = option.model;\n\t\tconst command = editor.commands.get( commandName as string )!;\n\n\t\tcommands.push( command );\n\n\t\tmodel.set( { commandName } );\n\n\t\tmodel.bind( 'isEnabled' ).to( command );\n\n\t\tif ( bindIsOn ) {\n\t\t\tmodel.bind( 'isOn' ).to( command, 'value' );\n\t\t}\n\n\t\tmodel.set( {\n\t\t\twithText: true\n\t\t} );\n\t}\n\n\titemDefinitions.add( option );\n}\n","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M3 6v3h4V6H3zm0 4v3h4v-3H3zm0 4v3h4v-3H3zm5 3h4v-3H8v3zm5 0h4v-3h-4v3zm4-4v-3h-4v3h4zm0-4V6h-4v3h4zm1.5 8a1.5 1.5 0 0 1-1.5 1.5H3A1.5 1.5 0 0 1 1.5 17V4c.222-.863 1.068-1.5 2-1.5h13c.932 0 1.778.637 2 1.5v13zM12 13v-3H8v3h4zm0-4V6H8v3h4z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2.5 1h15A1.5 1.5 0 0 1 19 2.5v15a1.5 1.5 0 0 1-1.5 1.5h-15A1.5 1.5 0 0 1 1 17.5v-15A1.5 1.5 0 0 1 2.5 1zM2 2v16h16V2H2z\\\" opacity=\\\".6\\\"/><path d=\\\"M18 7v1H2V7h16zm0 5v1H2v-1h16z\\\" opacity=\\\".6\\\"/><path d=\\\"M14 1v18a1 1 0 0 1-1 1H7a1 1 0 0 1-1-1V1a1 1 0 0 1 1-1h6a1 1 0 0 1 1 1zm-2 1H8v4h4V2zm0 6H8v4h4V8zm0 6H8v4h4v-4z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2.5 1h15A1.5 1.5 0 0 1 19 2.5v15a1.5 1.5 0 0 1-1.5 1.5h-15A1.5 1.5 0 0 1 1 17.5v-15A1.5 1.5 0 0 1 2.5 1zM2 2v16h16V2H2z\\\" opacity=\\\".6\\\"/><path d=\\\"M7 2h1v16H7V2zm5 0h1v16h-1V2z\\\" opacity=\\\".6\\\"/><path d=\\\"M1 6h18a1 1 0 0 1 1 1v6a1 1 0 0 1-1 1H1a1 1 0 0 1-1-1V7a1 1 0 0 1 1-1zm1 2v4h4V8H2zm6 0v4h4V8H8zm6 0v4h4V8h-4z\\\"/></svg>\";","export default \"<svg viewBox=\\\"0 0 20 20\\\" xmlns=\\\"http://www.w3.org/2000/svg\\\"><path d=\\\"M2.5 1h15A1.5 1.5 0 0 1 19 2.5v15a1.5 1.5 0 0 1-1.5 1.5h-15A1.5 1.5 0 0 1 1 17.5v-15A1.5 1.5 0 0 1 2.5 1zM2 2v16h16V2H2z\\\" opacity=\\\".6\\\"/><path d=\\\"M7 2h1v16H7V2zm5 0h1v7h-1V2zm6 5v1H2V7h16zM8 12v1H2v-1h6z\\\" opacity=\\\".6\\\"/><path d=\\\"M7 7h12a1 1 0 0 1 1 1v11a1 1 0 0 1-1 1H7a1 1 0 0 1-1-1V8a1 1 0 0 1 1-1zm1 2v9h10V9H8z\\\"/></svg>\";","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./tableselection.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tableselection\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { type EventInfo, first } from 'ckeditor5/src/utils';\n\nimport type {\n\tElement,\n\tDocumentFragment,\n\tSelection,\n\tDowncastWriter,\n\tViewElement,\n\tModelDeleteContentEvent\n} from 'ckeditor5/src/engine';\n\nimport type {\n\tViewDocumentInsertTextEvent,\n\tInsertTextEventData\n} from 'ckeditor5/src/typing';\n\nimport TableWalker from './tablewalker';\nimport TableUtils from './tableutils';\n\nimport { cropTableToDimensions, adjustLastRowIndex, adjustLastColumnIndex } from './utils/structure';\n\nimport '../theme/tableselection.css';\n\n/**\n * This plugin enables the advanced table cells, rows and columns selection.\n * It is loaded automatically by the {@link module:table/table~Table} plugin.\n */\nexport default class TableSelection extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TableSelection' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ TableUtils, TableUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst model = editor.model;\n\t\tconst view = editor.editing.view;\n\n\t\tthis.listenTo<ModelDeleteContentEvent>(\n\t\t\tmodel,\n\t\t\t'deleteContent',\n\t\t\t( evt, args ) => this._handleDeleteContent( evt, args ),\n\t\t\t{ priority: 'high' }\n\t\t);\n\n\t\tthis.listenTo<ViewDocumentInsertTextEvent>(\n\t\t\tview.document,\n\t\t\t'insertText',\n\t\t\t( evt, data ) => this._handleInsertTextEvent( evt, data ),\n\t\t\t{ priority: 'high' }\n\t\t);\n\n\t\tthis._defineSelectionConverter();\n\t\tthis._enablePluginDisabling(); // sic!\n\t}\n\n\t/**\n\t * Returns the currently selected table cells or `null` if it is not a table cells selection.\n\t */\n\tpublic getSelectedTableCells(): Array<Element> | null {\n\t\tconst tableUtils = this.editor.plugins.get( TableUtils );\n\t\tconst selection = this.editor.model.document.selection;\n\n\t\tconst selectedCells = tableUtils.getSelectedTableCells( selection );\n\n\t\tif ( selectedCells.length == 0 ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// This should never happen, but let's know if it ever happens.\n\t\t// @if CK_DEBUG //\tif ( selectedCells.length != selection.rangeCount ) {\n\t\t// @if CK_DEBUG //\t\tconsole.warn( 'Mixed selection warning. The selection contains table cells and some other ranges.' );\n\t\t// @if CK_DEBUG //\t}\n\n\t\treturn selectedCells;\n\t}\n\n\t/**\n\t * Returns the selected table fragment as a document fragment.\n\t */\n\tpublic getSelectionAsFragment(): DocumentFragment | null {\n\t\tconst tableUtils = this.editor.plugins.get( TableUtils );\n\t\tconst selectedCells = this.getSelectedTableCells();\n\n\t\tif ( !selectedCells ) {\n\t\t\treturn null;\n\t\t}\n\n\t\treturn this.editor.model.change( writer => {\n\t\t\tconst documentFragment = writer.createDocumentFragment();\n\n\t\t\tconst { first: firstColumn, last: lastColumn } = tableUtils.getColumnIndexes( selectedCells );\n\t\t\tconst { first: firstRow, last: lastRow } = tableUtils.getRowIndexes( selectedCells );\n\n\t\t\tconst sourceTable = selectedCells[ 0 ].findAncestor( 'table' )!;\n\n\t\t\tlet adjustedLastRow = lastRow;\n\t\t\tlet adjustedLastColumn = lastColumn;\n\n\t\t\t// If the selection is rectangular there could be a case of all cells in the last row/column spanned over\n\t\t\t// next row/column so the real lastRow/lastColumn should be updated.\n\t\t\tif ( tableUtils.isSelectionRectangular( selectedCells ) ) {\n\t\t\t\tconst dimensions = {\n\t\t\t\t\tfirstColumn,\n\t\t\t\t\tlastColumn,\n\t\t\t\t\tfirstRow,\n\t\t\t\t\tlastRow\n\t\t\t\t};\n\n\t\t\t\tadjustedLastRow = adjustLastRowIndex( sourceTable, dimensions );\n\t\t\t\tadjustedLastColumn = adjustLastColumnIndex( sourceTable, dimensions );\n\t\t\t}\n\n\t\t\tconst cropDimensions = {\n\t\t\t\tstartRow: firstRow,\n\t\t\t\tstartColumn: firstColumn,\n\t\t\t\tendRow: adjustedLastRow,\n\t\t\t\tendColumn: adjustedLastColumn\n\t\t\t};\n\n\t\t\tconst table = cropTableToDimensions( sourceTable, cropDimensions, writer );\n\n\t\t\twriter.insert( table, documentFragment, 0 );\n\n\t\t\treturn documentFragment;\n\t\t} );\n\t}\n\n\t/**\n\t * Sets the model selection based on given anchor and target cells (can be the same cell).\n\t * Takes care of setting the backward flag.\n\t *\n\t * ```ts\n\t * const modelRoot = editor.model.document.getRoot();\n\t * const firstCell = modelRoot.getNodeByPath( [ 0, 0, 0 ] );\n\t * const lastCell = modelRoot.getNodeByPath( [ 0, 0, 1 ] );\n\t *\n\t * const tableSelection = editor.plugins.get( 'TableSelection' );\n\t * tableSelection.setCellSelection( firstCell, lastCell );\n\t * ```\n\t */\n\tpublic setCellSelection( anchorCell: Element, targetCell: Element ): void {\n\t\tconst cellsToSelect = this._getCellsToSelect( anchorCell, targetCell );\n\n\t\tthis.editor.model.change( writer => {\n\t\t\twriter.setSelection(\n\t\t\t\tcellsToSelect.cells.map( cell => writer.createRangeOn( cell ) ),\n\t\t\t\t{ backward: cellsToSelect.backward }\n\t\t\t);\n\t\t} );\n\t}\n\n\t/**\n\t * Returns the focus cell from the current selection.\n\t */\n\tpublic getFocusCell(): Element | null {\n\t\tconst selection = this.editor.model.document.selection;\n\t\tconst focusCellRange = [ ...selection.getRanges() ].pop()!;\n\t\tconst element = focusCellRange.getContainedElement();\n\n\t\tif ( element && element.is( 'element', 'tableCell' ) ) {\n\t\t\treturn element;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Returns the anchor cell from the current selection.\n\t */\n\tpublic getAnchorCell(): Element | null {\n\t\tconst selection = this.editor.model.document.selection;\n\t\tconst anchorCellRange = first( selection.getRanges() )!;\n\t\tconst element = anchorCellRange.getContainedElement();\n\n\t\tif ( element && element.is( 'element', 'tableCell' ) ) {\n\t\t\treturn element;\n\t\t}\n\n\t\treturn null;\n\t}\n\n\t/**\n\t * Defines a selection converter which marks the selected cells with a specific class.\n\t *\n\t * The real DOM selection is put in the last cell. Since the order of ranges is dependent on whether the\n\t * selection is backward or not, the last cell will usually be close to the \"focus\" end of the selection\n\t * (a selection has anchor and focus).\n\t *\n\t * The real DOM selection is then hidden with CSS.\n\t */\n\tprivate _defineSelectionConverter() {\n\t\tconst editor = this.editor;\n\t\tconst highlighted: Set<ViewElement> = new Set();\n\n\t\teditor.conversion.for( 'editingDowncast' ).add( dispatcher => dispatcher.on( 'selection', ( evt, data, conversionApi ) => {\n\t\t\tconst viewWriter = conversionApi.writer;\n\n\t\t\tclearHighlightedTableCells( viewWriter );\n\n\t\t\tconst selectedCells = this.getSelectedTableCells();\n\n\t\t\tif ( !selectedCells ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tfor ( const tableCell of selectedCells ) {\n\t\t\t\tconst viewElement = conversionApi.mapper.toViewElement( tableCell )!;\n\n\t\t\t\tviewWriter.addClass( 'ck-editor__editable_selected', viewElement );\n\t\t\t\thighlighted.add( viewElement );\n\t\t\t}\n\n\t\t\tconst lastViewCell = conversionApi.mapper.toViewElement( selectedCells[ selectedCells.length - 1 ] );\n\t\t\tviewWriter.setSelection( lastViewCell, 0 );\n\t\t}, { priority: 'lowest' } ) );\n\n\t\tfunction clearHighlightedTableCells( viewWriter: DowncastWriter ) {\n\t\t\tfor ( const previouslyHighlighted of highlighted ) {\n\t\t\t\tviewWriter.removeClass( 'ck-editor__editable_selected', previouslyHighlighted );\n\t\t\t}\n\n\t\t\thighlighted.clear();\n\t\t}\n\t}\n\n\t/**\n\t * Creates a listener that reacts to changes in {@link #isEnabled} and, if the plugin was disabled,\n\t * it collapses the multi-cell selection to a regular selection placed inside a table cell.\n\t *\n\t * This listener helps features that disable the table selection plugin bring the selection\n\t * to a clear state they can work with (for instance, because they don't support multiple cell selection).\n\t */\n\tprivate _enablePluginDisabling() {\n\t\tconst editor = this.editor;\n\n\t\tthis.on( 'change:isEnabled', () => {\n\t\t\tif ( !this.isEnabled ) {\n\t\t\t\tconst selectedCells = this.getSelectedTableCells();\n\n\t\t\t\tif ( !selectedCells ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\teditor.model.change( writer => {\n\t\t\t\t\tconst position = writer.createPositionAt( selectedCells[ 0 ], 0 );\n\t\t\t\t\tconst range = editor.model.schema.getNearestSelectionRange( position );\n\n\t\t\t\t\twriter.setSelection( range );\n\t\t\t\t} );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Overrides the default `model.deleteContent()` behavior over a selected table fragment.\n\t *\n\t * @param args Delete content method arguments.\n\t */\n\tprivate _handleDeleteContent( event: EventInfo, args: Array<unknown> ) {\n\t\tconst tableUtils = this.editor.plugins.get( TableUtils );\n\t\tconst selection = args[ 0 ] as Selection;\n\t\tconst options = args[ 1 ] as { direction?: string };\n\t\tconst model = this.editor.model;\n\t\tconst isBackward = !options || options.direction == 'backward';\n\t\tconst selectedTableCells = tableUtils.getSelectedTableCells( selection );\n\n\t\tif ( !selectedTableCells.length ) {\n\t\t\treturn;\n\t\t}\n\n\t\tevent.stop();\n\n\t\tmodel.change( writer => {\n\t\t\tconst tableCellToSelect = selectedTableCells[ isBackward ? selectedTableCells.length - 1 : 0 ];\n\n\t\t\tmodel.change( writer => {\n\t\t\t\tfor ( const tableCell of selectedTableCells ) {\n\t\t\t\t\tmodel.deleteContent( writer.createSelection( tableCell, 'in' ) );\n\t\t\t\t}\n\t\t\t} );\n\n\t\t\tconst rangeToSelect = model.schema.getNearestSelectionRange( writer.createPositionAt( tableCellToSelect, 0 ) );\n\n\t\t\t// Note: we ignore the case where rangeToSelect may be null because deleteContent() will always (unless someone broke it)\n\t\t\t// create an empty paragraph to accommodate the selection.\n\n\t\t\tif ( selection.is( 'documentSelection' ) ) {\n\t\t\t\twriter.setSelection( rangeToSelect );\n\t\t\t} else {\n\t\t\t\tselection.setTo( rangeToSelect );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * This handler makes it possible to remove the content of all selected cells by starting to type.\n\t * If you take a look at {@link #_defineSelectionConverter} you will find out that despite the multi-cell selection being set\n\t * in the model, the view selection is collapsed in the last cell (because most browsers are unable to render multi-cell selections;\n\t * yes, it's a hack).\n\t *\n\t * When multiple cells are selected in the model and the user starts to type, the\n\t * {@link module:engine/view/document~Document#event:insertText} event carries information provided by the\n\t * beforeinput DOM event, that in turn only knows about this collapsed DOM selection in the last cell.\n\t *\n\t * As a result, the selected cells have no chance to be cleaned up. To fix this, this listener intercepts\n\t * the event and injects the custom view selection in the data that translates correctly to the actual state\n\t * of the multi-cell selection in the model.\n\t *\n\t * @param data Insert text event data.\n\t */\n\tprivate _handleInsertTextEvent( evt: EventInfo, data: InsertTextEventData ) {\n\t\tconst editor = this.editor;\n\t\tconst selectedCells = this.getSelectedTableCells();\n\n\t\tif ( !selectedCells ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst view = editor.editing.view;\n\t\tconst mapper = editor.editing.mapper;\n\t\tconst viewRanges = selectedCells.map( tableCell => view.createRangeOn( mapper.toViewElement( tableCell )! ) );\n\n\t\tdata.selection = view.createSelection( viewRanges );\n\t}\n\n\t/**\n\t * Returns an array of table cells that should be selected based on the\n\t * given anchor cell and target (focus) cell.\n\t *\n\t * The cells are returned in a reverse direction if the selection is backward.\n\t */\n\tprivate _getCellsToSelect( anchorCell: Element, targetCell: Element ) {\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( 'TableUtils' );\n\t\tconst startLocation = tableUtils.getCellLocation( anchorCell );\n\t\tconst endLocation = tableUtils.getCellLocation( targetCell );\n\n\t\tconst startRow = Math.min( startLocation.row, endLocation.row );\n\t\tconst endRow = Math.max( startLocation.row, endLocation.row );\n\n\t\tconst startColumn = Math.min( startLocation.column, endLocation.column );\n\t\tconst endColumn = Math.max( startLocation.column, endLocation.column );\n\n\t\t// 2-dimensional array of the selected cells to ease flipping the order of cells for backward selections.\n\t\tconst selectionMap: Array<Array<Element>> = new Array( endRow - startRow + 1 ).fill( null ).map( () => [] );\n\n\t\tconst walkerOptions = {\n\t\t\tstartRow,\n\t\t\tendRow,\n\t\t\tstartColumn,\n\t\t\tendColumn\n\t\t};\n\n\t\tfor ( const { row, cell } of new TableWalker( anchorCell.findAncestor( 'table' )!, walkerOptions ) ) {\n\t\t\tselectionMap[ row - startRow ].push( cell );\n\t\t}\n\n\t\tconst flipVertically = endLocation.row < startLocation.row;\n\t\tconst flipHorizontally = endLocation.column < startLocation.column;\n\n\t\tif ( flipVertically ) {\n\t\t\tselectionMap.reverse();\n\t\t}\n\n\t\tif ( flipHorizontally ) {\n\t\t\tselectionMap.forEach( row => row.reverse() );\n\t\t}\n\n\t\treturn {\n\t\t\tcells: selectionMap.flat(),\n\t\t\tbackward: flipVertically || flipHorizontally\n\t\t};\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tableclipboard\n */\n\nimport type {\n\tClipboardEventData,\n\tViewDocumentClipboardOutputEvent,\n\tViewDocumentCopyEvent,\n\tViewDocumentCutEvent\n} from 'ckeditor5/src/clipboard';\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport type {\n\tDocumentFragment,\n\tDocumentSelection,\n\tDomEventData,\n\tElement,\n\tItem,\n\tModel,\n\tModelInsertContentEvent,\n\tPosition,\n\tSelection,\n\tWriter\n} from 'ckeditor5/src/engine';\n\nimport type { EventInfo } from 'ckeditor5/src/utils';\n\nimport TableSelection from './tableselection';\nimport TableWalker, { type TableSlot } from './tablewalker';\nimport TableUtils from './tableutils';\nimport {\n\tcropTableToDimensions,\n\tgetHorizontallyOverlappingCells,\n\tgetVerticallyOverlappingCells,\n\tremoveEmptyRowsColumns,\n\tsplitHorizontally,\n\tsplitVertically,\n\ttrimTableCellIfNeeded,\n\tadjustLastRowIndex,\n\tadjustLastColumnIndex\n} from './utils/structure';\n\n/**\n * This plugin adds support for copying/cutting/pasting fragments of tables.\n * It is loaded automatically by the {@link module:table/table~Table} plugin.\n */\nexport default class TableClipboard extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TableClipboard' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ TableSelection, TableUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst viewDocument = editor.editing.view.document;\n\n\t\tthis.listenTo<ViewDocumentCopyEvent>( viewDocument, 'copy', ( evt, data ) => this._onCopyCut( evt, data ) );\n\t\tthis.listenTo<ViewDocumentCutEvent>( viewDocument, 'cut', ( evt, data ) => this._onCopyCut( evt, data ) );\n\t\tthis.listenTo<ModelInsertContentEvent>(\n\t\t\teditor.model,\n\t\t\t'insertContent',\n\t\t\t( evt, [ content, selectable ] ) => this._onInsertContent( evt, content, selectable ),\n\t\t\t{ priority: 'high' }\n\t\t);\n\n\t\tthis.decorate( '_replaceTableSlotCell' );\n\t}\n\n\t/**\n\t * Copies table content to a clipboard on \"copy\" & \"cut\" events.\n\t *\n\t * @param evt An object containing information about the handled event.\n\t * @param data Clipboard event data.\n\t */\n\tprivate _onCopyCut( evt: EventInfo<'copy' | 'cut'>, data: DomEventData<ClipboardEvent> & ClipboardEventData ) {\n\t\tconst tableSelection = this.editor.plugins.get( TableSelection );\n\n\t\tif ( !tableSelection.getSelectedTableCells() ) {\n\t\t\treturn;\n\t\t}\n\n\t\tif ( evt.name == 'cut' && !this.editor.model.canEditAt( this.editor.model.document.selection ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tdata.preventDefault();\n\t\tevt.stop();\n\n\t\tconst dataController = this.editor.data;\n\t\tconst viewDocument = this.editor.editing.view.document;\n\n\t\tconst content = dataController.toView( tableSelection.getSelectionAsFragment()! );\n\n\t\tviewDocument.fire<ViewDocumentClipboardOutputEvent>( 'clipboardOutput', {\n\t\t\tdataTransfer: data.dataTransfer,\n\t\t\tcontent,\n\t\t\tmethod: evt.name\n\t\t} );\n\t}\n\n\t/**\n\t * Overrides default {@link module:engine/model/model~Model#insertContent `model.insertContent()`} method to handle pasting table inside\n\t * selected table fragment.\n\t *\n\t * Depending on selected table fragment:\n\t * - If a selected table fragment is smaller than paste table it will crop pasted table to match dimensions.\n\t * - If dimensions are equal it will replace selected table fragment with a pasted table contents.\n\t *\n\t * @param content The content to insert.\n\t * @param selectable The selection into which the content should be inserted.\n\t * If not provided the current model document selection will be used.\n\t */\n\tprivate _onInsertContent( evt: EventInfo, content: DocumentFragment | Item, selectable: Selection | DocumentSelection | undefined ) {\n\t\tif ( selectable && !selectable.is( 'documentSelection' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst model = this.editor.model;\n\t\tconst tableUtils = this.editor.plugins.get( TableUtils );\n\n\t\t// We might need to crop table before inserting so reference might change.\n\t\tlet pastedTable = this.getTableIfOnlyTableInContent( content, model )!;\n\n\t\tif ( !pastedTable ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst selectedTableCells = tableUtils.getSelectionAffectedTableCells( model.document.selection );\n\n\t\tif ( !selectedTableCells.length ) {\n\t\t\tremoveEmptyRowsColumns( pastedTable, tableUtils );\n\n\t\t\treturn;\n\t\t}\n\n\t\t// Override default model.insertContent() handling at this point.\n\t\tevt.stop();\n\n\t\tmodel.change( writer => {\n\t\t\tconst pastedDimensions = {\n\t\t\t\twidth: tableUtils.getColumns( pastedTable ),\n\t\t\t\theight: tableUtils.getRows( pastedTable )\n\t\t\t};\n\n\t\t\t// Prepare the table for pasting.\n\t\t\tconst selection = prepareTableForPasting( selectedTableCells, pastedDimensions, writer, tableUtils );\n\n\t\t\t// Beyond this point we operate on a fixed content table with rectangular selection and proper last row/column values.\n\n\t\t\tconst selectionHeight = selection.lastRow - selection.firstRow + 1;\n\t\t\tconst selectionWidth = selection.lastColumn - selection.firstColumn + 1;\n\n\t\t\t// Crop pasted table if:\n\t\t\t// - Pasted table dimensions exceeds selection area.\n\t\t\t// - Pasted table has broken layout (ie some cells sticks out by the table dimensions established by the first and last row).\n\t\t\t//\n\t\t\t// Note: The table dimensions are established by the width of the first row and the total number of rows.\n\t\t\t// It is possible to programmatically create a table that has rows which would have cells anchored beyond first row width but\n\t\t\t// such table will not be created by other editing solutions.\n\t\t\tconst cropDimensions = {\n\t\t\t\tstartRow: 0,\n\t\t\t\tstartColumn: 0,\n\t\t\t\tendRow: Math.min( selectionHeight, pastedDimensions.height ) - 1,\n\t\t\t\tendColumn: Math.min( selectionWidth, pastedDimensions.width ) - 1\n\t\t\t};\n\n\t\t\tpastedTable = cropTableToDimensions( pastedTable, cropDimensions, writer );\n\n\t\t\t// Content table to which we insert a pasted table.\n\t\t\tconst selectedTable = selectedTableCells[ 0 ].findAncestor( 'table' )!;\n\n\t\t\tconst cellsToSelect = this._replaceSelectedCellsWithPasted( pastedTable, pastedDimensions, selectedTable, selection, writer );\n\n\t\t\tif ( this.editor.plugins.get( 'TableSelection' ).isEnabled ) {\n\t\t\t\t// Selection ranges must be sorted because the first and last selection ranges are considered\n\t\t\t\t// as anchor/focus cell ranges for multi-cell selection.\n\t\t\t\tconst selectionRanges = tableUtils.sortRanges( cellsToSelect.map( cell => writer.createRangeOn( cell ) ) );\n\n\t\t\t\twriter.setSelection( selectionRanges );\n\t\t\t} else {\n\t\t\t\t// Set selection inside first cell if multi-cell selection is disabled.\n\t\t\t\twriter.setSelection( cellsToSelect[ 0 ], 0 );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Replaces the part of selectedTable with pastedTable.\n\t */\n\tprivate _replaceSelectedCellsWithPasted(\n\t\tpastedTable: Element,\n\t\tpastedDimensions: Record<string, number>,\n\t\tselectedTable: Element,\n\t\tselection: Record<string, number>,\n\t\twriter: Writer\n\t) {\n\t\tconst { width: pastedWidth, height: pastedHeight } = pastedDimensions;\n\n\t\t// Holds two-dimensional array that is addressed by [ row ][ column ] that stores cells anchored at given location.\n\t\tconst pastedTableLocationMap = createLocationMap( pastedTable, pastedWidth, pastedHeight );\n\n\t\tconst selectedTableMap = [ ...new TableWalker( selectedTable, {\n\t\t\tstartRow: selection.firstRow,\n\t\t\tendRow: selection.lastRow,\n\t\t\tstartColumn: selection.firstColumn,\n\t\t\tendColumn: selection.lastColumn,\n\t\t\tincludeAllSlots: true\n\t\t} ) ];\n\n\t\t// Selection must be set to pasted cells (some might be removed or new created).\n\t\tconst cellsToSelect: Array<Element> = [];\n\n\t\t// Store next cell insert position.\n\t\tlet insertPosition: Position;\n\n\t\t// Content table replace cells algorithm iterates over a selected table fragment and:\n\t\t//\n\t\t// - Removes existing table cells at current slot (location).\n\t\t// - Inserts cell from a pasted table for a matched slots.\n\t\t//\n\t\t// This ensures proper table geometry after the paste\n\t\tfor ( const tableSlot of selectedTableMap ) {\n\t\t\tconst { row, column } = tableSlot;\n\n\t\t\t// Save the insert position for current row start.\n\t\t\tif ( column === selection.firstColumn ) {\n\t\t\t\tinsertPosition = tableSlot.getPositionBefore();\n\t\t\t}\n\n\t\t\t// Map current table slot location to an pasted table slot location.\n\t\t\tconst pastedRow = row - selection.firstRow;\n\t\t\tconst pastedColumn = column - selection.firstColumn;\n\t\t\tconst pastedCell = pastedTableLocationMap[ pastedRow % pastedHeight ][ pastedColumn % pastedWidth ];\n\n\t\t\t// Clone cell to insert (to duplicate its attributes and children).\n\t\t\t// Cloning is required to support repeating pasted table content when inserting to a bigger selection.\n\t\t\tconst cellToInsert = pastedCell ? writer.cloneElement( pastedCell ) : null;\n\n\t\t\t// Replace the cell from the current slot with new table cell.\n\t\t\tconst newTableCell = this._replaceTableSlotCell( tableSlot, cellToInsert, insertPosition!, writer );\n\n\t\t\t// The cell was only removed.\n\t\t\tif ( !newTableCell ) {\n\t\t\t\tcontinue;\n\t\t\t}\n\n\t\t\t// Trim the cell if it's row/col-spans would exceed selection area.\n\t\t\ttrimTableCellIfNeeded( newTableCell, row, column, selection.lastRow, selection.lastColumn, writer );\n\n\t\t\tcellsToSelect.push( newTableCell );\n\n\t\t\tinsertPosition = writer.createPositionAfter( newTableCell );\n\t\t}\n\n\t\t// If there are any headings, all the cells that overlap from heading must be splitted.\n\t\tconst headingRows = parseInt( selectedTable.getAttribute( 'headingRows' ) as string || '0' );\n\t\tconst headingColumns = parseInt( selectedTable.getAttribute( 'headingColumns' ) as string || '0' );\n\n\t\tconst areHeadingRowsIntersectingSelection = selection.firstRow < headingRows && headingRows <= selection.lastRow;\n\t\tconst areHeadingColumnsIntersectingSelection = selection.firstColumn < headingColumns && headingColumns <= selection.lastColumn;\n\n\t\tif ( areHeadingRowsIntersectingSelection ) {\n\t\t\tconst columnsLimit = { first: selection.firstColumn, last: selection.lastColumn };\n\t\t\tconst newCells = doHorizontalSplit( selectedTable, headingRows, columnsLimit, writer, selection.firstRow ) as Array<Element>;\n\n\t\t\tcellsToSelect.push( ...newCells );\n\t\t}\n\n\t\tif ( areHeadingColumnsIntersectingSelection ) {\n\t\t\tconst rowsLimit = { first: selection.firstRow, last: selection.lastRow };\n\t\t\tconst newCells = doVerticalSplit( selectedTable, headingColumns, rowsLimit, writer ) as Array<Element>;\n\n\t\t\tcellsToSelect.push( ...newCells );\n\t\t}\n\n\t\treturn cellsToSelect;\n\t}\n\n\t/**\n\t * Replaces a single table slot.\n\t *\n\t * @returns Inserted table cell or null if slot should remain empty.\n\t * @private\n\t */\n\tpublic _replaceTableSlotCell(\n\t\ttableSlot: TableSlot,\n\t\tcellToInsert: Element | null,\n\t\tinsertPosition: Position,\n\t\twriter: Writer\n\t): Element | null {\n\t\tconst { cell, isAnchor } = tableSlot;\n\n\t\t// If the slot is occupied by a cell in a selected table - remove it.\n\t\t// The slot of this cell will be either:\n\t\t// - Replaced by a pasted table cell.\n\t\t// - Spanned by a previously pasted table cell.\n\t\tif ( isAnchor ) {\n\t\t\twriter.remove( cell );\n\t\t}\n\n\t\t// There is no cell to insert (might be spanned by other cell in a pasted table) - advance to the next content table slot.\n\t\tif ( !cellToInsert ) {\n\t\t\treturn null;\n\t\t}\n\n\t\twriter.insert( cellToInsert, insertPosition );\n\n\t\treturn cellToInsert;\n\t}\n\n\t/**\n\t * Extracts the table for pasting into a table.\n\t *\n\t * @param content The content to insert.\n\t * @param model The editor model.\n\t */\n\tpublic getTableIfOnlyTableInContent( content: DocumentFragment | Item, model: Model ): Element | null {\n\t\tif ( !content.is( 'documentFragment' ) && !content.is( 'element' ) ) {\n\t\t\treturn null;\n\t\t}\n\n\t\t// Table passed directly.\n\t\tif ( content.is( 'element', 'table' ) ) {\n\t\t\treturn content;\n\t\t}\n\n\t\t// We do not support mixed content when pasting table into table.\n\t\t// See: https://github.com/ckeditor/ckeditor5/issues/6817.\n\t\tif ( content.childCount == 1 && content.getChild( 0 )!.is( 'element', 'table' ) ) {\n\t\t\treturn content.getChild( 0 ) as Element;\n\t\t}\n\n\t\t// If there are only whitespaces around a table then use that table for pasting.\n\n\t\tconst contentRange = model.createRangeIn( content );\n\n\t\tfor ( const element of contentRange.getItems() ) {\n\t\t\tif ( element.is( 'element', 'table' ) ) {\n\t\t\t\t// Stop checking if there is some content before table.\n\t\t\t\tconst rangeBefore = model.createRange( contentRange.start, model.createPositionBefore( element ) );\n\n\t\t\t\tif ( model.hasContent( rangeBefore, { ignoreWhitespaces: true } ) ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\t// Stop checking if there is some content after table.\n\t\t\t\tconst rangeAfter = model.createRange( model.createPositionAfter( element ), contentRange.end );\n\n\t\t\t\tif ( model.hasContent( rangeAfter, { ignoreWhitespaces: true } ) ) {\n\t\t\t\t\treturn null;\n\t\t\t\t}\n\n\t\t\t\t// There wasn't any content neither before nor after.\n\t\t\t\treturn element;\n\t\t\t}\n\t\t}\n\n\t\treturn null;\n\t}\n}\n\n/**\n * Prepares a table for pasting and returns adjusted selection dimensions.\n */\nfunction prepareTableForPasting(\n\tselectedTableCells: Array<Element>,\n\tpastedDimensions: {\n\t\theight: number;\n\t\twidth: number;\n\t},\n\twriter: Writer,\n\ttableUtils: TableUtils\n) {\n\tconst selectedTable = selectedTableCells[ 0 ].findAncestor( 'table' )!;\n\n\tconst columnIndexes = tableUtils.getColumnIndexes( selectedTableCells );\n\tconst rowIndexes = tableUtils.getRowIndexes( selectedTableCells );\n\n\tconst selection = {\n\t\tfirstColumn: columnIndexes.first,\n\t\tlastColumn: columnIndexes.last,\n\t\tfirstRow: rowIndexes.first,\n\t\tlastRow: rowIndexes.last\n\t};\n\n\t// Single cell selected - expand selection to pasted table dimensions.\n\tconst shouldExpandSelection = selectedTableCells.length === 1;\n\n\tif ( shouldExpandSelection ) {\n\t\tselection.lastRow += pastedDimensions.height - 1;\n\t\tselection.lastColumn += pastedDimensions.width - 1;\n\n\t\texpandTableSize( selectedTable, selection.lastRow + 1, selection.lastColumn + 1, tableUtils );\n\t}\n\n\t// In case of expanding selection we do not reset the selection so in this case we will always try to fix selection\n\t// like in the case of a non-rectangular area. This might be fixed by re-setting selected cells array but this shortcut is safe.\n\tif ( shouldExpandSelection || !tableUtils.isSelectionRectangular( selectedTableCells ) ) {\n\t\t// For a non-rectangular selection (ie in which some cells sticks out from a virtual selection rectangle) we need to create\n\t\t// a table layout that has a rectangular selection. This will split cells so the selection become rectangular.\n\t\t// Beyond this point we will operate on fixed content table.\n\t\tsplitCellsToRectangularSelection( selectedTable, selection, writer );\n\t}\n\t// However a selected table fragment might be invalid if examined alone. Ie such table fragment:\n\t//\n\t// +---+---+---+---+\n\t// 0 | a | b | c | d |\n\t// + + +---+---+\n\t// 1 | | e | f | g |\n\t// + +---+ +---+\n\t// 2 | | h | | i | <- last row, each cell has rowspan = 2,\n\t// + + + + + so we need to return 3, not 2\n\t// 3 | | | | |\n\t// +---+---+---+---+\n\t//\n\t// is invalid as the cells \"h\" and \"i\" have rowspans.\n\t// This case needs only adjusting the selection dimension as the rest of the algorithm operates on empty slots also.\n\telse {\n\t\tselection.lastRow = adjustLastRowIndex( selectedTable, selection );\n\t\tselection.lastColumn = adjustLastColumnIndex( selectedTable, selection );\n\t}\n\n\treturn selection;\n}\n\n/**\n * Expand table (in place) to expected size.\n */\nfunction expandTableSize( table: Element, expectedHeight: number, expectedWidth: number, tableUtils: TableUtils ) {\n\tconst tableWidth = tableUtils.getColumns( table );\n\tconst tableHeight = tableUtils.getRows( table );\n\n\tif ( expectedWidth > tableWidth ) {\n\t\ttableUtils.insertColumns( table, {\n\t\t\tat: tableWidth,\n\t\t\tcolumns: expectedWidth - tableWidth\n\t\t} );\n\t}\n\n\tif ( expectedHeight > tableHeight ) {\n\t\ttableUtils.insertRows( table, {\n\t\t\tat: tableHeight,\n\t\t\trows: expectedHeight - tableHeight\n\t\t} );\n\t}\n}\n\n/**\n * Returns two-dimensional array that is addressed by [ row ][ column ] that stores cells anchored at given location.\n *\n * At given row & column location it might be one of:\n *\n * * cell - cell from pasted table anchored at this location.\n * * null - if no cell is anchored at this location.\n *\n * For instance, from a table below:\n *\n * +----+----+----+----+\n * | 00 | 01 | 02 | 03 |\n * + +----+----+----+\n * | | 11 | 13 |\n * +----+ +----+\n * | 20 | | 23 |\n * +----+----+----+----+\n *\n * The method will return an array (numbers represents cell element):\n *\n * ```ts\n * const map = [\n * [ '00', '01', '02', '03' ],\n * [ null, '11', null, '13' ],\n * [ '20', null, null, '23' ]\n * ]\n * ```\n *\n * This allows for a quick access to table at give row & column. For instance to access table cell \"13\" from pasted table call:\n *\n * ```ts\n * const cell = map[ 1 ][ 3 ]\n * ```\n */\nfunction createLocationMap( table: Element, width: number, height: number ) {\n\t// Create height x width (row x column) two-dimensional table to store cells.\n\tconst map = new Array( height ).fill( null )\n\t\t.map( () => new Array( width ).fill( null ) );\n\n\tfor ( const { column, row, cell } of new TableWalker( table ) ) {\n\t\tmap[ row ][ column ] = cell;\n\t}\n\n\treturn map;\n}\n\n/**\n * Make selected cells rectangular by splitting the cells that stand out from a rectangular selection.\n *\n * In the table below a selection is shown with \"::\" and slots with anchor cells are named.\n *\n * +----+----+----+----+----+ +----+----+----+----+----+\n * | 00 | 01 | 02 | 03 | | 00 | 01 | 02 | 03 |\n * + +----+ +----+----+ | ::::::::::::::::----+\n * | | 11 | | 13 | 14 | | ::11 | | 13:: 14 | <- first row\n * +----+----+ + +----+ +----::---| | ::----+\n * | 20 | 21 | | | 24 | select cells: | 20 ::21 | | :: 24 |\n * +----+----+ +----+----+ 11 -> 33 +----::---| |---::----+\n * | 30 | | 33 | 34 | | 30 :: | | 33:: 34 | <- last row\n * + + +----+ + | :::::::::::::::: +\n * | | | 43 | | | | | 43 | |\n * +----+----+----+----+----+ +----+----+----+----+----+\n * ^ ^\n * first & last columns\n *\n * Will update table to:\n *\n * +----+----+----+----+----+\n * | 00 | 01 | 02 | 03 |\n * + +----+----+----+----+\n * | | 11 | | 13 | 14 |\n * +----+----+ + +----+\n * | 20 | 21 | | | 24 |\n * +----+----+ +----+----+\n * | 30 | | | 33 | 34 |\n * + +----+----+----+ +\n * | | | | 43 | |\n * +----+----+----+----+----+\n *\n * In th example above:\n * - Cell \"02\" which have `rowspan = 4` must be trimmed at first and at after last row.\n * - Cell \"03\" which have `rowspan = 2` and `colspan = 2` must be trimmed at first column and after last row.\n * - Cells \"00\", \"03\" & \"30\" which cannot be cut by this algorithm as they are outside the trimmed area.\n * - Cell \"13\" cannot be cut as it is inside the trimmed area.\n */\nfunction splitCellsToRectangularSelection( table: Element, dimensions: Record<string, number>, writer: Writer ) {\n\tconst { firstRow, lastRow, firstColumn, lastColumn } = dimensions;\n\n\tconst rowIndexes = { first: firstRow, last: lastRow };\n\tconst columnIndexes = { first: firstColumn, last: lastColumn };\n\n\t// 1. Split cells vertically in two steps as first step might create cells that needs to split again.\n\tdoVerticalSplit( table, firstColumn, rowIndexes, writer );\n\tdoVerticalSplit( table, lastColumn + 1, rowIndexes, writer );\n\n\t// 2. Split cells horizontally in two steps as first step might create cells that needs to split again.\n\tdoHorizontalSplit( table, firstRow, columnIndexes, writer );\n\tdoHorizontalSplit( table, lastRow + 1, columnIndexes, writer, firstRow );\n}\n\nfunction doHorizontalSplit( table: Element, splitRow: number, limitColumns: Record<string, number>, writer: Writer, startRow: number = 0 ) {\n\t// If selection starts at first row then no split is needed.\n\tif ( splitRow < 1 ) {\n\t\treturn;\n\t}\n\n\tconst overlappingCells = getVerticallyOverlappingCells( table, splitRow, startRow );\n\n\t// Filter out cells that are not touching insides of the rectangular selection.\n\tconst cellsToSplit = overlappingCells.filter( ( { column, cellWidth } ) => isAffectedBySelection( column, cellWidth, limitColumns ) );\n\n\treturn cellsToSplit.map( ( { cell } ) => splitHorizontally( cell, splitRow, writer ) );\n}\n\nfunction doVerticalSplit( table: Element, splitColumn: number, limitRows: Record<string, number>, writer: Writer ) {\n\t// If selection starts at first column then no split is needed.\n\tif ( splitColumn < 1 ) {\n\t\treturn;\n\t}\n\n\tconst overlappingCells = getHorizontallyOverlappingCells( table, splitColumn );\n\n\t// Filter out cells that are not touching insides of the rectangular selection.\n\tconst cellsToSplit = overlappingCells.filter( ( { row, cellHeight } ) => isAffectedBySelection( row, cellHeight, limitRows ) );\n\n\treturn cellsToSplit.map( ( { cell, column } ) => splitVertically( cell, column, splitColumn, writer ) );\n}\n\n/**\n * Checks if cell at given row (column) is affected by a rectangular selection defined by first/last column (row).\n *\n * The same check is used for row as for column.\n */\nfunction isAffectedBySelection( index: number, span: number, limit: Record<string, number> ) {\n\tconst endIndex = index + span - 1;\n\tconst { first, last } = limit;\n\n\tconst isInsideSelection = index >= first && index <= last;\n\tconst overlapsSelectionFromOutside = index < first && endIndex >= first;\n\n\treturn isInsideSelection || overlapsSelectionFromOutside;\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tablekeyboard\n */\n\nimport TableSelection from './tableselection';\nimport TableWalker from './tablewalker';\nimport TableUtils from './tableutils';\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport {\n\tgetLocalizedArrowKeyCodeDirection,\n\ttype EventInfo,\n\ttype ArrowKeyCodeDirection,\n\ttype KeystrokeInfo\n} from 'ckeditor5/src/utils';\n\nimport type {\n\tBubblingEventInfo,\n\tDocumentSelection,\n\tDomEventData,\n\tElement,\n\tSelection,\n\tViewDocumentArrowKeyEvent,\n\tViewDocumentTabEvent\n} from 'ckeditor5/src/engine';\n\n/**\n * This plugin enables keyboard navigation for tables.\n * It is loaded automatically by the {@link module:table/table~Table} plugin.\n */\nexport default class TableKeyboard extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TableKeyboard' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ TableSelection, TableUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst view = this.editor.editing.view;\n\t\tconst viewDocument = view.document;\n\n\t\tthis.listenTo<ViewDocumentArrowKeyEvent>(\n\t\t\tviewDocument,\n\t\t\t'arrowKey',\n\t\t\t( ...args ) => this._onArrowKey( ...args ),\n\t\t\t{ context: 'table' }\n\t\t);\n\n\t\tthis.listenTo<ViewDocumentTabEvent>(\n\t\t\tviewDocument,\n\t\t\t'tab',\n\t\t\t( ...args ) => this._handleTabOnSelectedTable( ...args ),\n\t\t\t{ context: 'figure' }\n\t\t);\n\n\t\tthis.listenTo<ViewDocumentTabEvent>(\n\t\t\tviewDocument,\n\t\t\t'tab',\n\t\t\t( ...args ) => this._handleTab( ...args ),\n\t\t\t{ context: [ 'th', 'td' ] }\n\t\t);\n\t}\n\n\t/**\n\t * Handles {@link module:engine/view/document~Document#event:tab tab} events for the <kbd>Tab</kbd> key executed\n\t * when the table widget is selected.\n\t */\n\tprivate _handleTabOnSelectedTable( bubblingEventInfo: BubblingEventInfo, domEventData: DomEventData ) {\n\t\tconst editor = this.editor;\n\t\tconst selection = editor.model.document.selection;\n\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\tif ( !selectedElement || !selectedElement.is( 'element', 'table' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tdomEventData.preventDefault();\n\t\tdomEventData.stopPropagation();\n\t\tbubblingEventInfo.stop();\n\n\t\teditor.model.change( writer => {\n\t\t\twriter.setSelection( writer.createRangeIn( ( selectedElement.getChild( 0 ) as Element ).getChild( 0 ) as Element ) );\n\t\t} );\n\t}\n\n\t/**\n\t * Handles {@link module:engine/view/document~Document#event:tab tab} events for the <kbd>Tab</kbd> key executed\n\t * inside table cells.\n\t */\n\tprivate _handleTab( bubblingEventInfo: BubblingEventInfo, domEventData: DomEventData & KeystrokeInfo ) {\n\t\tconst editor = this.editor;\n\t\tconst tableUtils: TableUtils = this.editor.plugins.get( TableUtils );\n\t\tconst tableSelection: TableSelection = this.editor.plugins.get( 'TableSelection' );\n\n\t\tconst selection = editor.model.document.selection;\n\t\tconst isForward = !domEventData.shiftKey;\n\n\t\tlet tableCell: Element | null = tableUtils.getTableCellsContainingSelection( selection )[ 0 ];\n\n\t\tif ( !tableCell ) {\n\t\t\ttableCell = tableSelection.getFocusCell();\n\t\t}\n\n\t\tif ( !tableCell ) {\n\t\t\treturn;\n\t\t}\n\n\t\tdomEventData.preventDefault();\n\t\tdomEventData.stopPropagation();\n\t\tbubblingEventInfo.stop();\n\n\t\tconst tableRow = tableCell.parent as Element;\n\t\tconst table = tableRow.parent as Element;\n\n\t\tconst currentRowIndex = table.getChildIndex( tableRow )!;\n\t\tconst currentCellIndex = tableRow.getChildIndex( tableCell )!;\n\n\t\tconst isFirstCellInRow = currentCellIndex === 0;\n\n\t\tif ( !isForward && isFirstCellInRow && currentRowIndex === 0 ) {\n\t\t\t// Set the selection over the whole table if the selection was in the first table cell.\n\t\t\teditor.model.change( writer => {\n\t\t\t\twriter.setSelection( writer.createRangeOn( table ) );\n\t\t\t} );\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst isLastCellInRow = currentCellIndex === tableRow.childCount - 1;\n\t\tconst isLastRow = currentRowIndex === tableUtils.getRows( table ) - 1;\n\n\t\tif ( isForward && isLastRow && isLastCellInRow ) {\n\t\t\teditor.execute( 'insertTableRowBelow' );\n\n\t\t\t// Check if the command actually added a row. If `insertTableRowBelow` execution didn't add a row (because it was disabled\n\t\t\t// or it got overwritten) set the selection over the whole table to mirror the first cell case.\n\t\t\tif ( currentRowIndex === tableUtils.getRows( table ) - 1 ) {\n\t\t\t\teditor.model.change( writer => {\n\t\t\t\t\twriter.setSelection( writer.createRangeOn( table ) );\n\t\t\t\t} );\n\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\n\t\tlet cellToFocus: Element;\n\n\t\t// Move to the first cell in the next row.\n\t\tif ( isForward && isLastCellInRow ) {\n\t\t\tconst nextRow = table.getChild( currentRowIndex + 1 ) as Element;\n\n\t\t\tcellToFocus = nextRow.getChild( 0 ) as Element;\n\t\t}\n\t\t// Move to the last cell in the previous row.\n\t\telse if ( !isForward && isFirstCellInRow ) {\n\t\t\tconst previousRow = table.getChild( currentRowIndex - 1 ) as Element;\n\n\t\t\tcellToFocus = previousRow.getChild( previousRow.childCount - 1 ) as Element;\n\t\t}\n\t\t// Move to the next/previous cell.\n\t\telse {\n\t\t\tcellToFocus = tableRow.getChild( currentCellIndex + ( isForward ? 1 : -1 ) ) as Element;\n\t\t}\n\n\t\teditor.model.change( writer => {\n\t\t\twriter.setSelection( writer.createRangeIn( cellToFocus ) );\n\t\t} );\n\t}\n\n\t/**\n\t * Handles {@link module:engine/view/document~Document#event:keydown keydown} events.\n\t */\n\tprivate _onArrowKey( eventInfo: EventInfo, domEventData: DomEventData & KeystrokeInfo ) {\n\t\tconst editor = this.editor;\n\t\tconst keyCode = domEventData.keyCode;\n\n\t\tconst direction = getLocalizedArrowKeyCodeDirection( keyCode, editor.locale.contentLanguageDirection )!;\n\t\tconst wasHandled = this._handleArrowKeys( direction, domEventData.shiftKey );\n\n\t\tif ( wasHandled ) {\n\t\t\tdomEventData.preventDefault();\n\t\t\tdomEventData.stopPropagation();\n\t\t\teventInfo.stop();\n\t\t}\n\t}\n\n\t/**\n\t * Handles arrow keys to move the selection around the table.\n\t *\n\t * @param direction The direction of the arrow key.\n\t * @param expandSelection If the current selection should be expanded.\n\t * @returns Returns `true` if key was handled.\n\t */\n\tprivate _handleArrowKeys( direction: ArrowKeyCodeDirection, expandSelection: boolean ) {\n\t\tconst tableUtils = this.editor.plugins.get( TableUtils );\n\t\tconst tableSelection: TableSelection = this.editor.plugins.get( 'TableSelection' );\n\t\tconst model = this.editor.model;\n\t\tconst selection = model.document.selection;\n\t\tconst isForward = [ 'right', 'down' ].includes( direction );\n\n\t\t// In case one or more table cells are selected (from outside),\n\t\t// move the selection to a cell adjacent to the selected table fragment.\n\t\tconst selectedCells = tableUtils.getSelectedTableCells( selection );\n\n\t\tif ( selectedCells.length ) {\n\t\t\tlet focusCell;\n\n\t\t\tif ( expandSelection ) {\n\t\t\t\tfocusCell = tableSelection.getFocusCell();\n\t\t\t} else {\n\t\t\t\tfocusCell = isForward ? selectedCells[ selectedCells.length - 1 ] : selectedCells[ 0 ];\n\t\t\t}\n\n\t\t\tthis._navigateFromCellInDirection( focusCell!, direction, expandSelection );\n\n\t\t\treturn true;\n\t\t}\n\n\t\t// Abort if we're not in a table cell.\n\t\tconst tableCell = selection.focus!.findAncestor( 'tableCell' );\n\n\t\t/* istanbul ignore if: paranoid check -- @preserve */\n\t\tif ( !tableCell ) {\n\t\t\treturn false;\n\t\t}\n\n\t\t// When the selection is not collapsed.\n\t\tif ( !selection.isCollapsed ) {\n\t\t\tif ( expandSelection ) {\n\t\t\t\t// Navigation is in the opposite direction than the selection direction so this is shrinking of the selection.\n\t\t\t\t// Selection for sure will not approach cell edge.\n\t\t\t\t//\n\t\t\t\t// With a special case when all cell content is selected - then selection should expand to the other cell.\n\t\t\t\t// Note: When the entire cell gets selected using CTRL+A, the selection is always forward.\n\t\t\t\tif ( selection.isBackward == isForward && !selection.containsEntireContent( tableCell ) ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tconst selectedElement = selection.getSelectedElement();\n\n\t\t\t\t// It will collapse for non-object selected so it's not going to move to other cell.\n\t\t\t\tif ( !selectedElement || !model.schema.isObject( selectedElement ) ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// Let's check if the selection is at the beginning/end of the cell.\n\t\tif ( this._isSelectionAtCellEdge( selection, tableCell, isForward ) ) {\n\t\t\tthis._navigateFromCellInDirection( tableCell, direction, expandSelection );\n\n\t\t\treturn true;\n\t\t}\n\n\t\treturn false;\n\t}\n\n\t/**\n\t * Returns `true` if the selection is at the boundary of a table cell according to the navigation direction.\n\t *\n\t * @param selection The current selection.\n\t * @param tableCell The current table cell element.\n\t * @param isForward The expected navigation direction.\n\t */\n\tprivate _isSelectionAtCellEdge( selection: Selection | DocumentSelection, tableCell: Element, isForward: boolean ) {\n\t\tconst model = this.editor.model;\n\t\tconst schema = this.editor.model.schema;\n\n\t\tconst focus = isForward ? selection.getLastPosition()! : selection.getFirstPosition()!;\n\n\t\t// If the current limit element is not table cell we are for sure not at the cell edge.\n\t\t// Also `modifySelection` will not let us out of it.\n\t\tif ( !schema.getLimitElement( focus ).is( 'element', 'tableCell' ) ) {\n\t\t\tconst boundaryPosition = model.createPositionAt( tableCell, isForward ? 'end' : 0 );\n\n\t\t\treturn boundaryPosition.isTouching( focus );\n\t\t}\n\n\t\tconst probe = model.createSelection( focus );\n\n\t\tmodel.modifySelection( probe, { direction: isForward ? 'forward' : 'backward' } );\n\n\t\t// If there was no change in the focus position, then it's not possible to move the selection there.\n\t\treturn focus.isEqual( probe.focus! );\n\t}\n\n\t/**\n\t * Moves the selection from the given table cell in the specified direction.\n\t *\n\t * @param focusCell The table cell that is current multi-cell selection focus.\n\t * @param direction Direction in which selection should move.\n\t * @param expandSelection If the current selection should be expanded. Default value is false.\n\t */\n\tprotected _navigateFromCellInDirection( focusCell: Element, direction: ArrowKeyCodeDirection, expandSelection = false ): void {\n\t\tconst model = this.editor.model;\n\n\t\tconst table = focusCell.findAncestor( 'table' )!;\n\t\tconst tableMap = [ ...new TableWalker( table, { includeAllSlots: true } ) ];\n\t\tconst { row: lastRow, column: lastColumn } = tableMap[ tableMap.length - 1 ];\n\n\t\tconst currentCellInfo = tableMap.find( ( { cell } ) => cell == focusCell )!;\n\t\tlet { row, column } = currentCellInfo;\n\n\t\tswitch ( direction ) {\n\t\t\tcase 'left':\n\t\t\t\tcolumn--;\n\t\t\t\tbreak;\n\n\t\t\tcase 'up':\n\t\t\t\trow--;\n\t\t\t\tbreak;\n\n\t\t\tcase 'right':\n\t\t\t\tcolumn += currentCellInfo.cellWidth;\n\t\t\t\tbreak;\n\n\t\t\tcase 'down':\n\t\t\t\trow += currentCellInfo.cellHeight;\n\t\t\t\tbreak;\n\t\t}\n\n\t\tconst isOutsideVertically = row < 0 || row > lastRow;\n\t\tconst isBeforeFirstCell = column < 0 && row <= 0;\n\t\tconst isAfterLastCell = column > lastColumn && row >= lastRow;\n\n\t\t// Note that if the table cell at the end of a row is row-spanned then isAfterLastCell will never be true.\n\t\t// However, we don't know if user was navigating on the last row or not, so let's stay in the table.\n\n\t\tif ( isOutsideVertically || isBeforeFirstCell || isAfterLastCell ) {\n\t\t\tmodel.change( writer => {\n\t\t\t\twriter.setSelection( writer.createRangeOn( table ) );\n\t\t\t} );\n\n\t\t\treturn;\n\t\t}\n\n\t\tif ( column < 0 ) {\n\t\t\tcolumn = expandSelection ? 0 : lastColumn;\n\t\t\trow--;\n\t\t} else if ( column > lastColumn ) {\n\t\t\tcolumn = expandSelection ? lastColumn : 0;\n\t\t\trow++;\n\t\t}\n\n\t\tconst cellToSelect = tableMap.find( cellInfo => cellInfo.row == row && cellInfo.column == column )!.cell;\n\t\tconst isForward = [ 'right', 'down' ].includes( direction );\n\t\tconst tableSelection: TableSelection = this.editor.plugins.get( 'TableSelection' );\n\n\t\tif ( expandSelection && tableSelection.isEnabled ) {\n\t\t\tconst anchorCell = tableSelection.getAnchorCell() || focusCell;\n\n\t\t\ttableSelection.setCellSelection( anchorCell, cellToSelect );\n\t\t} else {\n\t\t\tconst positionToSelect = model.createPositionAt( cellToSelect, isForward ? 0 : 'end' );\n\n\t\t\tmodel.change( writer => {\n\t\t\t\twriter.setSelection( positionToSelect );\n\t\t\t} );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tablemouse/mouseeventsobserver\n */\n\nimport { DomEventObserver, type DomEventData, type View } from 'ckeditor5/src/engine';\n\n/**\n * The mouse selection event observer.\n *\n * It registers listeners for the following DOM events:\n *\n * - `'mousemove'`\n * - `'mouseleave'`\n *\n * Note that this observer is disabled by default. To enable this observer, it needs to be added to\n * {@link module:engine/view/view~View} using the {@link module:engine/view/view~View#addObserver} method.\n *\n * The observer is registered by the {@link module:table/tableselection~TableSelection} plugin.\n */\nexport default class MouseEventsObserver extends DomEventObserver<'mousemove' | 'mouseleave'> {\n\tpublic readonly domEventType = [\n\t\t'mousemove', 'mouseleave'\n\t] as const;\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic onDomEvent( domEvent: MouseEvent ): void {\n\t\tthis.fire( domEvent.type, domEvent );\n\t}\n}\n\n/**\n * Fired when the mouse is moved over one of the editables.\n *\n * Introduced by {@link module:table/tablemouse/mouseeventsobserver~MouseEventsObserver}.\n *\n * Note that this event is not available by default. To make it available,\n * {@link module:table/tablemouse/mouseeventsobserver~MouseEventsObserver} needs to be added\n * to {@link module:engine/view/view~View} using the {@link module:engine/view/view~View#addObserver} method.\n *\n * @see module:table/tablemouse/mouseeventsobserver~MouseEventsObserver\n * @eventName module:engine/view/document~Document#mousemove\n * @param data Event data.\n */\nexport type ViewDocumentMouseMoveEvent = {\n\tname: 'mousemove';\n\targs: [ data: DomEventData<MouseEvent> ];\n};\n\n/**\n * Fired when the mouse is moved out of one of the editables.\n *\n * Introduced by {@link module:table/tablemouse/mouseeventsobserver~MouseEventsObserver}.\n *\n * Note that this event is not available by default. To make it available,\n * {@link module:table/tablemouse/mouseeventsobserver~MouseEventsObserver} needs to be added\n * to {@link module:engine/view/view~View} using the {@link module:engine/view/view~View#addObserver} method.\n *\n * @see module:table/tablemouse/mouseeventsobserver~MouseEventsObserver\n * @eventName module:engine/view/document~Document#mouseleave\n * @param data Event data.\n */\nexport type ViewDocumentMouseLeaveEvent = {\n\tname: 'mouseleave';\n\targs: [ data: DomEventData<MouseEvent> ];\n};\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tablemouse\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport TableSelection from './tableselection';\nimport MouseEventsObserver from './tablemouse/mouseeventsobserver';\nimport TableUtils from './tableutils';\nimport type { DomEventData, Element } from 'ckeditor5/src/engine';\n\n/**\n * This plugin enables a table cells' selection with the mouse.\n * It is loaded automatically by the {@link module:table/table~Table} plugin.\n */\nexport default class TableMouse extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TableMouse' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ TableSelection, TableUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\t// Currently the MouseObserver only handles `mousedown` and `mouseup` events.\n\t\t// TODO move to the engine?\n\t\teditor.editing.view.addObserver( MouseEventsObserver );\n\n\t\tthis._enableShiftClickSelection();\n\t\tthis._enableMouseDragSelection();\n\t}\n\n\t/**\n\t * Enables making cells selection by <kbd>Shift</kbd>+click. Creates a selection from the cell which previously held\n\t * the selection to the cell which was clicked. It can be the same cell, in which case it selects a single cell.\n\t */\n\tprivate _enableShiftClickSelection() {\n\t\tconst editor = this.editor;\n\t\tconst tableUtils = editor.plugins.get( TableUtils );\n\t\tlet blockSelectionChange = false;\n\n\t\tconst tableSelection = editor.plugins.get( TableSelection );\n\n\t\tthis.listenTo( editor.editing.view.document, 'mousedown', ( evt, domEventData ) => {\n\t\t\tconst selection = editor.model.document.selection;\n\n\t\t\tif ( !this.isEnabled || !tableSelection.isEnabled ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( !domEventData.domEvent.shiftKey ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst anchorCell = tableSelection.getAnchorCell() || tableUtils.getTableCellsContainingSelection( selection )[ 0 ];\n\n\t\t\tif ( !anchorCell ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst targetCell = this._getModelTableCellFromDomEvent( domEventData );\n\n\t\t\tif ( targetCell && haveSameTableParent( anchorCell, targetCell ) ) {\n\t\t\t\tblockSelectionChange = true;\n\t\t\t\ttableSelection.setCellSelection( anchorCell, targetCell );\n\n\t\t\t\tdomEventData.preventDefault();\n\t\t\t}\n\t\t} );\n\n\t\tthis.listenTo( editor.editing.view.document, 'mouseup', () => {\n\t\t\tblockSelectionChange = false;\n\t\t} );\n\n\t\t// We need to ignore a `selectionChange` event that is fired after we render our new table cells selection.\n\t\t// When downcasting table cells selection to the view, we put the view selection in the last selected cell\n\t\t// in a place that may not be natively a \"correct\" location. This is we put it directly in the `<td>` element.\n\t\t// All browsers fire the native `selectionchange` event.\n\t\t// However, all browsers except Safari return the selection in the exact place where we put it\n\t\t// (even though it's visually normalized). Safari returns `<td><p>^foo` that makes our selection observer\n\t\t// fire our `selectionChange` event (because the view selection that we set in the first step differs from the DOM selection).\n\t\t// Since `selectionChange` is fired, we automatically update the model selection that moves it that paragraph.\n\t\t// This breaks our dear cells selection.\n\t\t//\n\t\t// Theoretically this issue concerns only Safari that is the only browser that do normalize the selection.\n\t\t// However, to avoid code branching and to have a good coverage for this event blocker, I enabled it for all browsers.\n\t\t//\n\t\t// Note: I'm keeping the `blockSelectionChange` state separately for shift+click and mouse drag (exact same logic)\n\t\t// so I don't have to try to analyze whether they don't overlap in some weird cases. Probably they don't.\n\t\t// But I have other things to do, like writing this comment.\n\t\tthis.listenTo( editor.editing.view.document, 'selectionChange', evt => {\n\t\t\tif ( blockSelectionChange ) {\n\t\t\t\t// @if CK_DEBUG // console.log( 'Blocked selectionChange to avoid breaking table cells selection.' );\n\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { priority: 'highest' } );\n\t}\n\n\t/**\n\t * Enables making cells selection by dragging.\n\t *\n\t * The selection is made only on mousemove. Mouse tracking is started on mousedown.\n\t * However, the cells selection is enabled only after the mouse cursor left the anchor cell.\n\t * Thanks to that normal text selection within one cell works just fine. However, you can still select\n\t * just one cell by leaving the anchor cell and moving back to it.\n\t */\n\tprivate _enableMouseDragSelection() {\n\t\tconst editor = this.editor;\n\t\tlet anchorCell: Element | null, targetCell: Element | null;\n\t\tlet beganCellSelection = false;\n\t\tlet blockSelectionChange = false;\n\n\t\tconst tableSelection = editor.plugins.get( TableSelection );\n\n\t\tthis.listenTo( editor.editing.view.document, 'mousedown', ( evt, domEventData ) => {\n\t\t\tif ( !this.isEnabled || !tableSelection.isEnabled ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\t// Make sure to not conflict with the shift+click listener and any other possible handler.\n\t\t\tif ( domEventData.domEvent.shiftKey || domEventData.domEvent.ctrlKey || domEventData.domEvent.altKey ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tanchorCell = this._getModelTableCellFromDomEvent( domEventData );\n\t\t} );\n\n\t\tthis.listenTo( editor.editing.view.document, 'mousemove', ( evt, domEventData ) => {\n\t\t\tif ( !domEventData.domEvent.buttons ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( !anchorCell ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tconst newTargetCell = this._getModelTableCellFromDomEvent( domEventData );\n\n\t\t\tif ( newTargetCell && haveSameTableParent( anchorCell, newTargetCell ) ) {\n\t\t\t\ttargetCell = newTargetCell;\n\n\t\t\t\t// Switch to the cell selection mode after the mouse cursor left the anchor cell.\n\t\t\t\t// Switch off only on mouseup (makes selecting a single cell possible).\n\t\t\t\tif ( !beganCellSelection && targetCell != anchorCell ) {\n\t\t\t\t\tbeganCellSelection = true;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// Yep, not making a cell selection yet. See method docs.\n\t\t\tif ( !beganCellSelection ) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tblockSelectionChange = true;\n\t\t\ttableSelection.setCellSelection( anchorCell, targetCell! );\n\n\t\t\tdomEventData.preventDefault();\n\t\t} );\n\n\t\tthis.listenTo( editor.editing.view.document, 'mouseup', () => {\n\t\t\tbeganCellSelection = false;\n\t\t\tblockSelectionChange = false;\n\t\t\tanchorCell = null;\n\t\t\ttargetCell = null;\n\t\t} );\n\n\t\t// See the explanation in `_enableShiftClickSelection()`.\n\t\tthis.listenTo( editor.editing.view.document, 'selectionChange', evt => {\n\t\t\tif ( blockSelectionChange ) {\n\t\t\t\t// @if CK_DEBUG // console.log( 'Blocked selectionChange to avoid breaking table cells selection.' );\n\n\t\t\t\tevt.stop();\n\t\t\t}\n\t\t}, { priority: 'highest' } );\n\t}\n\n\t/**\n\t * Returns the model table cell element based on the target element of the passed DOM event.\n\t *\n\t * @returns Returns the table cell or `undefined`.\n\t */\n\tprivate _getModelTableCellFromDomEvent( domEventData: DomEventData ) {\n\t\t// Note: Work with positions (not element mapping) because the target element can be an attribute or other non-mapped element.\n\t\tconst viewTargetElement = domEventData.target;\n\t\tconst viewPosition = this.editor.editing.view.createPositionAt( viewTargetElement, 0 );\n\t\tconst modelPosition = this.editor.editing.mapper.toModelPosition( viewPosition );\n\t\tconst modelElement = modelPosition.parent as Element;\n\n\t\treturn modelElement.findAncestor( 'tableCell', { includeSelf: true } );\n\t}\n}\n\nfunction haveSameTableParent( cellA: Element, cellB: Element ) {\n\treturn cellA.parent!.parent == cellB.parent!.parent;\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./table.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/utils/ui/widget\n */\n\nimport type { ViewDocumentFragment, ViewDocumentSelection, ViewElement, ViewNode } from 'ckeditor5/src/engine';\n\nimport { isWidget } from 'ckeditor5/src/widget';\n\n/**\n * Returns a table widget editing view element if one is selected.\n */\nexport function getSelectedTableWidget( selection: ViewDocumentSelection ): ViewElement | null {\n\tconst viewElement = selection.getSelectedElement();\n\n\tif ( viewElement && isTableWidget( viewElement ) ) {\n\t\treturn viewElement;\n\t}\n\n\treturn null;\n}\n\n/**\n * Returns a table widget editing view element if one is among the selection's ancestors.\n */\nexport function getTableWidgetAncestor( selection: ViewDocumentSelection ): ViewElement | null {\n\tconst selectionPosition = selection.getFirstPosition();\n\n\tif ( !selectionPosition ) {\n\t\treturn null;\n\t}\n\n\tlet parent: ViewNode | ViewDocumentFragment | null = selectionPosition.parent;\n\n\twhile ( parent ) {\n\t\tif ( parent.is( 'element' ) && isTableWidget( parent ) ) {\n\t\t\treturn parent;\n\t\t}\n\n\t\tparent = parent.parent;\n\t}\n\n\treturn null;\n}\n\n/**\n * Checks if a given view element is a table widget.\n */\nfunction isTableWidget( viewElement: ViewElement ): boolean {\n\treturn !!viewElement.getCustomProperty( 'table' ) && isWidget( viewElement );\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./colorinput.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./formrow.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./form.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./tableform.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./tablecellproperties.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/utils/ui/contextualballoon\n */\n\nimport { Rect, type PositionOptions } from 'ckeditor5/src/utils';\nimport { BalloonPanelView, type ContextualBalloon } from 'ckeditor5/src/ui';\n\nimport { getTableWidgetAncestor } from './widget';\nimport type { Editor } from 'ckeditor5/src/core';\nimport type { Element, Position, Range } from 'ckeditor5/src/engine';\n\nconst DEFAULT_BALLOON_POSITIONS = BalloonPanelView.defaultPositions;\n\nconst BALLOON_POSITIONS = [\n\tDEFAULT_BALLOON_POSITIONS.northArrowSouth,\n\tDEFAULT_BALLOON_POSITIONS.northArrowSouthWest,\n\tDEFAULT_BALLOON_POSITIONS.northArrowSouthEast,\n\tDEFAULT_BALLOON_POSITIONS.southArrowNorth,\n\tDEFAULT_BALLOON_POSITIONS.southArrowNorthWest,\n\tDEFAULT_BALLOON_POSITIONS.southArrowNorthEast,\n\tDEFAULT_BALLOON_POSITIONS.viewportStickyNorth\n];\n\n/**\n * A helper utility that positions the\n * {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon contextual balloon} instance\n * with respect to the table in the editor content, if one is selected.\n *\n * @param editor The editor instance.\n * @param target Either \"cell\" or \"table\". Determines the target the balloon will be attached to.\n */\nexport function repositionContextualBalloon( editor: Editor, target: string ): void {\n\tconst balloon: ContextualBalloon = editor.plugins.get( 'ContextualBalloon' );\n\n\tif ( getTableWidgetAncestor( editor.editing.view.document.selection ) ) {\n\t\tlet position;\n\n\t\tif ( target === 'cell' ) {\n\t\t\tposition = getBalloonCellPositionData( editor );\n\t\t} else {\n\t\t\tposition = getBalloonTablePositionData( editor );\n\t\t}\n\n\t\tballoon.updatePosition( position );\n\t}\n}\n\n/**\n * Returns the positioning options that control the geometry of the\n * {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon contextual balloon} with respect\n * to the selected table in the editor content.\n *\n * @param editor The editor instance.\n */\nexport function getBalloonTablePositionData( editor: Editor ): Partial<PositionOptions> {\n\tconst firstPosition = editor.model.document.selection.getFirstPosition()!;\n\tconst modelTable = firstPosition.findAncestor( 'table' )!;\n\tconst viewTable = editor.editing.mapper.toViewElement( modelTable )!;\n\n\treturn {\n\t\ttarget: editor.editing.view.domConverter.mapViewToDom( viewTable )!,\n\t\tpositions: BALLOON_POSITIONS\n\t};\n}\n\n/**\n * Returns the positioning options that control the geometry of the\n * {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon contextual balloon} with respect\n * to the selected table cell in the editor content.\n *\n * @param editor The editor instance.\n */\nexport function getBalloonCellPositionData( editor: Editor ): Partial<PositionOptions> {\n\tconst mapper = editor.editing.mapper;\n\tconst domConverter = editor.editing.view.domConverter;\n\tconst selection = editor.model.document.selection;\n\n\tif ( selection.rangeCount > 1 ) {\n\t\treturn {\n\t\t\ttarget: () => createBoundingRect( selection.getRanges(), editor ),\n\t\t\tpositions: BALLOON_POSITIONS\n\t\t};\n\t}\n\n\tconst modelTableCell = getTableCellAtPosition( selection.getFirstPosition()! );\n\tconst viewTableCell = mapper.toViewElement( modelTableCell )!;\n\n\treturn {\n\t\ttarget: domConverter.mapViewToDom( viewTableCell ),\n\t\tpositions: BALLOON_POSITIONS\n\t};\n}\n\n/**\n * Returns the first selected table cell from a multi-cell or in-cell selection.\n *\n * @param position Document position.\n */\nfunction getTableCellAtPosition( position: Position ): Element {\n\tconst isTableCellSelected = position.nodeAfter && position.nodeAfter.is( 'element', 'tableCell' );\n\n\treturn isTableCellSelected ? position.nodeAfter : position.findAncestor( 'tableCell' )!;\n}\n\n/**\n * Returns bounding rectangle for given model ranges.\n *\n * @param ranges Model ranges that the bounding rect should be returned for.\n * @param editor The editor instance.\n */\nfunction createBoundingRect( ranges: Iterable<Range>, editor: Editor ): Rect {\n\tconst mapper = editor.editing.mapper;\n\tconst domConverter = editor.editing.view.domConverter;\n\tconst rects = Array.from( ranges ).map( range => {\n\t\tconst modelTableCell = getTableCellAtPosition( range.start );\n\t\tconst viewTableCell = mapper.toViewElement( modelTableCell )!;\n\t\treturn new Rect( domConverter.mapViewToDom( viewTableCell )! );\n\t} );\n\n\treturn Rect.getBoundingRect( rects )!;\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./tableproperties.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./tablecaption.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./tablecolumnresize.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module cloud-services/token/token\n */\n\n/* globals XMLHttpRequest, setTimeout, clearTimeout, atob */\n\nimport { ObservableMixin, CKEditorError } from 'ckeditor5/src/utils';\nimport type { TokenUrl } from '../cloudservicesconfig';\n\nconst DEFAULT_OPTIONS = { autoRefresh: true };\nconst DEFAULT_TOKEN_REFRESH_TIMEOUT_TIME = 3600000;\n\n/**\n * Class representing the token used for communication with CKEditor Cloud Services.\n * Value of the token is retrieving from the specified URL and is refreshed every 1 hour by default.\n */\nexport default class Token extends ObservableMixin() {\n\t/**\n\t * Value of the token.\n\t * The value of the token is undefined if `initValue` is not provided or `init` method was not called.\n\t * `create` method creates token with initialized value from url.\n\t *\n\t * @see module:cloud-services/token/token~InitializedToken\n\t * @observable\n\t * @readonly\n\t */\n\tdeclare public value: string | undefined;\n\n\t/**\n\t * Base refreshing function.\n\t */\n\tprivate _refresh: () => Promise<string>;\n\n\tprivate _options: { initValue?: string; autoRefresh: boolean };\n\n\tprivate _tokenRefreshTimeout?: ReturnType<typeof setTimeout>;\n\n\t/**\n\t * Creates `Token` instance.\n\t * Method `init` should be called after using the constructor or use `create` method instead.\n\t *\n\t * @param tokenUrlOrRefreshToken Endpoint address to download the token or a callback that provides the token. If the\n\t * value is a function it has to match the {@link module:cloud-services/token/token~Token#refreshToken} interface.\n\t */\n\tconstructor( tokenUrlOrRefreshToken: TokenUrl, options: TokenOptions = {} ) {\n\t\tsuper();\n\n\t\tif ( !tokenUrlOrRefreshToken ) {\n\t\t\t/**\n\t\t\t * A `tokenUrl` must be provided as the first constructor argument.\n\t\t\t *\n\t\t\t * @error token-missing-token-url\n\t\t\t */\n\t\t\tthrow new CKEditorError(\n\t\t\t\t'token-missing-token-url',\n\t\t\t\tthis\n\t\t\t);\n\t\t}\n\n\t\tif ( options.initValue ) {\n\t\t\tthis._validateTokenValue( options.initValue );\n\t\t}\n\n\t\tthis.set( 'value', options.initValue );\n\n\t\tif ( typeof tokenUrlOrRefreshToken === 'function' ) {\n\t\t\tthis._refresh = tokenUrlOrRefreshToken;\n\t\t} else {\n\t\t\tthis._refresh = () => defaultRefreshToken( tokenUrlOrRefreshToken );\n\t\t}\n\n\t\tthis._options = { ...DEFAULT_OPTIONS, ...options };\n\t}\n\n\t/**\n\t * Initializes the token.\n\t */\n\tpublic init(): Promise<InitializedToken> {\n\t\treturn new Promise( ( resolve, reject ) => {\n\t\t\tif ( !this.value ) {\n\t\t\t\tthis.refreshToken()\n\t\t\t\t\t.then( resolve )\n\t\t\t\t\t.catch( reject );\n\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tif ( this._options.autoRefresh ) {\n\t\t\t\tthis._registerRefreshTokenTimeout();\n\t\t\t}\n\n\t\t\tresolve( this as InitializedToken );\n\t\t} );\n\t}\n\n\t/**\n\t * Refresh token method. Useful in a method form as it can be override in tests.\n\t */\n\tpublic refreshToken(): Promise<InitializedToken> {\n\t\treturn this._refresh()\n\t\t\t.then( value => {\n\t\t\t\tthis._validateTokenValue( value );\n\t\t\t\tthis.set( 'value', value );\n\n\t\t\t\tif ( this._options.autoRefresh ) {\n\t\t\t\t\tthis._registerRefreshTokenTimeout();\n\t\t\t\t}\n\n\t\t\t\treturn this as InitializedToken;\n\t\t\t} );\n\t}\n\n\t/**\n\t * Destroys token instance. Stops refreshing.\n\t */\n\tpublic destroy(): void {\n\t\tclearTimeout( this._tokenRefreshTimeout );\n\t}\n\n\t/**\n\t * Checks whether the provided token follows the JSON Web Tokens (JWT) format.\n\t *\n\t * @param tokenValue The token to validate.\n\t */\n\tprivate _validateTokenValue( tokenValue: string ) {\n\t\t// The token must be a string.\n\t\tconst isString = typeof tokenValue === 'string';\n\n\t\t// The token must be a plain string without quotes (\"\").\n\t\tconst isPlainString = !/^\".*\"$/.test( tokenValue );\n\n\t\t// JWT token contains 3 parts: header, payload, and signature.\n\t\t// Each part is separated by a dot.\n\t\tconst isJWTFormat = isString && tokenValue.split( '.' ).length === 3;\n\n\t\tif ( !( isPlainString && isJWTFormat ) ) {\n\t\t\t/**\n\t\t\t * The provided token must follow the [JSON Web Tokens](https://jwt.io/introduction/) format.\n\t\t\t *\n\t\t\t * @error token-not-in-jwt-format\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'token-not-in-jwt-format', this );\n\t\t}\n\t}\n\n\t/**\n\t * Registers a refresh token timeout for the time taken from token.\n\t */\n\tprivate _registerRefreshTokenTimeout() {\n\t\tconst tokenRefreshTimeoutTime = this._getTokenRefreshTimeoutTime();\n\n\t\tclearTimeout( this._tokenRefreshTimeout );\n\n\t\tthis._tokenRefreshTimeout = setTimeout( () => {\n\t\t\tthis.refreshToken();\n\t\t}, tokenRefreshTimeoutTime );\n\t}\n\n\t/**\n\t * Returns token refresh timeout time calculated from expire time in the token payload.\n\t *\n\t * If the token parse fails or the token payload doesn't contain, the default DEFAULT_TOKEN_REFRESH_TIMEOUT_TIME is returned.\n\t */\n\tprivate _getTokenRefreshTimeoutTime() {\n\t\ttry {\n\t\t\tconst [ , binaryTokenPayload ] = this.value!.split( '.' );\n\t\t\tconst { exp: tokenExpireTime } = JSON.parse( atob( binaryTokenPayload ) );\n\n\t\t\tif ( !tokenExpireTime ) {\n\t\t\t\treturn DEFAULT_TOKEN_REFRESH_TIMEOUT_TIME;\n\t\t\t}\n\n\t\t\tconst tokenRefreshTimeoutTime = Math.floor( ( ( tokenExpireTime * 1000 ) - Date.now() ) / 2 );\n\n\t\t\treturn tokenRefreshTimeoutTime;\n\t\t} catch ( err ) {\n\t\t\treturn DEFAULT_TOKEN_REFRESH_TIMEOUT_TIME;\n\t\t}\n\t}\n\n\t/**\n\t * Creates a initialized {@link module:cloud-services/token/token~Token} instance.\n\t *\n\t * @param tokenUrlOrRefreshToken Endpoint address to download the token or a callback that provides the token. If the\n\t * value is a function it has to match the {@link module:cloud-services/token/token~Token#refreshToken} interface.\n\t */\n\tpublic static create( tokenUrlOrRefreshToken: TokenUrl, options: TokenOptions = {} ): Promise<InitializedToken> {\n\t\tconst token = new Token( tokenUrlOrRefreshToken, options );\n\n\t\treturn token.init();\n\t}\n}\n\n/**\n * A {@link ~Token} instance that has been initialized.\n */\nexport type InitializedToken = Token & { value: string };\n\n/**\n * Options for creating tokens.\n */\nexport interface TokenOptions {\n\n\t/**\n\t * Initial value of the token.\n\t */\n\tinitValue?: string;\n\n\t/**\n\t * Specifies whether to start the refresh automatically.\n\t *\n\t * @default true\n\t */\n\tautoRefresh?: boolean;\n}\n\n/**\n * This function is called in a defined interval by the {@link ~Token} class. It also can be invoked manually.\n * It should return a promise, which resolves with the new token value.\n * If any error occurs it should return a rejected promise with an error message.\n */\nfunction defaultRefreshToken( tokenUrl: string ) {\n\treturn new Promise<string>( ( resolve, reject ) => {\n\t\tconst xhr = new XMLHttpRequest();\n\n\t\txhr.open( 'GET', tokenUrl );\n\n\t\txhr.addEventListener( 'load', () => {\n\t\t\tconst statusCode = xhr.status;\n\t\t\tconst xhrResponse = xhr.response;\n\n\t\t\tif ( statusCode < 200 || statusCode > 299 ) {\n\t\t\t\t/**\n\t\t\t\t * Cannot download new token from the provided url.\n\t\t\t\t *\n\t\t\t\t * @error token-cannot-download-new-token\n\t\t\t\t */\n\t\t\t\treturn reject(\n\t\t\t\t\tnew CKEditorError( 'token-cannot-download-new-token', null )\n\t\t\t\t);\n\t\t\t}\n\n\t\t\treturn resolve( xhrResponse );\n\t\t} );\n\n\t\txhr.addEventListener( 'error', () => reject( new Error( 'Network Error' ) ) );\n\t\txhr.addEventListener( 'abort', () => reject( new Error( 'Abort' ) ) );\n\n\t\txhr.send();\n\t} );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module cloud-services/uploadgateway/fileuploader\n */\n\n/* globals XMLHttpRequest, FormData, Blob, atob */\n\nimport type { UploadResponse } from 'ckeditor5/src/upload';\nimport { EmitterMixin, CKEditorError } from 'ckeditor5/src/utils';\nimport type { InitializedToken } from '../token/token';\n\nconst BASE64_HEADER_REG_EXP = /^data:(\\S*?);base64,/;\n\n/**\n * FileUploader class used to upload single file.\n */\nexport default class FileUploader extends EmitterMixin() {\n\t/**\n\t * A file that is being uploaded.\n\t */\n\tpublic readonly file: Blob;\n\n\tpublic xhr?: XMLHttpRequest;\n\n\t/**\n\t * CKEditor Cloud Services access token.\n\t */\n\tprivate readonly _token: InitializedToken;\n\n\t/**\n\t * CKEditor Cloud Services API address.\n\t */\n\tprivate readonly _apiAddress: string;\n\n\t/**\n\t * Creates `FileUploader` instance.\n\t *\n\t * @param fileOrData A blob object or a data string encoded with Base64.\n\t * @param token Token used for authentication.\n\t * @param apiAddress API address.\n\t */\n\tconstructor(\n\t\tfileOrData: string | Blob,\n\t\ttoken: InitializedToken,\n\t\tapiAddress: string\n\t) {\n\t\tsuper();\n\n\t\tif ( !fileOrData ) {\n\t\t\t/**\n\t\t\t * File must be provided as the first argument.\n\t\t\t *\n\t\t\t * @error fileuploader-missing-file\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'fileuploader-missing-file', null );\n\t\t}\n\n\t\tif ( !token ) {\n\t\t\t/**\n\t\t\t * Token must be provided as the second argument.\n\t\t\t *\n\t\t\t * @error fileuploader-missing-token\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'fileuploader-missing-token', null );\n\t\t}\n\n\t\tif ( !apiAddress ) {\n\t\t\t/**\n\t\t\t * Api address must be provided as the third argument.\n\t\t\t *\n\t\t\t * @error fileuploader-missing-api-address\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'fileuploader-missing-api-address', null );\n\t\t}\n\n\t\tthis.file = _isBase64( fileOrData ) ? _base64ToBlob( fileOrData ) : fileOrData;\n\n\t\tthis._token = token;\n\t\tthis._apiAddress = apiAddress;\n\t}\n\n\t/**\n\t * Registers callback on `progress` event.\n\t */\n\tpublic onProgress( callback: ( status: { total: number; uploaded: number } ) => void ): this {\n\t\tthis.on<FileUploaderProgressErrorEvent>( 'progress', ( event, data ) => callback( data ) );\n\n\t\treturn this;\n\t}\n\n\t/**\n\t * Registers callback on `error` event. Event is called once when error occurs.\n\t */\n\tpublic onError( callback: ( error: string ) => void ): this {\n\t\tthis.once<FileUploaderErrorEvent>( 'error', ( event, data ) => callback( data ) );\n\n\t\treturn this;\n\t}\n\n\t/**\n\t * Aborts upload process.\n\t */\n\tpublic abort(): void {\n\t\tthis.xhr!.abort();\n\t}\n\n\t/**\n\t * Sends XHR request to API.\n\t */\n\tpublic send(): Promise<UploadResponse> {\n\t\tthis._prepareRequest();\n\t\tthis._attachXHRListeners();\n\n\t\treturn this._sendRequest();\n\t}\n\n\t/**\n\t * Prepares XHR request.\n\t */\n\tprivate _prepareRequest() {\n\t\tconst xhr = new XMLHttpRequest();\n\n\t\txhr.open( 'POST', this._apiAddress );\n\t\txhr.setRequestHeader( 'Authorization', this._token.value );\n\t\txhr.responseType = 'json';\n\n\t\tthis.xhr = xhr;\n\t}\n\n\t/**\n\t * Attaches listeners to the XHR.\n\t */\n\tprivate _attachXHRListeners() {\n\t\tconst xhr = this.xhr!;\n\n\t\tconst onError = ( message: string ) => {\n\t\t\treturn () => this.fire<FileUploaderErrorEvent>( 'error', message );\n\t\t};\n\n\t\txhr.addEventListener( 'error', onError( 'Network Error' ) );\n\t\txhr.addEventListener( 'abort', onError( 'Abort' ) );\n\n\t\t/* istanbul ignore else -- @preserve */\n\t\tif ( xhr.upload ) {\n\t\t\txhr.upload.addEventListener( 'progress', event => {\n\t\t\t\tif ( event.lengthComputable ) {\n\t\t\t\t\tthis.fire<FileUploaderProgressErrorEvent>( 'progress', {\n\t\t\t\t\t\ttotal: event.total,\n\t\t\t\t\t\tuploaded: event.loaded\n\t\t\t\t\t} );\n\t\t\t\t}\n\t\t\t} );\n\t\t}\n\n\t\txhr.addEventListener( 'load', () => {\n\t\t\tconst statusCode = xhr.status;\n\t\t\tconst xhrResponse = xhr.response;\n\n\t\t\tif ( statusCode < 200 || statusCode > 299 ) {\n\t\t\t\treturn this.fire<FileUploaderErrorEvent>( 'error', xhrResponse.message || xhrResponse.error );\n\t\t\t}\n\t\t} );\n\t}\n\n\t/**\n\t * Sends XHR request.\n\t */\n\tprivate _sendRequest() {\n\t\tconst formData = new FormData();\n\t\tconst xhr = this.xhr!;\n\n\t\tformData.append( 'file', this.file );\n\n\t\treturn new Promise<UploadResponse>( ( resolve, reject ) => {\n\t\t\txhr.addEventListener( 'load', () => {\n\t\t\t\tconst statusCode = xhr.status;\n\t\t\t\tconst xhrResponse = xhr.response;\n\n\t\t\t\tif ( statusCode < 200 || statusCode > 299 ) {\n\t\t\t\t\tif ( xhrResponse.message ) {\n\t\t\t\t\t\t/**\n\t\t\t\t\t\t * Uploading file failed.\n\t\t\t\t\t\t *\n\t\t\t\t\t\t * @error fileuploader-uploading-data-failed\n\t\t\t\t\t\t */\n\t\t\t\t\t\treturn reject( new CKEditorError(\n\t\t\t\t\t\t\t'fileuploader-uploading-data-failed',\n\t\t\t\t\t\t\tthis,\n\t\t\t\t\t\t\t{ message: xhrResponse.message }\n\t\t\t\t\t\t) );\n\t\t\t\t\t}\n\n\t\t\t\t\treturn reject( xhrResponse.error );\n\t\t\t\t}\n\n\t\t\t\treturn resolve( xhrResponse );\n\t\t\t} );\n\n\t\t\txhr.addEventListener( 'error', () => reject( new Error( 'Network Error' ) ) );\n\t\t\txhr.addEventListener( 'abort', () => reject( new Error( 'Abort' ) ) );\n\n\t\t\txhr.send( formData );\n\t\t} );\n\t}\n}\n\n/**\n * Fired when error occurs.\n *\n * @eventName ~FileUploader#error\n * @param error Error message\n */\nexport type FileUploaderErrorEvent = {\n\tname: 'error';\n\targs: [ error: string ];\n};\n\n/**\n * Fired on upload progress.\n *\n * @eventName ~FileUploader#progress\n * @param status Total and uploaded status\n */\nexport type FileUploaderProgressErrorEvent = {\n\tname: 'progress';\n\targs: [ status: { total: number; uploaded: number } ];\n};\n\n/**\n * Transforms Base64 string data into file.\n *\n * @param base64 String data.\n */\nfunction _base64ToBlob( base64: string, sliceSize = 512 ) {\n\ttry {\n\t\tconst contentType = base64.match( BASE64_HEADER_REG_EXP )![ 1 ];\n\t\tconst base64Data = atob( base64.replace( BASE64_HEADER_REG_EXP, '' ) );\n\n\t\tconst byteArrays: Array<Uint8Array> = [];\n\n\t\tfor ( let offset = 0; offset < base64Data.length; offset += sliceSize ) {\n\t\t\tconst slice = base64Data.slice( offset, offset + sliceSize );\n\t\t\tconst byteNumbers = new Array<number>( slice.length );\n\n\t\t\tfor ( let i = 0; i < slice.length; i++ ) {\n\t\t\t\tbyteNumbers[ i ] = slice.charCodeAt( i );\n\t\t\t}\n\n\t\t\tbyteArrays.push( new Uint8Array( byteNumbers ) );\n\t\t}\n\n\t\treturn new Blob( byteArrays, { type: contentType } );\n\t} catch ( error ) {\n\t\t/**\n\t\t * Problem with decoding Base64 image data.\n\t\t *\n\t\t * @error fileuploader-decoding-image-data-error\n\t\t */\n\t\tthrow new CKEditorError( 'fileuploader-decoding-image-data-error', null );\n\t}\n}\n\n/**\n * Checks that string is Base64.\n */\nfunction _isBase64( string: string | Blob ): string is string {\n\tif ( typeof string !== 'string' ) {\n\t\treturn false;\n\t}\n\n\tconst match = string.match( BASE64_HEADER_REG_EXP );\n\treturn !!( match && match.length );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module cloud-services/uploadgateway/uploadgateway\n */\n\nimport FileUploader from './fileuploader';\nimport type { InitializedToken } from '../token/token';\nimport { CKEditorError } from 'ckeditor5/src/utils';\n\n/**\n * UploadGateway abstracts file uploads to CKEditor Cloud Services.\n */\nexport default class UploadGateway {\n\t/**\n\t * CKEditor Cloud Services access token.\n\t */\n\tprivate readonly _token: InitializedToken;\n\n\t/**\n\t * CKEditor Cloud Services API address.\n\t */\n\tprivate readonly _apiAddress: string;\n\n\t/**\n\t * Creates `UploadGateway` instance.\n\t *\n\t * @param token Token used for authentication.\n\t * @param apiAddress API address.\n\t */\n\tconstructor( token: InitializedToken, apiAddress: string ) {\n\t\tif ( !token ) {\n\t\t\t/**\n\t\t\t * Token must be provided.\n\t\t\t *\n\t\t\t * @error uploadgateway-missing-token\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'uploadgateway-missing-token', null );\n\t\t}\n\n\t\tif ( !apiAddress ) {\n\t\t\t/**\n\t\t\t * Api address must be provided.\n\t\t\t *\n\t\t\t * @error uploadgateway-missing-api-address\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'uploadgateway-missing-api-address', null );\n\t\t}\n\n\t\tthis._token = token;\n\t\tthis._apiAddress = apiAddress;\n\t}\n\n\t/**\n\t * Creates a {@link module:cloud-services/uploadgateway/fileuploader~FileUploader} instance that wraps\n\t * file upload process. The file is being sent at a time when the\n\t * {@link module:cloud-services/uploadgateway/fileuploader~FileUploader#send} method is called.\n\t *\n\t * ```ts\n\t * const token = await Token.create( 'https://token-endpoint' );\n\t * new UploadGateway( token, 'https://example.org' )\n\t * \t.upload( 'FILE' )\n\t * \t.onProgress( ( data ) => console.log( data ) )\n\t * \t.send()\n\t * \t.then( ( response ) => console.log( response ) );\n\t * ```\n\t *\n\t * @param {Blob|String} fileOrData A blob object or a data string encoded with Base64.\n\t * @returns {module:cloud-services/uploadgateway/fileuploader~FileUploader} Returns `FileUploader` instance.\n\t */\n\tpublic upload( fileOrData: string | Blob ): FileUploader {\n\t\treturn new FileUploader( fileOrData, this._token, this._apiAddress );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module cloud-services/cloudservicescore\n */\n\nimport { ContextPlugin } from 'ckeditor5/src/core';\nimport type { TokenUrl } from './cloudservicesconfig';\nimport Token, { type InitializedToken, type TokenOptions } from './token/token';\nimport UploadGateway from './uploadgateway/uploadgateway';\n\n/**\n * The `CloudServicesCore` plugin exposes the base API for communication with CKEditor Cloud Services.\n */\nexport default class CloudServicesCore extends ContextPlugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CloudServicesCore' as const;\n\t}\n\n\t/**\n\t * Creates the {@link module:cloud-services/token/token~Token} instance.\n\t *\n\t * @param tokenUrlOrRefreshToken Endpoint address to download the token or a callback that provides the token. If the\n\t * value is a function it has to match the {@link module:cloud-services/token/token~Token#refreshToken} interface.\n\t * @param options.initValue Initial value of the token.\n\t * @param options.autoRefresh Specifies whether to start the refresh automatically.\n\t */\n\tpublic createToken( tokenUrlOrRefreshToken: TokenUrl, options?: TokenOptions ): Token {\n\t\treturn new Token( tokenUrlOrRefreshToken, options );\n\t}\n\n\t/**\n\t * Creates the {@link module:cloud-services/uploadgateway/uploadgateway~UploadGateway} instance.\n\t *\n\t * @param token Token used for authentication.\n\t * @param apiAddress API address.\n\t */\n\tpublic createUploadGateway( token: InitializedToken, apiAddress: string ): UploadGateway {\n\t\treturn new UploadGateway( token, apiAddress );\n\t}\n}\n","import api from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n import content from \"!!../../../node_modules/css-loader/dist/cjs.js!../../../node_modules/postcss-loader/dist/cjs.js??ruleSet[1].rules[1].use[2]!./theme.css\";\n\nvar options = {\"injectType\":\"singletonStyleTag\",\"attributes\":{\"data-cke\":true}};\n\noptions.insert = \"head\";\noptions.singleton = true;\n\nvar update = api(content, options);\n\n\n\nexport default content.locals || {};","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n// The editor creator to use.\nimport { BalloonEditor as BalloonEditorBase } from '@ckeditor/ckeditor5-editor-balloon';\n\nimport { Essentials } from '@ckeditor/ckeditor5-essentials';\nimport { UploadAdapter } from '@ckeditor/ckeditor5-adapter-ckfinder';\nimport { Autoformat } from '@ckeditor/ckeditor5-autoformat';\nimport { BlockToolbar } from '@ckeditor/ckeditor5-ui';\nimport { Bold, Italic } from '@ckeditor/ckeditor5-basic-styles';\nimport { BlockQuote } from '@ckeditor/ckeditor5-block-quote';\nimport { CKBox } from '@ckeditor/ckeditor5-ckbox';\nimport { CKFinder } from '@ckeditor/ckeditor5-ckfinder';\nimport { EasyImage } from '@ckeditor/ckeditor5-easy-image';\nimport { Heading } from '@ckeditor/ckeditor5-heading';\nimport { Image, ImageCaption, ImageStyle, ImageToolbar, ImageUpload, PictureEditing } from '@ckeditor/ckeditor5-image';\nimport { Indent } from '@ckeditor/ckeditor5-indent';\nimport { Link } from '@ckeditor/ckeditor5-link';\nimport { List } from '@ckeditor/ckeditor5-list';\nimport { MediaEmbed } from '@ckeditor/ckeditor5-media-embed';\nimport { Paragraph } from '@ckeditor/ckeditor5-paragraph';\nimport { PasteFromOffice } from '@ckeditor/ckeditor5-paste-from-office';\nimport { Table, TableToolbar } from '@ckeditor/ckeditor5-table';\nimport { TextTransformation } from '@ckeditor/ckeditor5-typing';\nimport { CloudServices } from '@ckeditor/ckeditor5-cloud-services';\n\n/**\n * @internal\n */\nimport '../theme/theme.css';\n\nexport default class BalloonEditor extends BalloonEditorBase {\n\tpublic static override builtinPlugins = [\n\t\tEssentials,\n\t\tUploadAdapter,\n\t\tAutoformat,\n\t\tBlockToolbar,\n\t\tBold,\n\t\tItalic,\n\t\tBlockQuote,\n\t\tCKBox,\n\t\tCKFinder,\n\t\tCloudServices,\n\t\tEasyImage,\n\t\tHeading,\n\t\tImage,\n\t\tImageCaption,\n\t\tImageStyle,\n\t\tImageToolbar,\n\t\tImageUpload,\n\t\tIndent,\n\t\tLink,\n\t\tList,\n\t\tMediaEmbed,\n\t\tParagraph,\n\t\tPasteFromOffice,\n\t\tPictureEditing,\n\t\tTable,\n\t\tTableToolbar,\n\t\tTextTransformation\n\t];\n\n\tpublic static override defaultConfig = {\n\t\tblockToolbar: [\n\t\t\t'undo', 'redo',\n\t\t\t'|', 'heading',\n\t\t\t'|', 'uploadImage', 'insertTable', 'blockQuote', 'mediaEmbed',\n\t\t\t'|', 'bulletedList', 'numberedList', 'outdent', 'indent'\n\t\t],\n\t\ttoolbar: {\n\t\t\titems: [\n\t\t\t\t'bold', 'italic', 'link'\n\t\t\t]\n\t\t},\n\t\timage: {\n\t\t\ttoolbar: [\n\t\t\t\t'imageStyle:inline',\n\t\t\t\t'imageStyle:block',\n\t\t\t\t'imageStyle:side',\n\t\t\t\t'|',\n\t\t\t\t'toggleImageCaption',\n\t\t\t\t'imageTextAlternative'\n\t\t\t]\n\t\t},\n\t\ttable: {\n\t\t\tcontentToolbar: [\n\t\t\t\t'tableColumn',\n\t\t\t\t'tableRow',\n\t\t\t\t'mergeTableCells'\n\t\t\t]\n\t\t},\n\t\t// This value must be kept in sync with the language defined in webpack.config.js.\n\t\tlanguage: 'en'\n\t};\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module essentials/essentials\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport { Clipboard } from 'ckeditor5/src/clipboard';\nimport { Enter, ShiftEnter } from 'ckeditor5/src/enter';\nimport { SelectAll } from 'ckeditor5/src/select-all';\nimport { Typing } from 'ckeditor5/src/typing';\nimport { Undo } from 'ckeditor5/src/undo';\n\n/**\n * A plugin including all essential editing features. It represents a set of features that enables similar functionalities\n * to a `<textarea>` element.\n *\n * It includes:\n *\n * * {@link module:clipboard/clipboard~Clipboard},\n * * {@link module:enter/enter~Enter},\n * * {@link module:select-all/selectall~SelectAll},\n * * {@link module:enter/shiftenter~ShiftEnter},\n * * {@link module:typing/typing~Typing},\n * * {@link module:undo/undo~Undo}.\n *\n * This plugin set does not define any block-level containers (such as {@link module:paragraph/paragraph~Paragraph}).\n * If your editor is supposed to handle block content, make sure to include it.\n */\nexport default class Essentials extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ Clipboard, Enter, SelectAll, ShiftEnter, Typing, Undo ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Essentials' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module basic-styles/bold\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport BoldEditing from './bold/boldediting';\nimport BoldUI from './bold/boldui';\n\n/**\n * The bold feature.\n *\n * For a detailed overview check the {@glink features/basic-styles Basic styles feature} guide\n * and the {@glink api/basic-styles package page}.\n *\n * This is a \"glue\" plugin which loads the {@link module:basic-styles/bold/boldediting~BoldEditing bold editing feature}\n * and {@link module:basic-styles/bold/boldui~BoldUI bold UI feature}.\n */\nexport default class Bold extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ BoldEditing, BoldUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Bold' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module basic-styles/italic\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport ItalicEditing from './italic/italicediting';\nimport ItalicUI from './italic/italicui';\n\n/**\n * The italic feature.\n *\n * For a detailed overview check the {@glink features/basic-styles Basic styles feature} guide\n * and the {@glink api/basic-styles package page}.\n *\n * This is a \"glue\" plugin which loads the {@link module:basic-styles/italic/italicediting~ItalicEditing} and\n * {@link module:basic-styles/italic/italicui~ItalicUI} plugins.\n */\nexport default class Italic extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ItalicEditing, ItalicUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Italic' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module block-quote/blockquote\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport BlockQuoteEditing from './blockquoteediting';\nimport BlockQuoteUI from './blockquoteui';\n\n/**\n * The block quote plugin.\n *\n * For more information about this feature check the {@glink api/block-quote package page}.\n *\n * This is a \"glue\" plugin which loads the {@link module:block-quote/blockquoteediting~BlockQuoteEditing block quote editing feature}\n * and {@link module:block-quote/blockquoteui~BlockQuoteUI block quote UI feature}.\n *\n * @extends module:core/plugin~Plugin\n */\nexport default class BlockQuote extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ BlockQuoteEditing, BlockQuoteUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'BlockQuote' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ckbox/ckbox\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport CKBoxUI from './ckboxui';\nimport CKBoxEditing from './ckboxediting';\n\n/**\n * The CKBox feature, a bridge between the CKEditor 5 WYSIWYG editor and the CKBox file manager and uploader.\n *\n * This is a \"glue\" plugin which enables:\n *\n * * {@link module:ckbox/ckboxediting~CKBoxEditing},\n * * {@link module:ckbox/ckboxui~CKBoxUI},\n *\n * See the {@glink features/file-management/ckbox CKBox integration} guide to learn how to configure and use this feature.\n *\n * Check out the {@glink features/images/image-upload/image-upload Image upload} guide to learn about other ways to upload\n * images into CKEditor 5.\n */\nexport default class CKBox extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CKBox' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ CKBoxEditing, CKBoxUI ] as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module ckfinder/ckfinder\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport CKFinderUI from './ckfinderui';\nimport CKFinderEditing from './ckfinderediting';\n\n/**\n * The CKFinder feature, a bridge between the CKEditor 5 WYSIWYG editor and the\n * [CKFinder](https://ckeditor.com/ckfinder) file manager and uploader.\n *\n * This is a \"glue\" plugin which enables:\n *\n * * {@link module:ckfinder/ckfinderediting~CKFinderEditing},\n * * {@link module:ckfinder/ckfinderui~CKFinderUI},\n * * {@link module:adapter-ckfinder/uploadadapter~CKFinderUploadAdapter}.\n *\n * See the {@glink features/file-management/ckfinder \"CKFinder integration\" guide} to learn how to configure\n * and use this feature.\n *\n * Check out the {@glink features/images/image-upload/image-upload comprehensive \"Image upload\" guide} to learn about\n * other ways to upload images into CKEditor 5.\n */\nexport default class CKFinder extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CKFinder' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ 'Link', 'CKFinderUploadAdapter', CKFinderEditing, CKFinderUI ] as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module cloud-services/cloudservices\n */\n\nimport { ContextPlugin } from 'ckeditor5/src/core';\nimport { CKEditorError } from 'ckeditor5/src/utils';\nimport CloudServicesCore from './cloudservicescore';\nimport type { CloudServicesConfig, TokenUrl } from './cloudservicesconfig';\nimport type { InitializedToken } from './token/token';\n\n/**\n * Plugin introducing the integration between CKEditor 5 and CKEditor Cloud Services .\n *\n * It initializes the token provider based on\n * the {@link module:cloud-services/cloudservicesconfig~CloudServicesConfig `config.cloudService`}.\n */\nexport default class CloudServices extends ContextPlugin implements CloudServicesConfig {\n\t/**\n\t * The authentication token URL for CKEditor Cloud Services or a callback to the token value promise. See the\n\t * {@link module:cloud-services/cloudservicesconfig~CloudServicesConfig#tokenUrl} for more details.\n\t */\n\tpublic readonly tokenUrl?: TokenUrl;\n\n\t/**\n\t * The URL to which the files should be uploaded.\n\t */\n\tpublic readonly uploadUrl?: string;\n\n\t/**\n\t * The URL for web socket communication, used by the `RealTimeCollaborativeEditing` plugin. Every customer (organization in the CKEditor\n\t * Ecosystem dashboard) has their own, unique URLs to communicate with CKEditor Cloud Services. The URL can be found in the\n\t * CKEditor Ecosystem customer dashboard.\n\t *\n\t * Note: Unlike most plugins, `RealTimeCollaborativeEditing` is not included in any CKEditor 5 build and needs to be installed manually.\n\t * Check [Collaboration overview](https://ckeditor.com/docs/ckeditor5/latest/features/collaboration/overview.html) for more details.\n\t */\n\tpublic readonly webSocketUrl?: string;\n\n\t/**\n\t * An optional parameter used for integration with CKEditor Cloud Services when uploading the editor build to cloud services.\n\t *\n\t * Whenever the editor build or the configuration changes, this parameter should be set to a new, unique value to differentiate\n\t * the new bundle (build + configuration) from the old ones.\n\t */\n\tpublic readonly bundleVersion?: string;\n\n\t/**\n\t * Other plugins use this token for the authorization process. It handles token requesting and refreshing.\n\t * Its value is `null` when {@link module:cloud-services/cloudservicesconfig~CloudServicesConfig#tokenUrl} is not provided.\n\t *\n\t * @readonly\n\t */\n\tpublic token: InitializedToken | null = null;\n\n\t/**\n\t * A map of token object instances keyed by the token URLs.\n\t */\n\tprivate readonly _tokens = new Map<TokenUrl, InitializedToken>();\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'CloudServices' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ CloudServicesCore ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic async init(): Promise<void> {\n\t\tconst config = this.context.config;\n\n\t\tconst options = config.get( 'cloudServices' ) || {};\n\n\t\tfor ( const [ key, value ] of Object.entries( options ) as Array<[key: keyof CloudServicesConfig, value: any]> ) {\n\t\t\t( this as CloudServicesConfig )[ key ] = value;\n\t\t}\n\n\t\tif ( !this.tokenUrl ) {\n\t\t\tthis.token = null;\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst cloudServicesCore: CloudServicesCore = this.context.plugins.get( 'CloudServicesCore' );\n\n\t\tthis.token = await cloudServicesCore.createToken( this.tokenUrl ).init();\n\n\t\tthis._tokens.set( this.tokenUrl, this.token );\n\t}\n\n\t/**\n\t * Registers an additional authentication token URL for CKEditor Cloud Services or a callback to the token value promise. See the\n\t * {@link module:cloud-services/cloudservicesconfig~CloudServicesConfig#tokenUrl} for more details.\n\t *\n\t * @param tokenUrl The authentication token URL for CKEditor Cloud Services or a callback to the token value promise.\n\t */\n\tpublic async registerTokenUrl( tokenUrl: TokenUrl ): Promise<InitializedToken> {\n\t\t// Reuse the token instance in case of multiple features using the same token URL.\n\t\tif ( this._tokens.has( tokenUrl ) ) {\n\t\t\treturn this.getTokenFor( tokenUrl );\n\t\t}\n\n\t\tconst cloudServicesCore: CloudServicesCore = this.context.plugins.get( 'CloudServicesCore' );\n\t\tconst token = await cloudServicesCore.createToken( tokenUrl ).init();\n\n\t\tthis._tokens.set( tokenUrl, token );\n\n\t\treturn token;\n\t}\n\n\t/**\n\t * Returns an authentication token provider previously registered by {@link #registerTokenUrl}.\n\t *\n\t * @param tokenUrl The authentication token URL for CKEditor Cloud Services or a callback to the token value promise.\n\t */\n\tpublic getTokenFor( tokenUrl: TokenUrl ): InitializedToken {\n\t\tconst token = this._tokens.get( tokenUrl );\n\n\t\tif ( !token ) {\n\t\t\t/**\n\t\t\t * The provided `tokenUrl` was not registered by {@link module:cloud-services/cloudservices~CloudServices#registerTokenUrl}.\n\t\t\t *\n\t\t\t * @error cloudservices-token-not-registered\n\t\t\t */\n\t\t\tthrow new CKEditorError( 'cloudservices-token-not-registered', this );\n\t\t}\n\n\t\treturn token;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic override destroy(): void {\n\t\tsuper.destroy();\n\n\t\tfor ( const token of this._tokens.values() ) {\n\t\t\ttoken.destroy();\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module easy-image/easyimage\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { logWarning } from 'ckeditor5/src/utils';\n\nimport CloudServicesUploadAdapter from './cloudservicesuploadadapter';\n\n/**\n * The Easy Image feature, which makes the image upload in CKEditor 5 possible with virtually zero\n * server setup. A part of the [CKEditor Cloud Services](https://ckeditor.com/ckeditor-cloud-services/)\n * family.\n *\n * This is a \"glue\" plugin which enables:\n *\n * * {@link module:easy-image/cloudservicesuploadadapter~CloudServicesUploadAdapter}.\n *\n * This plugin requires plugin to be present in the editor configuration:\n *\n * * {@link module:image/image~Image},\n * * {@link module:image/imageupload~ImageUpload},\n *\n * See the {@glink features/images/image-upload/easy-image \"Easy Image integration\" guide} to learn how to configure\n * and use this feature.\n *\n * Check out the {@glink features/images/image-upload/image-upload comprehensive \"Image upload\" guide} to learn about\n * other ways to upload images into CKEditor 5.\n *\n * **Note**: After enabling the Easy Image plugin you need to configure the\n * [CKEditor Cloud Services](https://ckeditor.com/ckeditor-cloud-services/)\n * integration through {@link module:cloud-services/cloudservicesconfig~CloudServicesConfig `config.cloudServices`}.\n */\nexport default class EasyImage extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'EasyImage' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ CloudServicesUploadAdapter, 'ImageUpload' ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\n\t\tif ( !editor.plugins.has( 'ImageBlockEditing' ) && !editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\t\t/**\n\t\t\t * The Easy Image feature requires one of the following plugins to be loaded to work correctly:\n\t\t\t *\n\t\t\t * * {@link module:image/imageblock~ImageBlock},\n\t\t\t * * {@link module:image/imageinline~ImageInline},\n\t\t\t * * {@link module:image/image~Image} (loads both `ImageBlock` and `ImageInline`)\n\t\t\t *\n\t\t\t * Please make sure your editor configuration is correct.\n\t\t\t *\n\t\t\t * @error easy-image-image-feature-missing\n\t\t\t * @param {module:core/editor/editor~Editor} editor\n\t\t\t */\n\t\t\tlogWarning( 'easy-image-image-feature-missing', editor );\n\t\t}\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module heading/heading\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport HeadingEditing from './headingediting';\nimport HeadingUI from './headingui';\n\nimport '../theme/heading.css';\n\n/**\n * The headings feature.\n *\n * For a detailed overview, check the {@glink features/headings Headings feature} guide\n * and the {@glink api/heading package page}.\n *\n * This is a \"glue\" plugin which loads the {@link module:heading/headingediting~HeadingEditing heading editing feature}\n * and {@link module:heading/headingui~HeadingUI heading UI feature}.\n *\n * @extends module:core/plugin~Plugin\n */\nexport default class Heading extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ HeadingEditing, HeadingUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Heading' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/image\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport ImageBlock from './imageblock';\nimport ImageInline from './imageinline';\n\nimport '../theme/image.css';\n\n/**\n * The image plugin.\n *\n * For a detailed overview, check the {@glink features/images/images-overview image feature} documentation.\n *\n * This is a \"glue\" plugin which loads the following plugins:\n *\n * * {@link module:image/imageblock~ImageBlock},\n * * {@link module:image/imageinline~ImageInline},\n *\n * Usually, it is used in conjunction with other plugins from this package. See the {@glink api/image package page}\n * for more information.\n */\nexport default class Image extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageBlock, ImageInline ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Image' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagecaption\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport ImageCaptionEditing from './imagecaption/imagecaptionediting';\nimport ImageCaptionUI from './imagecaption/imagecaptionui';\n\nimport '../theme/imagecaption.css';\n\n/**\n * The image caption plugin.\n *\n * For a detailed overview, check the {@glink features/images/images-captions image caption} documentation.\n */\nexport default class ImageCaption extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageCaptionEditing, ImageCaptionUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageCaption' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagestyle\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport ImageStyleEditing from './imagestyle/imagestyleediting';\nimport ImageStyleUI from './imagestyle/imagestyleui';\n\n/**\n * The image style plugin.\n *\n * For a detailed overview of the image styles feature, check the {@glink features/images/images-styles documentation}.\n *\n * This is a \"glue\" plugin which loads the following plugins:\n * * {@link module:image/imagestyle/imagestyleediting~ImageStyleEditing},\n * * {@link module:image/imagestyle/imagestyleui~ImageStyleUI}\n *\n * It provides a default configuration, which can be extended or overwritten.\n * Read more about the {@link module:image/imageconfig~ImageConfig#styles image styles configuration}.\n */\nexport default class ImageStyle extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageStyleEditing, ImageStyleUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageStyle' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imagetoolbar\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { WidgetToolbarRepository } from 'ckeditor5/src/widget';\n\nimport ImageUtils from './imageutils';\nimport type { ImageStyleDropdownDefinition } from './imageconfig';\n\nimport { isObject } from 'lodash-es';\n\n/**\n * The image toolbar plugin. It creates and manages the image toolbar (the toolbar displayed when an image is selected).\n *\n * For an overview, check the {@glink features/images/images-overview#image-contextual-toolbar image contextual toolbar} documentation.\n *\n * Instances of toolbar components (e.g. buttons) are created using the editor's\n * {@link module:ui/componentfactory~ComponentFactory component factory}\n * based on the {@link module:image/imageconfig~ImageConfig#toolbar `image.toolbar` configuration option}.\n *\n * The toolbar uses the {@link module:ui/panel/balloon/contextualballoon~ContextualBalloon}.\n */\nexport default class ImageToolbar extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ WidgetToolbarRepository, ImageUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageToolbar' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst widgetToolbarRepository = editor.plugins.get( WidgetToolbarRepository );\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\t\twidgetToolbarRepository.register( 'image', {\n\t\t\tariaLabel: t( 'Image toolbar' ),\n\t\t\titems: normalizeDeclarativeConfig( editor.config.get( 'image.toolbar' ) || [] ),\n\t\t\tgetRelatedElement: selection => imageUtils.getClosestSelectedImageWidget( selection )!\n\t\t} );\n\t}\n}\n\n/**\n * Convert the dropdown definitions to their keys registered in the ComponentFactory.\n * The registration precess should be handled by the plugin which handles the UI of a particular feature.\n */\nfunction normalizeDeclarativeConfig( config: Array<string | ImageStyleDropdownDefinition> ): Array<string> {\n\treturn config.map( item => isObject( item ) ? item.name : item );\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/imageupload\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport ImageUploadUI from './imageupload/imageuploadui';\nimport ImageUploadProgress from './imageupload/imageuploadprogress';\nimport ImageUploadEditing from './imageupload/imageuploadediting';\n\n/**\n * The image upload plugin.\n *\n * For a detailed overview, check the {@glink features/images/image-upload/image-upload image upload feature} documentation.\n *\n * This plugin does not do anything directly, but it loads a set of specific plugins to enable image uploading:\n *\n * * {@link module:image/imageupload/imageuploadediting~ImageUploadEditing},\n * * {@link module:image/imageupload/imageuploadui~ImageUploadUI},\n * * {@link module:image/imageupload/imageuploadprogress~ImageUploadProgress}.\n */\nexport default class ImageUpload extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'ImageUpload' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageUploadEditing, ImageUploadUI, ImageUploadProgress ] as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module indent/indent\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport IndentEditing from './indentediting';\nimport IndentUI from './indentui';\n\n/**\n * The indent feature.\n *\n * This plugin acts as a single entry point plugin for other features that implement indentation of elements like lists or paragraphs.\n *\n * The compatible features are:\n *\n * * The {@link module:list/list~List} or {@link module:list/list/listediting~ListEditing} feature for list indentation.\n * * The {@link module:indent/indentblock~IndentBlock} feature for block indentation.\n *\n * This is a \"glue\" plugin that loads the following plugins:\n *\n * * The {@link module:indent/indentediting~IndentEditing indent editing feature}.\n * * The {@link module:indent/indentui~IndentUI indent UI feature}.\n *\n * The dependent plugins register the `'indent'` and `'outdent'` commands and introduce the `'indent'` and `'outdent'` buttons\n * that allow to increase or decrease text indentation of supported elements.\n *\n * **Note**: In order for the commands and buttons to work, at least one of compatible features is required.\n */\nexport default class Indent extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Indent' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ IndentEditing, IndentUI ] as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module link/link\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport LinkEditing from './linkediting';\nimport LinkUI from './linkui';\nimport AutoLink from './autolink';\n\n/**\n * The link plugin.\n *\n * This is a \"glue\" plugin that loads the {@link module:link/linkediting~LinkEditing link editing feature}\n * and {@link module:link/linkui~LinkUI link UI feature}.\n */\nexport default class Link extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ LinkEditing, LinkUI, AutoLink ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Link' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module list/list\n */\n\nimport ListEditing from './list/listediting';\nimport ListUI from './list/listui';\n\nimport { Plugin } from 'ckeditor5/src/core';\n\n/**\n * The list feature.\n *\n * This is a \"glue\" plugin that loads the {@link module:list/list/listediting~ListEditing list editing feature}\n * and {@link module:list/list/listui~ListUI list UI feature}.\n */\nexport default class List extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ListEditing, ListUI ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'List' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module media-embed/mediaembed\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { Widget } from 'ckeditor5/src/widget';\n\nimport MediaEmbedEditing from './mediaembedediting';\nimport AutoMediaEmbed from './automediaembed';\nimport MediaEmbedUI from './mediaembedui';\n\nimport '../theme/mediaembed.css';\n\n/**\n * The media embed plugin.\n *\n * For a detailed overview, check the {@glink features/media-embed Media Embed feature documentation}.\n *\n * This is a \"glue\" plugin which loads the following plugins:\n *\n * * The {@link module:media-embed/mediaembedediting~MediaEmbedEditing media embed editing feature},\n * * The {@link module:media-embed/mediaembedui~MediaEmbedUI media embed UI feature} and\n * * The {@link module:media-embed/automediaembed~AutoMediaEmbed auto-media embed feature}.\n */\nexport default class MediaEmbed extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ MediaEmbedEditing, MediaEmbedUI, AutoMediaEmbed, Widget ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'MediaEmbed' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module paste-from-office/pastefromoffice\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport { ClipboardPipeline } from 'ckeditor5/src/clipboard';\n\nimport MSWordNormalizer from './normalizers/mswordnormalizer';\nimport GoogleDocsNormalizer from './normalizers/googledocsnormalizer';\nimport GoogleSheetsNormalizer from './normalizers/googlesheetsnormalizer';\n\nimport { parseHtml } from './filters/parse';\nimport type { Normalizer, NormalizerData } from './normalizer';\n\n/**\n * The Paste from Office plugin.\n *\n * This plugin handles content pasted from Office apps and transforms it (if necessary)\n * to a valid structure which can then be understood by the editor features.\n *\n * Transformation is made by a set of predefined {@link module:paste-from-office/normalizer~Normalizer normalizers}.\n * This plugin includes following normalizers:\n * * {@link module:paste-from-office/normalizers/mswordnormalizer~MSWordNormalizer Microsoft Word normalizer}\n * * {@link module:paste-from-office/normalizers/googledocsnormalizer~GoogleDocsNormalizer Google Docs normalizer}\n *\n * For more information about this feature check the {@glink api/paste-from-office package page}.\n */\nexport default class PasteFromOffice extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'PasteFromOffice' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ClipboardPipeline ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic init(): void {\n\t\tconst editor = this.editor;\n\t\tconst clipboardPipeline: ClipboardPipeline = editor.plugins.get( 'ClipboardPipeline' );\n\t\tconst viewDocument = editor.editing.view.document;\n\t\tconst normalizers: Array<Normalizer> = [];\n\n\t\tnormalizers.push( new MSWordNormalizer( viewDocument ) );\n\t\tnormalizers.push( new GoogleDocsNormalizer( viewDocument ) );\n\t\tnormalizers.push( new GoogleSheetsNormalizer( viewDocument ) );\n\n\t\tclipboardPipeline.on(\n\t\t\t'inputTransformation',\n\t\t\t( evt, data: NormalizerData ) => {\n\t\t\t\tif ( data._isTransformedWithPasteFromOffice ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst codeBlock = editor.model.document.selection.getFirstPosition()!.parent;\n\n\t\t\t\tif ( codeBlock.is( 'element', 'codeBlock' ) ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tconst htmlString = data.dataTransfer.getData( 'text/html' );\n\t\t\t\tconst activeNormalizer = normalizers.find( normalizer => normalizer.isActive( htmlString ) );\n\n\t\t\t\tif ( activeNormalizer ) {\n\t\t\t\t\tif ( !data._parsedData ) {\n\t\t\t\t\t\tdata._parsedData = parseHtml( htmlString, viewDocument.stylesProcessor );\n\t\t\t\t\t}\n\n\t\t\t\t\tactiveNormalizer.execute( data );\n\n\t\t\t\t\tdata._isTransformedWithPasteFromOffice = true;\n\t\t\t\t}\n\t\t\t},\n\t\t\t{ priority: 'high' }\n\t\t);\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module image/pictureediting\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\n\nimport ImageEditing from './image/imageediting';\nimport ImageUtils from './imageutils';\nimport { downcastSourcesAttribute, upcastPicture } from './image/converters';\nimport type { default as ImageUploadEditing, ImageUploadCompleteEvent } from './imageupload/imageuploadediting';\n\n/**\n * This plugin enables the [`<picture>`](https://developer.mozilla.org/en-US/docs/Web/HTML/Element/picture) element support in the editor.\n *\n * * It enables the `sources` model attribute on `imageBlock` and `imageInline` model elements\n * (brought by {@link module:image/imageblock~ImageBlock} and {@link module:image/imageinline~ImageInline}, respectively).\n * * It translates the `sources` model element to the view (also: data) structure that may look as follows:\n *\n * ```html\n * <p>Inline image using picture:\n * \t<picture>\n * \t\t<source media=\"(min-width: 800px)\" srcset=\"image-large.webp\" type=\"image/webp\">\n * \t\t<source media=\"(max-width: 800px)\" srcset=\"image-small.webp\" type=\"image/webp\">\n * \t\t<!-- Other sources as specified in the \"sources\" model attribute... -->\n * \t\t<img src=\"image.png\" alt=\"An image using picture\" />\n * \t</picture>\n * </p>\n *\n * <p>Block image using picture:</p>\n * <figure class=\"image\">\n * \t<picture>\n * \t\t<source media=\"(min-width: 800px)\" srcset=\"image-large.webp\" type=\"image/webp\">\n * \t\t<source media=\"(max-width: 800px)\" srcset=\"image-small.webp\" type=\"image/webp\">\n * \t\t<!-- Other sources as specified in the \"sources\" model attribute... -->\n * \t\t<img src=\"image.png\" alt=\"An image using picture\" />\n * \t</picture>\n * \t<figcaption>Caption of the image</figcaption>\n * </figure>\n * ```\n *\n * **Note:** The value of the `sources` {@glink framework/architecture/editing-engine#changing-the-model model attribute}\n * in both examples equals:\n *\n * ```css\n * [\n * \t{\n * \t\tmedia: '(min-width: 800px)',\n * \t\tsrcset: 'image-large.webp',\n * \t\ttype: 'image/webp'\n * \t},\n * \t{\n * \t\tmedia: '(max-width: 800px)',\n * \t\tsrcset: 'image-small.webp',\n * \t\ttype: 'image/webp'\n * \t}\n * ]\n * ```\n *\n * * It integrates with the {@link module:image/imageupload~ImageUpload} plugin so images uploaded in the editor\n * automatically render using `<picture>` if the {@glink features/images/image-upload/image-upload upload adapter}\n * supports image sources and provides neccessary data.\n *\n * @private\n */\nexport default class PictureEditing extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ ImageEditing, ImageUtils ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'PictureEditing' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\tconst editor = this.editor;\n\n\t\tif ( editor.plugins.has( 'ImageBlockEditing' ) ) {\n\t\t\teditor.model.schema.extend( 'imageBlock', {\n\t\t\t\tallowAttributes: [ 'sources' ]\n\t\t\t} );\n\t\t}\n\n\t\tif ( editor.plugins.has( 'ImageInlineEditing' ) ) {\n\t\t\teditor.model.schema.extend( 'imageInline', {\n\t\t\t\tallowAttributes: [ 'sources' ]\n\t\t\t} );\n\t\t}\n\n\t\tthis._setupConversion();\n\t\tthis._setupImageUploadEditingIntegration();\n\t}\n\n\t/**\n\t * Configures conversion pipelines to support upcasting and downcasting images using the `<picture>` view element\n\t * and the model `sources` attribute.\n\t */\n\tprivate _setupConversion(): void {\n\t\tconst editor = this.editor;\n\t\tconst conversion = editor.conversion;\n\t\tconst imageUtils: ImageUtils = editor.plugins.get( 'ImageUtils' );\n\n\t\tconversion.for( 'upcast' ).add( upcastPicture( imageUtils ) );\n\t\tconversion.for( 'downcast' ).add( downcastSourcesAttribute( imageUtils ) );\n\t}\n\n\t/**\n\t * Makes it possible for uploaded images to get the `sources` model attribute and the `<picture>...</picture>`\n\t * view structure out-of-the-box if relevant data is provided along the\n\t * {@link module:image/imageupload/imageuploadediting~ImageUploadEditing#event:uploadComplete} event.\n\t */\n\tprivate _setupImageUploadEditingIntegration(): void {\n\t\tconst editor = this.editor;\n\n\t\tif ( !editor.plugins.has( 'ImageUploadEditing' ) ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst imageUploadEditing: ImageUploadEditing = editor.plugins.get( 'ImageUploadEditing' );\n\n\t\tthis.listenTo<ImageUploadCompleteEvent>(\n\t\t\timageUploadEditing,\n\t\t\t'uploadComplete',\n\t\t\t( evt, { imageElement, data } ) => {\n\t\t\t\tconst sources = data.sources;\n\n\t\t\t\tif ( !sources ) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\teditor.model.change( writer => {\n\t\t\t\t\twriter.setAttributes( {\n\t\t\t\t\t\tsources\n\t\t\t\t\t}, imageElement );\n\t\t\t\t} );\n\t\t\t} );\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/table\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { Widget } from 'ckeditor5/src/widget';\n\nimport TableEditing from './tableediting';\nimport TableUI from './tableui';\nimport TableSelection from './tableselection';\nimport TableClipboard from './tableclipboard';\nimport TableKeyboard from './tablekeyboard';\nimport TableMouse from './tablemouse';\n\nimport '../theme/table.css';\n\n/**\n * The table plugin.\n *\n * For a detailed overview, check the {@glink features/tables/tables Table feature documentation}.\n *\n * This is a \"glue\" plugin that loads the following table features:\n *\n * * {@link module:table/tableediting~TableEditing editing feature},\n * * {@link module:table/tableselection~TableSelection selection feature},\n * * {@link module:table/tablekeyboard~TableKeyboard keyboard navigation feature},\n * * {@link module:table/tablemouse~TableMouse mouse selection feature},\n * * {@link module:table/tableclipboard~TableClipboard clipboard feature},\n * * {@link module:table/tableui~TableUI UI feature}.\n */\nexport default class Table extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ TableEditing, TableUI, TableSelection, TableMouse, TableKeyboard, TableClipboard, Widget ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'Table' as const;\n\t}\n}\n","/**\n * @license Copyright (c) 2003-2023, CKSource Holding sp. z o.o. All rights reserved.\n * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license\n */\n\n/**\n * @module table/tabletoolbar\n */\n\nimport { Plugin } from 'ckeditor5/src/core';\nimport { WidgetToolbarRepository } from 'ckeditor5/src/widget';\nimport { getSelectedTableWidget, getTableWidgetAncestor } from './utils/ui/widget';\n\n/**\n * The table toolbar class. It creates toolbars for the table feature and its content (for now only for the table cell content).\n *\n * The table toolbar shows up when a table widget is selected. Its components (e.g. buttons) are created based on the\n * {@link module:table/tableconfig~TableConfig#tableToolbar `table.tableToolbar` configuration option}.\n *\n * Table content toolbar shows up when the selection is inside the content of a table. It creates its component based on the\n * {@link module:table/tableconfig~TableConfig#contentToolbar `table.contentToolbar` configuration option}.\n */\nexport default class TableToolbar extends Plugin {\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get requires() {\n\t\treturn [ WidgetToolbarRepository ] as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic static get pluginName() {\n\t\treturn 'TableToolbar' as const;\n\t}\n\n\t/**\n\t * @inheritDoc\n\t */\n\tpublic afterInit(): void {\n\t\tconst editor = this.editor;\n\t\tconst t = editor.t;\n\t\tconst widgetToolbarRepository = editor.plugins.get( WidgetToolbarRepository );\n\n\t\tconst tableContentToolbarItems = editor.config.get( 'table.contentToolbar' );\n\n\t\tconst tableToolbarItems = editor.config.get( 'table.tableToolbar' );\n\n\t\tif ( tableContentToolbarItems ) {\n\t\t\twidgetToolbarRepository.register( 'tableContent', {\n\t\t\t\tariaLabel: t( 'Table toolbar' ),\n\t\t\t\titems: tableContentToolbarItems,\n\t\t\t\tgetRelatedElement: getTableWidgetAncestor\n\t\t\t} );\n\t\t}\n\n\t\tif ( tableToolbarItems ) {\n\t\t\twidgetToolbarRepository.register( 'table', {\n\t\t\t\tariaLabel: t( 'Table toolbar' ),\n\t\t\t\titems: tableToolbarItems,\n\t\t\t\tgetRelatedElement: getSelectedTableWidget\n\t\t\t} );\n\t\t}\n\t}\n}\n"],"names":["root","factory","exports","module","define","amd","self","cssKeywords","reverseKeywords","key","Object","keys","convert","rgb","channels","labels","hsl","hsv","hwb","cmyk","xyz","lab","lch","hex","keyword","ansi16","ansi256","hcg","apple","gray","model","Error","length","defineProperty","value","r","g","b","min","Math","max","delta","h","s","l","rdif","gdif","bdif","v","diff","diffc","c","w","k","reversed","currentClosestKeyword","currentClosestDistance","Infinity","distance","y","x","z","t2","t3","val","t1","i","smin","lmin","hi","floor","f","p","q","t","vmin","sl","wh","bl","ratio","n","m","y2","x2","z2","a","atan2","PI","sqrt","hr","cos","sin","args","saturation","round","ansi","color","mult","rem","string","toString","toUpperCase","substring","match","colorString","split","map","char","join","integer","parseInt","chroma","grayscale","hue","pure","mg","conversions","route","forEach","fromModel","routes","toModel","fn","wrappedFn","arg0","result","len","conversion","wrapRounded","raw","wrapRaw","deriveBFS","graph","models","parent","buildGraph","queue","current","pop","adjacents","adjacent","node","unshift","link","from","to","wrapConversion","path","cur","___CSS_LOADER_EXPORT___","push","id","cssWithMappingToString","list","this","item","content","concat","modules","mediaQuery","dedupe","alreadyImportedModules","_i","_slicedToArray","arr","Array","isArray","_arrayWithHoles","Symbol","iterator","_s","_e","_arr","_n","_d","call","next","done","err","_iterableToArrayLimit","o","minLen","_arrayLikeToArray","prototype","slice","constructor","name","test","_unsupportedIterableToArray","TypeError","_nonIterableRest","arr2","_item","cssMapping","btoa","base64","unescape","encodeURIComponent","JSON","stringify","data","sourceMapping","sourceURLs","sources","source","sourceRoot","memo","isOldIE","Boolean","window","document","all","atob","getTarget","target","styleTarget","querySelector","HTMLIFrameElement","contentDocument","head","e","stylesInDom","getIndexByIdentifier","identifier","modulesToDom","options","idCountMap","identifiers","base","count","index","obj","css","media","sourceMap","references","updater","addStyle","insertStyleElement","style","createElement","attributes","nonce","setAttribute","insert","appendChild","textStore","replaceText","replacement","filter","applyToSingletonTag","remove","styleSheet","cssText","cssNode","createTextNode","childNodes","removeChild","insertBefore","applyToTag","removeAttribute","firstChild","singleton","singletonCounter","update","styleIndex","bind","parentNode","removeStyleElement","newObj","lastIdentifiers","newList","newLastIdentifiers","_index","splice","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","undefined","__webpack_modules__","getter","__esModule","d","definition","enumerable","get","globalThis","Function","prop","hasOwnProperty","nc","userAgent","navigator","toLowerCase","getUserAgent","isMac","isWindows","indexOf","isGecko","isSafari","isiOS","maxTouchPoints","isAndroid","isBlink","features","isRegExpUnicodePropertySupported","isSupported","search","RegExp","error","fastDiff","cmp","atomicChanges","arrayA","arrayB","changeIndexes","arr1","firstIndex","findFirstDifferenceIndex","lastIndexOld","lastIndexNew","oldArrayReversed","cutAndReverse","newArrayReversed","lastIndex","findChangeBoundaryIndexes","newLength","fill","changeIndexesToAtomicChanges","newArray","type","values","howMany","changeIndexesToChanges","reverse","aLength","bLength","_insert","_delete","tmp","es","fp","snake","y1","dir","spy","called","EventInfo","stop","off","HEX_NUMBERS","_","r1","random","r2","r3","r4","priority","normal","highest","high","low","lowest","insertToPriorityArray","objects","objectToInsert","DOCUMENTATION_URL","errorName","context","super","processedObjects","WeakSet","circularReferencesReplacer","has","add","stringifiedData","documentationLink","getLinkToDocumentationMessage","getErrorMessage","is","rethrowUnexpectedError","message","stack","console","warn","formatConsoleArguments","logError","documentationMessage","version","releaseDate","Date","windowOrGlobal","CKEDITOR_VERSION","_listeningTo","_emitterId","_delegations","defaultEmitterClass","EmitterMixin","on","event","callback","listenTo","once","wasFired","stopListening","emitter","emitterInfo","eventCallbacks","emitters","_getEmitterId","_setEmitterId","emitterId","callbacks","listener","_addEventListener","addEventListener","removeEventListener","fire","eventOrInfo","eventInfo","getCallbacksForEvent","callbackArgs","apply","_removeEventListener","delegations","destinations","passAllDestinations","fireDelegatedEvents","return","delegate","events","nameOrFunction","Map","eventName","set","stopDelegating","delete","clear","getEvents","childEventName","newEventNodes","childEvents","substr","lastIndexOf","createEventNamespace","lists","getCallbacksListsForNamespace","callbackDefinition","_events","eventNode","callbacksLists","childCallbacksLists","fireArgs","delegatedInfo","observablePropertiesSymbol","boundObservablesSymbol","boundPropertiesSymbol","decoratedMethods","decoratedOriginal","defaultObservableClass","ObservableMixin","property","initObservable","properties","configurable","oldValue","newValue","bindProperties","isStringArray","Set","size","boundProperties","propertyName","bindings","binding","bindTo","toMany","bindToMany","_observable","_bindProperties","_to","_bindings","unbind","unbindProperties","boundObservables","toObservable","toProperty","toProperties","toPropertyBindings","boundObservable","decorate","methodName","originalMethod","object","evt","observable","parsedArgs","parsed","lastObservable","parseBindToArgs","bindingsKeys","numberOfBindings","updateBoundObservableProperty","chain","toPropertyName","bindingsToObservable","updateBoundObservables","updateBindToBound","observables","attribute","observableAndAttributePairs","getBindingTargets","every","propertyValue","iterable","compareArrays","isIterable","global","freeSelf","objectProto","nativeObjectToString","symToStringTag","toStringTag","isOwn","tag","unmasked","doc","children","namespace","xmlns","element","createElementNS","child","func","transform","arg","getPrototypeOf","funcProto","funcToString","objectCtorString","proto","Ctor","__data__","other","array","ListCache","entries","entry","maskSrcKey","uid","exec","IE_PROTO","reIsHostCtor","reIsNative","replace","Hash","MapCache","pairs","LARGE_ARRAY_SIZE","Stack","iteratee","objValue","props","customizer","isNew","propertyIsEnumerable","arguments","freeExports","nodeType","freeModule","Buffer","isBuffer","reIsUint","typedArrayTags","freeProcess","process","types","require","nodeIsTypedArray","isTypedArray","inherited","isArr","isArg","isBuff","isType","skipIndexes","String","isProto","allocUnsafe","buffer","isDeep","copy","predicate","resIndex","nativeGetSymbols","getOwnPropertySymbols","symbol","offset","keysFunc","symbolsFunc","promiseTag","dataViewCtorString","mapCtorString","promiseCtorString","setCtorString","weakMapCtorString","getTag","ArrayBuffer","resolve","ctorString","input","Uint8Array","arrayBuffer","byteLength","dataView","byteOffset","reFlags","regexp","symbolProto","symbolValueOf","valueOf","typedArray","objectCreate","create","nodeIsMap","isMap","nodeIsSet","isSet","cloneableTags","baseClone","bitmask","isFlat","isFull","isFunc","stacked","subValue","Config","configurations","defaultConfigurations","_config","cloneConfig","_setObjectToTarget","_setToTarget","_getFromSource","names","isDefine","parts","part","configuration","leaveDOMReferences","isNode","defaultView","Document","ownerDocument","Node","isWindow","stringifiedObject","proxyOptions","capture","useCapture","passive","usePassive","proxyEmitter","_getProxyEmitter","ProxyEmitter","proxyEmitters","_getAllProxyEmitters","proxy","listeningEmitter","listenedToEmitterId","listeningTo","_getEmitterListenedTo","getProxyEmitterId","_domNode","_options","attach","_domListeners","domListener","_createDomListener","detach","removeListener","domEvt","getNodeUID","option","sort","globalVar","isText","isRange","getBorderWidths","getComputedStyle","top","borderTopWidth","right","borderRightWidth","bottom","borderBottomWidth","left","borderLeftWidth","rectProperties","isSourceRange","_source","writable","isDomElement","rangeRects","getDomRangeRects","copyRectProperties","getBoundingRect","getBoundingClientRect","innerWidth","innerHeight","width","height","clone","moveTo","moveBy","getIntersection","anotherRect","rect","getIntersectionArea","getArea","getVisible","visibleRect","isBody","absolutelyPositionedChildElement","commonAncestorContainer","HTMLElement","getElementPosition","overflow","parentRect","intersectionRect","isEqual","contains","intersectRect","excludeScrollbarsAndBorders","scrollBarWidth","scrollBarHeight","direction","documentElement","clientWidth","clientHeight","borderWidths","offsetWidth","offsetHeight","range","rects","clientRects","getClientRects","startContainer","boundingRectData","Number","POSITIVE_INFINITY","NEGATIVE_INFINITY","rectangleCount","body","position","_observerInstance","_createObserver","_element","_callback","_addElementCallback","observe","destroy","_deleteElementCallback","_elementCallbacks","_getElementCallbacks","unobserve","ResizeObserver","setDataInElement","el","HTMLTextAreaElement","innerHTML","toUnit","unit","previousSibling","insertAt","parentElement","nodeToInsert","isComment","COMMENT_NODE","isVisible","getOptimalPosition","positions","limiter","fitInViewport","viewportOffsetConfig","positionedElementAncestor","offsetParent","getPositionedAncestor","elementRect","targetRect","bestPosition","viewportRect","assign","getConstrainedViewportRect","positionOptions","limiterRect","elementRectArea","positionInstances","positioningFunction","PositionObject","maxFitFactor","limiterIntersectionArea","viewportIntersectionArea","fitFactor","getBestPosition","getRectForAbsolutePositioning","scrollX","scrollY","positioningFunctionOutput","config","_positioningFunctionCorrdinates","_absoluteRect","_rect","limiterViewportIntersectRect","_cachedRect","_cachedAbsoluteRect","ancestorPosition","ancestorBorderWidths","moveX","moveY","scrollLeft","scrollTop","shiftRectToCompensatePositionedAncestor","scrollWindowToShowRect","alignToTop","forceScroll","viewportOffset","targetShiftedDownRect","targetShiftedUpRect","forceScrollToTop","allRectsFitInViewport","initialScrollX","initialScrollY","isAbove","isBelow","isLeftOf","isRightOf","scrollTo","scrollAncestorsToShowRect","getRect","ancestorOffset","limiterElement","parentWindow","getWindow","targetFitsInTarget","firstRect","secondRect","elementOrRange","getParentElement","getRectRelativeToWindow","relativeWindow","targetWindow","currentWindow","frame","frameElement","frameRect","modifiersToGlyphsMac","ctrl","cmd","alt","shift","modifiersToGlyphsNonMac","keyCodes","arrowleft","arrowup","arrowright","arrowdown","backspace","enter","space","esc","tab","code","fromCharCode","charCodeAt","generateKnownKeyCodes","keyCodeNames","fromEntries","charAt","keyCode","altKey","ctrlKey","shiftKey","metaKey","parseKeystroke","keystroke","trim","splitKeystrokeText","endsWith","getEnvKeyCode","reduce","sum","getEnvKeystrokeText","keystrokeCode","modifiers","glyph","contentLanguageDirection","isLtrContent","_translate","language","quantity","numberOfLanguages","CKEDITOR_TRANSLATIONS","messageId","dictionary","hasTranslation","plural","getPluralForm","translation","RTL_LANGUAGE_CODES","getLanguageDirection","languageCode","includes","Locale","uiLanguage","contentLanguage","uiLanguageDirection","_t","interpolateString","initialItemsOrOptions","hasInitialItems","_items","_itemMap","_idProperty","idProperty","_bindToExternalToInternalMap","WeakMap","_bindToInternalToExternalMap","_skippedIndexesFromExternal","_getItemIdBeforeAdding","first","last","addMany","items","itemId","currentItemIndex","added","removed","idOrIndex","itemOrId","getIndex","subject","_remove","ctx","find","_bindToCollection","removedItems","externalCollection","as","Class","_setUpBindToBinding","using","callbackOrProperty","addItem","externalItem","isExternalBoundToThis","externalItemBound","finalIndex","skipped","itemDoesNotExist","iteratorItem","_elements","_nextEventLoopTimeout","_focus","_blur","focusedElement","clearTimeout","isFocused","setTimeout","_listener","keyEvtData","preventDefault","stopPropagation","press","objectToMap","wait","timer","delayed","cancel","isInsideSurrogatePair","character","isLowSurrogateHalf","isInsideCombinedSymbol","EMOJI_PATTERN","flagSequence","emoji","buildEmojiRegexp","isInsideEmojiSequence","matches","matchAll","some","editor","_disableStack","forceDisabled","forceDisable","isEnabled","clearForceDisabled","isContextPlugin","_affectsData","_isEnabledBasedOnSelection","refresh","affectsData","selection","canEditAtSelection","getFirstPosition","rootName","canEditAt","isReadOnly","execute","MultiCommand","_childCommandsDefinitions","command","_getFirstEnabledCommand","registerChildCommand","_checkEnabled","commandDefinition","PluginCollection","availablePlugins","contextPlugins","_plugins","_context","_availablePlugins","PluginConstructor","pluginName","_contextPlugins","pluginInstance","plugin","init","plugins","pluginsToRemove","pluginsSubstitutions","that","findAvailablePluginConstructors","processed","isPluginConstructor","requires","validatePlugins","pluginConstructors","getPluginConstructors","isPluginRemoved","pluginItem","pluginToReplace","indexInPluginConstructors","substitutePlugins","pluginInstances","_add","initPlugins","then","removedPlugin","getPluginName","parentPluginConstructor","missingPlugin","requiredBy","checkMissingPlugin","checkContextPlugin","checkRemovedPlugin","method","promise","Promise","promises","plugin1","plugin2","Context","_contextOwner","defaultConfig","builtinPlugins","languageConfig","locale","ui","editors","Plugin","_addEditor","isContextOwner","_removeEditor","_getEditorConfig","ContextPlugin","EditingKeystrokeHandler","commandName","evtData","locals","documentPlaceholders","hasDisplayedPlaceholderDeprecationWarning","view","text","isDirectHost","keepOnFocus","setPlaceholder","hostElement","change","writer","updateDocumentPlaceholders","registerPostFixer","evtInfo","placeholder","showPlaceholderTextDeprecationWarning","hasClass","removeClass","placeholders","directHostElements","wasViewModified","updatePlaceholder","getChildPlaceholderHostSubstitute","getAttribute","childCount","isAttached","getChildren","selectionAnchor","anchor","isComposing","addClass","getChild","TypeCheckable","pos","getChildIndex","nextSibling","getPath","getAncestors","ancestors","includeSelf","parentFirst","getCommonAncestor","ancestorsA","ancestorsB","isBefore","thisPath","nodePath","isAfter","_removeChildren","_fireChange","toJSON","json","Text","_textData","_data","isSimilar","otherNode","_clone","TextProxy","textNode","offsetInText","offsetSize","isPartial","pattern","_patterns","singleElement","isElementMatching","results","getElementName","matchName","patterns","attributeKeys","getAttributeKeys","class","matchPatterns","matchAttributes","classes","getClassNames","matchClasses","styles","getStyleNames","getStyle","matchStyles","valueGetter","normalizedPatterns","normalizePatterns","normalizedItems","patternKey","patternValue","itemKey","isKeyMatched","itemValue","isValueMatched","reIsDeepProp","reIsPlainProp","memoize","resolver","memoized","cache","Cache","rePropName","reEscapeChar","stringToPath","number","quote","subString","symbolToString","baseToString","start","end","defaultValue","fromRight","srcIndex","mergeFunc","srcValue","isCommon","isTyped","baseMerge","thisArg","nativeMax","otherArgs","nativeNow","now","lastCalled","stamp","remaining","assigner","guard","nested","StylesMap","styleProcessor","_styles","_styleProcessor","isEmpty","setTo","inlineStyle","parsedStyles","stylesString","quoteType","propertyNameStart","propertyValueStart","stylesMap","parseInlineStyles","toNormalizedForm","propertyDescriptor","getReducedForm","nameOrObject","valueOrObject","toPath","_cleanEmptyObjectsOnPath","getNormalized","_getStylesEntries","getAsString","expand","pathParts","parentPath","parentObject","StylesProcessor","_normalizers","_extractors","_reducers","_consumables","appendStyleValue","normalizer","extractor","normalizedValue","reducer","expandedStyleNames","styleNamesKeysSet","getRelatedStyles","setNormalizer","setExtractor","callbackOrPath","setReducer","setStyleRelation","shorthandName","styleNames","_mapStyleNames","alsoName","stylesObject","nameOrPath","valueToSet","attrs","_unsafeAttributesToRender","_customProperties","_attrs","attrsMap","parseAttributes","_children","_insertChild","_classes","classString","parseClasses","stylesProcessor","getAttributes","hasAttribute","otherElement","className","getNormalizedStyle","hasStyle","findAncestor","matcher","getCustomProperty","getCustomProperties","getIdentity","shouldRenderUnsafeAttribute","attributeName","deep","childrenClone","cloned","getFillerOffset","_appendChild","nodes","normalize","_setAttribute","stringValue","_removeAttribute","_addClass","_removeClass","_setStyle","_removeStyle","_setCustomProperty","_removeCustomProperty","classesSet","classesString","classArray","ContainerElement","lastChild","EditableElement","editableElement","rootNameSymbol","RootEditableElement","_name","TreeWalker","boundaries","startPosition","_position","Position","_createAt","singleCharacters","shallow","ignoreElementEnd","_boundaryStartParent","_boundaryEndParent","skip","nextResult","prevPosition","_next","_previous","previousPosition","isAtEnd","_createAfter","_formatReturnValue","charactersCount","textLength","textProxy","isAtStart","_createBefore","startOffset","nextPosition","nodeAfter","nodeBefore","endOffset","editable","getShiftedBy","shifted","getLastMatchingPosition","treeWalker","otherPosition","compareWith","otherPath","getWalker","itemOrPosition","Range","isCollapsed","getEnlarged","enlargeTrimSkip","getTrimmed","nodeAfterStart","nodeBeforeEnd","otherRange","containsPosition","containsRange","loose","containsStart","containsEnd","getDifference","ranges","isIntersecting","commonRangeStart","commonRangeEnd","getContainedElement","getItems","getPositions","_createFromParentsAndOffsets","startElement","endElement","_createFromPositionAndShift","_createIn","_createOn","Selection","_ranges","_lastRangeBackward","_isFake","_fakeSelectionLabel","isFake","fakeSelectionLabel","focus","rangeCount","isBackward","getRanges","getFirstRange","getLastRange","firstRange","getLastPosition","lastRange","otherSelection","thisRange","found","numOfRangesA","rangeA","rangeB","getSelectedElement","selectable","placeOrOffset","_setRanges","_setFakeOptions","DocumentSelection","fake","label","backward","setFocus","newFocus","_addRange","newRanges","isLastBackward","_pushRange","storedRange","addedRange","intersectingRange","_selection","_setTo","_setFocus","BubblingEventInfo","startRange","_eventPhase","_currentTarget","eventPhase","currentTarget","contextsSymbol","BubblingEmitterMixin","eventArgs","eventContexts","getBubblingContexts","updateEventInfo","fireListenerFor","selectedElement","isCustomContext","getCustomContext","startParent","endParent","startPath","endPath","getDeeperRangeParent","contexts","mixin","_postFixers","roots","getRoot","postFixer","_callPostFixers","wasFixed","AttributeElement","_priority","_id","_clonesGroup","getElementsWithSameId","nonUiChildrenCount","DEFAULT_PRIORITY","EmptyElement","UIElement","render","domDocument","domConverter","toDomElement","domElement","injectUiElementHandling","domSelection","domTarget","getSelection","domSelectionCollapsed","getRangeAt","collapsed","domParent","focusNode","domOffset","focusOffset","viewPosition","domPositionToView","jumpedOverAnyUiElement","nextViewPosition","newDomPosition","viewPositionToDom","collapse","extend","jumpOverUiElement","RawElement","DocumentFragment","_cloneGroups","_slotFactory","setSelection","setSelectionFocus","createDocumentFragment","createText","createAttributeElement","attributeElement","renderUnsafeAttributes","createContainerElement","childrenOrOptions","containerElement","createEditableElement","createEmptyElement","emptyElement","createUIElement","renderFunction","uiElement","createRawElement","rawElement","setStyle","removeStyle","setCustomProperty","removeCustomProperty","breakAttributes","positionOrRange","_breakAttributes","_breakAttributesRange","breakContainer","newElement","sourceRange","targetPosition","move","mergeAttributes","positionOffset","positionParent","_removeFromClonedElementsGroup","mergeTextNodes","mergeContainers","prev","newPosition","validateNodesToInsert","nodeGroups","groups","lastGroup","_insertNodes","rangeOrItem","validateRangeContainer","breakStart","breakEnd","parentContainer","mergePosition","walker","rangeToRemove","ancestor","countBefore","wrap","_wrapPosition","viewSelection","_wrapRange","unwrap","newRange","_unwrapChildren","rename","newName","viewElement","clearClonedElementsGroup","groupName","createPositionAt","createPositionAfter","createPositionBefore","createRange","createRangeOn","createRangeIn","createSelection","createSlot","modeOrFilter","_registerSlotFactory","slotFactory","_clearSlotFactory","insertionPosition","getParentContainer","breakTextNode","_addToClonedElementsGroup","endPosition","_wrapChildren","wrapElement","wrapPositions","isAttribute","_wrapAttributeElement","shouldABeOutsideB","newAttribute","offsetChange","unwrapElement","unwrapPositions","unwrapped","_unwrapAttributeElement","movePositionToTextNode","fakeElement","wrapRange","wrapper","toWrap","canBeJoined","toUnwrap","forceSplitText","rangeStart","rangeEnd","isContainerOrFragment","offsetAfter","clonedNode","nodesToMove","group","textToMove","nodeBeforeLength","validNodesToInsert","errorContext","validNode","endContainer","NBSP_FILLER","MARKED_NBSP_FILLER","span","dataset","ckeFiller","innerText","BR_FILLER","fillerBr","INLINE_FILLER_LENGTH","INLINE_FILLER","repeat","startsWithFiller","domNode","isInlineFiller","domText","getDataWithoutFiller","jumpOverInlineFiller","Renderer","domDocuments","markedAttributes","markedChildren","markedTexts","_inlineFiller","_fakeSelectionContainer","isSelecting","markToSync","mapViewToDom","inlineFillerPosition","isInlineFillerRenderingPossible","_updateChildrenMappings","_isSelectionInInlineFiller","_removeInlineFiller","_getInlineFillerPosition","_needsInlineFillerAtSelection","_updateAttrs","_updateChildren","_updateText","fillerDomPosition","addInlineFiller","_updateFocus","_updateSelection","actualDomChildren","expectedDomChildren","viewChildrenToDom","withChildren","_diffNodeLists","actions","_findUpdateActions","areSimilarElements","counter","equal","action","insertIndex","deleteIndex","viewChild","_updateElementMappings","unbindDomElement","bindElements","firstPos","selectionPosition","domFillerNode","selectionParent","selectionOffset","isEditable","viewText","findCorrespondingDomText","expectedText","viewToDom","filler","updateTextNode","domAttrKeys","attr","viewAttrKeys","setDomElementAttribute","removeDomElementAttribute","previousDomNode","areTextNodes","nodesToUnbind","_markDescendantTextToSync","domToView","domChildList","fakeSelectionContainer","childList","filterOutFakeSelectionContainer","sameNodes","actualDom","expectedDom","comparator","newActions","actualSlice","expectedSlice","viewNode","_removeDomSelection","_removeFakeSelection","domRoot","_updateFakeSelection","isConnected","_updateDomSelection","container","textContent","createFakeSelectionContainer","bindFakeSelection","_fakeSelectionNeedsUpdate","domRange","removeAllRanges","selectNodeContents","addRange","_domSelectionNeedsUpdate","setBaseAndExtent","ELEMENT_NODE","childAtOffset","tagName","fixGeckoSelectionAfterBr","isDomSelectionCorrect","oldViewSelection","domSelectionToView","anchorNode","activeDomElement","activeElement","mapDomToView","domParentOrArray","nodeAfterFiller","fillerNode","node1","node2","actualDomChild","expectedDomChild","isBlockFiller","actualText","insertData","deleteData","BR_FILLER_REF","NBSP_FILLER_REF","MARKED_NBSP_FILLER_REF","UNSAFE_ATTRIBUTE_NAME_PREFIX","UNSAFE_ELEMENT_REPLACEMENT_ATTRIBUTE","DomConverter","blockFillerMode","renderingMode","_domToViewMapping","_viewToDomMapping","_fakeSelectionMapping","_rawContentElementMatcher","_inlineObjectElementMatcher","preElements","blockElements","inlineObjectElements","unsafeElements","_domDocument","implementation","createHTMLDocument","viewDocumentSelection","fakeSelectionToView","bindDocumentFragments","domFragment","viewFragment","shouldRenderAttribute","attributeKey","attributeValue","elementName","startsWith","setContentOf","html","DOMParser","parseFromString","fragment","bodyChildNodes","createTreeWalker","NodeFilter","SHOW_ELEMENT","currentNode","nextNode","getAttributeNames","_shouldRenameElement","_logUnsafeElement","replaceWith","_createReplacementDomElement","append","textData","_processDataFromViewText","createComment","relatedViewElement","createAttribute","isValidAttributeName","fillerPositionOffset","childView","_getBlockFiller","transparentRendering","viewRangeToDom","viewRange","domStart","domEnd","setStart","setEnd","viewParent","domBefore","domAfter","inlineNodes","generator","_domToView","_processDomInlineNodes","domChildrenToView","domChild","_isBlockViewElement","isDomSelectionBackward","viewRanges","domRangeToView","viewStart","viewEnd","findCorrespondingViewText","viewBefore","domElementOrDocumentFragment","getHostViewElement","isElement","documentFragmentOrElement","viewEditable","domEditable","scrollPositions","forEachDomElementAncestor","_clearDomSelection","newViewSelection","isDocumentFragment","DOCUMENT_FRAGMENT_NODE","isEqualNode","hasBlockParent","isNBSP","isNbspBlockFiller","anchorOffset","DOCUMENT_NODE","_isDomSelectionPositionCorrect","registerRawContentMatcher","registerInlineObjectMatcher","skipComments","_isInlineObjectElement","_createViewElement","_isViewElementWithRawContent","nestedInlineNodes","inlineNode","_isBlockDomElement","prevNodeEndsWithSpace","nodeEndsWithSpace","_hasViewParentOfType","prevNode","shouldLeftTrim","shouldRightTrim","isNextNodeInlineObjectElement","isNextNodeStartingWithSpace","_getTouchingInlineViewNode","_nodeEndsWithSpace","nextStartsWithSpace","getNext","viewName","keepOriginalCase","originalDomElement","newDomElement","Observer","_isEnabled","enable","disable","checkShouldIgnoreEventFromTarget","DomEventData","domEvent","additionalData","DomEventObserver","domEventType","onDomEvent","stopObserving","eventType","KeyObserver","reWhitespace","reTrimStart","reIsBadHex","reIsBinary","reIsOctal","freeParseInt","isBinary","nativeMin","lastArgs","lastThis","maxWait","timerId","lastCallTime","lastInvokeTime","leading","maxing","trailing","invokeFunc","time","shouldInvoke","timeSinceLastCall","timerExpired","trailingEdge","timeWaiting","remainingWait","debounced","isInvoking","leadingEdge","flush","FakeSelectionObserver","_fireSelectionChangeDoneDebounced","_handleSelectionMove","newSelection","oldSelection","SetCache","equalFunc","arrLength","othLength","arrStacked","othStacked","seen","arrValue","othValue","compared","othIndex","objProps","objLength","objStacked","skipCtor","objCtor","othCtor","objIsArr","othIsArr","objTag","othTag","objIsObj","othIsObj","isSameTag","objIsWrapped","othIsWrapped","objUnwrapped","othUnwrapped","baseIsEqual","MutationObserver","characterData","subtree","renderer","_renderer","_domElements","_mutationObserver","_onMutations","takeRecords","disconnect","domMutations","mutatedTextNodes","elementsWithMutatedChildren","mutation","_isBogusBrMutation","hasMutations","viewChildren","newViewChildren","forceRender","addedNode","removedNodes","addedNodes","child1","child2","FocusObserver","_isFocusChanging","_renderTimeoutId","selectedEditable","SelectionObserver","mutationObserver","getObserver","focusObserver","_documents","_clearInfiniteLoopInterval","setInterval","_clearInfiniteLoop","_documentIsSelectingInactivityTimeoutDebounced","_loopbackCounter","endDocumentIsSelecting","_handleSelectionChange","clearInterval","_reportInfiniteLoop","hasDomSelection","CompositionObserver","DataTransfer","nativeDataTransfer","_files","cacheFiles","getFiles","_native","files","getData","setData","effectAllowed","dropEffect","setDragImage","image","isCanceled","mozUserCancelled","kind","getAsFile","InputObserver","domTargetRanges","getTargetRanges","viewDocument","dataTransfer","targetRanges","inputType","partTargetRanges","dataPart","ArrowKeysObserver","TabObserver","domRoots","_initialDomRootAttributes","_observers","_ongoingChange","_postFixersInProgress","_renderingDisabled","_hasChangedSinceTheLastRendering","_writer","addObserver","_render","relatedTarget","attachDomRoot","viewRoot","initialDomRootAttributes","updateContenteditableAttribute","observer","detachDomRoot","getDomRoot","ObserverConstructor","disableObservers","enableObservers","scrollToTheSelection","originalArgs","currentFrame","normalizeViewportOffset","firstAncestorToScroll","scrollViewportToShowTarget","isRenderingInProgress","callbackResult","_disableRendering","flag","getChildStartOffset","_deep","_setAttributesTo","_clearAttributes","NodeList","_nodes","maxOffset","getNode","getNodeIndex","getNodeStartOffset","indexToOffset","offsetToIndex","totalOffset","nodeList","newTarget","spliceArray","_removeNodes","indexStart","fromJSON","getNodeByPath","relativePath","parentName","stickiness","_visitedParent","prevVisitedParent","textNodeAtPosition","getTextNodeAtPosition","getNodeAfterPosition","formatReturnValue","offsetInTextNode","getNodeBeforePosition","newOffset","getParentPath","getCommonPath","diffAt","isTouching","commonLevel","level","checkTouchingBranch","checkOnlyZeroes","hasSameParentAs","getTransformedByOperation","operation","_getTransformedByInsertOperation","_getTransformedByMoveOperation","_getTransformedBySplitOperation","_getTransformedByMergeOperation","_getTransformedByInsertion","_getTransformedByMove","sourcePosition","movedRange","_getCombined","splitPosition","moveTargetPosition","graveyardPosition","_getTransformedByDeletion","deletionPosition","deletePosition","transformed","insertPosition","combined","graveyard","idx","checkOnlyMaxOffset","containsItem","getJoined","shouldJoin","getMinimalFlatRanges","posParent","getTransformedByOperations","operations","j","spread","newPos","moveRange","differenceSet","difference","common","transformedCommon","newStart","newEnd","_createFromRanges","ref","refIndex","Mapper","_modelToViewMapping","_viewToModelMapping","_viewToModelLengthCallbacks","_markerNameToElements","_elementToMarkerNames","_deferredBindingRemovals","_unboundMarkerNames","viewContainer","modelPosition","findPositionIn","viewBlock","findMappedViewAncestor","modelParent","modelOffset","_toModelOffset","modelElement","unbindViewElement","toModelElement","markerName","defer","unbindModelElement","toViewElement","bindElementToMarker","elements","unbindElementFromMarkerName","nameToElements","elementToNames","flushUnboundMarkerNames","markerNames","flushDeferredBindings","clearBindings","toModelRange","toModelPosition","toViewRange","modelRange","toViewPosition","mapper","isPhantom","markerNameToElements","boundElements","registerViewToModelLength","viewElementName","lengthCallback","viewOffset","getModelLength","expectedOffset","lastLength","_moveViewPositionToTextNode","ModelConsumable","_consumable","_textProxyRegistry","_normalizeConsumableType","_getSymbolForTextProxy","consume","itemConsumables","revert","verifyAllConsumed","eventGroup","consumables","canConsume","eventPrefix","description","startMap","endMap","_addSymbolForTextProxy","DowncastDispatcher","conversionApi","_conversionApi","dispatcher","_firedEventsMap","convertChanges","differ","markers","_createConversionApi","getRefreshedItems","getMarkersToRemove","_convertMarkerRemove","changes","_reduceChanges","getChanges","_convertInsert","_convertReinsert","_convertRemove","_convertAttribute","attributeOldValue","attributeNewValue","markerRange","getRange","_convertMarkerAdd","getMarkersToAdd","consumable","convertSelection","modelRoot","markersAtSelection","getMarkersAtPosition","_addConsumablesForSelection","marker","shouldMarkerChangeBeConverted","doNotAddConsumables","_addConsumablesForInsert","walkerValueToEventData","_testAndFire","_addConsumablesForRange","walkerValues","reconversion","getEventName","eventsFiredForConversion","eventsFiredForItem","_testAndFireAddAttributes","refreshedItems","convertItem","convertChildren","convertAttributes","canReuseView","anyNewRange","oldRange","_replaceAllRanges","directChange","_popRange","getSelectedBlocks","visited","startBlock","getParentBlock","isStartBlockSelected","block","isUnvisitedTopBlock","endBlock","isEndBlockSelected","containsEntireContent","limitStartPosition","limitEndPosition","_checkRange","_removeAllRanges","isUnvisitedBlock","schema","isBlock","isTopBlockInRange","hasParentLimit","isLimit","parentBlock","findAncestorBlock","bindWithDocument","toRange","fromRange","isDocumentOperation","boundariesChanged","contentChanged","doesOperationChangeRangeContent","storePrefix","LiveSelection","hasOwnRange","isGravityOverridden","updateMarkers","_updateAttributes","observeMarkers","prefixOrName","_getStoredAttributes","getStoredAttributes","_overrideGravity","overrideGravity","_restoreGravity","restoreGravity","_getStoreAttributeKey","_isStoreAttributeKey","_attributePriority","_selectionRestorePosition","_hasChangedRange","_overriddenGravityRegister","_observedMarkers","_model","_document","_fixGraveyardSelection","_validateSelectionRanges","_updateMarker","batch","changeParent","enqueueChange","storedAttributes","clearAttributesStoredInElement","_getDefaultRange","overrideUid","liveRange","_prepareRange","_validateSelectionRange","changed","markerGroup","selectionRange","oldMarkers","hasMarker","contained","clearAll","newAttributes","_getSurroundingAttributes","oldAttributes","newKey","oldKey","realKey","getTextAttributes","isObject","getNearestSelectionRange","isInline","checkAttribute","getAttributeProperties","copyFromObject","ConversionHelpers","dispatchers","_dispatchers","conversionHelper","DowncastHelpers","elementToElement","normalizeModelElementConfig","normalizeToElementConfig","elementCreator","consumer","defaultConsumer","preflight","reinsertOrConvertNodes","insertElement","createConsumer","converterPriority","createChangeReducer","downcastElementToElement","elementToStructure","checkChild","slotsMap","slot","createSlotFactory","childrenInSlots","flat","uniqueChildrenInSlots","validateSlotsChildren","toViewPositionMapping","currentSlot","currentSlotNodes","fillSlots","downcastElementToStructure","attributeToElement","modelValue","getFromAttributeCreator","oldViewElement","newViewElement","viewWriter","downcastAttributeToElement","attributeToAttribute","normalizeToAttributeConfig","attributeCreator","oldAttribute","downcastAttributeToAttribute","markerToElement","isOpening","viewStartElement","viewEndElement","downcastMarkerToElement","markerToHighlight","highlightDescriptor","descriptor","prepareDescriptor","createViewElementFromHighlightDescriptor","rangeAfterWrap","addHighlightCallback","highlightElement","viewHighlightElement","removeHighlightCallback","removeHighlight","downcastMarkerToHighlight","markerToData","viewCreator","viewMarkerData","handleMarkerBoundary","viewData","removeMarkerFromAttribute","removeMarkerData","downcastMarkerToData","isStart","elementAfter","elementBefore","insertMarkerAsAttribute","insertMarkerAsElement","viewElementType","modelData","viewElementDefinition","createViewElementFromDefinition","modelAttributeValue","shouldReplace","createChangeReducerCallback","reducedChanges","reconvertedElements","changeIndex","positionRelation","modelNodes","modelChildNode","reinsertNode","modelNode","viewChildNode","autoParagraphEmptyRoots","getRoots","isParagraphable","nodeOrType","createContext","wrapInParagraph","paragraph","UpcastHelpers","upcastElementToElement","elementToAttribute","normalizeModelAttributeConfig","converter","prepareToAttributeConverter","getViewElementNameFromConfig","upcastElementToAttribute","viewKey","normalized","normalizeViewAttributeKeyValueConfig","upcastAttributeToAttribute","elementToMarker","normalizeElementToMarkerModelConfig","upcastElementToMarker","dataToMarker","normalizedConfig","converterStart","prepareToElementConverter","normalizeDataToMarkerConfig","converterEnd","basePriority","maxPriority","priorityFactor","attrName","addMarkerElements","markerViewNames","markerViewName","modelCursor","viewItem","upcastAttributeToMarker","upcastDataToMarker","viewConfig","matcherResult","getModelElement","safeInsert","updateConversionResult","viewAttributeKeyToCopy","defaultModelValue","configToTest","onlyViewNameIsDefined","modelKey","attributeWasSet","modelAttribute","setAttributeOn","injectSelectionPostFixer","correctedRange","tryFixingRange","rangesToMerge","rangeIndexesToRemove","currentRangeIndex","currentRange","previousRanges","previousRangeIndex","previousRange","mergedRange","mergeIntersectingRanges","selectionPostFixer","originalPosition","nearestSelectionRange","ancestorObject","fixedPosition","tryFixingCollapsedRange","isTextAllowedOnStart","isTextAllowedOnEnd","startLimitElement","getLimitElement","endLimitElement","startIsOnBlock","endIsOnBlock","checkSelectionOnNonLimitElements","fixedStart","isSelectable","fixedEnd","isStartInLimit","isEndInLimit","bothInSameParent","expandStart","expandEnd","findOutermostLimitAncestor","tryFixingNonCollapsedRage","startingNode","isLimitNode","EditingController","downcastDispatcher","modelSelection","convertSelectionChange","modelEnd","brokenPosition","reconvertMarker","markerOrName","currentMarker","_refresh","reconvertItem","_refreshItem","ViewConsumable","elementConsumables","ViewElementConsumables","consumablesFromElement","createFrom","instance","CONSUMABLE_TYPES","_canConsumeName","_test","_consume","_revert","consumableName","toConsume","Schema","_sourceDefinitions","_attributeProperties","SchemaContext","getDefinition","register","itemName","_clearCache","getDefinitions","_compiledDefinitions","_compile","isRegistered","def","isContent","_checkContextMatch","allowAttributes","checkMerge","positionOrBaseElement","elementToMerge","addChildCheck","childDef","retValue","addAttributeCheck","setAttributeProperties","selectionOrRangeOrPosition","rangeCommonAncestor","checkAttributeInSelection","getValidRanges","convertToMinimalFlatRanges","_getValidRangesForRange","backwardWalker","forwardWalker","limitElement","forward","step","combineWalkers","findAllowedParent","setAllowedAttributes","removeDisallowedAttributes","removeDisallowedAttributeFromNode","positionsInRange","getAttributesWithProperty","attributeProperties","compiledDefinitions","sourceRules","itemNames","compileBaseItemRule","compileAllowChildren","compileAllowContentOf","compileAllowWhere","compileAllowAttributesOf","compileInheritPropertiesFrom","cleanUpAllowIn","setupAllowChildren","cleanUpAllowAttributes","contextItemIndex","contextItem","getItem","allowIn","parentRule","mapContextItem","getNames","query","sourceItemRules","itemRule","allowContentOf","allowWhere","allowAttributesOf","allowChildren","inheritTypesFrom","sourceItemRule","typeNames","copyTypes","copyProperty","inheritFrom","inheritAllFrom","makeInheritAllWork","allowChildrenItem","allowedChildren","allowContentOfItemName","getAllowedChildren","allowedItem","allowWhereItemName","allowedIn","allowAttributeOfItem","inheritAttributes","inheritPropertiesOfItem","existingItems","itemToCheck","allowedParentItemName","ctxItem","UpcastDispatcher","_splitParts","_cursorParents","_modelCursor","_emptyElementsToKeep","store","_convertItem","positionOrElement","_convertChildren","_safeInsert","_updateConversionResult","splitToAllowedParent","_splitToAllowedParent","getSplitParts","_getSplitParts","keepEmptyElement","_keepEmptyElement","contextDefinition","createContextTree","documentFragment","_removeEmptyElements","modelItem","markerElements","markerElement","currentPosition","extractMarkersFromModelFragment","elementOrModelCursor","nextModelCursor","splitResult","savedCursorParent","allowedParent","treeWalkerValue","originalPart","splitPart","_registerSplitPair","cursorParent","anyRemoved","BasicHtmlWriter","getHtml","HtmlDataProcessor","domParser","htmlWriter","toData","toView","_toDom","useFillerType","DataController","upcastDispatcher","htmlProcessor","processor","_viewWriter","isUndoable","_checkIfRootsExists","hasContent","ignoreWhitespaces","modelElementOrFragment","viewDocumentFragment","elementRange","isMarkerCollapsed","isMarkerAtElementBoundary","updatedMarkerRange","n1","n2","localeCompare","_getMarkersRelativeToElement","initialData","main","parse","newData","batchType","removeSelectionAttribute","viewElementOrFragment","addStyleProcessorRules","rootNames","Conversion","downcastDispatchers","upcastDispatchers","_helpers","_downcast","_createConversionHelpers","isDowncast","_upcast","addAlias","alias","for","_getAllUpcastDefinitions","helpers","upcastAlso","_getUpcastDefinition","upcastAlsoItem","Operation","baseVersion","_validate","__className","normalizedNodes","_normalizeNodes","_splitNodeAtPosition","_mergeNodesAtIndex","_move","_haveSameAttributes","mergedNode","offsetDiff","firstPart","secondPart","nodeA","nodeB","iteratorA","iteratorB","MoveOperation","affectedSelectable","getMovedRangeStart","getReversed","newTargetPosition","sourceElement","targetElement","sourceOffset","targetOffset","_execute","InsertOperation","shouldReceiveAttributes","gyPosition","originalNodes","SplitOperation","MergeOperation","splitElement","getInsertionPosition","mergedElement","MarkerOperation","_markers","_set","AttributeOperation","NoOperation","RenameOperation","oldName","RootAttributeOperation","RootOperation","isAdd","createRoot","_isAttached","OperationFactory","transformations","setTransformation","OperationA","OperationB","transformationFunction","aGroup","noUpdateTransformation","getTransformation","transformSets","operationsA","operationsB","contextFactory","ContextFactory","useRelations","forceWeakRemove","setOriginalOperations","originalOperations","nextTransformIndex","op","nextBaseVersionA","nextBaseVersionB","originalOperationsACount","originalOperationsBCount","opA","indexB","opB","newOpsA","getContext","newOpsB","updateRelation","newOpA","padWithNoOps","brokenOperationsACount","brokenOperationsBCount","updateBaseVersions","_history","history","_useRelations","_forceWeakRemove","_relations","takeFrom","originalOperation","_setRelation","affectedLeft","affectedRight","side","wasInLeftElement","wasStartBeforeMergedElement","wasEndBeforeMergedElement","wasInRightElement","aIsStrong","aWasUndone","_wasUndone","bWasUndone","abRelation","_getRelation","baRelation","originalOp","wasUndone","isUndoneOperation","origB","undoneB","getUndoneOperation","origA","relationsA","relation","_getComplementaryAttributeOperations","insertOperation","insertValue","_moveTargetIntoMovedRange","_makeMoveOperationsFromRanges","moveOp","diffs","_breakRangeByMoveOperation","aNewRange","aToGraveyard","bToGraveyard","removedRange","mergeInside","mergeSplittingElement","aCompB","shouldSpread","rightRange","movesGraveyardElement","gyMoveSource","splitNodesMoveSource","gyMoveTarget","gyMove","splitNodesMoveTargetPath","splitNodesMoveTarget","splitNodesMove","extraRename","splitPath","additionalSplit","rangeToMove","gyElementMoved","newParentPosition","newTargetPath","splitAtTarget","howManyRemoved","aInGraveyard","bInGraveyard","newPositionPath","toPosition","fromPosition","oldPosition","Batch","isLocal","isUndo","isTyping","addOperation","Differ","markerCollection","_changesInElement","_elementSnapshots","_changedMarkers","_changedRoots","_changeCount","_cachedChanges","_cachedChangesWithGraveyard","_refreshedItems","_markerCollection","bufferOperation","operationToBuffer","_isInInsertedElement","_markInsert","_markAttribute","sourceParentInserted","targetParentInserted","_markRemove","getMarkersIntersectingRange","markerData","bufferMarkerChange","graveyardParent","mergedIntoElement","_isLoaded","_bufferRootStateChange","_bufferRootAttributeChange","oldMarkerData","newMarkerData","buffered","getChangedMarkers","hasDataChanges","markerAdded","markerRemoved","markerChanged","includeChangesInGraveyard","diffSet","snapshotChildren","elementChildren","_getChildrenSnapshot","_generateActionsFromChanges","_getInsertDiff","_getRemoveDiff","elementAttributes","snapshotAttributes","_getAttributesDiff","changeCount","prevIndex","prevDiff","thisDiff","isConsecutiveTextRemove","isConsecutiveTextAdd","isConsecutiveAttributeChange","_changesInGraveyardFilter","getChangedRoots","diffItem","state","reset","attrEntry","_bufferRootLoad","changeItem","_markChange","_removeAllNestedChanges","_makeSnapshot","_getChangesForElement","_handleChange","inc","nodesToHandle","old","incEnd","oldEnd","intersectionLength","howManyAfter","attributePart","elementSnapshot","snapshot","oldChildrenLength","oldChildrenHandled","posInGy","rangeInGy","History","_operations","_undoPairs","_undoneOperations","_baseVersionToOperationIndex","_version","_gaps","lastOperation","historyVersion","getOperations","fromBaseVersion","toBaseVersion","firstOperation","inclusiveTo","gapFrom","gapTo","fromIndex","toIndex","getOperation","operationIndex","setOperationAsUndone","undoneOperation","undoingOperation","isUndoingOperation","RootElement","graveyardName","_hasSelectionChangedFromTheLastChangeBlock","removeMarker","getRootNames","includeDetached","_handleChangeBlock","_hasDocumentChangedFromTheLastChangeBlock","_getDefaultRoot","defaultRoot","createPositionFromPath","validateTextNodePosition","rangeBoundary","MarkerCollection","Marker","managedUsingOperations","oldMarker","hasChanged","_attachLiveRange","_managedUsingOperations","_destroyMarker","getMarkersGroup","prefix","_detachLiveRange","_liveRange","getStart","getEnd","DetachOperation","Writer","cloneElement","_assertWriterUsedCorrectly","isSameTree","applyOperation","rangeRootPosition","usingOperation","updateMarker","addMarker","insertText","itemOrPositionOrOffset","appendText","appendElement","itemOrRange","setAttributeOnRange","setAttributeOnItem","setAttributes","clearAttributes","removeAttributesFromItem","_addOperationForAffectedMarkers","applyRemoveOperation","merge","_merge","_mergeDetached","renameOperation","firstSplitElement","firstCopyElement","elementOrString","shiftedRange","applyMarkerOperation","hasUsingOperationDefined","affectsDataDefined","updatedRange","addRoot","detachRoot","rootOrName","setSelectionAttribute","keyOrObjectOrIterable","_setSelectionAttribute","keyOrIterableOfKeys","_removeSelectionAttribute","overrideSelectionGravity","restoreSelectionGravity","storeKey","_currentWriter","isAffected","affectedInLeftElement","affectedInRightElement","affectedAfterLeftElement","affectedBeforeRightElement","valueBefore","valueAfter","lastSplitPosition","previousValue","rootA","rootB","deleteContent","selRange","doNotResetEntireContent","shouldEntireContentBeReplacedWithParagraph","insertParagraph","replaceEntireContentWithParagraph","attributesForAutoparagraph","doNotAutoparagraph","ignoreMarkers","modifySelection","newEndPosition","skippedRange","getLivePositionsForSelectedBlocks","leaveUnmerged","checkShouldMerge","startAncestor","endAncestor","positionA","positionB","getAncestorsJustBelowCommonAncestor","mergeBranchesRight","mergeBranchesLeft","mergeBranches","collapseSelectionAt","isTextAllowed","isParagraphAllowed","shouldAutoparagraph","commonAncestor","parentToRemove","mergeRight","leftPos","rightPos","rangeToCheck","isCrossingLimitElement","removeRangeContent","parentsToCheck","itemRange","parentToCheck","removeRange","Insertion","_firstNode","_lastNode","_lastAutoParagraph","_filterAttributesOf","_affectedStart","_affectedEnd","_nodeToSelect","canMergeWith","_documentFragment","_documentFragmentPosition","handleNodes","_handleNode","_insertPartialFragment","_updateLastNodeFromAutoParagraph","_mergeOnRight","positionAfterLastNode","positionAfterNode","_setAffectedBoundaries","getSelectionRange","getAffectedRange","_handleObject","isAllowed","_checkAndAutoParagraphToAllowedPosition","_checkAndSplitToAllowedPosition","_appendToFragment","_handleDisallowedNode","livePosition","_mergeOnLeft","_tryAutoparagraphing","_canMergeLeft","mergePosLeft","_canMergeRight","mergePosRight","_getAllowedIn","tempPos","contextElement","childNode","findOptimalInsertionRange","place","firstBlock","positionAfter","insertObject","originalSelection","insertionSelection","findOptimalPosition","firstSelectedBlock","attributesToCopy","elementToInsert","insertionPositionParent","affectedRange","insertContent","paragraphAttributes","nextElement","canSetSelection","updateSelection","wordBoundaryCharacters","tryExtendingTo","isForward","treatEmojiAsSingleUnit","isAtNodeBoundary","isAtWordBoundary","getCorrectWordBreakPosition","getCorrectPosition","getSearchRange","searchEnd","offsetToCheck","_pendingChanges","childDefinition","insertion","fakeMarkerElements","nodesToInsert","markersPosition","posA","posB","isAtBeginning","selectionLiveRange","markersData","isStartBoundary","elementPosition","rangeOnInsertion","_runPendingChanges","batchOrType","rest","normalizeSelectable","getSelectedContent","frag","commonPath","commonParent","flatSubtreeRange","leftExcessRange","rangeOrElement","intersectingMarker","createBatch","createOperationFromJSON","ret","currentBatch","callbackReturnValue","ClickObserver","insertChild","removeChildren","oldElement","valueOrElement","CommandCollection","_commands","commandParams","commands","Editor","_readOnlyLocks","editing","keystrokes","enableReadOnlyMode","lockId","disableReadOnlyMode","removePlugins","extraPlugins","readyPromise","DataApiMixin","ElementApiMixin","updateSourceElement","shouldUpdateSourceElement","isSourceElementTextArea","PendingActions","_actions","hasAny","bold","caption","check","cog","eraser","lowVision","importExport","plus","alignBottom","alignMiddle","alignTop","alignLeft","alignCenter","alignRight","alignJustify","objectLeft","objectCenter","objectRight","objectFullWidth","objectInline","objectBlockLeft","objectBlockRight","objectSizeFull","objectSizeLarge","objectSizeSmall","objectSizeMedium","pencil","pilcrow","threeVerticalDots","activator","contextElements","composedPath","contextElementsList","CssTransitionDisablerMixin","disableCssTransitions","_isCssTransitionsDisabled","enableCssTransitions","initializeCssTransitionDisablerMixin","extendTemplate","bindTemplate","if","initialItems","_renderViewIntoCollectionParent","_parentElement","setParent","elementOrDocFragment","dest","evtName","isRendered","_viewCollections","_unboundChildren","createCollection","collection","_bindTemplate","views","registerChild","deregisterChild","setTemplate","template","getViews","_revertData","_isRendered","_renderNode","intoFragment","isApplying","revertData","_revertTemplateFromNode","isView","isTemplate","eventNameOrFunctionOrAttribute","TemplateToBinding","eventNameOrFunction","valueIfTrue","TemplateIfBinding","isInvalid","_renderText","_renderElement","ns","_renderAttributes","_renderElementChildren","_setUpListeners","hasTemplateBinding","_bindToObservable","getTextUpdater","domAttrValue","attrValue","attrNs","isNamespaced","valueToBind","shouldExtend","getAttributeUpdater","_renderStyleAttribute","arrayValueReducer","isFalsy","setAttributeNS","styleName","styleValue","getStyleUpdater","childIndex","isViewCollection","childRevertData","eventListeners","revertBindings","schemaItem","domEvtName","domSelector","activateDomEventListener","syncValueSchemaValue","templateBinding","activateAttributeListener","revertBinding","TemplateBinding","getValue","getValueSchemaValue","removeAttributeNS","normalizePlainTextDefinition","normalizeTextDefinition","listeners","arrayify","normalizeListeners","normalizeAttributes","extendObjectValueArray","ext","BodyCollection","bodyCollectionContainer","_bodyCollectionContainer","attachToDom","detachFromDom","childElementCount","viewBox","_updateXMLContent","_colorFillPaths","svg","presentationalAttributeNames","fillColor","querySelectorAll","IconView","_focusDelayed","ariaLabelUid","labelView","_createLabelView","iconView","keystrokeView","_createKeystrokeView","_getTooltipString","role","tabindex","isToggleable","click","mousedown","mouseup","icon","withKeystroke","ariaLabelledBy","tooltip","toggleSwitchView","_createToggleView","viewUid","statusUid","fieldView","statusView","_createStatusView","fieldWrapperChildren","errorText","infoText","focusTracker","readonly","inputmode","_updateIsEmpty","_setDomElementValue","select","DropdownPanelView","selectstart","toLocaleLowerCase","dropdownPanel","focusLast","buttonView","panelView","isOpen","panelPosition","_getOptimalPosition","_panelPositions","closeDropdown","south","north","southEast","southWest","northEast","northWest","southMiddleEast","southMiddleWest","northMiddleEast","northMiddleWest","defaultPanelPositions","DropdownView","buttonRect","panelRect","arrowView","_createArrowView","focusables","keystrokeHandler","isFocusable","_getFocusableItem","previous","viewIndex","focused","focusFirst","focusNext","focusPrevious","collectionLength","ToolbarSeparatorView","ToolbarLineBreakView","normalizeToolbarConfig","removeItems","NESTED_TOOLBAR_ICONS","itemsView","ItemsView","isRtl","_focusCycler","shouldGroupWhenFull","isFloating","maxWidth","_behavior","DynamicGrouping","StaticLayout","fillFromConfig","itemsOrConfig","_buildItemsFromConfig","normalizedRemoveItems","_cleanItemsConfiguration","_createNestedToolbarDropdown","filteredItems","_cleanSeparatorsAndLineBreaks","nonSeparatorPredicate","firstCommandItemIndex","findIndex","lastCommandItemIndex","componentFactory","withText","dropdownView","addToolbarToDropdown","toolbarView","resizeObserver","cachedPadding","shouldUpdateGroupingOnNextResize","viewFocusables","viewItemsView","viewFocusTracker","viewLocale","ungroupedItems","groupedItems","groupedItemsDropdown","_createGroupedItemsDropdown","_updateFocusCycleableItems","changeData","removedItem","currentIndex","addedItem","_updateGrouping","_enableGroupingOnResize","_enableGroupingOnMaxWidthChange","initialGroupedItemsCount","wereItemsGrouped","_areItemsOverflowing","_groupLastItem","_ungroupFirstItem","lastChildRect","toolbarRect","computedStyle","paddingProperty","previousWidth","contentRect","dropdown","tooltipPosition","ListView","ListItemView","ListSeparatorView","actionView","_createActionView","ButtonClass","closeDropdownOnClickOutside","closeDropdownOnExecute","closeDropdownOnBlur","focusDropdownContentsOnArrows","focusDropdownButtonOnClose","focusDropdownPanelOnOpen","addDefaultBehavior","buttonsOrCallback","addToolbarToOpenDropdown","enableActiveItemFocusOnDropdownOpen","isOn","buttons","ariaLabel","isCompact","isVertical","itemsOrCallback","addListToOpenDropdown","listView","listItemView","childSelectorCallback","childToFocus","labeledFieldView","inputView","ariaDescribedById","digits","pow","hexToRgba","hsvaToHsla","hh","hsvaToHslString","hsva","hsvaToRgba","format","rgbaToHex","alphaHex","rgbaToHsva","equalColorObjects","second","tpl","detail","dispatchEvent","CustomEvent","bubbles","hasTouched","isTouch","pointerMove","pointer","touches","getMove","pageX","pageXOffset","pageY","pageYOffset","Slider","aria","xy","cloneNode","dragging","toggleEvent","handleEvent","isValid","button","keyMove","setProperty","Hue","Saturation","$isSame","$color","$hsva","$update","$parts","$css","$sliders","ColorPicker","observedAttributes","newColor","newHsva","colorModel","toHsva","attachShadow","mode","slider","connectedCallback","defaultColor","attributeChangedCallback","_attr","_oldVal","newVal","fromAttr","oldHsva","fromHsva","HexBase","customElements","ComponentFactory","_components","originalName","toPx","defaultLimiterElement","_pinWhenIsVisibleCallback","show","hide","attachTo","defaultPositions","southArrowNorth","southArrowNorthMiddleWest","southArrowNorthMiddleEast","southArrowNorthWest","southArrowNorthEast","northArrowSouth","northArrowSouthMiddleWest","northArrowSouthMiddleEast","northArrowSouthWest","northArrowSouthEast","viewportStickyNorth","optimalPosition","withArrow","pin","unpin","_startPinning","_stopPinning","getDomElement","scrollTarget","isWithinScrollTarget","isLimiterWithinScrollTarget","BalloonPanelView","generatePositions","sideOffset","arrowSideOffset","heightOffset","arrowHeightOffset","stickyVerticalOffset","northWestArrowSouthWest","balloonRect","getNorthTop","northWestArrowSouthMiddleWest","northWestArrowSouth","northWestArrowSouthMiddleEast","northWestArrowSouthEast","northEastArrowSouthWest","northEastArrowSouthMiddleWest","northEastArrowSouth","northEastArrowSouthMiddleEast","northEastArrowSouthEast","southWestArrowNorthWest","getSouthTop","southWestArrowNorthMiddleWest","southWestArrowNorth","southWestArrowNorthMiddleEast","southWestArrowNorthEast","southEastArrowNorthWest","southEastArrowNorthMiddleWest","southEastArrowNorth","southEastArrowNorthMiddleEast","southEastArrowNorthEast","westArrowEast","eastArrowWest","BALLOON_CLASS","_currentElementWithTooltip","_currentTooltipPosition","_resizeObserver","_editors","_instance","tooltipTextView","balloonPanelView","_pinTooltipDebounced","_pinTooltip","_onEnterOrFocus","_onLeaveOrBlur","_onScroll","_watchdogExcluded","editorBodyViewCollection","_unpinTooltip","getPositioningFunctions","defaultBalloonPositions","sw","se","elementWithTooltipAttribute","getDescendantWithTooltip","ckeTooltipText","ckeTooltipPosition","cssClass","ckeTooltipClass","descendantWithTooltip","relatedDescendantWithTooltip","targetDomElement","bodyViewCollection","_updateTooltipPosition","TooltipManager","closest","NARROW_ROOT_HEIGHT_THRESHOLD","NARROW_ROOT_WIDTH_THRESHOLD","DEFAULT_LABEL","OFF_THE_SCREEN_POSITION","PoweredBy","_balloonView","_lastFocusedEditableElement","_showBalloonThrottled","_showBalloon","_handleEditorReady","balloon","token","oldTokenCheck","decryptedData","splittedDecryptedData","firstElement","secondElement","decryptedSecondElement","year","monthIndex","day","date","isNaN","verifyLicense","_updateLastFocusedEditableElement","_hideBalloon","_createBalloonView","poweredByConfig","getNormalizedConfig","PoweredByView","attachOptions","focusedEditableElement","getLowerCornerPosition","rootRect","horizontalOffset","getLowerRightCornerPosition","getLowerLeftCornerPosition","getBalloonAttachOptions","editableEditorElements","getEditableElementsNames","getEditableElement","isColorInherited","ICON_WIDTH","ICON_HEIGHT","href","dragstart","getBalloonLeft","editableElementRect","visibleEditableElementRect","balloonTop","verticalOffset","balloonLeft","newBalloonRect","firstScrollableEditableElementAncestor","overflowY","firstScrollableEditableElementAncestorRect","notVisibleVertically","notVisibleHorizontally","userConfig","EditorUI","isReady","_editableElementsMap","_focusableToolbarDefinitions","editingView","tooltipManager","poweredBy","_readViewportOffsetFromConfig","_handleScrollToTheSelection","_initFocusTracking","ckeditorInstance","setEditableElement","setUpKeystrokeHandler","removeEditableElement","addToolbar","_editableElements","editorUI","legacyOffsetConfig","lastFocusedForeignElement","candidateDefinitions","currentFocusedToolbarDefinition","_getCurrentFocusedToolbarDefinition","_getFocusableCandidateToolbarDefinitions","candidateDefinition","_focusFocusableCandidateToolbar","afterBlur","focusedToolbarDef","definitions","toolbarDef","beforeFocus","defA","defB","getToolbarDefinitionWeight","candidateToolbarDefinition","configuredViewportOffset","weight","isContextual","EditableUIView","lang","_editableElement","_hasExternalElement","_editingView","_updateIsFocusedClasses","hasExternalElement","updateAfterRender","InlineEditableUIView","_generateLabel","Notification","alert","showSuccess","_showNotification","title","showInfo","showWarning","_viewToStack","_idToStack","_view","_rotatorView","_fakePanelsView","positionLimiter","_createPanelView","hasView","stackId","_numberOfStacks","_visibleStack","singleViewMode","showStack","_showView","_singleViewMode","visibleView","_showNextStack","hideView","_getStackId","updatePosition","_getBalloonPosition","visibleStack","_createRotatorView","_createFakePanelsView","stacks","nextIndex","_showPrevStack","RotatorView","isSingleViewMode","numberOfStacks","buttonNextView","buttonPrevView","FakePanelsView","balloonClassName","showView","_createButtonView","_balloonPanelView","_addPanels","_removePanels","numberOfPanels","BalloonToolbar","_balloonConfig","_createToolbarView","_balloon","_fireSelectionChangeDebounced","isToolbarVisible","shouldNotGroupWhenFull","_updatePosition","afterInit","showForCollapsedSelection","selectionContainsOnlyMultipleSelectables","_getBalloonPositionData","_getBalloonPositions","visualViewport","scale","BlockButtonView","Watchdog","crashes","_now","_crashNumberLimit","crashNumberLimit","_minimumNonErrorTimePeriod","minimumNonErrorTimePeriod","_boundErrorHandler","reason","_handleError","_listeners","_restart","_stopErrorHandling","cb","_fire","_startErrorHandling","_shouldReactToError","filename","ErrorEvent","lineno","colno","causesRestart","_shouldRestart","_isErrorComingFromThisItem","excludedProperties","subNodes","nodeIndex","shouldNodeBeIncluded","typeOfNode","EventTarget","Event","areConnectedThroughProperties","target1","target2","excludedNodes","structure","subNodes1","subNodes2","EditorWatchdog","watchdogConfig","_editor","_initUsingData","_editables","_throttledSave","_save","saveInterval","_creator","elementOrData","_destructor","setCreator","creator","setDestructor","destructor","_destroy","catch","existingRoots","lazyRoots","oldRootsAttributes","rootsAttributes","rootData","isLoaded","updatedConfig","_watchdogInitialData","EditorWatchdogInitPlugin","_elementOrData","_cloneEditorConfiguration","_lastDocumentVersion","_getData","_getEditables","_setExcludedProperties","_excludedProps","commentsRepository","trackChanges","commentThreads","suggestions","rangeJSON","getCommentThreads","skipNotAttached","getSuggestions","editables","_restoreCollaborationData","_restoreEditorData","_createNode","jsonNode","parsedNodes","parsedAttributes","rootElement","markerOptions","parsedCommentThreads","parsedSuggestions","commentThreadData","channelId","hasCommentThread","threadId","getCommentThread","addCommentThread","suggestionData","trackChangesEditing","hasSuggestion","getSuggestion","addSuggestionData","mainQueueId","ActionQueues","_onEmptyCallbacks","_queues","_activeActions","onEmpty","onEmptyCallback","enqueue","queueId","isMainAction","queueWithAction","nonErrorQueue","finally","elementOrArray","BalloonEditorUI","editingRoot","_initPlaceholder","placeholderText","BalloonEditorUIView","editableView","sourceElementOrData","getInitialData","secureSourceElement","isTextArea","form","originalSubmit","onSubmit","submit","attachToForm","ContextWatchdog","_watchdogs","_contextProps","_actionQueues","_watchdogConfig","contextConfig","_contextConfig","_create","_getWatchdog","getItemState","itemConfigurationOrItemConfigurations","itemConfigurations","watchdog","res","rethrowRestartEventOnce","itemIdOrItemIds","itemIds","handleInput","dropRange","clipboardData","domDoc","clientX","clientY","caretRangeFromPoint","rangeParent","rangeOffset","getDropViewRange","smallPaddingElements","viewToPlainText","childText","_setupPasteDrop","_setupCopyCut","contentData","fullMatch","spaces","modelFragment","resultRange","modelDocument","onCopyCut","ChangeBuffer","limit","_batch","_size","_isLocked","_changeCallback","_reset","_selectionChangeCallback","isLocked","lock","unlock","ignoreLock","InsertTextCommand","undoStepSize","_buffer","textInsertions","TYPING_INPUT_TYPES","InsertTextObserver","Input","insertTextCommand","viewResultRange","modelRanges","selectedText","rangeText","insertTextCommandData","deleteSelectionContent","DeleteCommand","sequence","_shouldEntireContentBeReplacedWithParagraph","_replaceEntireContentWithParagraph","_shouldReplaceFirstBlockWithParagraph","limitElementFirstChild","DELETE_WORD","DELETE_SELECTION","DELETE_BACKWARD","DELETE_FORWARD","DELETE_EVENT_TYPES","deleteContentBackward","deleteWordBackward","deleteHardLineBackward","deleteSoftLineBackward","deleteContentForward","deleteWordForward","deleteHardLineForward","deleteSoftLineForward","DeleteObserver","deleteEventSpec","selectionToRemove","shouldUseTargetRanges","pressedKeyCode","beforeInputReceived","isDeleteKeyCode","getDeleteDirection","shouldFireDeleteEvent","targetRange","enableChromeWorkaround","_undoOnBackspace","deleteForwardCommand","commandData","requestUndoOnBackspace","Typing","getLastTextLine","TextWatcher","testCallback","_hasMatch","_startListening","hasMatch","_evaluateTextBeforeSelection","suffix","rangeBeforeSelection","testResult","eventData","_overrideUid","arrowRightPressed","arrowLeftPressed","contentDirection","isMovementHandled","_handleForwardMovement","_handleBackwardMovement","_isNextGravityRestorationSkipped","_isGravityOverridden","isBetweenDifferentAttributes","registerAttribute","hasAnyAttribute","preventCaretMovement","setSelectionAttributesFromTheNodeBefore","positionBefore","isStepAfterAnyAttributeBoundary","observedAttribute","isInlineObject","attrBefore","reHasRegExpChar","TRANSFORMATIONS","copyright","registeredTrademark","trademark","oneHalf","oneThird","twoThirds","oneForth","threeQuarters","lessThanOrEqual","greaterThanOrEqual","notEqual","arrowLeft","arrowRight","horizontalEllipsis","enDash","emDash","quotesPrimary","buildQuotesRegExp","quotesSecondary","quotesPrimaryEnGb","quotesSecondaryEnGb","quotesPrimaryPl","quotesSecondaryPl","TRANSFORMATION_GROUPS","symbols","mathematical","typography","quotes","DEFAULT_TRANSFORMATIONS","normalizeFrom","normalizeTo","getTextAttributesAfterPosition","quoteCharacter","findAttributeRange","findAttributeRangeBound","lookBack","lastNode","getCopyOnEnterAttributes","allAttributes","copyOnEnter","EnterCommand","enterBlock","isSelectionEmpty","splitBlock","isContainedWithinOneElement","splitPos","ENTER_EVENT_TYPES","isSoft","insertLineBreak","EnterObserver","shiftPressed","enterEventSpec","ShiftEnterCommand","insertBreak","softBreakAction","anchorPos","isInsideLimitElement","breakLineElement","ShiftEnter","HighlightStack","_stack","oldTop","_insertDescriptor","newTop","compareDescriptors","oldDescriptor","newDescriptor","_removeDescriptor","shouldABeBeforeB","classesToString","WIDGET_CLASS_NAME","WIDGET_SELECTED_CLASS_NAME","toWidget","labelOrCreator","widgetLabel","setLabel","hasSelectionHandle","widgetElement","selectionHandle","addSelectionHandle","setHighlightHandling","addHighlight","typeAroundFakeCaretPosition","getTypeAroundFakeCaretPosition","TYPE_AROUND_SELECTION_ATTRIBUTE","isTypeAroundWidget","POSSIBLE_INSERTION_POSITIONS","RETURN_ARROW_ICON_ELEMENT","PLUGIN_DISABLED_EDITING_ROOT_CLASS","WidgetTypeAround","_currentFakeCaretModelElement","_enableTypeAroundUIInjection","_enableInsertingParagraphsOnButtonClick","_enableInsertingParagraphsOnEnterKeypress","_enableInsertingParagraphsOnTypingKeystroke","_enableTypeAroundFakeCaretActivationUsingKeyboardArrows","_enableDeleteIntegration","_enableInsertContentIntegration","_enableInsertObjectIntegration","_enableDeleteContentIntegration","_insertParagraph","widgetModelElement","_listenToIfEnabled","_insertParagraphAccordingToFakeCaretPosition","selectedModelElement","buttonTitles","before","after","widgetViewElement","typeAroundWrapper","wrapperDomElement","buttonTemplate","importNode","injectButtons","caretTemplate","injectFakeCaret","injectUIIntoWidget","positionToWidgetCssClass","domEventData","_handleArrowKeyPress","selectedViewElement","localizedKeyCodeDirection","isForwardArrowKeyCode","shouldStopAndPreventDefault","_handleArrowKeyPressOnSelectedWidget","_handleArrowKeyPressWhenSelectionNextToAWidget","_handleArrowKeyPressWhenNonCollapsedSelection","widgetPlugin","modelElementNextToSelection","_getObjectElementNextToSelection","_setSelectionOverElement","selectedModelNode","buttonPosition","classList","getTypeAroundButtonPosition","widgetDomElement","getClosestWidgetViewElement","wasHandled","selectedModelWidget","isDeleteForward","probe","deepestEmptyRangeAncestor","deepestEmptyAncestor","getDeepestEmptyElementAncestor","documentSelection","verticalNavigationHandler","arrowUpPressed","arrowDownPressed","expandSelection","selectionWillShrink","getNearestNonInlineLimit","lastRangePosition","getNearestTextPosition","firstRangePosition","findTextRangeFromSelection","boundaryVerticalPosition","isSingleLineRange","walkerValueType","_previouslySelected","_clearPreviouslySelectedWidgets","lastMarked","isChild","_onMousedown","_handleSelectionChangeOnArrowKeyPress","_preventDefaultOnArrowKeyPress","_handleDelete","currentElement","isInsideNestedEditable","objectElement","isVerticalNavigation","firstPosition","lastPosition","firstSelectedNode","lastSelectedNode","objectElementNextToSelection","previousNode","nodeToRemove","widget","_toolbarDefinitions","balloonToolbar","isWidgetSelected","_updateToolbarsVisibility","toolbarConfig","toolbarId","getRelatedElement","toolbarDefinition","itemsConfig","initialized","relatedElement","_showToolbar","_hideToolbar","maxRelatedElementDepth","deepestRelatedElement","deepestToolbarDefinition","relatedElementDepth","_isToolbarVisible","_isToolbarInBalloon","getBalloonPositionData","toolbar","DragDrop","_draggedRange","_draggingUid","_draggableElement","_updateDropMarkerThrottled","_updateDropMarker","_removeDropMarkerDelayed","_removeDropMarker","_clearDraggableAttributesDelayed","_clearDraggableAttributes","_setupDragging","_setupContentInsertionIntegration","_setupClipboardInputIntegration","_setupDropMarker","_setupDraggableAttributeHandling","_finalizeDragging","draggableWidget","findDraggableWidget","canEditAtDraggedRange","draggedSelection","findDropTargetRange","getFinalDropEffect","clipboardPipeline","isMove","isSuccess","draggableElement","moved","targetViewRanges","targetViewElement","targetViewPosition","findDropTargetRangeOnWidget","targetModelElement","getClosestMappedModelElement","targetModelPosition","positionAtElementStart","betweenBlocksPosition","findDropTargetRangeBetweenBlocks","findDropTargetRangeOnAncestorObject","findDropTargetRangeInElement","PastePlainText","isPlainTextFragment","textAttributes","isFormatting","SelectAllCommand","scopeElement","isSelectAllScope","SELECT_ALL_KEYSTROKE","SelectAllEditing","SelectAllUI","SelectAll","BaseCommand","_createdBatches","clearStack","createdBatches","addBatch","docSelection","_restoreSelection","selectionRanges","transformedRangeGroups","allRanges","rangeGroup","isRangeContainedByAnyOtherRange","normalizeRanges","_undo","batchToUndo","undoingBatch","operationsToUndo","operationToUndo","nextBaseVersion","historyOperations","reversedOperations","joinedRange","UndoCommand","batchIndex","RedoCommand","redoingBatch","UndoEditing","_batchRegistry","_undoCommand","_redoCommand","isRedoBatch","isUndoBatch","undoneBatch","UndoUI","localizedUndoIcon","undo","redo","localizedRedoIcon","_addButton","Icon","FileReader","reader","_reader","onprogress","loaded","read","file","total","reject","onload","onerror","onabort","readAsDataURL","abort","loaders","_loadersMap","_pendingAction","_updatePendingAction","uploaded","getLoader","fileOrPromise","createLoader","createUploadAdapter","loader","FileLoader","aggregatedUploaded","aggregatedTotal","uploadTotal","destroyLoader","fileOrPromiseOrLoader","pendingActions","getMessage","uploadedPercent","filePromise","uploadAdapterCreator","_filePromiseWrapper","_createFilePromiseWrapper","_adapter","status","upload","uploadResponse","isFulfilled","rejecter","FileDialogButtonView","_fileInputView","FileInputView","open","accept","multiple","TOKEN_COOKIE_NAME","tokenCharset","getCsrfToken","cookie","pair","decodeURIComponent","getCookie","randValues","crypto","getRandomValues","generateToken","UploadAdapter","url","_initRequest","_initListeners","_sendRequest","xhr","XMLHttpRequest","responseType","genericError","response","default","lengthComputable","FormData","send","blockAutoformatEditing","callbackOrCommand","blockToFormat","firstNode","firstNodeRange","blockRange","inlineAutoformatEditing","testRegexpOrCallback","formatCallback","regExp","leftDel","rightDel","delStart","delEnd","getTextAfterCode","testOutput","rangesToFormat","testOutputToRanges","rangesToRemove","arrays","getCallbackFunctionForInlineAutoformat","validRanges","_getValueFromFirstAllowedNode","forceValue","BOLD","BoldEditing","fontWeight","BoldUI","ITALIC","ItalicEditing","ItalicUI","BlockQuoteCommand","_getValue","blocks","blocksToQuote","findQuote","checkCanBeQuoted","_applyQuote","_removeQuote","getRangesOfBlockGroups","groupRange","quotesToMerge","currentQuote","nextQuote","elementOrPosition","nextBlock","isBQAllowed","isBlockAllowedInBQ","BlockQuoteEditing","blockQuoteCommand","BlockQuoteUI","CKBoxUI","getImageUrls","imageUrls","responsiveUrls","imageSources","srcset","sizes","imageFallbackUrl","CKBoxCommand","_chosenAssets","_wrapper","imageCommand","linkCommand","_prepareOptions","ckboxConfig","theme","tokenUrl","serviceOrigin","dialog","onClose","assets","onChoose","shouldInsertDataId","CKBox","mount","assetsToProcess","isImageAllowed","isLinkAllowed","asset","metadata","isImage","prepareImageAssetAttributes","prepareLinkAssetAttributes","prepareAssets","isLastAsset","_insertAsset","_insertImage","_insertLink","imageTextAlternative","src","linkName","linkHref","selectionAttributes","getAssetUrl","URL","searchParams","CKBoxUploadAdapter","CKBoxEditing","hasConfiguration","isLibraryLoaded","fileRepository","ckboxEditing","getToken","imageUploadEditing","imageElement","ckboxImageId","controller","AbortController","getWorkspaceId","cannotAccessDefaultWorkspaceError","defaultWorkspaceId","workspaceId","binaryTokenPayload","payload","workspaces","auth","ckbox","aud","getAvailableCategories","categoryUrl","_sendHttpRequest","totalCount","offsetItems","signal","throwIfAborted","getCategoryIdForFile","extension","extensionRegExp","getFileExtension","allCategories","defaultCategories","userCategory","category","serverCategory","extensions","cannotFindCategoryError","uploadUrl","formData","requestConfig","onUploadProgress","setRequestHeader","abortCallback","statusCode","_initConfig","cloudServicesCore","ckboxTokenUrl","cloudServices","_token","createToken","_initSchema","_initConversion","_initFixers","defaultUploadCategories","ignoreDataId","linkInImage","createLinkElement","shouldUpcastAttributeForNode","ckboxCommand","isLinkHrefAttributeRemoval","findAssetsForItem","syncDataIdPostFixer","isImageElement","isLinkElement","CKFinderUI","CKFinderCommand","openerMethod","chooseFiles","originalOnInit","onInit","finder","toArray","links","images","linkFile","getUrl","imagesUrls","request","insertImages","resizedUrl","notification","CKFinder","urls","CKFinderEditing","CloudServicesUploadAdapter","_uploadGateway","createUploadGateway","uploadGateway","fileUploader","ParagraphCommand","checkCanBecomeParagraph","InsertParagraphCommand","_findPositionToInsertParagraph","paragraphLikeElements","HeadingCommand","modelElements","heading","checkCanBecomeHeading","defaultModelElement","HeadingEditing","_addDefaultH1Conversion","enterCommand","HeadingUI","localizedTitles","defaultTitle","accessibleLabel","titles","itemDefinitions","headingCommand","paragraphCommand","commandValue","areEnabled","para","whichModel","createBlockImageViewElement","getImgViewElementMatcher","matchImageType","imageUtils","areBothImagePluginsLoaded","isInlineImageView","getPositiveMatchPattern","isBlockImageView","determineImageTypeForInsertionAtSelection","isInlineImage","isBlockImage","insertImage","imageType","determineImageTypeForInsertion","getClosestSelectedImageWidget","isImageWidget","getClosestSelectedImageElement","insertionRange","getInsertImageParent","isImageAllowedInParent","isNotInsideImage","toImageWidget","altText","findViewImgElement","figureView","configImageInsertType","ImageTextAlternativeCommand","ImageTextAlternativeEditing","TextAlternativeFormView","labeledInput","_createLabeledInputView","saveButtonView","_createButton","cancelButtonView","_focusables","ImageTextAlternativeUI","_form","_showForm","_createForm","_hideForm","_isVisible","_isInBalloon","focusEditable","ImageTextAlternative","downcastSrcsetAttribute","img","downcastImageAttribute","_fireEvents","InsertImageCommand","sourceDefinitions","sourceDefinition","ReplaceImageSourceCommand","ImageEditing","viewImage","insertImageCommand","replaceImageSourceCommand","ImageTypeCommand","modelElementName","_modelElementName","uploadId","newElementRange","ImageBlockEditing","_setupConversion","_setupClipboardIntegration","modelImage","upcastImageFigure","docFragmentChildren","blockViewImages","inlineViewImage","ImageBlock","ImageInlineEditing","createInlineImageViewElement","inlineViewImages","blockViewImage","ImageInline","ImageCaptionUtils","getCaptionFromImageModelElement","imageModelElement","getCaptionFromModelSelection","captionElement","matchImageCaptionViewElement","ToggleImageCaptionCommand","imageCaptionUtils","ancestorCaptionElement","focusCaptionOnShow","_hideImageCaption","_showImageCaption","imageCaptionEditing","selectedImage","savedCaption","_getSavedCaption","newCaptionElement","_saveCaption","ImageCaptionEditing","_savedCaptionsMap","_setupImageTypeCommandsIntegration","_registerCaptionReconversion","figcaptionElement","imageAlt","imageTypeInlineCommand","imageTypeBlockCommand","handleImageTypeChange","oldCaptionElement","savedOldElementCaption","jsonObject","ImageCaptionUI","modelCaptionElement","createImageTypeRegExp","regExpSafeNames","fetchLocalImage","imageSrc","fetch","resource","blob","mimeType","getImageMimeType","File","canvas","drawImage","toBlob","getBlobFromCanvas","convertLocalImageOnCanvas","ImageUploadUI","componentCreator","imageTypes","imageTypesRegExp","acceptedType","allowMultipleFiles","imagesToUpload","ImageUploadProgress","uploadStatusChange","viewFigure","_startAppearEffect","_showPlaceholder","_hidePlaceholder","progressBar","_createProgressBar","_showProgressBar","viewImg","_displayLocalImage","completeIcon","_showCompleteIcon","_removeUIElement","_hideProgressBar","_stopAppearEffect","_getUIElement","_createPlaceholder","imageFigure","uniqueProperty","UploadImageCommand","_uploadImage","ImageUploadEditing","_uploadImageElements","uploadImageCommand","fetchableImages","isLocalImage","fetchableImage","insertedImagesIds","isInsertedInGraveyard","getImagesFromChangeItem","_readAndUpload","_parseAndSetSrcsetAttributeOnImage","imageUploadElements","domFigure","originalDisplay","display","_ckHack","clean","srcsetAttribute","ImageStyleCommand","_defaultStyles","imageBlock","imageInline","isDefault","requestedStyle","shouldConvertImageType","DEFAULT_OPTIONS","inline","alignBlockLeft","alignBlockRight","DEFAULT_ICONS","full","center","inlineLeft","inlineRight","DEFAULT_DROPDOWN_DEFINITIONS","defaultItem","warnInvalidStyle","info","normalizeStyles","configuredStyles","arrangement","extendedStyle","extendStyle","normalizeDefinition","isBlockPluginLoaded","isInlinePluginLoaded","supportedElements","missingPlugins","isValidOption","getDefaultStylesConfiguration","getDefaultDropdownDefinitions","pluginCollection","getStyleDefinitionByName","ImageStyleEditing","utils","normalizedStyles","_setupPostFixer","modelToViewConverter","newStyle","oldStyle","viewToModelConverter","nonDefaultStyles","modelImageElement","viewToModelStyleAttribute","imageStyle","imageStyleDefinition","ImageStyleUI","localizedDefaultStylesTitles","definedStyles","translateStyles","styleConfig","definedDropdowns","dropdownConfig","_createDropdown","defaultButton","buttonViews","getUIComponentName","buttonName","splitButtonView","splitButtonViewArrow","getDropdownButtonTitle","areOn","buttonConfig","_executeCommand","dropdownTitle","buttonTitle","IndentEditing","IndentUI","localizedIndentIcon","indent","outdent","localizedOutdentIcon","_defineButton","AutomaticDecorators","_definitions","getDispatcher","getDispatcherForLinkedImage","reHasUnicode","rsAstral","rsCombo","rsFitz","rsNonAstral","rsRegional","rsSurrPair","reOptMod","rsOptVar","rsSeq","rsSymbol","reUnicode","strSymbols","chr","ATTRIBUTE_WHITESPACES","SAFE_URL","EMAIL_REG_EXP","PROTOCOL_REG_EXP","linkElement","ensureSafeUrl","urlString","normalizedUrl","isSafeUrl","isLinkableElement","addLinkProtocolIfApplicable","defaultProtocol","protocol","isProtocolNeeded","linkHasProtocol","openLink","LinkCommand","manualDecorators","automaticDecorators","restoreManualDecoratorStates","manualDecorator","_getDecoratorStateFromModel","manualDecoratorIds","truthyManualDecorators","falsyManualDecorators","linkText","extractTextFromSelection","linkRange","_updateLinkContent","allowedRanges","rangesToUpdate","_isRangeToUpdate","decoratorName","allowedRange","rangeItems","UnlinkCommand","rangesToUnlink","ManualDecorator","_createPattern","DECORATOR_AUTOMATIC","EXTERNAL_LINKS_REGEXP","addTargetToExternalLinks","linkDecorators","decorators","localizedDecoratorsLabels","decorator","getLocalizedDecorators","retArray","normalizeDecorators","_enableAutomaticDecorators","_enableManualDecorators","highlightedElements","_enableLinkOpen","_enableInsertContentSelectionAttributesFixer","_enableClickingAfterLink","_enableTypingOverLink","_handleDeleteContentAfterLink","_enableClipboardIntegration","automaticDecoratorDefinitions","rel","manualDecoratorDefinitions","decoratorDefinition","manualDecoratorValue","clickedElement","removeLinkAttributesFromSelection","getLinkAttributesAllowedOnText","clicked","deletedContent","nodeAtFirstPosition","nodeAtLastPosition","shouldCopyAttributes","shouldPreserveAttributes","hasBackspacePressed","newLink","linkAttributes","LinkFormView","urlInputView","_createUrlInput","_manualDecoratorSwitches","_createManualDecoratorSwitches","_createFormChildren","getDecoratorSwitchesState","accumulator","switchButton","switches","decoratorValue","additionalButtonsView","LinkActionsView","previewButtonView","_createPreviewButton","unlinkButtonView","editButtonView","VISUAL_SELECTION_MARKER_NAME","actionsView","formView","_createToolbarLinkButton","_enableBalloonActivators","_createViews","_createActionsView","_createFormView","_enableUserBalloonInteractions","unlinkCommand","_addFormView","_hideUI","parsedUrl","_closeFormView","_showUI","_getSelectedLinkElement","_areActionsVisible","_isUIVisible","_isUIInPanel","_addActionsView","_areActionsInPanel","_isFormInPanel","_removeFormView","_hideFakeVisualSelection","forceVisible","_showFakeVisualSelection","_startUpdatingUI","prevSelectedLink","prevSelectionParent","getSelectionParent","selectedLink","markerViewElements","targetLink","findLinkElementAncestor","startLink","endLink","URL_REG_EXP","AutoLink","_enableTypingHandling","_enableEnterHandling","_enableShiftEnterHandling","watcher","isSingleSpaceAtTheEnd","getUrlAtTextEnd","linkEnd","linkStart","_applyAutoLink","_checkAndApplyAutoLinkOnRange","shiftEnterCommand","fullUrl","isLinkAllowedOnRange","linkIsAlreadySet","_persistAutoLink","deletePlugin","listType","getListItemFillerOffset","createViewListItemElement","viewList","injectedItem","injectedList","refItem","sameIndent","smallerIndent","listIndent","prevItem","mappedViewAncestor","nestedList","prevView","breakPosition","mergeViewLists","nextViewList","lastSubChild","modelChild","firstList","secondList","itemIndent","listItem","walkerOptions","limitIndent","BULLETED_LIST_STYLE_TYPES","NUMBERED_LIST_STYLE_TYPES","hasOnlyLists","LIST_STYLE_TO_LIST_TYPE","LIST_STYLE_TO_TYPE_ATTRIBUTE","TYPE_ATTRIBUTE_TO_LIST_STYLE","LIST_STYLE_TYPES","listStyle","typeAttribute","IndentCommand","indentDirection","_indentBy","itemsToChange","lastItem","checkCanBecomeListItem","turnOff","currentIndent","newIndent","lowestIndent","_fixType","startingItem","ListUtils","getListTypeFromListStyleType","listStyleType","getSelectedListItems","listItems","getSiblingNodes","listName","modelViewMergeAfterChangeType","modelViewSplitOnInsert","removeStart","removeEnd","previousList","mergePos","modelViewMergeAfter","viewItemPrev","viewItemNext","viewModelConverter","listItemModel","convertedChild","findNextListItem","viewToModelListItemChildrenConverter","cleanList","isList","cleanListItem","foundList","modelToViewPosition","topmostViewList","modelIndentPasteFixer","indentChange","hoistNestedLists","nextIndent","modelRemoveStartPosition","viewRemoveStartPosition","viewRemovedItem","prevModelItem","prevIndent","prevViewList","itemToListHead","applied","_addListToFix","innerItem","listHead","_fixListIndents","_fixListTypes","maxIndent","fixBy","typesStack","getViewListItemLength","viewPos","modelLength","viewListPrev","modelViewChangeIndent","modelViewRemove","modelToViewUrlAttributeConverter","registry","figure","mediaContentElement","mediaViewElement","getMediaViewElement","createMediaFigureElement","getSelectedMediaModelWidget","insertMedia","mediaElement","MediaEmbedCommand","selectedMedia","isMediaSelected","isAllowedInParent","MediaRegistry","providers","extraProviders","removedProviders","removeProviders","providerDefinitions","provider","hasMedia","_getMedia","getViewElement","Media","previewRenderer","subPattern","_getUrlMatches","rawUrl","_getValidUrl","_locale","_match","_previewRenderer","renderForEditingView","renderMediaPreview","mediaHtml","_getPreviewHtml","_getPlaceholderHtml","outerHTML","MediaEmbedEditing","toMediaWidget","viewMedia","URL_REGEXP","AutoMediaEmbed","_timeoutId","_positionToInsert","leftLivePosition","rightLivePosition","_embedMediaBetweenPositions","leftPosition","rightPosition","mediaRegistry","urlRange","MediaFormView","validators","_validators","resetFormStatus","validator","_urlInputViewInfoDefault","inputField","_urlInputViewInfoTip","mediaURLInputValue","MediaEmbedUI","_setUpDropdown","getFormValidators","transformListItemLikeElementsIntoLists","itemLikeElements","itemLikeElementsMatcher","itemData","getListItemData","order","findAllItemLikeElements","currentList","currentIndentation","itemLikeElement","isDifferentList","previousItem","currentItem","isNewListNeeded","previousItemLikeElement","indentationDifference","listLikeItem","listStyleRegexp","listStyleTypeRegex","listStartIndexRegex","listStyleMatch","startIndex","listStyleTypeMatch","bulletedStyle","listMarkerElement","textNodeOrElement","findListMarkerNode","listMarker","findBulletedListStyle","listStartIndexMatch","mapListStyleDefinition","detectListStyle","lastListItem","lastListItemChild","insertNewEmptyList","differentIndentation","listElement","parentList","levelChange","findParentListAtLevel","bulletMatcher","removeBulletElement","transformElementIntoListItem","idMatch","orderMatch","indentMatch","replaceImagesSourceWithBase64","rtfData","upcastWriter","shapesIds","shapeElementsMatcher","prevSiblingName","findAllShapesIds","imageElementsMatcher","imgs","shapes","shape","removeAllImgElementsRepresentingShapes","shapeIds","containsMatchingImg","findSrc","insertMissingImgs","removeAllShapeElements","findAllImageElementsWithLocalSource","imageElements","imagesHexSources","newSrc","_convertHexToBase64","replaceImagesFileSourceWithInlineRepresentation","regexPictureHeader","regexPicture","extractImageDataFromRtf","hexString","msWordMatch1","msWordMatch2","MSWordNormalizer","isActive","htmlString","_parsedData","align","setTableAlignment","findSibling","isBlockViewElement","googleDocsMatch","GoogleDocsNormalizer","removeBoldWrapper","unwrapParagraphInListItem","elementsToReplace","nextSiblingIsBlock","transformBlockBrsToParagraphs","googleSheetsMatch","GoogleSheetsNormalizer","removeGoogleSheetsTag","removeXmlns","removeInvalidTableWidth","removeStyleBlock","normalizeSafariSpaceSpans","parseHtml","normalizedHtml","normalizeSpacing","bodyCloseTag","htmlCloseTag","bodyCloseIndex","htmlCloseIndex","cleanContentAfterBody","htmlDocument","htmlElement","innerTextLength","normalizeSpacerunSpans","bodyString","bodyView","documentToView","styleTags","getElementsByTagName","sheet","cssRules","extractStyles","updateNumericAttribute","createEmptyTableCell","tableCell","isHeadingColumnCell","tableUtils","table","headingColumns","column","getCellLocation","upcastTable","viewTable","rows","headingRows","headRows","bodyRows","firstTheadElement","tableChild","trs","tr","headingCols","scanRowForHeadingColumns","scanTable","row","ensureParagraphInTableCell","th","_table","_startRow","startRow","_endRow","endRow","_startColumn","startColumn","_endColumn","endColumn","_includeAllSlots","includeAllSlots","_skipRows","_row","_rowIndex","_column","_cellIndex","_spannedCells","_nextCellAtColumn","_isOverEndRow","_isOverEndColumn","_advanceToNextRow","outValue","spanData","_getSpanned","_shouldSkipSlot","_formatOutValue","cell","colspan","rowspan","_recordSpans","skipRow","anchorRow","anchorColumn","TableSlot","rowIsMarkedAsSkipped","rowIsBeforeStartRow","columnIsBeforeStartColumn","columnIsAfterEndColumn","rowMap","rowToUpdate","columnToUpdate","_markSpannedCell","tableWalker","cellAnchorRow","cellAnchorColumn","isAnchor","cellWidth","cellHeight","rowIndex","getPositionBefore","downcastTable","tableElement","figureElement","getRows","additionalSlots","asWidget","toTableWidget","downcastCell","tableRow","tableSlot","cellElementName","convertParagraphInTableCell","isSingleParagraphWithoutAttributes","InsertTableCommand","validParent","defaultRows","defaultColumns","createTable","InsertRowCommand","isAnyCellSelected","getSelectionAffectedTableCells","insertAbove","affectedTableCells","rowIndexes","getRowIndexes","insertRows","at","copyStructureFromAbove","InsertColumnCommand","columnIndexes","getColumnIndexes","insertColumns","columns","SplitCellCommand","selectedCells","splitCellHorizontally","splitCellVertically","cropTableToDimensions","sourceTable","cropDimensions","croppedTable","cropHeight","tableMap","sourceRow","sourceColumn","rowInCroppedTable","tableCellCopy","trimTableCellIfNeeded","addHeadingsToCroppedTable","getVerticallyOverlappingCells","overlapRow","cells","slotInfo","splitHorizontally","splitRow","newRowspan","newCellAttributes","newCellRowSpan","columnIndex","newCell","getHorizontallyOverlappingCells","overlapColumn","cellsToSplit","splitVertically","splitColumn","newColspan","newCellColSpan","cellRow","cellColumn","limitRow","limitColumn","removeEmptyColumns","getColumns","columnsMap","emptyColumns","cellsCount","emptyColumn","removeColumns","removeEmptyRows","emptyRows","tableRowCount","emptyRow","removeRows","removeEmptyRowsColumns","adjustLastRowIndex","dimensions","lastRowMap","firstColumn","lastColumn","lastRow","rowspanAdjustment","adjustLastColumnIndex","lastColumnMap","firstRow","colspanAdjustment","MergeCellCommand","isHorizontal","cellToMerge","_getMergeableCell","getTableCellsContainingSelection","isMergeNext","cellToExpand","cellToRemove","removedTableCellRow","mergeTableCells","spanAttribute","cellSpan","cellToMergeSpan","horizontalCell","hasHeadingColumns","cellOnLeft","cellOnRight","leftCellColumn","rightCellColumn","leftCellSpan","isCellOnLeftInHeadingColumn","isCellOnRightInHeadingColumn","getHorizontalCell","isMergeWithBodyCell","isMergeWithHeadCell","currentCellRowSpan","rowOfCellToMerge","currentCellData","mergeColumn","cellToMergeData","getVerticalCell","firstTableChild","RemoveRowCommand","firstCell","lastRowIndex","selectedRowIndexes","areAllRowsSelected","referenceCells","removedRowIndexes","columnIndexToFocus","rowsToRemove","cellToFocus","removedRowIndex","columnToFocus","getCellToFocus","RemoveColumnCommand","tableColumnCount","lastCell","returnValue","getBoundaryCells","removedColumnIndexes","columnsToRemove","SetHeaderRowCommand","isInTable","_isInHeading","headingRowsToSet","currentHeadingRows","overlappingCells","SetHeaderColumnCommand","headingColumnsToSet","createEmptyRows","rowsToInsert","isCopyStructure","copyStructureFrom","walkerEndRow","tableIterator","rowColSpansMap","lastCellRow","isReferenceRow","cellIndex","abs","columnsToInsert","tableColumns","createCells","rowCount","indexesObject","cellsToMove","cellsToTrim","lastRowOfCell","rowSpanToSet","getCellsToMoveAndTrimOnRemoveRow","targetRowIndex","tableRowMap","previousCell","cellToMove","moveCellsToRow","updateHeadingRows","headingsRemoved","adjustHeadingColumns","removedColumnIndex","numberOfCells","newCellsSpan","updatedSpan","breakSpanEvenly","newCellsAttributes","cellsToInsert","splitCellColumn","cellsToUpdate","splitCellRow","rowspanToSet","createTableWalker","getSelectedTableCells","sortRanges","cellWithSelection","tableCells","indexes","_getFirstLastIndexesObject","isSelectionRectangular","selectedTableCells","_areCellInTheSameTableSection","areaOfSelectedCells","areaOfValidSelection","rowsIndexes","getBiggestRectangleArea","compareRangeOrder","allIndexesSorted","indexA","_areIndexesInSameSection","headingSectionSize","tableCellToInsert","MergeCellsCommand","firstTableCell","mergeWidth","mergeHeight","maxWidthOffset","maxHeightOffset","getMaxOffset","firstCellRow","firstCellColumn","getMergeDimensions","cellBeingMerged","targetCell","currentMaxOffset","which","dimensionValue","SelectRowCommand","rangesToSelect","SelectColumnCommand","startLocation","endLocation","cellInfo","injectTableLayoutPostFixer","analyzedTables","isTableAttributeEntry","fixTableCellsRowspan","fixTableRowsSizes","tableLayoutPostFixer","maxRows","rowLimit","findCellsToTrim","childrenLengths","lengths","getChildrenLengths","rowsLengths","tableSize","maxColumns","injectTableCellParagraphPostFixer","fixTable","fixTableRow","fixTableCellContent","checkTableCellChange","tableCellContentsPostFixer","textNodes","shouldRefresh","_additionalSlots","figureChild","getViewTableFromFigure","modelTable","upcastCellSpan","isRowChange","expectedElementName","tableHeadingsRefreshHandler","cellsToCheck","paragraphsToRefresh","tableCellRefreshHandler","registerAdditionalSlot","slotHandler","InsertTableView","_createGridCollection","_highlightGridBoxes","gridItems","numberOfColumns","getNumberOfColumns","getGridItemFocuser","getIndexToFocus","focusedElementIndex","nextIndexToFocus","getRightElementIndex","elementIndex","getLeftElementIndex","boxView","_createGridButton","boxes","TableUI","isContentLtr","insertTableView","bindIsOn","_prepareDropdown","_prepareMergeSplitButtonDropdown","_fillDropdownWithListOptions","mergeCommandName","mergeCommand","addListOption","TableSelection","_handleDeleteContent","_handleInsertTextEvent","_defineSelectionConverter","_enablePluginDisabling","getSelectionAsFragment","adjustedLastRow","adjustedLastColumn","setCellSelection","anchorCell","cellsToSelect","_getCellsToSelect","getFocusCell","getAnchorCell","highlighted","previouslyHighlighted","clearHighlightedTableCells","lastViewCell","tableCellToSelect","rangeToSelect","selectionMap","flipVertically","flipHorizontally","TableClipboard","_onCopyCut","_onInsertContent","tableSelection","dataController","pastedTable","getTableIfOnlyTableInContent","pastedDimensions","selectedTable","shouldExpandSelection","expectedHeight","expectedWidth","tableWidth","tableHeight","expandTableSize","doVerticalSplit","doHorizontalSplit","splitCellsToRectangularSelection","prepareTableForPasting","selectionHeight","selectionWidth","_replaceSelectedCellsWithPasted","pastedWidth","pastedHeight","pastedTableLocationMap","createLocationMap","selectedTableMap","pastedRow","pastedColumn","pastedCell","cellToInsert","newTableCell","_replaceTableSlotCell","areHeadingRowsIntersectingSelection","areHeadingColumnsIntersectingSelection","newCells","contentRange","rangeBefore","rangeAfter","limitColumns","isAffectedBySelection","limitRows","endIndex","TableKeyboard","_onArrowKey","_handleTabOnSelectedTable","_handleTab","bubblingEventInfo","currentRowIndex","currentCellIndex","isFirstCellInRow","isLastCellInRow","isLastRow","nextRow","previousRow","_handleArrowKeys","focusCell","_navigateFromCellInDirection","_isSelectionAtCellEdge","currentCellInfo","cellToSelect","positionToSelect","TableMouse","_enableShiftClickSelection","_enableMouseDragSelection","blockSelectionChange","_getModelTableCellFromDomEvent","haveSameTableParent","beganCellSelection","newTargetCell","viewTargetElement","cellA","cellB","getSelectedTableWidget","isTableWidget","DEFAULT_BALLOON_POSITIONS","autoRefresh","DEFAULT_TOKEN_REFRESH_TIMEOUT_TIME","Token","tokenUrlOrRefreshToken","initValue","_validateTokenValue","defaultRefreshToken","xhrResponse","_registerRefreshTokenTimeout","refreshToken","_tokenRefreshTimeout","tokenValue","isString","isPlainString","isJWTFormat","tokenRefreshTimeoutTime","_getTokenRefreshTimeoutTime","exp","tokenExpireTime","BASE64_HEADER_REG_EXP","FileUploader","fileOrData","apiAddress","_isBase64","sliceSize","contentType","base64Data","byteArrays","byteNumbers","Blob","_base64ToBlob","_apiAddress","onProgress","onError","_prepareRequest","_attachXHRListeners","UploadGateway","CloudServicesCore","BalloonEditor","_addListAutoformats","_addBasicStylesAutoformats","_addHeadingAutoformats","_addBlockQuoteAutoformats","_addCodeBlockAutoformats","_addHorizontalLineAutoformats","boldCallback","italicCallback","codeCallback","strikethroughCallback","modelName","usePreviousLanguageChoice","_blockToolbarConfig","_hidePanel","_updateButton","_showPanel","_hideButton","modelTarget","_setupToolbarResize","_attachButtonToElement","wasVisible","_getSelectedEditableElement","_getToolbarMaxWidth","selectedModelRootName","contentStyles","editableRect","contentPaddingTop","paddingTop","contentLineHeight","lineHeight","fontSize","_tokens","registerTokenUrl","getTokenFor","widgetToolbarRepository","normalizers","_isTransformedWithPasteFromOffice","activeNormalizer","_setupImageUploadEditingIntegration","sourceAttributeNames","pictureViewElement","childSourceElement","imgViewElement","conversionResult","upcastPicture","imgElement","pictureElement","sourceAttributes","attributeElements","downcastSourcesAttribute","tableContentToolbarItems","tableToolbarItems","include","_enableTransformationWatchers","normalizedTransformations","extra","isNotRemoved","transformation","definedTransformations","transformationOrGroup","expandGroupsAndRemoveDuplicates","normalizeTransformations","normalizedTransformation","replaces","matchedRange","replacePosition","replaceRange","blockToolbar","contentToolbar"],"sourceRoot":""}