From e04494ecad843b64e81c6effc437ab8208f62330 Mon Sep 17 00:00:00 2001 From: Justin Lee <91563628+jjstnlee@users.noreply.github.com> Date: Tue, 23 Apr 2024 21:49:37 -0700 Subject: [PATCH] [saved] Implement the favorited stories button (#90) * Added favorite button and updated styling * fixing storyObjects bug * Add pubsub to favorites * Run prettier --------- Co-authored-by: Kyle Ramachandran <156966341+kylezryr@users.noreply.github.com> Co-authored-by: Kyle Ramachandran Co-authored-by: Aditya Pawar <34043950+adityapawar1@users.noreply.github.com> Co-authored-by: Aditya Pawar --- assets/icons.tsx | 1 - assets/save_story.png | Bin 1052 -> 0 bytes assets/saved_story.png | Bin 12911 -> 0 bytes package-lock.json | 22 +++++ package.json | 5 +- src/app/(tabs)/library/index.tsx | 21 ++++- src/app/(tabs)/story/index.tsx | 31 ++++--- src/app/(tabs)/story/styles.ts | 5 ++ .../FavoriteStoryButton.tsx | 76 ++++++++++++++++++ .../SaveStoryButton/SaveStoryButton.tsx | 36 ++++++--- src/queries/savedStories.tsx | 20 ++++- 11 files changed, 187 insertions(+), 30 deletions(-) delete mode 100644 assets/save_story.png delete mode 100644 assets/saved_story.png create mode 100644 src/components/FavoriteStoryButton/FavoriteStoryButton.tsx diff --git a/assets/icons.tsx b/assets/icons.tsx index 44728bb1..df93d8cc 100644 --- a/assets/icons.tsx +++ b/assets/icons.tsx @@ -110,7 +110,6 @@ const IconSvgs: Record = { `} /> ), - home_inactive: ( Px#L}ge>W=%~1DgXcg2mk?xX#fNO00031000^Q000001E2u_0{{R30RRC20H6W@ z1ONa40RR91JfH&r1ONa40RR91JOBUy0E^%0TmS$AhDk(0RA>e5n@@7uFciimr$e`7 zhCjm&P7v1VhMpkVWYbNP6Eryi^aM>$(8;zeN=^{ENH^V(6Qo@wfn;E^B&pr+b1h;E2p>P%^ zpbQq?-QDePZEbx)!+h{wlCB94d0Rp7Dd53i@ClWUJ+Q82S)K8Cd|s(k*x2~g@AvBv zl1CH3Rpd;qVzKz~->L8O;o+ePUuKrbW`fVi zqM-RzouJ&`-(P#ciMQ+o$~yrUE$teUld(|NMSdNuoR(_(yQfn;M+^dj^sVI7We(n$i%k7S&sd<5^duoYdS(=jKw9*6>g(U-yqNOC) z$t;{?>qU`$<1GA2r@nU*mXIMN=))V14f=4r*LT@(7=xLG;_HeuMsDMM!AIJIj+ z-`aK91W&7fuzLz$U`I&ND}KE^SwwMKU=uto54IuPy_rRDh0p@0&C}8Xm)bw*$iH1o zujaJQW@&S55sxI9h2znyIlX2UHpdq6NTS^USeB<1wpTUrpfqe!iaLA_ZvBkld~lhZ zpF%t`^;e5_9bT5F6}DG3@u235?{~1Uza|=*>cZb3SXT?YEKe(JugVe5m2@KfhX)Ho z@W>Wszd=xStsT;;;yUxe!J7-$vX1B$+n4|4^{FA9&C=_%S$Gv%UgtA13$IiA zt9W*1;ZQ8bmrI=+zT z9%pIy_a(shlL!-0X8zg|h$sEc;L`8BK;WetRm=0VB}580$ppr4fxLNn~&jVDeU zEM9H+KsFT*Zf$RGk1(8%JPkB`<2~{0FeYC?Y7t~69vrU$oKh3CHY3NAPD{&z$sM_P z-@%0@)71V7V0ptG@80a)V&HhGRB|UaHZVDHrOD-TdyttEaD%eK!k;1k#DG)4tbieX zh|K@q4~vtSMgTu5m&+Z$@H4>bCc$B4&fHZTbyaS1&QS@A_V9h}K0ZF4)oQg-P~N|w WSfpXwp?D_%00005M}G(m zG(mT@JH+>VIR4Znc`1SMqsgbeN)=4c|xLCJoX0j?MOq< z)JT2Gp1OqU;9TAF!w+H>lbWL_C-Lt2t~k};T91+Ud36exmo(-_)CIg|`VbnM@!|O0 zusHpt!R3;BSF#>c_o|ZfXAulKYqV{R{nx2dCxy$Luw}4*xB#UFF6s7V)K5cB=V2S^ zpK`@lj@2a1<2a8(WJ$wH@M@KSq&=|^-*!T0Q!YHwogSI@u3Y{;*X6*%2Ea<%r0A>H z6ADmSE9Sb~VT;J^skV9xFXkJz2P}X8Ha~ruj$jBQRDh-BH+?>PI<*dTNf+on~ zsf(TwO;3WV^YyApN*MJXnIvy|ZN+PSO@o((#eNWbOl#QuswP?-vfatOIgBGY;;ZIW z`_d!oXvyBG%fC{C;@8b~jqAgaqF7umUgjUW9uIN)v7MbiX~fWg(XU38A17Qm$6MA- z3_(Niq_vcjrJZm^R*oM|?C)1UXZUy65PWbY zjC1>F-6D;+8;4JYPj*8qfiq5ijFVmNI9Qjk z%PL?0WC@Er-)p7tW}89lSLaiglqXBL2h^t27!pIxIV&1TUAd&v-M2|Yx|wCmdQKWZ zJIJ+L9!*4SMldAG*>0VA`}W*+k@4`+5rIC5-pOZv(cWS7)9~piXQfc4X?r2&lHV8G zG`mXg(?3I@5!lAP>6Mj%i?i=zNSYaHgaWaE_JQ?V0MCLj8Ki8*x3&zC(}jCZwp=aE z3w`y*HZ#D9-6zhqA2)bsP@zG{z+{Z2GA5rSgL-9~88eq91szR*(hEyEW*|M*$FWE% zT->NQrPJl_6fDQ(3PZroMr@L5g)`BCJ}&m}}1S zIi}y|8SZijVh0>mBmTUn1x^}=5i+guIeuD+%ccqHp|Ba1jc1L%N*YN@@wC61oZ`o2 zLJ+2ilnsPtV8IN<3CC>S3`NvKlo^#6M{4QStC$^-u%{8apIKUPH+dySu3@v&DrXwk z7^4U>s}&o>V@fPiZgUwJl4xJ*P9PQU#fj0Ty&YO1#fLN7VbQ*5Eer$-b9*tKZia8m z8*HB6;?8&}p|kwz^YENVSW6!3@$_rEmn0?hY^NIkNH2sRhVJ2%BR(ht2FIULtG9BG z6o8x&GAlw7jEQnGJ*wPY@VU(U138Oec9!=ht|1Cg%Hdi@n^{0g0)^zGvnsDhJXmWM%99!6@@R;Ln z)_#@Tb4!lmJRQ6AKk~IP1*=@?)2!dm8!hP|5v!~~*SEL%@w7yXYks&?=97^aWjT6e z=69{=T|CRuMNWvAU0+YA=6tpq#_jUtX$?+5>#Wpgi_{xB;yNqmn@_LP_UYB1`P_Fb zq@<&A#WYRKggN`Hq;j-S`@IFdzj>cVReTp?-q%ektmJ&klQ+~~l3?s9)Njsn(@^H> z2#$Ox*1q(dt!#cYB4LE4wK3r%SI&a42e5I&jiEgko*)!U8{1-D@cab=zK2VOB{d!= zgbCgm>vnpKKxzTfItD zwh%gKsrt)G<9jKZbb+ z?x?)g{r4)~JVBT{(!VTA%MrKO@R9Q_C?jU4^9WzmdP4peL_=?oihq1=9e!2&BPB5q z$JHyq((RO%JOTsGknu_N2`q4y^Yp6!E%#BS_>fleH4Dxjr6Cbr9s+f8oPdI6Msnaj)lF-6uJDPBjcoZ@AI;msZuQ=LT_rC`#lf7ERs;k z7gvt?O-K2uxf+sj=S#@VJ1_9C1fX^QqdY`O|s7iSI=sDcRura8q8Dc>X*&nQwS+)O{ze71e$Ds39UnKc@c9z<8GHy#ka? zh&k8YV*#~qS~EA9efq5=Z_2WLOb9|z#Cb$MErcg3W+NS17r(uGqpoYfx2JSEv$sLBvx`EqIVGRVR}})G2b_nEvNS)#H<4)Olddys z3NRP69d8{~G3K2VE$MKv{ZmG1lfJ7>u=zXwen;WAdMx+&*1qYQDf(c%fiJ&D%*R{n zwX$8Se_2bdp(cz(qpS{G;4hxUTp6LuqX^!NlSvWAd`wJLVqJ^ zNXzkVMvggO(&Fu*dnUoPNhOwL_QdA#}EaqJ+$P?mZ1*RnZ z$vL%syao12iqbb*nfdTnw*#xIl^*B+Izd@tjaILGR|xdE6ujkpS_t1Mk@Ou9h|l-l z4E$ag6a*lt@(xe2;MRKsd!4G&b#}Rrv+}9ujm20D-{j*D;&Jb`#n&zm#rqsiD9j6`p2ch9E+<4LCgYYptro1RmEaDuyEu(B$BYrJa(Hp z7IQNdj{r|j)D^N2{4YS>7=LpVkY|Ly1RWi24DKCvYPkSm2! z@VF;ykfQrHh|&+=kUipv$Eg}nP-DUbiM4DoK8Ej%^?kOhxk>jh*VoM;7S`Jo36 zi0KZI9Ld8|6|UyP7x_?)0SEb@qRGFedD6pQ;@`IPa1#0yTAF`#@syix$U|3A0s!J& zs2D!<@&sg^#@hs`h{Z&0j3V%_6`uZb_%T}~Fn||yV?{m(Jo9Sdy{F(WfFZgOGJeMb zJQa=c;{`XMOdO*T$>YG9YeD}W^JeP5+BCQN*J!i0{vUlaCw+zeI@P{t` z3Xu|}d;{R{b3?7qx0emB-j=@xKmUvwix;3|!d$C=HfPRCY~_RHu(crT&FW46UJCzz zUkd*>_wB!UJp=#uS_*;jZePjI?aX1qkBFKispZW#W>Qt5e@z>OTR$#@SHvDhZUR!o+z zE{z*_M2e3}(;^wOifDS9j4qW=Yu7{znO%)Lj6;>S?&iH;*GrAcfPt-5)Pz1>-Uhdc zk^#0TzBo!{M_%4o%uO8pki_VLxohUHlQgILcIH7r3gsG-E&=9V)ea|VCb%NKpz|>= zB3`XGeZfwfyG7FM>)WNq9VvtyLYWHFx;RTcklZZsr;;TRjC03QZgM%Bk$fXZyEt&R z8AQjW`Qp6O4oSh!o3Lm5ko5a6PDV=75A z<>;3i&{33CuxJZUGL>CCoaMx2BRN`Btu zP+ph91nl<@JUnB8OZ+>qWv z(w^k}68(F-?-C-kw?8InzJ4%rw8;9#Yh8N@mdb7a+5k9%cGZd91pd3S zk-W{AnC`Z=g+|g89ZI%R#ME6_Y(8269p57BNCsK;`dC{wsip2c`yd>y*mWEQqq5QW zHB;oHGQWJt^~|Oc?RUOb08gW{v~3_VK60l?GXzb$zcbsS0!`Pt`J;RdjG9n~LkP0S zs3WvKhBZ2jaA^;kzK9CBq=@N&9|2SEej2bMe*u<$ zMdFcdhDyIR!Q%i14N1s&wX^ONoN=b!%iCF);FlE4nMIi{ugeZz%_7r|2)7lH&}XroN1^v8 z#u*5Ehhu=88m4u3!{)pVblM}ePGb5~lK;%(?4&t`DWhMD_H{jlmp>9~2CfPrO@H(% zP}$E%jNXhP7<(}(Cr(fjsqD>Y^*51>0j)S|&E(HE4N4e`fb*0^NjhwuiJBV1Lx`|s zlHjY^eP4=pr6_s&E}XH=N@M>Nl^DO5{K=XkO=ag=E6L9G{W2zo)i~B;F|B{SrH+fz zQG;qJ+i|m`;vK-{-#QI2DaP#Kz};1cak=Mm?|xLmoLp`1tw`CnhS37%2lPoB6N8F6 zT{mG`4F}eai}PNuq+C{xCj?8SH0vAQyR#J$3|))RdnG7>B1>;F9d=|bA?eEI0rGO& z23ng;-7JztF1W7X+e2lC-?7}C$ZVk!`@rh?z2?Sja`h?OWCvXCar#NAO_-ar_68Ot za~3%Ve3$E%F(!6Sy-$u4qYvvZQ`&Z5*1^AMGEj%MVAjb2bTaZ%%OJP7KK8KOfwA3X zF1!g(8iTnxTqfIUrR?^~1Y1%r=? ziN;<@$+p&fIFekijH&Yug5?Pah^a!o0B1Ve%){|y!2$A}??}@co%-4BRQ9p0K0R<` zBw93fF~(G7Y-FU>0$48scra2KE9<2tnnB*BT^Qm3G6;$QcSNo6;2uF|K=+6}$Snk} z0qjswNV78$8)!|*vNx5%1D|&%n8G6&H62lzo4cNjWm1W)pGtIKa1T+@*jrXam0Y)= z*G>^6w2!hLz&EC#;J~PxSjHQTw+_G!9iBkd5()(o$(Y(3l}XXQ)oh>yj93(c%6MjX zO#*f^luVWfvUz!JbqwuwNx{Dz%~PJWD0}sUsMP*-!EYL}~ITojOM#Y4wCv z&;gE+b^t1^jN>tBj>`>r(q1pr>PGwbbv%X5`j{g66 zpsx}>1g*Ks2cF%vACK$&yDl2y$%*6v@l=^M+>%1l%s%6!CI~AMyJHUYeyT*hM#JuR z;+5n_83iLdFFtkAjRlurY+exndn_H_0(Oe(*eu-vp8p-|Avsa-#eh~vYPRsd_S-s; z{t|zS0qzT~`(3mI=?9*mRJL2|0j8p~!>2t#R;6U$dg%UehGo5AgS1w6af#n@_wrZ} zXroxpJNCKd!9F}L_eA`yy+B1fU|U`gZ?7!MtD$EllBdQIjD`B%cS53f%JBJs&%}jS zU&}7Ca<-=52F6T!2P=tE%pkw3*hv7GK-V3G!h~A5hme^u@v*v%ts5&R_nhNeAZ0D_ zJ0alkT1gf5a%hlcLP6KlGYwoTBv7Ua@B;jcCKl| z+v~86M@bWa79f93YArB&2wNbgl9&#VK)WnyB`z$LHhq6kU8WomS}O;3IuKL+?i7Nd z`r3mChQ8T~vjI#^*B#1g1y@uL(32pn60vv%)WGMF$s{cB1?>K8=qcKb&{_wW>JGAe zIQu6WIpmn`3Fa#sd{s@7qUXxynL;83As-;VtK0wizZZ=V23-Zl?EWE3op@l3qW7Ai zK#Exd%LCFsk^LWqiaXDkD_k|D?;^Y65?S%;34p0B2uW}zRb52M5ny*V+qflr<|~+& zjz>g-0*noaqvA~pK>EN*;w?b(laMdU`z7j($I%Qo{jR(G&$`V2&WTT}s(yZ~*z>XA zlv)m74tQ)Pw`S+WnT_K-3LURe+pyHkYcBxil>{|N2@yJqC7>V@NX{5QcE)RlE?1&> zY&OW|9&u9V{{k65bclW;zhZPA)h$BkJGM0WH70t*ugx|Znh$}MW&e!K?WFvkP$Q`K z;KlCjO}zQxcuR`Hui;%Rm95Vp{DnZ|7yy~CXZ?ZuL6c8N28+_YFZpJD-xPOu_9lO$ z7sz4}m$Zumy)_>vcY!M5ow7d(Q|vb^w`HVv`Df*DmsPJuOLYCUZ7sy)5P1YMf!p_zAg!xjBWA=_j zuRpd^j6S_~uLPfXg_*(1@nS;*v-kr^7b%scY%tJ+)1P#(P^;o&HKtlzg2%Za41s`3 z%le?z=&D}ux3?dojRN-M_eb8~H)1>yny8L>Izvln*!50JTK%+FM2QjVAb=^?q|VO4$Z&ondZoeW$)sch=e@C7DBf2pP{{l0$(KP7w@wh8NXW(u-W*f4%0Z7;HAB9 zU082S+I7vcCI3tq%owk|5Q<>TDHTYsGuZrl{qx9)YR8tg2|4<)-S0rd0^bTVKpTWQ z=)ox2`u#w=N!+^yKK(Ib&d|Ryr(~MOo|~KV_AvQfCO;w)B+J_ipPmD`{O+x;Izj1I zleB$yOnT-W#CbIMj@&#=+hKj~x2xvz)M2s)^nXZa#lASlx6cC zD}Ho&$TY`pIldc`<#l%<*Qz#tmh8;@HpY~`LxGM$JHMe1M;=@?^6wtp_+?q6 zS6UBKg1ESOj<9QbGan*|r11~e>ljkKmimkRZbu`CPAm0l4QIO_N^!G3z#i6%L%m>? z;ZN2mmbQe{y^rSaF9DmZX!PcMGpoG#WDVnKSA|0L$BRpEmnx;_kMr$=X-xXQH#V== zJ5ALV7-HRpUX_4-v3$2!FW$b?_Ur9xMj$0s`w5%BY85QBFVW}=Y3Kr@%8513r81u* z=EBNn??n*zm=BP9Wje=r8rICB4~ z2bY&e*O+d75q1_hu~@ARX&0WA8rnd##d%06 z%Leppj)hV|g0LT=4pKD7s4*&dn#R?6-Q-zsG~Qe#R|``jyyv?Drx!w-=)1qW#ZZXD zCcsk2p`oFrag(`I(gzGqhi?1eRbr#gZ!6S8#5xD(K@G=4wd)d$Zb@{}MCqNcA{j%K z{dg`r);IAVty^|ffp$b$+mhGnyo8dH&{+`Ei_;?+_Y8)xyn#{7`x#_AaO&LSg-&p; zBkS0tTd5>EgG2w);zQ{FU5ReOi)xL5hny%vYWU?`%l3OrJ{cp~vAN)x2!Z`}`@S_;3+4Fq3{$@rwIi=%Q zORk-}BSzO?%za=|3dmZ*b!TT9=2FSu z98Cx^nceX6)tgyl0*n0EM(~U`=jbCBu)zl_QZrr&Q-{N--|D{1aa{Ub3}VjuDeYG@ zNO9?UlsPs1*p-Q_;Wu})?F`+omy;VX`SH0Dr>4#$?zOmz^LoJdQ7S29wrk6j>cLqV zwJ@6n=Q9s-CeJx*U+FsQ>@UadhungYzc5!j3_9%9y)wqD!&8KHsV~bMNt{<6J|!ha1xm9WwmiTc2OZisON_R<{#@Y%E^i#S zv++}x?Fztkp1<={h{i>M2ELF8fvtxY!3}De_w6%ekjxB@qv_ShjA(TU#l6* zHQt3P%LApH^PITq<)P)4%6_hwP%X83cjzwj=$s7+Od&C)B{uTUb zBbcDbouE%~w@UVkH3*!;QddK0cQhU#QMT0-0fj=Y>IAChI JufUi@{VzM@4g>%I diff --git a/package-lock.json b/package-lock.json index 4bd8250b..65702c95 100644 --- a/package-lock.json +++ b/package-lock.json @@ -73,6 +73,7 @@ }, "devDependencies": { "@babel/core": "^7.20.0", + "@iconify/react": "^4.1.1", "@types/react": "~18.2.14", "@types/react-native": "^0.72.3", "@types/react-native-htmlview": "^0.16.1", @@ -3773,6 +3774,27 @@ "integrity": "sha512-ZnQMnLV4e7hDlUvw8H+U8ASL02SS2Gn6+9Ac3wGGLIe7+je2AeAOxPY+izIPJDfFDb7eDjev0Us8MO1iFRN8hA==", "dev": true }, + "node_modules/@iconify/react": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/@iconify/react/-/react-4.1.1.tgz", + "integrity": "sha512-jed14EjvKjee8mc0eoscGxlg7mSQRkwQG3iX3cPBCO7UlOjz0DtlvTqxqEcHUJGh+z1VJ31Yhu5B9PxfO0zbdg==", + "dev": true, + "dependencies": { + "@iconify/types": "^2.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/cyberalien" + }, + "peerDependencies": { + "react": ">=16" + } + }, + "node_modules/@iconify/types": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@iconify/types/-/types-2.0.0.tgz", + "integrity": "sha512-+wluvCrRhXrhyOmRDJ3q8mux9JkKy5SJ/v8ol2tu4FVjyYvtEzkc/3pK15ET6RKg4b4w4BmTk1+gsCUhf21Ykg==", + "dev": true + }, "node_modules/@jest/create-cache-key-function": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/@jest/create-cache-key-function/-/create-cache-key-function-29.7.0.tgz", diff --git a/package.json b/package.json index d058a0c8..bea0a195 100644 --- a/package.json +++ b/package.json @@ -41,6 +41,7 @@ "expo": "~49.0.11", "expo-constants": "~14.4.2", "expo-font": "~11.4.0", + "expo-image": "~1.3.5", "expo-linking": "~5.0.2", "expo-router": "^2.0.0", "expo-status-bar": "~1.6.0", @@ -72,11 +73,11 @@ "react-native-vector-icons": "^10.0.2", "react-scroll-to-top": "^3.0.0", "use-debounce": "^10.0.0", - "validator": "^13.11.0", - "expo-image": "~1.3.5" + "validator": "^13.11.0" }, "devDependencies": { "@babel/core": "^7.20.0", + "@iconify/react": "^4.1.1", "@types/react": "~18.2.14", "@types/react-native": "^0.72.3", "@types/react-native-htmlview": "^0.16.1", diff --git a/src/app/(tabs)/library/index.tsx b/src/app/(tabs)/library/index.tsx index 2542da9e..3d50558a 100644 --- a/src/app/(tabs)/library/index.tsx +++ b/src/app/(tabs)/library/index.tsx @@ -13,7 +13,7 @@ import { fetchUserStoriesReadingList, } from '../../../queries/savedStories'; import { FlatList } from 'react-native-gesture-handler'; -import { usePubSub } from '../../../utils/PubSubContext'; +import { Channel, usePubSub } from '../../../utils/PubSubContext'; function LibraryScreen() { const { user } = useSession(); @@ -25,6 +25,7 @@ function LibraryScreen() { ); const { channels } = usePubSub(); let updateReadingListTimeout: NodeJS.Timeout | null = null; + let updateFavoritesListTimeout: NodeJS.Timeout | null = null; const favoritesPressed = () => { setFavoritesSelected(true); @@ -60,6 +61,20 @@ function LibraryScreen() { ); }; + useEffect(() => { + if (updateFavoritesListTimeout) { + clearTimeout(updateFavoritesListTimeout); + } + + updateFavoritesListTimeout = setTimeout( + () => + fetchUserStoriesFavorites(user?.id).then(favoriteStories => { + setFavoriteStories(favoriteStories); + }), + 4000, + ); + }, [channels[Channel.FAVORITES]]); + useEffect(() => { if (updateReadingListTimeout) { clearTimeout(updateReadingListTimeout); @@ -70,9 +85,9 @@ function LibraryScreen() { fetchUserStoriesReadingList(user?.id).then(readingList => { setReadingListStories(readingList); }), - 5000, + 4000, ); - }, [channels]); + }, [channels[Channel.SAVED_STORIES]]); useEffect(() => { (async () => { diff --git a/src/app/(tabs)/story/index.tsx b/src/app/(tabs)/story/index.tsx index 33ada971..281359d5 100644 --- a/src/app/(tabs)/story/index.tsx +++ b/src/app/(tabs)/story/index.tsx @@ -16,6 +16,8 @@ import { RenderHTML } from 'react-native-render-html'; import { SafeAreaView } from 'react-native-safe-area-context'; import styles from './styles'; +import FavoriteStoryButton from '../../../components/FavoriteStoryButton/FavoriteStoryButton'; +import SaveStoryButton from '../../../components/SaveStoryButton/SaveStoryButton'; import ReactionPicker from '../../../components/ReactionPicker/ReactionPicker'; import BackButton from '../../../components/BackButton/BackButton'; import { fetchStory } from '../../../queries/stories'; @@ -117,20 +119,23 @@ function StoryScreen() { )} /> - - + + Share Story + + + { + isStoryInFavorites(storyId, user?.id).then(storyInReadingList => { + setStoryIsFavorited(storyInReadingList); + }); + }, [storyId]); + + useEffect(() => { + isStoryInFavorites(storyId, user?.id).then(storyInFavorites => { + setStoryIsFavorited(storyInFavorites); + publish(Channel.FAVORITES, storyId, storyInFavorites); + }); + }, [storyId]); + + const favoriteStory = async (favorited: boolean) => { + setStoryIsFavorited(favorited); + + if (favorited) { + publish(Channel.FAVORITES, storyId, true); + await addUserStoryToFavorites(user?.id, storyId); + } else { + publish(Channel.FAVORITES, storyId, false); + await deleteUserStoryToFavorites(user?.id, storyId); + } + }; + + const renderFavoritedIcon = () => { + return ( + + + + ); + }; + + const renderNotFavoritedIcon = () => { + return ( + + + + ); + }; + + return ( + favoriteStory(!storyIsFavorited)}> + {storyIsFavorited ? renderFavoritedIcon() : renderNotFavoritedIcon()} + + ); +} diff --git a/src/components/SaveStoryButton/SaveStoryButton.tsx b/src/components/SaveStoryButton/SaveStoryButton.tsx index 307d1d7e..dbf88956 100644 --- a/src/components/SaveStoryButton/SaveStoryButton.tsx +++ b/src/components/SaveStoryButton/SaveStoryButton.tsx @@ -1,4 +1,7 @@ import { useEffect, useState } from 'react'; +import { TouchableOpacity } from 'react-native-gesture-handler'; +import Svg, { Path } from 'react-native-svg'; + import { addUserStoryToReadingList, deleteUserStoryToReadingList, @@ -6,17 +9,12 @@ import { } from '../../queries/savedStories'; import { Channel, usePubSub } from '../../utils/PubSubContext'; import { useSession } from '../../utils/AuthContext'; -import { Image } from 'expo-image'; -import { TouchableOpacity } from 'react-native-gesture-handler'; type SaveStoryButtonProps = { storyId: number; defaultState?: boolean | null; }; -const saveStoryImage = require('../../../assets/save_story.png'); -const savedStoryImage = require('../../../assets/saved_story.png'); - export default function SaveStoryButton({ storyId, defaultState = null, @@ -54,13 +52,31 @@ export default function SaveStoryButton({ } }; + const renderSavedStoryImage = () => { + return ( + + + + ); + }; + + const renderSaveStoryImage = () => { + return ( + + + + ); + }; + return ( saveStory(!storyIsSaved)}> - {storyIsSaved ? ( - - ) : ( - - )} + {storyIsSaved ? renderSavedStoryImage() : renderSaveStoryImage()} ); } diff --git a/src/queries/savedStories.tsx b/src/queries/savedStories.tsx index 5ab847d7..13aeae64 100644 --- a/src/queries/savedStories.tsx +++ b/src/queries/savedStories.tsx @@ -48,7 +48,7 @@ async function addUserStory( ) { const { error } = await supabase .from('saved_stories') - .upsert([{ user_id: user_id, story_id: story_id, name: name }]) + .upsert([{ user_id, story_id, name }]) .select(); if (error) { @@ -128,3 +128,21 @@ export async function isStoryInReadingList( return data; } + +export async function isStoryInFavorites( + storyId: number, + userId: string | undefined, +): Promise { + const { data, error } = await supabase.rpc('is_story_saved_for_user', { + list_name: 'favorites', + story_db_id: storyId, + user_uuid: userId, + }); + + if (error) { + console.error(error); + return false; + } + + return data; +}