C;f`g)n)A{odb1V9T%s^HoBif)Ewk-A
zvV>OK`$vT;(kqFhKV%(0J!wwdKWw)=TCVt7d4F0aqF-+x_ggBm!>o9iifvtPd!ILY
zZd25O?%`(lyV2jZ@G?*P?3(VvrE_#tcJR^ef;u6)_i1~3Xm40bXsLO#-0x=3yBYT6
z&MwMsp~~BZzYp3jl)Zkdy*!{9yF9(Sk2#v!uCMZNKZaTA?Niff!0o5Q&~%OcJVRAj
zdEBYb&GqEc>vy^Cd!yhX)SNw&UV9(CQ#m#B+-^Q=J~H$rxKFvzO*y+bLxF#0S5)tH
z@7ToWTXGAk?9Rka_)4;-9`Au)w|v(17*00C2X(`?ts~bEL@_j5!sOePg|2`WO>fsJ
zbe4Pg$NG6wgU1Va28^Mj=Dm&qBZbCG!mSJS9S?0Xm!yF*-BqDStEXSDuheNZ?-MHP
zjdVvWstH3F3oH3UBq~Vd4`%KBo}b>ddNLB!52C~k_FT=2$6R6J+!%ix4U9YHMzG1U
zgJs0llSB|Rfs2?bi{YB3;Y%(zr9?%rL|eRF0pnM(w)~*bI}wDWuT64Rns|}9j62NQ
zlY)A2#%Z#vk0PydcrpzKdjmWSz2^1*obEyHnydS>Ah
zOKxAh40HbEL`f;xSS06^?uYVv&O7V61^3odVXH-#WwDro!Yr@d`bPizNYI+pn;5pe
zkH6%Ex?@t!vP`FElk}Pdu*4kFHGXXUd)*9x=$?b%Lb-0gyb~4r4xoh1X_JYr;NBOWAa>HpU}I1IM&RnkSbx521eG**3}T;qy21Y
zI{fB3pcgUd_{`LrO5*!_`0-hL;#_wI9_IuxxQFIS4T|53b!gF#+2$dGBb`Q;tfux-
zf6#TbQqE9(iB(}NUp?g*C0n2GN;>eq-eLB72Zbpsd2z@x?d`9^;TUovE`UE}jHC@VyBy9H*`*vUVk9P{UInKX``aA8>A?`9K(z`t
ziMyF95f2z>$FJ*-sc*Hu#MQiFPtNCspKBH0Fkafv(QL#tj)(fGANsjjj|2oiTXsEf
zU!E}?j(Vn!=7c_L_jjr@Jd;O0_04oMbky?r
zd^gG=m*Dfh`kNw4pZj%1xDUY>eBhIy;|wHk)|D>4
zLctP-zH+8u&YC_Z@vobt$k%7zm8@Sc7N5FY5lU`4eWa}+Dv%9zCH4D+6O%d
z_2j8d1oeAw*IAm&_})$##wlwBjRLB!SyB{#4ImYnDnPAkeNI$4iGLt#*wH4WVzo@k
z4e>i)nBM`8*LB%Ak?)!>Jb*A$=~iMy8_6HVnl85iLW9ogKJz}%ce$2lxa=p&-tQF?t#f~xl{ApJlUw(8mJQc9qpdtvU+E}
zO%fUpm{t#L7q7=}H=p<3XYj^M1`u0*d}BcIX4deX)T)o1nA|%5Tw5bywiv;j^47Wh
z+Z2$)!TrWTwpy427b@0`w?4s7h|33+R!uf;vg;U{KmuBl2dhCFGHox|J-_LG{m4Y0h
zeJlArVZbp}DLMB&x!*U;Vw^W0X%gS_vvO^lg>LZ6uJNwEiX%RROTzjbilRX3CWnYP
zAwnHco<=vy#~TF1Ek(7`(HCh)L;cEgq9$q#r){()GAbAa+k*UngWkn~K8vQ3wSIb%S;%&fgUQm11wZm7jwTv;J9TKJ0vWP
zDKx|<0pX0BJLZxm@=!UZV^Lylt}F2Ps)m)I5eB1Y?iA6C;b=^+?=|@clXdfsn<>KF
zhbVGC>h>Qg<1xwWl^;tysZyb(eH;D4zinA5obe{-(B+N~?$`USnN;RGKZXeU1=H}b
zngjK51g8tM{BpEnujAPXJ8>NDC9Y0c($Dz;;9BJqZ2*zm8Cu!EE*{mYk)#L$Y2ZF0
zn)F<7$5whr3601a7Nd$xkNN{5L4St(44$?5ZFIjAHvjlp(62PIUNy8(4FC|I{&2Wo
zuVB8LvWDv*a!9V!Vq%@8mT=&+gLxdlVDCQq_sjsWJ}{KaW)!
zv3K9%z@-69V9G9r35&B+#cDWD;=?p;>$!4C4_UC+(yIGB5gnGid+W}D^n`azDZHa$
zxWVlx>i!Hz59!T*h*+&-`!d`T#nfr@ZaAYFGiS7J1D4PDL1u+Y$D#FC*8rqvt!I#U
zW|Ah7_nIz$a*o#iL>Uqn#}^baZ0{8EkT*P>6xtQ~V-&7$wX#Pe3{M*)h^UjBr6Rgk
zs7H4*=N{6ucRH_a{fLW0-hvJc#K1uUS(MF>I>(*SDq?E`G+*6wca#Grp;$hlqWuvc
zYUDpB|H5-t&3Np)mp$9nJSO0P^U_Y}B=#u$+3?7hHGjA4`uSWUF7mTg0?Aow{y4{v
ztH;WtU4DJdCtmq_*v~xst#%BzWr?|o`pV~iEOdM8*Wm)nmfTD&Li%IF1gni^bvO5~
zf3Zj4b)Ts$>83|_3cyMZejkmEly-hs@+N5E*PtMSAN(-`w{X-DVx`&FD;1IZGR%~OvkcV`BeomK?9O_0r;n&oSOvI!*y
za{f;dms!X%X5oCuqb`aJv5vPDsy}#`DSgV=ko&Uvsz_@_$Y9RZ83TAelz2#98Sn*W%}phfhQ}WJAV@
zWO4CelL|7Q3lGe{u>_w>PHqZmbY3Vi<~P_+6aOU5@zb9QVYLINWYx7qRvv>S^%PLX
z=HyUHuvwy|OyU+@VNETSuJwL7vSdeStfYLkZ^<4bu-$MZ9kYron@d?uaify*BW`L|0z%#2{oG-O2Sym&Ke=Ha0`3se*C_t@k(dT9%9E`nvDz7*~ZS
zeM*G$Lr6Du<(oSyhIVV&eaQ*yA+r42u;olzG*XqHj5j84ro^lin^3Q1&WkHV&H83C
z*Yu^&e&RiiO8gl2TwA67sIBMNS+I35lXKlX(tF7S`JFzz5LS;=Vn>lJZNoIxTJhdN
z=sE^oHq@Jp96Pvau%)xZ)wYOku}Fj9Lrgl1`IDc4vieD+Mns}Pn_2VslXuIPqIM=6
z@KNNWQz{?E`q^I>(%UhG^+k3IFqB~6bpI^XmuwoQ
zL)9E;Lwe0{t8?;bKs7OKp{mGxbf?`FO09v;EiS}R8eYmOV+MwT?5O`<
zde~V`x86^^RNs;}0Gg_LH2O#h##5#5*~mv6%&dUVKX{e5bGt&fFVut?t67$ci0sZy
z8VC-gLWY#TNc<`*=mAEH`#F5ox6ht7mfbmzi|>J6uT~`H5mWvgZ>{k}`n9P~uEIyh
zXF{V(R#IJ}D^N0LYXYA_y+_qLRt9~Ij~)_1NU>S)0XB=gpC`n^zuLC=ZV&b+j!bP`
zPM_4?`9`w5GSyerY2OU{6UFIUTA>g1)9+xS$OU9{N-4gaawHg(eIG-$5KG(?(yygx
zg|K0r0K?qMW9ArD;Bi8*H?jld+>?Kkmk~Vc$I!D-LNH_wRg&gd!Y+)Tw%tMUe3xlL
z{xHp-W@>X!Fnx(o#|s_%Yj(lL_TP{*`fM79Z=TPlm2G6;IWN=dO&!nsflDq7HunpP
z{lgo$22a*a6w-*36~~tO}
zBBq{<<`Cv6^SmTa$n@O1*woK3!1@Aof}X(!9#5hEyZkXFPK
z+BC_YrpkW3!xV?x)29rgSpwFv%8K+m-L~^CkFshd*D3w5QZpsC^ILFm{m2NcR03kb
z8)Kx5Fs6ugT2OS|N`z2MI(W==Afg
zT67Eq*dlhF!s-cX1ah2477IEE_98xsKy2Ex0*nbhRGojlY_PyQ@AS_K48j8*JWe(St
z8sbST3bdJjzkAf@IK1`dOTZ?%g=RI^gHZ^syiMe
z$>&;*Dhw?KskFtYZSupQf-C`y0RWsGaCu(sJ62FgkAH5vg~V=-^qSi4G0Hm6Aq9t}
z=+TA!-N`OaTkA3hZW&TTb}2N3Ab#(#@nv>tE*Q_Oze)6YI^RveCc8*!^0b^VD3$}S%%bat?--oqZq^2WHBboeo44-+uHp*O&k_@
z-z;wNiaK@|=rv&ihUK^nWYl#aSQ?4kOYO>sRc(9{SJOb%AmaW`FFyXz%Jtc|eual=
zPsbb&&9oy-x_8Pg048hxjQ#_1+*GTsQBYrYp|XLmn~79!*VpHZcgT+)eqDU@)f#a~
zJ}><#G<{!LAZNFwoswUD8*-K#7Ph%
z+jq3oYUWTE4iw5~chB#B4>!$$I7WJ8K33&7O2c7?C3mIJ*S>Yy#3Rw1>O@j_g-+kN
zCmjvgcA`10iJybVyPfkS2qOPM8qQq|+vlV?m3;Lp=<)-YEM!@s
zteD=Fn{%MpycF)as7tWIcQMB2lq@3iz~o$&(*lR(ggnuNpay=b%zVrqTZf3wYqgD&
z<@EQqCfuZmLSbSAR~dN%u&+!jU(J1Bc9xf#9EpFwNxFv`?XYr+rYk@BR~CY|fTdp%
z%7s&4(5c3@(gHoR(H{B?S={a-l{2#d$xOGUeWNa8*_$rsB&zNK_yf6&4}A3+M4jn9
zftbgMPjd5@hJBk6cwy0$Uo40SvEmWEE(^7xiKRl8l%-qcnI-vYYB5X&p1fIkK|k8>
z-##T{F2+v{HZW^O2KfeEyn9t#|D1$xsl3TAS+Q7BaKSjNg*zmVLzS=JomV@I6`naq
zmG-`F4SSE3!{EdYMSVGkd36aYpV
zFcWBOphM~Co~X`(x{xgy6Nrg_#Kivj@tU`~XS)ouw=9x)#4r)2RaMJ<vfVM(zrB-V4n6Dgm_wDje^Xx!v##W`*rk!YMIu+BuVoRYIhFg
z?9K8U723JF;q0$q9O_2}pr=`=6f$ncHDiz-f}Q&KiUNdiv}&)zA_nzNhMe<@`R8E)
z^@u(X!TW%|R3j@GwE0LcdSt=E@KbF+I9jcFrF7&HI9>ltvW$AP)|Mg_3>>WvHAHqN
zG8AGUmk8!lJ)X1Ul}O0=PDuj|%er{SS5x(5)QD64R}#xBOW=nq+X$cVf<#SLJf#Oo
zY-&6uI3LxDlzUa=l~STE1RW+3D>bjAV?iS)t(7a)L*U|KyGlaBKw=2h&|wef*6A(F
z!w0fj0-GfKSLSnC=*~!9%
zH+sv%T(?p+n8$(Ch1i|9XK12@RZsJEdIB(D|K0KSJMypsXH8%qu5IlFryy=)S`n@k
zw%9iqHO*Lz@@CZg;(W+8+jDu6nd9pt{nJW})jcwLkrF21>;?hNgD)G&OYDq70|C_u
z=g;&P6%KPoq-oiSL2?sB-3o8O{%Dc-1Of98?5s)6H%?b(`A$a&n9)m?<<3|&h`OX&
zQ&BG9_}>T!FuFhLrDQ**l_19ZtBt?+TmjG66X&zp;shVpTP*&H;3NQsMGYPsMk#ml
zq|GRyc7S+NG^s!?aO+0E8LsS*CM;Ge;=$u`ON~1dYSMxk
zEz+69#)L)El>C(ScdV_u$Abll8<%q
zhMdLDKPa4o^w5R|
zC8-0Z0y5J$KsZ7Tx}n5v$;jk7LXJGI&rF|MSedP0#A@8f83>qLP*_mm%@33yHwfX9
zFz-*#*rzz=C*zA$`KaJy8p~KfIv1JzdHJ{x9NoIKz5Q50|HA@NT>Rc``?dIoS?a`y
z#iQRy?p8y@j3_?*!-YQ0#}m_N@4E}gMw5=PXsN$@al)aReVFS$HrQRoO8rv4f{I?^
zOEQj_!=czjQkk`m?)a(&-M+#=miCPIdz5ARsx^kfYW4&_&;j{X@#IehPjUA9+ncn}
z*-zRb3|mfN?&zep<{r4tooaY+<}1c
z6W#*z^#mu?hnpmW)*T~rB`mK#X|s_s;iZZZlgukVG+$W0Iwp?Hu1qiygV^V{4$?EV
z1#uYqV$Tj#hjS75lN5H5rwdecVYe|Q0y&z;^^%7x>6Pa3zhgP+?FThle<^*repC?E
z#E@k7*faTFJTC1^IyF^Drb3#D;6Jou|42{H)Z!HV;j4WJh3KXPAck|o0+7jR%@EAf
zaa$;J&T)SsboA(N*^!jH|FjT{slzVm)1lX5pkO^FVu6~VzSHbH$HSCJNA(^lBrUUi
z4n1`-^7C#4%UN>WIdAK`V>4e_EeR%NBWCl~RpK2j%?|l8(xNsz`Y~sd3)J?uHBaAq~lqMOHhI1OzElOvhDhYwCx
z2h2z<&mjN})ZTO4oalITo&S)OGe1(zRl#n-qs06weGC%mr%PVEg4UEP0-;J}TzCFA
z^;jlK4h!abK?e%0NgYDd;b3!sIF5d}OflC_^GdNE+ox@Vh0`cQHRC@n*iyuiKBF9=R{pAjEqLg?V_t3A!N_U}^(0$%qlr
zTis+#EML})DcDt|fa#lS$t4X_5FQg!5&!Gj{5pu1umUv#;!%XGQ1LX#C@
z_x9n(Puj5Vcp1+6&KYp1fbPwfGT_b}qNYN+w`r0$=$Qn{k#eVYE4WUs7U88BGAqi^kqWHjW%$m47)oC7eqWeX7E8;dh<-PjuXE1T@anccDf2=U
z+6NFLf%x5ek@kSj4zujzfqdWku=M?7=w1n>O(phP*{|(hhaY#t(y&20Em8WyB@*}>
zJPOe1kMim?@Y==SXn^?U9&7GJ@-lmRerC=MM^rT;t(;QSHVhmQE&Pa;A?mYhBPNOL4_kfp;p
zHX<=#-{8uBn*M>xrom4(=rT2Yrhz@W>t;KKd((Ke6@h`SJ_R-Kapsn#R8oP@t1aKP
z!>-L9aXR2su4)E`)Wx|g?LeHW&m~X
z^7B-JPU(9VmPZsN`83o}J;W>bXApk0f_KNA`%u&)IPK`Tgt7&AG6_U0ibXe6!&vDT
zrOtjNkA&atvi!nWJxGQ-l;3q;?x2b38?_AZ{^s=&H49Jt#^pwp*j!m$+l_8Wnd6O;
z*Y*;7=|W%;kVz_nLWHK7usm6_F>x-xmH1<^QzI`>Cu*V
zkv*rJoHPCMzLkf@lWAF^6w>L%l9uo+Sp1adwR9EP?p<X&d3nX3YG_>(ss&mo>vtZRdcmFlD?~O(<6}lg~(IldBhnRRhBI~Lj7{NrOLO!A~
zHf>QO!K#%ZuOoEC#&4dP4gJz=m87aB=sw?80ut(WI4=D?Gmf}$sX4R(T{s0{@~ebB
zda{E261Ib(H-p~{qBWjea?1P8pBgSn(#;Y&XxjuUi~24lpgp9ZhVx-wg|z7J2J3HY
zy-;@BpYIP1Y}1}cel9O_K4&I(E_3~LeixB<_4FQPwQV#$=?%CAiB`EhqGV98DC1={
zlvhSijipLzP&oB5OQ=7CY0>bB`xFKHxCVDE6#xJRTZoA%N{fm8fV^Z%#{|4Ut6PX!Y6sqECV<3?C-76
zBL$FC*xJf}Vpd*zP7{qFz>K~>Jz82nBBhv=k*x&}0OT5nRjHWGY8Z@GUJ2|-B4I~<
zA7V43rb8Y*b-^SzI5y({dB2eTRbyBS?IgU9eF$W;=CW2SH^0)RKw>7LMJ-4}#5N
z7o)48>$|S|v8s!`*9^=lN%9UK)uybDl_8Dj^sR&Hl=I+gHs;cdMjoqt6sx7rouHIo
z$;nOfPFTo{hS)gbv_*!n)W5^VOFkP>O58hOJ`4A<#C@QgK_KAWQVKz6UPXKkTwc_^
z4R_$CBR~WjkZcXk~0RGC()6mhrK-!r{V6M;8&si2%
z*RV;1j`i(aU9K;?3y3UUb`faG$%2jTY?vV?c1EVm?l$%>y9@vTej#^zh_RKaGlh|<
zxrMC&)p2Vl6@`U~0F?%(97xVy%=Dv$l&7Pqil@A)v8R3NbrJQwk1d4rUOOgu8_+88gSy
zH-8}hh9Pe1Wb9~R?`&abOYsL2Vr1vyEI>u|Qcv*@|7`5#-2XPUie^fhuE{Q
zGJ{xbY*_wX!^v5~^#$bb4*g#>oK#;nW3VWjI@!568kbA#&5Nkhi&fVDFexc5r}$5eKNOf-*x3K2^+NW4NIF}X{fn&s@a<2{U*Y__
zBQNUz#QhKHf5iTm@Qak399Z1W*yWFV(&7SCf9!)z?2Ii;z<*tu8nJRhcukC$c-Xig
zOdJp%2or=A0%0-L*-ifqO4`=R8DeW}`UmO-oY~?9hm)O?8^p@Z&GZ6g
z#>8O)0Wlfz@|rTSavDQ8dD+Y$+&n!02BF|+@sgDg>wov^50uFZ6ptAQ!p>^S%f!J6
zGG*dmGXgR3@Nk$iadGi-aC5V4kX{6H``R
zQwS&bf1*3tnK`>b98E>cUm|^p<|TptiiU#rFDmK&Q`+sL=^vgzY^+SItW0cds;ul_
zHcl`bFB6Ch3<6QH{5@cnKfU@N5%aVBe>ma)OW@yzffv2MmAwovFQXO9KZmQobM}YE
z|A()?x5fWM3op?BGxA^Y`#-w=N7sMFz<(wDzwG)SUH=sW|CR9nvg`jhx={YL;4!s*
znFYDMEN4c*>vAtkAq1ljlHz|YiU7A_t$r_8$o5iNP5=NZ?waR?%hMIWqeq8p`dDiZFK1Ce9!|c2|
zMciHMkH8L_BNv`*>6i#@z3YJ=)jR3odd6MK*<&X`2;{w=l@!`Gi=#=~9IiC*%p|)#
z6uLQC?G1fWwUU#e!@^EbR+RC_6N&4bAtl2K2n=)@#)_NPjp72mQm@GBwE|$7HG1w&
zR07-&?t1`RE_WMtPt)C`e6IYhO-)sK2_e%&V0!cN+{A#1IapB}Vc}N*ePgPcOMB%2
z=-nfy{o2Dm6_$g&JOwqyzLnMXXtt2whDwb0-IN;%bkuM4bf-UA{^9W~YtvYXgz^(r
zpAsDD?LJD-C-heUagM7IVY0w%XsMRIVUgCv&2u0!3Z
zT8p1gXCx?w3?U%)vP*UyRQ!RZ_MhMM!5(`ZxP=P*uD^2A7u48Sluu!N?&7{-GBYs=
zwAR(t;eEqit2<8V0B!-uS?T-^=ZQypCIdobJXRZROy=ysrXxdRc}
zP7lx*^x5m^4aHbo+`qJJjafsI;TH8k=hoKXID#$0CFKX?Dtz<~H1|KF2!rJfP(gVM
zqYOihrUZu#Ur1Vq^mZc5dP5|PQT?iN;c*RhboqXeVcGLwrTe<25x(nK=z$MW{bdvG
zM_-74opOF}sdb05WxiXX;<<=>1_qQq!vj!+0L-@ltihqshoQky`8z?CQK%+v$jh-M
zun*@O&&JmM?lBdB?xqZcm+CD150agQ6?!dKl^W{mOT_)GYqX;A?%S^|NO{^;v>P*X
z$nl1Tk$hljFHoaACbvY346bd1EXQ=V3hAe@>y&pe5qwnexR*}5zZ%>W(qucbv*u)*
zUX!&QtyJqzi{a2n#piAI>Hdrw@7th}{WNw4JRGVHzof>^IJ>HxS29vI`DmW_gvtJf
z#Sz4gq3?F``(vF43sfi@jPh`5P9|vI8^!JG^gQ^Kf7=W%t~4p5_RXnInIMgW+e~?M
zzTo?m+oZ1DgBUp4+b8Oy~Xw
zp*w%_=hWNg;G|63&xwAWGBKsTJT#g*twEK~^E25zE_*_LSGeTQdGph7`hJ_Tc%>H=
z<0qwttHQ*uB*ZTCh*7wi%|FoOm)m`dP6|A@1Il*H+=Tv^(YDfcKVQ=`AI*pWq}Tv8
zH8uN2GWiaN8y7NcZEe?XPS*)+qUpI1lv|PxU2=;24tQ8g$@TSZ&EMtI$-x-%tuhED
zni3whonx42k$bS#@GZ4%4a!$7kCH@eIFMj|q3y}q)FVLK@};nPT(u9oz4ah}r*+nC
zxKe>6Jh#saA%poX4VcjXv~S1raP^wxyVq4Lwg#8(C@mGyxl1j-e206@2rJ_}B*|0-
z5StgBI)e;-)3g0_#Cd^1-qj&t$O>BrlR%W62!rQRfh-k%+Gp@U>3bF^b3%g%!TuS
zi@9E+(e;owjj)5YIXEM{!Z0F>c~-y?On91(d&kGr0le48&nJ>5CE`-zjy#CUD5%K2
zT_U(3j;_+ut_38`G+6w5{hy|0bM9@^6bE2uxIzV?nEd?wVgQfN9fvC^80Mp;dfVVe
zX@{%aO5B+*4?7>@1)R95dQ`kZCNg*%Wt)Tp8=rjzCQsI_T^$v;LAksIsm~isvCj^>qyCK(w0%i3P)up~)=ggV5Sa
z23{=yRFW~KTi~$78n!kt@RN|50R=dXV%UL*rkqnbPjR5HCF(oR0=dRSFFXs3$Ea%a
zqhmzp?}a%G4VdXc?A7enEwDleYZ{*fAf2~3kwL>y!axc1oO8h3G*X}#DjtAYhvEa&
z7v|zs6$o+Ru0x>ScoHs1#6JK}4CdRrim#7~H^TRnSg!!hkiNl5C1`(z0y?zXb!wi7
zavY(jf%=93DKktM1vCO1&qomsWkgU$e_O)j&7C;ok1s7VtRi`i8{$k5OB^AaJ_jcY
zQ7=Tup*>SeK-;0lplX;FRDy+3gBAf20BGetCQuj@L;{4aC%-S2v_ktJKJoeFCiN#?
zt&<95yn(f}ykf-nw;w$L3Ch7?P{4&UJZrDzMb&;r%rb_rZ6z>G8S`z}L
zl|_IdU1cz*-GMQTy=-%zX2UDq&CHnwo1ape#8TV8QL!rc8GujmS$GGjQ*Y**Dr2!1
zHXyMeXJyzp{=^mrwTzYE;*luC6gB_y0umd>6--DWI{kst+H!S&DpeW^cU-K@O#^5x
z2>ivoIlpO)q?(98jF5tygq+mghfRcwQa_@GgQYKv;guV$r
z(vEb}GnmlMg8}Ha)rAvZv)qt$5ux3|vU38ordd*Ev9YP816X|RgnT!)+M8U-!hX_&
zCWP^BtMUWAv^i_1p*0xKclj*qzx5;?=Hspv;JZ%OznyN{@dsw#?+8E(CZ2xMD?aF|
z`QfNSS1*6DwE3rF8k_+L)@vBk1w~Zw+DlE3TcMO_MsUyi=q9~+u0(ZRJ@6?C18zKn}rZS
zDq*P8#cSxZv3`^l+)%p?;y}^1@|OVyZeFhx`#rIH@6SK`N@%PB9%gPEtInPb;EFDu
z^t#2O?11m}Q(BBoA|15%cddXw+bm~Muy>`b!e%cF3ZQYokqThzme1zvEd+5>Bgw;Q
z6}uT=*@8Wv>eHPnnr8FX;&)f_6k5%nXeL&mGb7x*BDfqZEAt}wkt~7MX1ilSAodrY
zklJ4Svk?ErY5xF!QlCg_)>T(>#Fa6BIP;0@0a{EF)7CL7l2lkvx~iJ=t_AEd8yke&
z6GmaB$Q)HnDwv<@FJCR(K2#h++=$rKaCr_Sfjo8{b#>D33~e^LfAb<<6v-KDaFS(m
z2l})Yt0^nc2e&$8wu-j?m~+eA9)fc6ChUm?btfI>2&HuJEsS@DO5;Q-mASGS7vn
zz>3@YH!{Pb3JB?P=oeaP6bw#OkYd@M^=xdIF|c%!0@MMlymixt5Y6hZ`qSwhylI!7
z^5R?o#+Urij9kvA+z2iVffolrk=Yg|`0|<&maBUh(h2ZSi7tt
zWx=Li9%cYpr7||J&hZH3+xN)Agh1$svQ6j{^VowX4BmSTm3JRhnbRdZ*7~?2C@4|O
z0Mnd3UgTY2^<6c=^=3b*Pi=q%QSVC`(Y8s}ep7NsW;8l@o}^V0@H}-Zj(%C
z=2qeJBDS|E-&lr&CrDQVFSvEd-_Fg2%Ued>4q|97o=AUSL6~SN)8=FR_ysDFaF-7<
zMi;(t?a7&EjPyr(BZnvZ13IS!-}52$R6@;*ap6!Kchb(sd0<8KLX2QCQ1`_SBw@FgQY!aFDwDUTu_n
zL({RIn-3U%*PwCdC(Ssn47G4a-Rh^NH{CV9nV^P^^bac=MHANFC>?`dq|Cp`@1T7vIJIoS$X
zuBzt8)U@*MfH0O-g9a*w(S$2!*n{c1taI^1TEFbB{%AUR1xO?hlL6q7;PKMlM8~DY
z3iC)(*M9e7Mp{EUK$;%cticWF5{XJ?KRKiKUPABEj_4s0S28g}ppg6Z1=MV8ru-?k
z2?oO3KXMz7fj|M<)#FcTNNYp@N?_12MUpi%0*=p~VATNO{9*XeL&(=S`Xn|h5(&P1
znGU7DLJqFSpJoXgAw#I2CYDg*u|mhjb}gkW^MEx8Rcr%FVW3bH2dOeC98;u_+zqrD
z&QKR_KXa&Q3p
z-Bz3KAW;kfISihaD`Lv+?Q@E~O03QfRBX6cBSoCf%D|~?rzE2FTn1!SSP&kqU{=19
z1vhrQ0%-jJFhu}Gx6R(6oc4@r^n8+G|8^vorcT~Z7QCpId5T+s!KcRQ>A^4GcYLj;
zn$1=mYB};+iNOagdLDio@!JSa(lr-XNdJJnvCY=wzE?m*|3wHj}6Blx7lKs5WpT6+6ypxFcd_CssQ8x1S2R=m{`Ms
n^ZsX4@&?8bxaVgAzXxusenRbOyd(5KKgCE($ctBq8V39yEL=TL
literal 0
HcmV?d00001
diff --git a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/Signals.py b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/Signals.py
index 2336c26..fa7034b 100644
--- a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/Signals.py
+++ b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/Signals.py
@@ -20,7 +20,7 @@ def threaded(fn):
return wrapper
-class Signals(PaneMixin, WindowMixin):
+class Signals(WidgetFileActionMixin, PaneMixin, WindowMixin):
def __init__(self, settings):
self.settings = settings
self.builder = self.settings.builder
@@ -87,6 +87,30 @@ class Signals(PaneMixin, WindowMixin):
self.load_store(view, store)
+ def do_action_from_menu_controls(self, imagemenuitem, eventbutton):
+ action = imagemenuitem.get_name()
+ self.ctrlDown = True
+ self.hide_context_menu()
+
+ if action == "create":
+ self.create_file()
+ if action == "cut":
+ self.to_copy_files.clear()
+ self.cut_files()
+ if action == "copy":
+ self.to_cut_files.clear()
+ self.copy_files()
+ if action == "paste":
+ self.paste_files()
+ if action == "delete":
+ # self.delete_files()
+ self.trash_files()
+ if action == "trash":
+ self.trash_files()
+
+ self.ctrlDown = False
+
+
def global_key_press_controller(self, eve, user_data):
keyname = Gdk.keyval_name(user_data.keyval).lower()
if "control" in keyname or "alt" in keyname or "shift" in keyname:
@@ -110,13 +134,13 @@ class Signals(PaneMixin, WindowMixin):
if "shift" in keyname:
self.shiftDown = False
if "alt" in keyname:
- self.altDown = False
+ self.altDown = False
if (self.ctrlDown and keyname == "slash") or keyname == "home":
self.builder.get_object("go_home").released()
if self.ctrlDown and keyname == "r":
self.builder.get_object("refresh_view").released()
- if (self.ctrlDown and keyname == "up") or (self.ctrlDown and keyname == "up"):
+ if (self.ctrlDown and keyname == "up") or (self.ctrlDown and keyname == "u"):
self.builder.get_object("go_up").released()
if self.ctrlDown and keyname == "l":
self.builder.get_object("path_entry").grab_focus()
@@ -124,7 +148,6 @@ class Signals(PaneMixin, WindowMixin):
self.builder.get_object("create_tab").released()
if self.ctrlDown and keyname == "w":
self.keyboard_close_tab()
-
if self.ctrlDown and keyname == "h":
self.show_hide_hidden_files()
if self.ctrlDown and keyname == "c":
@@ -136,15 +159,18 @@ class Signals(PaneMixin, WindowMixin):
if self.ctrlDown and keyname == "v":
self.paste_files()
+ if self.ctrlDown and keyname == "o":
+ pass
+
+ if keyname == "delete":
+ self.trash_files()
+
if keyname == "f4":
wid, tid = self.window_controller.get_active_data()
view = self.get_fm_window(wid).get_view_by_id(tid)
dir = view.get_current_directory()
self.execute("terminator", dir)
- if keyname == "delete":
- self.trash_files()
-
def execute(self, option, start_dir=os.getenv("HOME")):
DEVNULL = open(os.devnull, 'w')
@@ -152,6 +178,23 @@ class Signals(PaneMixin, WindowMixin):
subprocess.Popen(command, cwd=start_dir, start_new_session=True, stdout=DEVNULL, stderr=DEVNULL)
+ def show_about_page(self, widget, eve):
+ about_page = self.builder.get_object("about_page")
+ response = about_page.run()
+ if response == -4:
+ self.hide_about_page()
+
+ def hide_about_page(self, widget=None, eve=None):
+ about_page = self.builder.get_object("about_page").hide()
+
+
+ def show_context_menu(self, widget=None, eve=None):
+ self.builder.get_object("context_menu").run()
+
+ def hide_context_menu(self, widget=None, eve=None):
+ self.builder.get_object("context_menu").hide()
+
+
def generate_windows(self, data = None):
if data:
for j, value in enumerate(data):
diff --git a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/TabMixin.py b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/TabMixin.py
index 2b768fa..2eccfa2 100644
--- a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/TabMixin.py
+++ b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/TabMixin.py
@@ -1,8 +1,6 @@
# Python imports
# Lib imports
-from gi.repository import GObject, Gio
-
# Application imports
from . import WidgetMixin
@@ -35,39 +33,7 @@ class TabMixin(WidgetMixin):
self.set_window_title()
self.set_file_watcher(view)
- def set_file_watcher(self, view):
- if view.get_dir_watcher():
- watcher = view.get_dir_watcher()
- watcher.cancel()
- if debug:
- print(f"Watcher Is Cancelled: {watcher.is_cancelled()}")
- dir_watcher = Gio.File.new_for_path(view.get_current_directory()) \
- .monitor_directory(Gio.FileMonitorFlags.WATCH_MOVES,
- Gio.Cancellable()
- )
-
- wid = view.get_wid()
- tid = view.get_tab_id()
- dir_watcher.connect("changed", self.dir_watch_updates, (f"{wid}|{tid}",))
- view.set_dir_watcher(dir_watcher)
-
- def dir_watch_updates(self, file_monitor, file, other_file=None, eve_type=None, data=None):
- if eve_type == Gio.FileMonitorEvent.CREATED or \
- eve_type == Gio.FileMonitorEvent.DELETED or \
- eve_type == Gio.FileMonitorEvent.RENAMED or \
- eve_type == Gio.FileMonitorEvent.MOVED_IN or \
- eve_type == Gio.FileMonitorEvent.MOVED_OUT:
- wid, tid = data[0].split("|")
- notebook = self.builder.get_object(f"window_{wid}")
- view = self.get_fm_window(wid).get_view_by_id(tid)
- iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
- store = iconview.get_model()
- _store, tab_label = self.get_store_and_label_from_notebook(notebook, f"{wid}|{tid}")
-
- view.load_directory()
- self.load_store(view, store)
- tab_label.set_label(view.get_end_of_path())
def close_tab(self, button, eve=None):
@@ -154,88 +120,3 @@ class TabMixin(WidgetMixin):
view.hide_hidden = not view.hide_hidden
view.load_directory()
self.builder.get_object("refresh_view").released()
-
-
- def create_file(self):
- pass
-
- def update_file(self):
- nFile = widget.get_text().strip()
- if data and data.keyval == 65293: # Enter key event
- view.update_file(nFile)
- elif data == None: # Save button 'event'
- view.update_file(nFile)
-
- def menu_bar_copy(self, widget, eve):
- self.copy_file()
-
- def copy_files(self):
- wid, tid = self.window_controller.get_active_data()
- iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
- store = iconview.get_model()
- uris = self.format_to_uris(store, wid, tid, self.selected_files)
- self.to_copy_files = uris
-
- def cut_files(self):
- wid, tid = self.window_controller.get_active_data()
- iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
- store = iconview.get_model()
- uris = self.format_to_uris(store, wid, tid, self.selected_files)
- self.to_cut_files = uris
-
- def paste_files(self):
- wid, tid = self.window_controller.get_active_data()
- view = self.get_fm_window(wid).get_view_by_id(tid)
- to_path = f"{view.get_current_directory()}"
-
- if len(self.to_copy_files) > 0:
- self.handle_file(self.to_copy_files, "copy", to_path)
- else:
- self.handle_file(self.to_cut_files, "move", to_path)
-
-
- def move_file(self, view, fFile, tFile):
- self.handle_file([fFile], "move", tFile)
-
- def delete_files(self):
- pass
-
- def trash_files(self):
- wid, tid = self.window_controller.get_active_data()
- view = self.get_fm_window(wid).get_view_by_id(tid)
- iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
- store = iconview.get_model()
- uris = self.format_to_uris(store, wid, tid, self.selected_files)
- self.handle_file(uris, "trash")
-
-
- # NOTE: Gio moves files by generating the target file path with name in it
- # We can't just give a base target directory and run with it.
- # Also, the display name is UTF-8 safe and meant for displaying in GUIs
- def handle_file(self, paths, action, base_dir=None):
- paths = self.preprocess_paths(paths)
- target = None
-
- for path in paths:
- try:
- f = Gio.File.new_for_uri(path)
- if base_dir:
- info = f.query_info("standard::display-name", 0, cancellable=None)
- _target = f"file://{base_dir}/{info.get_display_name()}"
- target = Gio.File.new_for_uri(_target)
-
- if action == "trash":
- f.trash(cancellable=None)
- if action == "copy":
- f.copy(target, flags=Gio.FileCopyFlags.OVERWRITE, cancellable=None)
- if action == "move":
- f.move(target, flags=Gio.FileCopyFlags.OVERWRITE, cancellable=None)
- except GObject.GError as e:
- raise OSError(e.message)
-
- def preprocess_paths(self, paths):
- if not isinstance(paths, list):
- paths = [paths]
- # Convert items such as pathlib paths to strings
- paths = [path.__fspath__() if hasattr(path, "__fspath__") else path for path in paths]
- return paths
diff --git a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/WidgetFileActionMixin.py b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/WidgetFileActionMixin.py
new file mode 100644
index 0000000..79fb617
--- /dev/null
+++ b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/WidgetFileActionMixin.py
@@ -0,0 +1,156 @@
+# Python imports
+
+# Lib imports
+from gi.repository import GObject, Gio
+
+# Application imports
+
+
+
+class WidgetFileActionMixin:
+ def set_file_watcher(self, view):
+ if view.get_dir_watcher():
+ watcher = view.get_dir_watcher()
+ watcher.cancel()
+ if debug:
+ print(f"Watcher Is Cancelled: {watcher.is_cancelled()}")
+
+ dir_watcher = Gio.File.new_for_path(view.get_current_directory()) \
+ .monitor_directory(Gio.FileMonitorFlags.WATCH_MOVES,
+ Gio.Cancellable()
+ )
+
+ wid = view.get_wid()
+ tid = view.get_tab_id()
+ dir_watcher.connect("changed", self.dir_watch_updates, (f"{wid}|{tid}",))
+ view.set_dir_watcher(dir_watcher)
+
+ def dir_watch_updates(self, file_monitor, file, other_file=None, eve_type=None, data=None):
+ if eve_type == Gio.FileMonitorEvent.CREATED or \
+ eve_type == Gio.FileMonitorEvent.DELETED or \
+ eve_type == Gio.FileMonitorEvent.RENAMED or \
+ eve_type == Gio.FileMonitorEvent.MOVED_IN or \
+ eve_type == Gio.FileMonitorEvent.MOVED_OUT:
+ wid, tid = data[0].split("|")
+ notebook = self.builder.get_object(f"window_{wid}")
+ view = self.get_fm_window(wid).get_view_by_id(tid)
+ iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
+ store = iconview.get_model()
+ _store, tab_label = self.get_store_and_label_from_notebook(notebook, f"{wid}|{tid}")
+
+ view.load_directory()
+ self.load_store(view, store)
+ tab_label.set_label(view.get_end_of_path())
+
+ def create_file(self):
+ file_name = self.builder.get_object("context_menu_fname").get_text().strip()
+ type = self.builder.get_object("context_menu_type_toggle").get_state()
+
+ wid, tid = self.window_controller.get_active_data()
+ view = self.get_fm_window(wid).get_view_by_id(tid)
+ target = f"{view.get_current_directory()}"
+
+ if file_name != "":
+ file_name = "file://" + target + "/" + file_name
+ if type == True: # Create File
+ self.handle_file([file_name], "create_file", target)
+ else: # Create Folder
+ self.handle_file([file_name], "create_dir")
+
+ def update_file(self):
+ pass
+
+ def menu_bar_copy(self, widget, eve):
+ self.copy_file()
+
+ def copy_files(self):
+ wid, tid = self.window_controller.get_active_data()
+ iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
+ store = iconview.get_model()
+ uris = self.format_to_uris(store, wid, tid, self.selected_files)
+ self.to_copy_files = uris
+
+ def cut_files(self):
+ wid, tid = self.window_controller.get_active_data()
+ iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
+ store = iconview.get_model()
+ uris = self.format_to_uris(store, wid, tid, self.selected_files)
+ self.to_cut_files = uris
+
+ def paste_files(self):
+ wid, tid = self.window_controller.get_active_data()
+ view = self.get_fm_window(wid).get_view_by_id(tid)
+ target = f"{view.get_current_directory()}"
+
+ if len(self.to_copy_files) > 0:
+ self.handle_file(self.to_copy_files, "copy", target)
+ elif len(self.to_cut_files) > 0:
+ self.handle_file(self.to_cut_files, "move", target)
+
+
+ def move_file(self, view, files, target):
+ self.handle_file([files], "move", target)
+
+ def delete_files(self):
+ wid, tid = self.window_controller.get_active_data()
+ iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
+ view = self.get_fm_window(wid).get_view_by_id(tid)
+ store = iconview.get_model()
+ uris = self.format_to_uris(store, wid, tid, self.selected_files)
+ self.handle_file(uris, "delete")
+
+ def trash_files(self):
+ wid, tid = self.window_controller.get_active_data()
+ iconview = self.builder.get_object(f"{wid}|{tid}|iconview")
+ view = self.get_fm_window(wid).get_view_by_id(tid)
+ store = iconview.get_model()
+ uris = self.format_to_uris(store, wid, tid, self.selected_files)
+ self.handle_file(uris, "trash")
+
+
+ # NOTE: Gio moves files by generating the target file path with name in it
+ # We can't just give a base target directory and run with it.
+ # Also, the display name is UTF-8 safe and meant for displaying in GUIs
+ def handle_file(self, paths, action, base_dir=None):
+ paths = self.preprocess_paths(paths)
+ target = None
+
+ for path in paths:
+ try:
+ f = Gio.File.new_for_uri(path)
+
+ if action == "create_file":
+ f.create(Gio.FileCreateFlags.NONE, cancellable=None)
+ break
+ if action == "create_dir":
+ f.make_directory(cancellable=None)
+ break
+
+
+ if base_dir:
+ info = f.query_info("standard::display-name", 0, cancellable=None)
+ _target = f"file://{base_dir}/{info.get_display_name()}"
+ target = Gio.File.new_for_uri(_target)
+
+ # See if dragging to same directory then break
+ if action != "trash" and action != "delete" and \
+ (f.get_parent().get_path() == target.get_parent().get_path()):
+ break
+
+ if action == "delete":
+ f.delete(cancellable=None)
+ if action == "trash":
+ f.trash(cancellable=None)
+ if action == "copy":
+ f.copy(target, flags=Gio.FileCopyFlags.OVERWRITE, cancellable=None)
+ if action == "move":
+ f.move(target, flags=Gio.FileCopyFlags.OVERWRITE, cancellable=None)
+ except GObject.GError as e:
+ raise OSError(e.message)
+
+ def preprocess_paths(self, paths):
+ if not isinstance(paths, list):
+ paths = [paths]
+ # Convert items such as pathlib paths to strings
+ paths = [path.__fspath__() if hasattr(path, "__fspath__") else path for path in paths]
+ return paths
diff --git a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/WindowMixin.py b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/WindowMixin.py
index 49c9d97..caee4ef 100644
--- a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/WindowMixin.py
+++ b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/WindowMixin.py
@@ -51,40 +51,14 @@ class WindowMixin(TabMixin):
try:
wid, tid = iconview.get_name().split("|")
self.window_controller.set_active_data(wid, tid)
+ self.set_path_text(wid, tid)
+ self.set_window_title()
if eve.type == Gdk.EventType.BUTTON_RELEASE and eve.button == 1: # l-click
- self.set_path_text(wid, tid)
- self.set_window_title()
-
if self.single_click_open: # FIXME: need to find a way to pass the model index
self.grid_icon_double_left_click(iconview)
elif eve.type == Gdk.EventType.BUTTON_RELEASE and eve.button == 3: # r-click
- pass
- # input = self.builder.get_object("filenameInput")
- # controls = self.builder.get_object("iconControlsWindow")
- # iconsButtonBox = self.builder.get_object("iconsButtonBox")
- # menuButtonBox = self.builder.get_object("menuButtonBox")
- #
- #
- # if len(self.selectedFiles) == 1:
- # parts = self.selectedFiles[0].split("/")
- # input.set_text(parts[len(parts) - 1])
- # input.show()
- # iconsButtonBox.show()
- # menuButtonBox.hide()
- # controls.show()
- # elif len(self.selectedFiles) > 1:
- # input.set_text("")
- # input.hide()
- # menuButtonBox.hide()
- # iconsButtonBox.show()
- # controls.show()
- # else:
- # input.set_text("")
- # input.show()
- # menuButtonBox.show()
- # iconsButtonBox.hide()
- # controls.show()
+ self.show_context_menu()
except Exception as e:
print(repr(e))
diff --git a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/__init__.py b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/__init__.py
index 1f74fca..cd23f8d 100644
--- a/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/__init__.py
+++ b/src/versions/pyfm-0.0.1/PyFM/new/pyfm/signal_classes/mixins/__init__.py
@@ -2,3 +2,4 @@ from .PaneMixin import PaneMixin
from .WidgetMixin import WidgetMixin
from .TabMixin import TabMixin
from .WindowMixin import WindowMixin
+from .WidgetFileActionMixin import WidgetFileActionMixin