@zezosoft/zezo-ott-react-native-ui-kit 1.1.1 → 1.1.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/module/Styles/globalStyles.js +0 -5
- package/lib/module/Styles/globalStyles.js.map +1 -1
- package/lib/module/components/Auth/AuthProvider/AuthProvider.js +64 -40
- package/lib/module/components/Auth/AuthProvider/AuthProvider.js.map +1 -1
- package/lib/module/components/Auth/ForgotPassword/ForgotPassword.js +5 -2
- package/lib/module/components/Auth/ForgotPassword/ForgotPassword.js.map +1 -1
- package/lib/module/components/Auth/Login/LoginWithEmail.js +17 -11
- package/lib/module/components/Auth/Login/LoginWithEmail.js.map +1 -1
- package/lib/module/components/Auth/Login/LoginWithPhone.js +12 -6
- package/lib/module/components/Auth/Login/LoginWithPhone.js.map +1 -1
- package/lib/module/components/Auth/OTP/OTP.js +7 -4
- package/lib/module/components/Auth/OTP/OTP.js.map +1 -1
- package/lib/module/components/Auth/QrLogin/QrLogin.js +357 -144
- package/lib/module/components/Auth/QrLogin/QrLogin.js.map +1 -1
- package/lib/module/components/Auth/QrLogin/components/QrViewArea.js +227 -110
- package/lib/module/components/Auth/QrLogin/components/QrViewArea.js.map +1 -1
- package/lib/module/components/Auth/SignUp/SignUp.js +19 -13
- package/lib/module/components/Auth/SignUp/SignUp.js.map +1 -1
- package/lib/module/components/Auth/SplashScreen/SplashScreen.js +54 -36
- package/lib/module/components/Auth/SplashScreen/SplashScreen.js.map +1 -1
- package/lib/module/components/Auth/SplashScreen/components/SplashImage/SplashImage.js +29 -11
- package/lib/module/components/Auth/SplashScreen/components/SplashImage/SplashImage.js.map +1 -1
- package/lib/module/components/Auth/SplashScreen/components/SplashLottie/SplashLottie.js +13 -7
- package/lib/module/components/Auth/SplashScreen/components/SplashLottie/SplashLottie.js.map +1 -1
- package/lib/module/components/Auth/SplashScreen/components/SplashVideo/SplashVideo.js +33 -21
- package/lib/module/components/Auth/SplashScreen/components/SplashVideo/SplashVideo.js.map +1 -1
- package/lib/module/components/BackgroundLayout/BackgroundLayout.js +26 -20
- package/lib/module/components/BackgroundLayout/BackgroundLayout.js.map +1 -1
- package/lib/module/components/BlogView/BlogView.js +36 -20
- package/lib/module/components/BlogView/BlogView.js.map +1 -1
- package/lib/module/components/Button/BackBtn.js +24 -20
- package/lib/module/components/Button/BackBtn.js.map +1 -1
- package/lib/module/components/Button/PrimaryBtn.js +19 -13
- package/lib/module/components/Button/PrimaryBtn.js.map +1 -1
- package/lib/module/components/Button/SecondaryBtn.js +19 -13
- package/lib/module/components/Button/SecondaryBtn.js.map +1 -1
- package/lib/module/components/Button/TextButton.js +19 -13
- package/lib/module/components/Button/TextButton.js.map +1 -1
- package/lib/module/components/Content/Card/Category/Category.js +151 -46
- package/lib/module/components/Content/Card/Category/Category.js.map +1 -1
- package/lib/module/components/Content/Card/NowWatching/NowWatching.js +237 -108
- package/lib/module/components/Content/Card/NowWatching/NowWatching.js.map +1 -1
- package/lib/module/components/Content/Card/Sliders/Styles/One.js +200 -133
- package/lib/module/components/Content/Card/Sliders/Styles/One.js.map +1 -1
- package/lib/module/components/Content/Card/Sliders/Styles/Two.js +192 -84
- package/lib/module/components/Content/Card/Sliders/Styles/Two.js.map +1 -1
- package/lib/module/components/Content/Card/Styles/Five.js +131 -48
- package/lib/module/components/Content/Card/Styles/Five.js.map +1 -1
- package/lib/module/components/Content/Card/Styles/Four.js +126 -59
- package/lib/module/components/Content/Card/Styles/Four.js.map +1 -1
- package/lib/module/components/Content/Card/Styles/One.js +125 -50
- package/lib/module/components/Content/Card/Styles/One.js.map +1 -1
- package/lib/module/components/Content/Card/Styles/RotateInOut.js +143 -65
- package/lib/module/components/Content/Card/Styles/RotateInOut.js.map +1 -1
- package/lib/module/components/Content/Card/Styles/Six.js +207 -115
- package/lib/module/components/Content/Card/Styles/Six.js.map +1 -1
- package/lib/module/components/Content/Card/Styles/Three.js +134 -79
- package/lib/module/components/Content/Card/Styles/Three.js.map +1 -1
- package/lib/module/components/Content/Card/Styles/TopTen.js +186 -171
- package/lib/module/components/Content/Card/Styles/TopTen.js.map +1 -1
- package/lib/module/components/Content/Card/Styles/Two.js +144 -64
- package/lib/module/components/Content/Card/Styles/Two.js.map +1 -1
- package/lib/module/components/Content/Card/components/AdsPoster.js +162 -0
- package/lib/module/components/Content/Card/components/AdsPoster.js.map +1 -0
- package/lib/module/components/Content/Card/components/CardPoster.js +139 -93
- package/lib/module/components/Content/Card/components/CardPoster.js.map +1 -1
- package/lib/module/components/Content/Card/components/RentOrBuyIcon.js +18 -17
- package/lib/module/components/Content/Card/components/RentOrBuyIcon.js.map +1 -1
- package/lib/module/components/Content/Card/components/ThumbnailCard.js +78 -28
- package/lib/module/components/Content/Card/components/ThumbnailCard.js.map +1 -1
- package/lib/module/components/Content/Card/components/index.js +4 -0
- package/lib/module/components/Content/Card/components/index.js.map +1 -0
- package/lib/module/components/Content/Content.js +99 -40
- package/lib/module/components/Content/Content.js.map +1 -1
- package/lib/module/components/Content/Sections.js +63 -34
- package/lib/module/components/Content/Sections.js.map +1 -1
- package/lib/module/components/ContentView/ContentView.js +70 -41
- package/lib/module/components/ContentView/ContentView.js.map +1 -1
- package/lib/module/components/ContentView/MoreContentList.js +74 -40
- package/lib/module/components/ContentView/MoreContentList.js.map +1 -1
- package/lib/module/components/ContentView/components/AboutSection.js +40 -19
- package/lib/module/components/ContentView/components/AboutSection.js.map +1 -1
- package/lib/module/components/ContentView/components/CastCard.js +6 -7
- package/lib/module/components/ContentView/components/CastCard.js.map +1 -1
- package/lib/module/components/ContentView/components/EpisodeCard.js +2 -2
- package/lib/module/components/ContentView/components/EpisodeCard.js.map +1 -1
- package/lib/module/components/ContentView/components/GenreTags.js +25 -13
- package/lib/module/components/ContentView/components/GenreTags.js.map +1 -1
- package/lib/module/components/ContentView/components/HeroBanner.js +39 -11
- package/lib/module/components/ContentView/components/HeroBanner.js.map +1 -1
- package/lib/module/components/ContentView/components/MiniInfo.js +84 -77
- package/lib/module/components/ContentView/components/MiniInfo.js.map +1 -1
- package/lib/module/components/ContentView/components/PlayButton.js +11 -7
- package/lib/module/components/ContentView/components/PlayButton.js.map +1 -1
- package/lib/module/components/ContentView/components/Title.js +12 -12
- package/lib/module/components/ContentView/components/Title.js.map +1 -1
- package/lib/module/components/ContentView/components/TrailerButton.js +12 -7
- package/lib/module/components/ContentView/components/TrailerButton.js.map +1 -1
- package/lib/module/components/Fallbacks/NoContentFallback.js +27 -22
- package/lib/module/components/Fallbacks/NoContentFallback.js.map +1 -1
- package/lib/module/components/Fallbacks/NotFoundFallback.js +6 -4
- package/lib/module/components/Fallbacks/NotFoundFallback.js.map +1 -1
- package/lib/module/components/Headers/AppHeader.js +22 -13
- package/lib/module/components/Headers/AppHeader.js.map +1 -1
- package/lib/module/components/Headers/Three.js +6 -5
- package/lib/module/components/Headers/Three.js.map +1 -1
- package/lib/module/components/Headers/Two.js +19 -14
- package/lib/module/components/Headers/Two.js.map +1 -1
- package/lib/module/components/Input/InputOne.js +46 -31
- package/lib/module/components/Input/InputOne.js.map +1 -1
- package/lib/module/components/Loader/Loader.js +2 -2
- package/lib/module/components/Loader/Loader.js.map +1 -1
- package/lib/module/components/Logo/Logo.js +17 -14
- package/lib/module/components/Logo/Logo.js.map +1 -1
- package/lib/module/components/Search/One.js +2 -1
- package/lib/module/components/Search/One.js.map +1 -1
- package/lib/module/components/Search/components/SearchCard.js +15 -19
- package/lib/module/components/Search/components/SearchCard.js.map +1 -1
- package/lib/module/components/Settings/AppSettings.js +97 -62
- package/lib/module/components/Settings/AppSettings.js.map +1 -1
- package/lib/module/components/Subscription/SubOne.js +313 -254
- package/lib/module/components/Subscription/SubOne.js.map +1 -1
- package/lib/module/components/Text/Text.js +15 -12
- package/lib/module/components/Text/Text.js.map +1 -1
- package/lib/module/components/User/DeviceSessions/DeviceSessions.js +25 -23
- package/lib/module/components/User/DeviceSessions/DeviceSessions.js.map +1 -1
- package/lib/module/components/User/ProfileUpdate/ProfileUpdate.js +23 -18
- package/lib/module/components/User/ProfileUpdate/ProfileUpdate.js.map +1 -1
- package/lib/module/components/User/PurchaseHistory/PurchaseHistory.js +38 -24
- package/lib/module/components/User/PurchaseHistory/PurchaseHistory.js.map +1 -1
- package/lib/module/components/User/WatchHistory/WatchHistory.js +21 -14
- package/lib/module/components/User/WatchHistory/WatchHistory.js.map +1 -1
- package/lib/module/components/User/WatchLater/WatchLater.js +30 -21
- package/lib/module/components/User/WatchLater/WatchLater.js.map +1 -1
- package/lib/module/components/User/components/UserAvatar.js +38 -19
- package/lib/module/components/User/components/UserAvatar.js.map +1 -1
- package/lib/module/components/User/components/UserSection.js +37 -17
- package/lib/module/components/User/components/UserSection.js.map +1 -1
- package/lib/module/components/View/View.js +7 -4
- package/lib/module/components/View/View.js.map +1 -1
- package/lib/module/components/index.js +0 -1
- package/lib/module/components/index.js.map +1 -1
- package/lib/module/constants/dummySections.js +44 -4
- package/lib/module/constants/dummySections.js.map +1 -1
- package/lib/module/hooks/Images/index.js +5 -0
- package/lib/module/hooks/Images/index.js.map +1 -0
- package/lib/module/hooks/Images/useImageLoader.js +168 -0
- package/lib/module/hooks/Images/useImageLoader.js.map +1 -0
- package/lib/module/hooks/Images/useImageValidation.js +36 -0
- package/lib/module/hooks/Images/useImageValidation.js.map +1 -0
- package/lib/module/hooks/index.js +20 -0
- package/lib/module/hooks/index.js.map +1 -0
- package/lib/module/hooks/useAdTracking.js +270 -0
- package/lib/module/hooks/useAdTracking.js.map +1 -0
- package/lib/module/hooks/useCards.js +164 -0
- package/lib/module/hooks/useCards.js.map +1 -0
- package/lib/module/hooks/useDebounce.js +25 -6
- package/lib/module/hooks/useDebounce.js.map +1 -1
- package/lib/module/hooks/useKeyboard.js +7 -4
- package/lib/module/hooks/useKeyboard.js.map +1 -1
- package/lib/module/hooks/useNavigationMode.js +10 -4
- package/lib/module/hooks/useNavigationMode.js.map +1 -1
- package/lib/module/hooks/usePaginatedSection.js +12 -7
- package/lib/module/hooks/usePaginatedSection.js.map +1 -1
- package/lib/module/hooks/usePrevious.js +30 -0
- package/lib/module/hooks/usePrevious.js.map +1 -0
- package/lib/module/hooks/useSafeCallback.js +33 -0
- package/lib/module/hooks/useSafeCallback.js.map +1 -0
- package/lib/module/hooks/useSkeletonItems.js +33 -0
- package/lib/module/hooks/useSkeletonItems.js.map +1 -0
- package/lib/module/hooks/useSplashCache.js +2 -1
- package/lib/module/hooks/useSplashCache.js.map +1 -1
- package/lib/module/hooks/useThemeColors.js +33 -0
- package/lib/module/hooks/useThemeColors.js.map +1 -0
- package/lib/module/theme/ThemeProvider.js +17 -11
- package/lib/module/theme/ThemeProvider.js.map +1 -1
- package/lib/module/theme/hook/useInternalTheme.js +18 -11
- package/lib/module/theme/hook/useInternalTheme.js.map +1 -1
- package/lib/module/theme/hook/useThemeStatusBar.js +10 -4
- package/lib/module/theme/hook/useThemeStatusBar.js.map +1 -1
- package/lib/typescript/src/Styles/globalStyles.d.ts +0 -5
- package/lib/typescript/src/Styles/globalStyles.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/AuthProvider/AuthProvider.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/ForgotPassword/ForgotPassword.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/Login/LoginWithEmail.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/Login/LoginWithPhone.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/OTP/OTP.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/QrLogin/QrLogin.d.ts +2 -15
- package/lib/typescript/src/components/Auth/QrLogin/QrLogin.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/QrLogin/components/QrViewArea.d.ts +0 -1
- package/lib/typescript/src/components/Auth/QrLogin/components/QrViewArea.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/SignUp/SignUp.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/SplashScreen/SplashScreen.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/SplashScreen/components/SplashImage/SplashImage.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/SplashScreen/components/SplashLottie/SplashLottie.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/SplashScreen/components/SplashVideo/SplashVideo.d.ts.map +1 -1
- package/lib/typescript/src/components/Auth/index.d.ts.map +1 -1
- package/lib/typescript/src/components/BackgroundLayout/BackgroundLayout.d.ts +1 -1
- package/lib/typescript/src/components/BackgroundLayout/BackgroundLayout.d.ts.map +1 -1
- package/lib/typescript/src/components/BlogView/BlogView.d.ts +2 -1
- package/lib/typescript/src/components/BlogView/BlogView.d.ts.map +1 -1
- package/lib/typescript/src/components/Button/BackBtn.d.ts +2 -2
- package/lib/typescript/src/components/Button/BackBtn.d.ts.map +1 -1
- package/lib/typescript/src/components/Button/PrimaryBtn.d.ts +2 -1
- package/lib/typescript/src/components/Button/PrimaryBtn.d.ts.map +1 -1
- package/lib/typescript/src/components/Button/SecondaryBtn.d.ts +2 -1
- package/lib/typescript/src/components/Button/SecondaryBtn.d.ts.map +1 -1
- package/lib/typescript/src/components/Button/TextButton.d.ts +2 -2
- package/lib/typescript/src/components/Button/TextButton.d.ts.map +1 -1
- package/lib/typescript/src/components/Button/index.d.ts +2 -2
- package/lib/typescript/src/components/Content/Card/Category/Category.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/NowWatching/NowWatching.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Sliders/Styles/One.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Sliders/Styles/Two.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Styles/Five.d.ts +13 -1
- package/lib/typescript/src/components/Content/Card/Styles/Five.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Styles/Four.d.ts +13 -1
- package/lib/typescript/src/components/Content/Card/Styles/Four.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Styles/One.d.ts +15 -3
- package/lib/typescript/src/components/Content/Card/Styles/One.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Styles/RotateInOut.d.ts +13 -1
- package/lib/typescript/src/components/Content/Card/Styles/RotateInOut.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Styles/Six.d.ts +1 -0
- package/lib/typescript/src/components/Content/Card/Styles/Six.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Styles/Three.d.ts +13 -5
- package/lib/typescript/src/components/Content/Card/Styles/Three.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Styles/TopTen.d.ts +1 -0
- package/lib/typescript/src/components/Content/Card/Styles/TopTen.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/Styles/Two.d.ts +13 -1
- package/lib/typescript/src/components/Content/Card/Styles/Two.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/components/AdsPoster.d.ts +26 -0
- package/lib/typescript/src/components/Content/Card/components/AdsPoster.d.ts.map +1 -0
- package/lib/typescript/src/components/Content/Card/components/CardPoster.d.ts +3 -1
- package/lib/typescript/src/components/Content/Card/components/CardPoster.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/components/RentOrBuyIcon.d.ts +2 -2
- package/lib/typescript/src/components/Content/Card/components/RentOrBuyIcon.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/components/ThumbnailCard.d.ts +2 -0
- package/lib/typescript/src/components/Content/Card/components/ThumbnailCard.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Card/components/index.d.ts +2 -0
- package/lib/typescript/src/components/Content/Card/components/index.d.ts.map +1 -0
- package/lib/typescript/src/components/Content/Card/index.d.ts +76 -6
- package/lib/typescript/src/components/Content/Card/index.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Content.d.ts +6 -3
- package/lib/typescript/src/components/Content/Content.d.ts.map +1 -1
- package/lib/typescript/src/components/Content/Sections.d.ts +22 -8
- package/lib/typescript/src/components/Content/Sections.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/ContentView.d.ts +6 -1
- package/lib/typescript/src/components/ContentView/ContentView.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/MoreContentList.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/AboutSection.d.ts +2 -1
- package/lib/typescript/src/components/ContentView/components/AboutSection.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/CastCard.d.ts +6 -1
- package/lib/typescript/src/components/ContentView/components/CastCard.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/EpisodeCard.d.ts +2 -1
- package/lib/typescript/src/components/ContentView/components/EpisodeCard.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/GenreTags.d.ts +1 -1
- package/lib/typescript/src/components/ContentView/components/GenreTags.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/HeroBanner.d.ts +2 -1
- package/lib/typescript/src/components/ContentView/components/HeroBanner.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/MiniInfo.d.ts +1 -1
- package/lib/typescript/src/components/ContentView/components/MiniInfo.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/PlayButton.d.ts +2 -1
- package/lib/typescript/src/components/ContentView/components/PlayButton.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/Title.d.ts +1 -1
- package/lib/typescript/src/components/ContentView/components/Title.d.ts.map +1 -1
- package/lib/typescript/src/components/ContentView/components/TrailerButton.d.ts +2 -1
- package/lib/typescript/src/components/ContentView/components/TrailerButton.d.ts.map +1 -1
- package/lib/typescript/src/components/Fallbacks/NoContentFallback.d.ts +2 -1
- package/lib/typescript/src/components/Fallbacks/NoContentFallback.d.ts.map +1 -1
- package/lib/typescript/src/components/Fallbacks/NotFoundFallback.d.ts +1 -1
- package/lib/typescript/src/components/Fallbacks/NotFoundFallback.d.ts.map +1 -1
- package/lib/typescript/src/components/Headers/AppHeader.d.ts +2 -2
- package/lib/typescript/src/components/Headers/AppHeader.d.ts.map +1 -1
- package/lib/typescript/src/components/Headers/Three.d.ts +2 -2
- package/lib/typescript/src/components/Headers/Three.d.ts.map +1 -1
- package/lib/typescript/src/components/Headers/Two.d.ts +2 -2
- package/lib/typescript/src/components/Headers/Two.d.ts.map +1 -1
- package/lib/typescript/src/components/Headers/index.d.ts +2 -2
- package/lib/typescript/src/components/Input/Input.d.ts +1 -1
- package/lib/typescript/src/components/Input/InputOne.d.ts +2 -2
- package/lib/typescript/src/components/Input/InputOne.d.ts.map +1 -1
- package/lib/typescript/src/components/Loader/Loader.d.ts.map +1 -1
- package/lib/typescript/src/components/Logo/Logo.d.ts +2 -1
- package/lib/typescript/src/components/Logo/Logo.d.ts.map +1 -1
- package/lib/typescript/src/components/Search/One.d.ts.map +1 -1
- package/lib/typescript/src/components/Search/components/SearchCard.d.ts +2 -5
- package/lib/typescript/src/components/Search/components/SearchCard.d.ts.map +1 -1
- package/lib/typescript/src/components/Search/index.d.ts +1 -1
- package/lib/typescript/src/components/Settings/AppSettings.d.ts +2 -2
- package/lib/typescript/src/components/Settings/AppSettings.d.ts.map +1 -1
- package/lib/typescript/src/components/Subscription/SubOne.d.ts.map +1 -1
- package/lib/typescript/src/components/Subscription/index.d.ts.map +1 -1
- package/lib/typescript/src/components/Text/Text.d.ts +2 -2
- package/lib/typescript/src/components/Text/Text.d.ts.map +1 -1
- package/lib/typescript/src/components/User/DeviceSessions/DeviceSessions.d.ts +1 -1
- package/lib/typescript/src/components/User/DeviceSessions/DeviceSessions.d.ts.map +1 -1
- package/lib/typescript/src/components/User/ProfileUpdate/ProfileUpdate.d.ts +1 -1
- package/lib/typescript/src/components/User/ProfileUpdate/ProfileUpdate.d.ts.map +1 -1
- package/lib/typescript/src/components/User/PurchaseHistory/PurchaseHistory.d.ts +1 -1
- package/lib/typescript/src/components/User/PurchaseHistory/PurchaseHistory.d.ts.map +1 -1
- package/lib/typescript/src/components/User/WatchHistory/WatchHistory.d.ts +1 -1
- package/lib/typescript/src/components/User/WatchHistory/WatchHistory.d.ts.map +1 -1
- package/lib/typescript/src/components/User/WatchLater/WatchLater.d.ts +1 -1
- package/lib/typescript/src/components/User/WatchLater/WatchLater.d.ts.map +1 -1
- package/lib/typescript/src/components/User/components/UserAvatar.d.ts +1 -1
- package/lib/typescript/src/components/User/components/UserAvatar.d.ts.map +1 -1
- package/lib/typescript/src/components/User/components/UserSection.d.ts +1 -1
- package/lib/typescript/src/components/User/components/UserSection.d.ts.map +1 -1
- package/lib/typescript/src/components/View/View.d.ts +2 -0
- package/lib/typescript/src/components/View/View.d.ts.map +1 -1
- package/lib/typescript/src/components/index.d.ts +0 -1
- package/lib/typescript/src/components/index.d.ts.map +1 -1
- package/lib/typescript/src/constants/dummySections.d.ts +5 -0
- package/lib/typescript/src/constants/dummySections.d.ts.map +1 -1
- package/lib/typescript/src/hooks/Images/index.d.ts +3 -0
- package/lib/typescript/src/hooks/Images/index.d.ts.map +1 -0
- package/lib/typescript/src/hooks/Images/useImageLoader.d.ts +36 -0
- package/lib/typescript/src/hooks/Images/useImageLoader.d.ts.map +1 -0
- package/lib/typescript/src/hooks/Images/useImageValidation.d.ts +17 -0
- package/lib/typescript/src/hooks/Images/useImageValidation.d.ts.map +1 -0
- package/lib/typescript/src/hooks/index.d.ts +17 -0
- package/lib/typescript/src/hooks/index.d.ts.map +1 -0
- package/lib/typescript/src/hooks/useAdTracking.d.ts +39 -0
- package/lib/typescript/src/hooks/useAdTracking.d.ts.map +1 -0
- package/lib/typescript/src/hooks/useCards.d.ts +36 -0
- package/lib/typescript/src/hooks/useCards.d.ts.map +1 -0
- package/lib/typescript/src/hooks/useDebounce.d.ts.map +1 -1
- package/lib/typescript/src/hooks/useKeyboard.d.ts.map +1 -1
- package/lib/typescript/src/hooks/useNavigationMode.d.ts.map +1 -1
- package/lib/typescript/src/hooks/usePaginatedSection.d.ts +12 -2
- package/lib/typescript/src/hooks/usePaginatedSection.d.ts.map +1 -1
- package/lib/typescript/src/hooks/usePrevious.d.ts +12 -0
- package/lib/typescript/src/hooks/usePrevious.d.ts.map +1 -0
- package/lib/typescript/src/hooks/useSafeCallback.d.ts +15 -0
- package/lib/typescript/src/hooks/useSafeCallback.d.ts.map +1 -0
- package/lib/typescript/src/hooks/useSkeletonItems.d.ts +11 -0
- package/lib/typescript/src/hooks/useSkeletonItems.d.ts.map +1 -0
- package/lib/typescript/src/hooks/useSplashCache.d.ts.map +1 -1
- package/lib/typescript/src/hooks/useThemeColors.d.ts +27 -0
- package/lib/typescript/src/hooks/useThemeColors.d.ts.map +1 -0
- package/lib/typescript/src/theme/ThemeProvider.d.ts.map +1 -1
- package/lib/typescript/src/theme/hook/useInternalTheme.d.ts.map +1 -1
- package/lib/typescript/src/theme/hook/useThemeStatusBar.d.ts.map +1 -1
- package/lib/typescript/src/types/sections/index.d.ts +7 -4
- package/lib/typescript/src/types/sections/index.d.ts.map +1 -1
- package/package.json +6 -3
- package/src/Styles/globalStyles.ts +0 -5
- package/src/components/Auth/AuthProvider/AuthProvider.tsx +288 -212
- package/src/components/Auth/ForgotPassword/ForgotPassword.tsx +11 -4
- package/src/components/Auth/Login/LoginWithEmail.tsx +266 -238
- package/src/components/Auth/Login/LoginWithPhone.tsx +18 -6
- package/src/components/Auth/OTP/OTP.tsx +21 -5
- package/src/components/Auth/QrLogin/QrLogin.tsx +477 -177
- package/src/components/Auth/QrLogin/components/QrViewArea.tsx +349 -139
- package/src/components/Auth/SignUp/SignUp.tsx +312 -293
- package/src/components/Auth/SplashScreen/SplashScreen.tsx +186 -131
- package/src/components/Auth/SplashScreen/components/SplashImage/SplashImage.tsx +32 -15
- package/src/components/Auth/SplashScreen/components/SplashLottie/SplashLottie.tsx +11 -5
- package/src/components/Auth/SplashScreen/components/SplashVideo/SplashVideo.tsx +34 -18
- package/src/components/BackgroundLayout/BackgroundLayout.tsx +43 -37
- package/src/components/BlogView/BlogView.tsx +66 -30
- package/src/components/Button/BackBtn.tsx +64 -34
- package/src/components/Button/PrimaryBtn.tsx +42 -25
- package/src/components/Button/SecondaryBtn.tsx +41 -25
- package/src/components/Button/TextButton.tsx +54 -32
- package/src/components/Content/Card/Category/Category.tsx +189 -64
- package/src/components/Content/Card/NowWatching/NowWatching.tsx +281 -136
- package/src/components/Content/Card/Sliders/Styles/One.tsx +270 -158
- package/src/components/Content/Card/Sliders/Styles/Two.tsx +239 -90
- package/src/components/Content/Card/Styles/Five.tsx +161 -62
- package/src/components/Content/Card/Styles/Four.tsx +164 -85
- package/src/components/Content/Card/Styles/One.tsx +161 -71
- package/src/components/Content/Card/Styles/RotateInOut.tsx +162 -67
- package/src/components/Content/Card/Styles/Six.tsx +242 -142
- package/src/components/Content/Card/Styles/Three.tsx +166 -133
- package/src/components/Content/Card/Styles/TopTen.tsx +230 -191
- package/src/components/Content/Card/Styles/Two.tsx +182 -79
- package/src/components/Content/Card/components/AdsPoster.tsx +202 -0
- package/src/components/Content/Card/components/CardPoster.tsx +166 -105
- package/src/components/Content/Card/components/RentOrBuyIcon.tsx +58 -38
- package/src/components/Content/Card/components/ThumbnailCard.tsx +122 -47
- package/src/components/Content/Card/components/index.ts +1 -0
- package/src/components/Content/Content.tsx +211 -106
- package/src/components/Content/Sections.tsx +148 -51
- package/src/components/ContentView/ContentView.tsx +194 -128
- package/src/components/ContentView/MoreContentList.tsx +215 -151
- package/src/components/ContentView/components/AboutSection.tsx +132 -93
- package/src/components/ContentView/components/CastCard.tsx +148 -138
- package/src/components/ContentView/components/EpisodeCard.tsx +160 -155
- package/src/components/ContentView/components/GenreTags.tsx +68 -53
- package/src/components/ContentView/components/HeroBanner.tsx +326 -284
- package/src/components/ContentView/components/MiniInfo.tsx +181 -159
- package/src/components/ContentView/components/PlayButton.tsx +27 -16
- package/src/components/ContentView/components/Title.tsx +89 -72
- package/src/components/ContentView/components/TrailerButton.tsx +35 -22
- package/src/components/Fallbacks/NoContentFallback.tsx +107 -103
- package/src/components/Fallbacks/NotFoundFallback.tsx +12 -4
- package/src/components/Headers/AppHeader.tsx +42 -26
- package/src/components/Headers/Three.tsx +12 -8
- package/src/components/Headers/Two.tsx +31 -10
- package/src/components/Input/InputOne.tsx +136 -108
- package/src/components/Loader/Loader.tsx +2 -2
- package/src/components/Logo/Logo.tsx +43 -27
- package/src/components/Search/One.tsx +2 -4
- package/src/components/Search/components/SearchCard.tsx +19 -13
- package/src/components/Settings/AppSettings.tsx +217 -128
- package/src/components/Subscription/SubOne.tsx +394 -317
- package/src/components/Text/Text.tsx +33 -22
- package/src/components/User/DeviceSessions/DeviceSessions.tsx +129 -102
- package/src/components/User/ProfileUpdate/ProfileUpdate.tsx +36 -17
- package/src/components/User/PurchaseHistory/PurchaseHistory.tsx +194 -140
- package/src/components/User/WatchHistory/WatchHistory.tsx +63 -36
- package/src/components/User/WatchLater/WatchLater.tsx +119 -81
- package/src/components/User/components/UserAvatar.tsx +49 -15
- package/src/components/User/components/UserSection.tsx +71 -37
- package/src/components/View/View.tsx +10 -2
- package/src/components/index.ts +0 -1
- package/src/constants/dummySections.ts +48 -1
- package/src/hooks/Images/index.ts +2 -0
- package/src/hooks/Images/useImageLoader.ts +206 -0
- package/src/hooks/Images/useImageValidation.ts +36 -0
- package/src/hooks/index.ts +17 -0
- package/src/hooks/useAdTracking.ts +349 -0
- package/src/hooks/useCards.ts +228 -0
- package/src/hooks/useDebounce.ts +25 -6
- package/src/hooks/useKeyboard.ts +12 -5
- package/src/hooks/useNavigationMode.ts +16 -5
- package/src/hooks/usePaginatedSection.ts +27 -8
- package/src/hooks/usePrevious.ts +28 -0
- package/src/hooks/useSafeCallback.ts +43 -0
- package/src/hooks/useSkeletonItems.ts +30 -0
- package/src/hooks/useSplashCache.ts +2 -1
- package/src/hooks/useThemeColors.ts +51 -0
- package/src/theme/ThemeProvider.tsx +25 -9
- package/src/theme/hook/useInternalTheme.ts +19 -13
- package/src/theme/hook/useThemeStatusBar.ts +14 -5
- package/src/types/sections/index.ts +7 -4
|
@@ -0,0 +1,206 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @author Naresh Dhamu
|
|
3
|
+
* @lastModified Wed 26 Nov 2025 at 02:19 PM
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { useState, useEffect, useMemo, useCallback, useRef } from 'react';
|
|
7
|
+
import FastImage from 'react-native-fast-image';
|
|
8
|
+
import { isValidImageUrl, isLocalFile } from './useImageValidation';
|
|
9
|
+
|
|
10
|
+
// Module-level cache to track successfully loaded images
|
|
11
|
+
const loadedImageCache = new Set<string>();
|
|
12
|
+
|
|
13
|
+
export type UseImageLoaderOptions = {
|
|
14
|
+
imageUri: string;
|
|
15
|
+
isLoading?: boolean;
|
|
16
|
+
enablePreload?: boolean;
|
|
17
|
+
onError?: () => void;
|
|
18
|
+
onLoad?: () => void;
|
|
19
|
+
onLoadStart?: () => void;
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
export type UseImageLoaderReturn = {
|
|
23
|
+
imageLoading: boolean;
|
|
24
|
+
imageError: boolean;
|
|
25
|
+
hasValidImage: boolean;
|
|
26
|
+
isImageCached: boolean;
|
|
27
|
+
showSkeleton: boolean;
|
|
28
|
+
showFallback: boolean;
|
|
29
|
+
imageSource: {
|
|
30
|
+
uri: string;
|
|
31
|
+
cache?: typeof FastImage.cacheControl.immutable;
|
|
32
|
+
priority?: typeof FastImage.priority.normal;
|
|
33
|
+
};
|
|
34
|
+
handleLoad: () => void;
|
|
35
|
+
handleError: () => void;
|
|
36
|
+
handleLoadStart?: () => void;
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Custom hook to manage image loading state, caching, and validation
|
|
41
|
+
* @param options - Configuration options for image loading
|
|
42
|
+
* @returns Object containing image state and handlers
|
|
43
|
+
*/
|
|
44
|
+
export const useImageLoader = ({
|
|
45
|
+
imageUri,
|
|
46
|
+
isLoading = false,
|
|
47
|
+
enablePreload = false,
|
|
48
|
+
onError,
|
|
49
|
+
onLoad,
|
|
50
|
+
onLoadStart,
|
|
51
|
+
}: UseImageLoaderOptions): UseImageLoaderReturn => {
|
|
52
|
+
const [imageLoading, setImageLoading] = useState(true);
|
|
53
|
+
const [imageError, setImageError] = useState(false);
|
|
54
|
+
const previousUriRef = useRef<string>('');
|
|
55
|
+
const onErrorRef = useRef(onError);
|
|
56
|
+
|
|
57
|
+
// Keep onError ref updated
|
|
58
|
+
useEffect(() => {
|
|
59
|
+
onErrorRef.current = onError;
|
|
60
|
+
}, [onError]);
|
|
61
|
+
|
|
62
|
+
// Validate image URL
|
|
63
|
+
const hasValidImage = useMemo(() => isValidImageUrl(imageUri), [imageUri]);
|
|
64
|
+
|
|
65
|
+
// Check if image is already cached
|
|
66
|
+
const isImageCached = useMemo(
|
|
67
|
+
() => loadedImageCache.has(imageUri),
|
|
68
|
+
[imageUri]
|
|
69
|
+
);
|
|
70
|
+
|
|
71
|
+
// Handle image state changes
|
|
72
|
+
useEffect(() => {
|
|
73
|
+
const uriChanged = previousUriRef.current !== imageUri;
|
|
74
|
+
|
|
75
|
+
// Update ref when URI changes
|
|
76
|
+
if (uriChanged) {
|
|
77
|
+
previousUriRef.current = imageUri;
|
|
78
|
+
// Reset error state when URI changes to allow new image to load
|
|
79
|
+
setImageError(false);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// Reset states when loading
|
|
83
|
+
if (isLoading) {
|
|
84
|
+
setImageLoading(true);
|
|
85
|
+
setImageError(false);
|
|
86
|
+
return;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Handle valid image URL
|
|
90
|
+
if (hasValidImage) {
|
|
91
|
+
const isLocal = isLocalFile(imageUri);
|
|
92
|
+
|
|
93
|
+
if (isImageCached) {
|
|
94
|
+
// Image is cached, mark as loaded
|
|
95
|
+
setImageLoading(false);
|
|
96
|
+
setImageError(false);
|
|
97
|
+
} else {
|
|
98
|
+
// Image not cached, start loading
|
|
99
|
+
setImageLoading(true);
|
|
100
|
+
setImageError(false);
|
|
101
|
+
// Preload image if enabled (only for remote URLs, not local files)
|
|
102
|
+
if (enablePreload && !isLocal && uriChanged) {
|
|
103
|
+
FastImage.preload([{ uri: imageUri }]);
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
} else {
|
|
107
|
+
// For empty or invalid URLs
|
|
108
|
+
if (!imageUri || imageUri.trim() === '') {
|
|
109
|
+
// Empty URI - show error immediately (no point in trying to load)
|
|
110
|
+
setImageLoading(false);
|
|
111
|
+
setImageError(true);
|
|
112
|
+
} else {
|
|
113
|
+
// Invalid URL format - attempt to load anyway
|
|
114
|
+
// FastImage might still be able to load it, so keep loading state
|
|
115
|
+
setImageLoading(true);
|
|
116
|
+
setImageError(false);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
}, [imageUri, hasValidImage, isLoading, isImageCached, enablePreload]);
|
|
120
|
+
|
|
121
|
+
// Determine if skeleton should be shown
|
|
122
|
+
const showSkeleton = useMemo(
|
|
123
|
+
() => isLoading || (imageLoading && !imageError),
|
|
124
|
+
[isLoading, imageLoading, imageError]
|
|
125
|
+
);
|
|
126
|
+
|
|
127
|
+
// Determine if fallback should be shown
|
|
128
|
+
// Show fallback when:
|
|
129
|
+
// 1. Not showing skeleton
|
|
130
|
+
// 2. AND (imageError is true OR URI is empty/invalid)
|
|
131
|
+
// This allows images with invalid format to attempt loading, but shows fallback for empty URIs
|
|
132
|
+
const showFallback = useMemo(() => {
|
|
133
|
+
if (showSkeleton) return false;
|
|
134
|
+
// Empty URI - show fallback immediately
|
|
135
|
+
if (!imageUri || imageUri.trim() === '') return true;
|
|
136
|
+
// Show fallback only after actual error (not just invalid format)
|
|
137
|
+
return imageError && !imageLoading;
|
|
138
|
+
}, [showSkeleton, imageError, imageLoading, imageUri]);
|
|
139
|
+
|
|
140
|
+
// Check if image is a local file
|
|
141
|
+
const isLocal = useMemo(() => isLocalFile(imageUri), [imageUri]);
|
|
142
|
+
|
|
143
|
+
// Image source configuration
|
|
144
|
+
// Local files don't need cache control, cloud URIs do
|
|
145
|
+
const imageSource = useMemo(() => {
|
|
146
|
+
const source: {
|
|
147
|
+
uri: string;
|
|
148
|
+
cache?: typeof FastImage.cacheControl.immutable;
|
|
149
|
+
priority?: typeof FastImage.priority.normal;
|
|
150
|
+
} = {
|
|
151
|
+
uri: imageUri,
|
|
152
|
+
};
|
|
153
|
+
|
|
154
|
+
// Only add cache control for remote URLs, not local files
|
|
155
|
+
if (!isLocal) {
|
|
156
|
+
source.cache = FastImage.cacheControl.immutable;
|
|
157
|
+
source.priority = FastImage.priority.normal;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
return source;
|
|
161
|
+
}, [imageUri, isLocal]);
|
|
162
|
+
|
|
163
|
+
// Image event handlers
|
|
164
|
+
const handleLoad = useCallback(() => {
|
|
165
|
+
setImageLoading(false);
|
|
166
|
+
setImageError(false);
|
|
167
|
+
if (imageUri) {
|
|
168
|
+
loadedImageCache.add(imageUri);
|
|
169
|
+
}
|
|
170
|
+
if (onLoad) {
|
|
171
|
+
onLoad();
|
|
172
|
+
}
|
|
173
|
+
}, [imageUri, onLoad]);
|
|
174
|
+
|
|
175
|
+
const handleError = useCallback(() => {
|
|
176
|
+
setImageLoading(false);
|
|
177
|
+
|
|
178
|
+
setImageError(true);
|
|
179
|
+
if (onError) {
|
|
180
|
+
onError();
|
|
181
|
+
}
|
|
182
|
+
}, [onError]);
|
|
183
|
+
|
|
184
|
+
const handleLoadStart = useCallback(() => {
|
|
185
|
+
if (!isImageCached) {
|
|
186
|
+
setImageLoading(true);
|
|
187
|
+
setImageError(false);
|
|
188
|
+
if (onLoadStart) {
|
|
189
|
+
onLoadStart();
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
}, [isImageCached, onLoadStart]);
|
|
193
|
+
|
|
194
|
+
return {
|
|
195
|
+
imageLoading,
|
|
196
|
+
imageError,
|
|
197
|
+
hasValidImage,
|
|
198
|
+
isImageCached,
|
|
199
|
+
showSkeleton,
|
|
200
|
+
showFallback,
|
|
201
|
+
imageSource,
|
|
202
|
+
handleLoad,
|
|
203
|
+
handleError,
|
|
204
|
+
handleLoadStart: enablePreload ? handleLoadStart : undefined,
|
|
205
|
+
};
|
|
206
|
+
};
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @author Naresh Dhamu
|
|
3
|
+
* @lastModified Wed 26 Nov 2025 at 02:19 PM
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
const IMAGE_URL_REGEX = /\.(jpg|jpeg|png|webp|gif|bmp)$/i;
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Utility function to check if URL is a local file path
|
|
10
|
+
* @param url - The URL to check
|
|
11
|
+
* @returns true if the URL is a local file path, false otherwise
|
|
12
|
+
*/
|
|
13
|
+
export const isLocalFile = (url: string): boolean => {
|
|
14
|
+
return url?.startsWith?.('file://') || url?.startsWith?.('/');
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Utility function to validate image URL (supports both cloud URIs and local file paths)
|
|
19
|
+
* @param url - The URL to validate
|
|
20
|
+
* @returns true if the URL is a valid image URL, false otherwise
|
|
21
|
+
*/
|
|
22
|
+
export const isValidImageUrl = (url: string | null | undefined): boolean => {
|
|
23
|
+
if (!url || typeof url !== 'string') return false;
|
|
24
|
+
const cleaned = url.trim();
|
|
25
|
+
|
|
26
|
+
// Check for local file paths
|
|
27
|
+
if (isLocalFile(cleaned)) {
|
|
28
|
+
return IMAGE_URL_REGEX.test(cleaned) || cleaned.length > 0;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// Check for cloud URIs (http/https)
|
|
32
|
+
return (
|
|
33
|
+
(cleaned?.startsWith?.('http://') || cleaned?.startsWith?.('https://')) &&
|
|
34
|
+
IMAGE_URL_REGEX.test(cleaned)
|
|
35
|
+
);
|
|
36
|
+
};
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @author Naresh Dhamu
|
|
3
|
+
* @lastModified Optimized for performance
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
export * from './useDebounce';
|
|
7
|
+
export * from './useKeyboard';
|
|
8
|
+
export * from './useNavigationMode';
|
|
9
|
+
export * from './usePaginatedSection';
|
|
10
|
+
export * from './useSplashCache';
|
|
11
|
+
export * from './usePrevious';
|
|
12
|
+
export * from './useSafeCallback';
|
|
13
|
+
export * from './useSkeletonItems';
|
|
14
|
+
export * from './useThemeColors';
|
|
15
|
+
export * from './useCards';
|
|
16
|
+
export * from './Images';
|
|
17
|
+
export * from './useAdTracking';
|
|
@@ -0,0 +1,349 @@
|
|
|
1
|
+
import { useRef, useMemo, useCallback, useEffect } from 'react';
|
|
2
|
+
import { type View, Dimensions } from 'react-native';
|
|
3
|
+
import type { IServeAd } from '@zezosoft/zezo-ott-api-client';
|
|
4
|
+
|
|
5
|
+
const CHECK_INTERVAL = 500;
|
|
6
|
+
const INITIAL_CHECK_DELAY = 100;
|
|
7
|
+
|
|
8
|
+
export interface ViewportOffsets {
|
|
9
|
+
top: number;
|
|
10
|
+
bottom: number;
|
|
11
|
+
left: number;
|
|
12
|
+
right: number;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export interface ScreenDimensions {
|
|
16
|
+
width: number;
|
|
17
|
+
height: number;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export interface UseAdTrackingOptions {
|
|
21
|
+
ad: IServeAd;
|
|
22
|
+
onDisplayAds?: (ad: IServeAd) => void;
|
|
23
|
+
isLoading?: boolean;
|
|
24
|
+
screenDimensions?: ScreenDimensions;
|
|
25
|
+
viewportOffsets?: ViewportOffsets;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
export interface UseAdTrackingReturn {
|
|
29
|
+
viewRef: React.RefObject<View | null>;
|
|
30
|
+
layoutRef: React.MutableRefObject<{
|
|
31
|
+
x: number;
|
|
32
|
+
y: number;
|
|
33
|
+
width: number;
|
|
34
|
+
height: number;
|
|
35
|
+
} | null>;
|
|
36
|
+
hasDisplayed: React.MutableRefObject<boolean>;
|
|
37
|
+
adUniqueId: string;
|
|
38
|
+
handleLayout: () => void;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// Global set to track displayed ads - prevents duplicate tracking
|
|
42
|
+
const globalDisplayedAds = new Set<string>();
|
|
43
|
+
// Lock to prevent race conditions when checking and adding ads
|
|
44
|
+
const trackingLock = new Map<string, boolean>();
|
|
45
|
+
|
|
46
|
+
export const extractToken = (trackingUrl?: string): string => {
|
|
47
|
+
if (!trackingUrl) return '';
|
|
48
|
+
try {
|
|
49
|
+
const url = new URL(trackingUrl);
|
|
50
|
+
return url.searchParams.get('token') || '';
|
|
51
|
+
} catch {
|
|
52
|
+
const match = trackingUrl.match(/[?&]token=([^&]+)/);
|
|
53
|
+
return match?.[1] || '';
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Creates a unique ID for an ad to prevent duplicate tracking
|
|
59
|
+
* Priority: _id > token > mediaUrl > fallback
|
|
60
|
+
*/
|
|
61
|
+
export const createAdUniqueId = (ad: IServeAd): string => {
|
|
62
|
+
// First try to use _id if available (most reliable)
|
|
63
|
+
if ('_id' in ad && ad._id) {
|
|
64
|
+
return `ad-${String(ad._id)}`;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// Then try token from tracking URL
|
|
68
|
+
const token = extractToken(ad.tracking?.impression);
|
|
69
|
+
if (token) {
|
|
70
|
+
return `ad-token-${token}`;
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// Then use mediaUrl
|
|
74
|
+
const mediaUrl = ad.mediaUrl || '';
|
|
75
|
+
if (mediaUrl) {
|
|
76
|
+
return `ad-media-${mediaUrl}`;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// Last resort: use a combination of available fields
|
|
80
|
+
const clickUrl = ad.tracking?.click || '';
|
|
81
|
+
if (clickUrl) {
|
|
82
|
+
return `ad-click-${clickUrl}`;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
// Fallback (should rarely happen)
|
|
86
|
+
return `ad-fallback-${Date.now()}-${Math.random()}`;
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
export const useAdTracking = (
|
|
90
|
+
options: UseAdTrackingOptions
|
|
91
|
+
): UseAdTrackingReturn => {
|
|
92
|
+
const {
|
|
93
|
+
ad,
|
|
94
|
+
onDisplayAds,
|
|
95
|
+
isLoading = false,
|
|
96
|
+
screenDimensions: propScreenDimensions,
|
|
97
|
+
viewportOffsets: propViewportOffsets,
|
|
98
|
+
} = options;
|
|
99
|
+
|
|
100
|
+
const viewRef = useRef<View>(null);
|
|
101
|
+
const layoutRef = useRef<{
|
|
102
|
+
x: number;
|
|
103
|
+
y: number;
|
|
104
|
+
width: number;
|
|
105
|
+
height: number;
|
|
106
|
+
} | null>(null);
|
|
107
|
+
const hasDisplayedRef = useRef<boolean>(false);
|
|
108
|
+
|
|
109
|
+
const onDisplayAdsRef = useRef(onDisplayAds);
|
|
110
|
+
const adRef = useRef(ad);
|
|
111
|
+
const isLoadingRef = useRef(isLoading);
|
|
112
|
+
|
|
113
|
+
useEffect(() => {
|
|
114
|
+
onDisplayAdsRef.current = onDisplayAds;
|
|
115
|
+
adRef.current = ad;
|
|
116
|
+
isLoadingRef.current = isLoading;
|
|
117
|
+
}, [onDisplayAds, ad, isLoading]);
|
|
118
|
+
|
|
119
|
+
const adUniqueId = useMemo(() => createAdUniqueId(ad), [ad]);
|
|
120
|
+
|
|
121
|
+
const screenDimensions = useMemo(
|
|
122
|
+
() => propScreenDimensions || Dimensions.get('window'),
|
|
123
|
+
[propScreenDimensions]
|
|
124
|
+
);
|
|
125
|
+
|
|
126
|
+
const viewportOffsets = useMemo(
|
|
127
|
+
() => ({
|
|
128
|
+
top: propViewportOffsets?.top ?? 50,
|
|
129
|
+
bottom: propViewportOffsets?.bottom ?? 100,
|
|
130
|
+
left: propViewportOffsets?.left ?? 20,
|
|
131
|
+
right: propViewportOffsets?.right ?? 20,
|
|
132
|
+
}),
|
|
133
|
+
[
|
|
134
|
+
propViewportOffsets?.top,
|
|
135
|
+
propViewportOffsets?.bottom,
|
|
136
|
+
propViewportOffsets?.left,
|
|
137
|
+
propViewportOffsets?.right,
|
|
138
|
+
]
|
|
139
|
+
);
|
|
140
|
+
|
|
141
|
+
const viewportBounds = useMemo(() => {
|
|
142
|
+
const { width: screenWidth, height: screenHeight } = screenDimensions;
|
|
143
|
+
return {
|
|
144
|
+
left: viewportOffsets.left,
|
|
145
|
+
right: screenWidth - viewportOffsets.right,
|
|
146
|
+
top: viewportOffsets.top,
|
|
147
|
+
bottom: screenHeight - viewportOffsets.bottom,
|
|
148
|
+
};
|
|
149
|
+
}, [screenDimensions, viewportOffsets]);
|
|
150
|
+
|
|
151
|
+
const checkVisibility = useCallback((): boolean => {
|
|
152
|
+
const layout = layoutRef.current;
|
|
153
|
+
if (!layout) {
|
|
154
|
+
return false;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
const { x, y, width, height } = layout;
|
|
158
|
+
|
|
159
|
+
const isVisible =
|
|
160
|
+
x + width > viewportBounds.left &&
|
|
161
|
+
x < viewportBounds.right &&
|
|
162
|
+
y + height > viewportBounds.top &&
|
|
163
|
+
y < viewportBounds.bottom;
|
|
164
|
+
|
|
165
|
+
return isVisible;
|
|
166
|
+
}, [viewportBounds]);
|
|
167
|
+
|
|
168
|
+
const measureInWindow = useCallback((): void => {
|
|
169
|
+
const view = viewRef.current;
|
|
170
|
+
if (!view) return;
|
|
171
|
+
|
|
172
|
+
view.measureInWindow((x, y, width, height) => {
|
|
173
|
+
layoutRef.current = { x, y, width, height };
|
|
174
|
+
});
|
|
175
|
+
}, []);
|
|
176
|
+
|
|
177
|
+
/**
|
|
178
|
+
* Atomically checks and tracks ad display to prevent duplicate tracking
|
|
179
|
+
* Uses a lock mechanism to ensure only one instance can track the same ad
|
|
180
|
+
*/
|
|
181
|
+
const trackAdDisplay = useCallback((): boolean => {
|
|
182
|
+
const currentOnDisplayAds = onDisplayAdsRef.current;
|
|
183
|
+
const currentIsLoading = isLoadingRef.current;
|
|
184
|
+
|
|
185
|
+
if (!currentOnDisplayAds || currentIsLoading) {
|
|
186
|
+
return false;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
// Early return if already displayed by this instance
|
|
190
|
+
if (hasDisplayedRef.current) {
|
|
191
|
+
return false;
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
// Atomic check-and-set: if ad is already being tracked or displayed, return false
|
|
195
|
+
if (globalDisplayedAds.has(adUniqueId)) {
|
|
196
|
+
// Mark this instance as displayed to prevent further checks
|
|
197
|
+
hasDisplayedRef.current = true;
|
|
198
|
+
return false;
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
// Set lock to prevent other instances from tracking the same ad
|
|
202
|
+
if (trackingLock.get(adUniqueId) === true) {
|
|
203
|
+
return false;
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
// Acquire lock
|
|
207
|
+
trackingLock.set(adUniqueId, true);
|
|
208
|
+
|
|
209
|
+
try {
|
|
210
|
+
// Double-check after acquiring lock (another instance might have added it)
|
|
211
|
+
if (globalDisplayedAds.has(adUniqueId)) {
|
|
212
|
+
hasDisplayedRef.current = true;
|
|
213
|
+
return false;
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
// Mark as displayed in both local and global state
|
|
217
|
+
hasDisplayedRef.current = true;
|
|
218
|
+
globalDisplayedAds.add(adUniqueId);
|
|
219
|
+
|
|
220
|
+
// Call the tracking callback
|
|
221
|
+
currentOnDisplayAds(adRef.current);
|
|
222
|
+
|
|
223
|
+
return true;
|
|
224
|
+
} finally {
|
|
225
|
+
// Release lock after a short delay to ensure callback completes
|
|
226
|
+
setTimeout(() => {
|
|
227
|
+
trackingLock.delete(adUniqueId);
|
|
228
|
+
}, 100);
|
|
229
|
+
}
|
|
230
|
+
}, [adUniqueId]);
|
|
231
|
+
|
|
232
|
+
const checkAndTrackVisibility = useCallback((): void => {
|
|
233
|
+
// Early exit if already displayed
|
|
234
|
+
if (hasDisplayedRef.current) {
|
|
235
|
+
return;
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
// Early exit if ad is already tracked globally or locked
|
|
239
|
+
if (
|
|
240
|
+
globalDisplayedAds.has(adUniqueId) ||
|
|
241
|
+
trackingLock.get(adUniqueId) === true
|
|
242
|
+
) {
|
|
243
|
+
// Mark this instance as displayed to prevent further checks
|
|
244
|
+
hasDisplayedRef.current = true;
|
|
245
|
+
return;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
// Check visibility before tracking
|
|
249
|
+
if (!checkVisibility()) {
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
// Attempt to track (will handle duplicate prevention internally)
|
|
254
|
+
trackAdDisplay();
|
|
255
|
+
}, [adUniqueId, checkVisibility, trackAdDisplay]);
|
|
256
|
+
|
|
257
|
+
const handleLayout = useCallback(() => {
|
|
258
|
+
const view = viewRef.current;
|
|
259
|
+
if (!view) return;
|
|
260
|
+
|
|
261
|
+
view.measureInWindow((x, y, width, height) => {
|
|
262
|
+
layoutRef.current = { x, y, width, height };
|
|
263
|
+
requestAnimationFrame(() => {
|
|
264
|
+
checkAndTrackVisibility();
|
|
265
|
+
});
|
|
266
|
+
});
|
|
267
|
+
}, [checkAndTrackVisibility]);
|
|
268
|
+
|
|
269
|
+
useEffect(() => {
|
|
270
|
+
if (isLoadingRef.current) {
|
|
271
|
+
return undefined;
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
let timeoutId: NodeJS.Timeout | null = null;
|
|
275
|
+
let rafId: number | null = null;
|
|
276
|
+
|
|
277
|
+
rafId = requestAnimationFrame(() => {
|
|
278
|
+
timeoutId = setTimeout(() => {
|
|
279
|
+
measureInWindow();
|
|
280
|
+
requestAnimationFrame(() => {
|
|
281
|
+
checkAndTrackVisibility();
|
|
282
|
+
});
|
|
283
|
+
}, INITIAL_CHECK_DELAY);
|
|
284
|
+
});
|
|
285
|
+
|
|
286
|
+
return () => {
|
|
287
|
+
if (rafId !== null) {
|
|
288
|
+
cancelAnimationFrame(rafId);
|
|
289
|
+
}
|
|
290
|
+
if (timeoutId !== null) {
|
|
291
|
+
clearTimeout(timeoutId);
|
|
292
|
+
}
|
|
293
|
+
};
|
|
294
|
+
}, [isLoading, checkAndTrackVisibility, measureInWindow]);
|
|
295
|
+
|
|
296
|
+
useEffect(() => {
|
|
297
|
+
if (
|
|
298
|
+
!onDisplayAdsRef.current ||
|
|
299
|
+
hasDisplayedRef.current ||
|
|
300
|
+
isLoadingRef.current
|
|
301
|
+
) {
|
|
302
|
+
return undefined;
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
let rafId: number | null = null;
|
|
306
|
+
let lastCheckTime = 0;
|
|
307
|
+
|
|
308
|
+
const checkWithThrottle = () => {
|
|
309
|
+
const now = Date.now();
|
|
310
|
+
|
|
311
|
+
if (now - lastCheckTime < CHECK_INTERVAL) {
|
|
312
|
+
rafId = requestAnimationFrame(checkWithThrottle);
|
|
313
|
+
return;
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
lastCheckTime = now;
|
|
317
|
+
|
|
318
|
+
if (hasDisplayedRef.current || !viewRef.current) {
|
|
319
|
+
return;
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
measureInWindow();
|
|
323
|
+
requestAnimationFrame(() => {
|
|
324
|
+
if (!hasDisplayedRef.current) {
|
|
325
|
+
checkAndTrackVisibility();
|
|
326
|
+
}
|
|
327
|
+
if (!hasDisplayedRef.current) {
|
|
328
|
+
rafId = requestAnimationFrame(checkWithThrottle);
|
|
329
|
+
}
|
|
330
|
+
});
|
|
331
|
+
};
|
|
332
|
+
|
|
333
|
+
rafId = requestAnimationFrame(checkWithThrottle);
|
|
334
|
+
|
|
335
|
+
return () => {
|
|
336
|
+
if (rafId !== null) {
|
|
337
|
+
cancelAnimationFrame(rafId);
|
|
338
|
+
}
|
|
339
|
+
};
|
|
340
|
+
}, [checkAndTrackVisibility, measureInWindow]);
|
|
341
|
+
|
|
342
|
+
return {
|
|
343
|
+
viewRef,
|
|
344
|
+
layoutRef,
|
|
345
|
+
hasDisplayed: hasDisplayedRef,
|
|
346
|
+
adUniqueId,
|
|
347
|
+
handleLayout,
|
|
348
|
+
};
|
|
349
|
+
};
|