From d8521fbae2721103095ee31ec2f5173a0f9f4e7c Mon Sep 17 00:00:00 2001 From: Akiva Berger Date: Mon, 11 Nov 2024 09:57:26 +0200 Subject: [PATCH 1/4] chore(analytics): expose global state --- StateManager.js | 58 ++++++++++++++++++++++++++++++++++--------------- 1 file changed, 41 insertions(+), 17 deletions(-) diff --git a/StateManager.js b/StateManager.js index f49900b1..2de143c3 100644 --- a/StateManager.js +++ b/StateManager.js @@ -192,6 +192,8 @@ const saveFieldToAsync = function (field, value) { AsyncStorage.setItem(field, JSON.stringify(value)); }; +let currentGlobalState = DEFAULT_STATE; + const reducer = function (state, action) { if (UPDATE_SETTINGS_ACTIONS[action.type] && !action.fromAsync) { AsyncStorage.setItem('lastSettingsUpdateTime', JSON.stringify(action.time)); @@ -201,109 +203,130 @@ const reducer = function (state, action) { //const theme = action.value === "white" ? themeWhite : themeBlack; //no need to save value in async if that's where it is coming from if (!action.fromAsync) { saveFieldToAsync('color', action.value); } - return { + newState = { ...state, //theme, themeStr: action.value, }; + break; case STATE_ACTIONS.setTextLanguage: if (!action.fromAsync) { saveFieldToAsync('textLanguage', action.value); } - return { + newState = { ...state, textLanguage: action.value, }; + break; case STATE_ACTIONS.setInterfaceLanguage: if (!action.fromAsync) { saveFieldToAsync('interfaceLanguage', action.value); } if (action.value == 'hebrew') { strings.setLanguage('he'); } else if (action.value == 'english') { strings.setLanguage('en'); } - return { + newState = { ...state, interfaceLanguage: action.value, } + break; case STATE_ACTIONS.setEmailFrequency: if (!action.fromAsync) { saveFieldToAsync('emailFrequency', action.value); } - return { + newState = { ...state, emailFrequency: action.value, } + break; case STATE_ACTIONS.setReadingHistory: if (!action.fromAsync) { if (!action.value && state.readingHistory) { Sefaria.history.deleteHistory(false); } saveFieldToAsync('readingHistory', action.value); } - return { + newState = { ...state, readingHistory: action.value, } + break; case STATE_ACTIONS.setPreferredCustom: if (!action.fromAsync) { saveFieldToAsync('preferredCustom', action.value); } - return { + newState = { ...state, preferredCustom: action.value, } + break; case STATE_ACTIONS.setFontSize: if (!action.fromAsync) { saveFieldToAsync('fontSize', action.value); } - return { + newState = { ...state, fontSize: action.value, } + break; case STATE_ACTIONS.setAliyot: if (!action.fromAsync) { saveFieldToAsync('showAliyot', action.value); } - return { + newState = { ...state, showAliyot: action.value, } + break; case STATE_ACTIONS.setVocalization: if (!action.fromAsync) { saveFieldToAsync('vocalization', action.value); } - return { + newState = { ...state, vocalization: action.value, - } + } + break; case STATE_ACTIONS.toggleDebugInterruptingMessage: // toggle if you didn't pass in debug, otherwise you're initializing the value const newDebug = action.value === undefined ? (!state.debugInterruptingMessage) : action.value; if (!action.fromAsync) { saveFieldToAsync('debugInterruptingMessage', newDebug); } - return { + newState = { ...state, debugInterruptingMessage: newDebug, } + break; case STATE_ACTIONS.setBiLayout: if (!action.fromAsync) { saveFieldToAsync('biLayout', action.value); } - return { + newState = { ...state, biLayout: action.value, } + break; case STATE_ACTIONS.setIsLoggedIn: // action can be passed either object or bool const isLoggedIn = !!action.value; if (isLoggedIn && !!action.value.uid) { Sefaria._auth = action.value; } - return { + newState = { ...state, isLoggedIn, } + break; case STATE_ACTIONS.setHasDismissedSyncModal: if (!action.fromAsync) { saveFieldToAsync('hasDismissedSyncModal', action.value); } - return { + newState = { ...state, hasDismissedSyncModal: action.value, } + break; case STATE_ACTIONS.setDownloadNetworkSetting: if (!action.fromAsync) { saveFieldToAsync('downloadNetworkSetting', action.value); } - return { + newState = { ...state, downloadNetworkSetting: action.value, }; + break; case STATE_ACTIONS.setGroggerActive: if (!action.fromAsync) { saveFieldToAsync('groggerActive', action.value); } - return { + newState = { ...state, groggerActive: action.value, }; + break; default: - return state; + newState = state; + break; } + + currentGlobalState = newState; + return newState; }; +const getCurrentGlobalState = () => currentGlobalState; + const initAsyncStorage = dispatch => { // Loads data from each field in `_data` stored in Async storage into local memory for sync access. // Returns a Promise that resolves when all fields are loaded. @@ -338,4 +361,5 @@ export { GlobalStateContext, DispatchContext, getTheme, + getCurrentGlobalState, }; From 1e6121e15fadad3b018cd585818ef70027474422 Mon Sep 17 00:00:00 2001 From: Akiva Berger Date: Mon, 11 Nov 2024 09:58:00 +0200 Subject: [PATCH 2/4] chore(analytics): add Analytics.js with ga4 tracking logic --- Analytics.js | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) create mode 100644 Analytics.js diff --git a/Analytics.js b/Analytics.js new file mode 100644 index 00000000..98f46706 --- /dev/null +++ b/Analytics.js @@ -0,0 +1,21 @@ +import { getCurrentGlobalState } from './StateManager'; +import Sefaria from './sefaria'; + +const trackEvent = (eventName, eventParams = {}) => { + const globalState = getCurrentGlobalState(); + + const isLoggedIn = globalState.isLoggedIn; + + const usesOfflinePackages = globalState.offlinePackages && globalState.offlinePackages.length > 0; + + const augmentedParams = { + ...eventParams, + is_logged_in: isLoggedIn, + user_uses_offline_packages: usesOfflinePackages, + }; + + Sefaria.track.event(eventName, augmentedParams); + console.log(`Event Tracked: ${eventName}`, augmentedParams); +}; + +export { trackEvent }; From 28be1c490e49974f51c001d0f4ae83980fbb090d Mon Sep 17 00:00:00 2001 From: YishaiGlasner Date: Wed, 13 Nov 2024 14:16:19 +0200 Subject: [PATCH 3/4] fix(StateManager): declare newState; --- StateManager.js | 1 + 1 file changed, 1 insertion(+) diff --git a/StateManager.js b/StateManager.js index 2de143c3..82ebdaf4 100644 --- a/StateManager.js +++ b/StateManager.js @@ -198,6 +198,7 @@ const reducer = function (state, action) { if (UPDATE_SETTINGS_ACTIONS[action.type] && !action.fromAsync) { AsyncStorage.setItem('lastSettingsUpdateTime', JSON.stringify(action.time)); } + let newState; switch (action.type) { case STATE_ACTIONS.setTheme: //const theme = action.value === "white" ? themeWhite : themeBlack; From 2c63440560ac87c757af83c24a93efe91953d85e Mon Sep 17 00:00:00 2001 From: YishaiGlasner Date: Wed, 13 Nov 2024 14:24:42 +0200 Subject: [PATCH 4/4] refactor(track): sefaria.track calls trackEvent which doing the analytics().logEvent after adding information from state. --- Analytics.js | 4 ++-- sefaria.js | 3 ++- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/Analytics.js b/Analytics.js index 98f46706..9624fb00 100644 --- a/Analytics.js +++ b/Analytics.js @@ -1,5 +1,5 @@ import { getCurrentGlobalState } from './StateManager'; -import Sefaria from './sefaria'; +import analytics from "@react-native-firebase/analytics"; const trackEvent = (eventName, eventParams = {}) => { const globalState = getCurrentGlobalState(); @@ -14,7 +14,7 @@ const trackEvent = (eventName, eventParams = {}) => { user_uses_offline_packages: usesOfflinePackages, }; - Sefaria.track.event(eventName, augmentedParams); + analytics().logEvent(eventName, augmentedParams); console.log(`Event Tracked: ${eventName}`, augmentedParams); }; diff --git a/sefaria.js b/sefaria.js index 66147709..049f7df2 100644 --- a/sefaria.js +++ b/sefaria.js @@ -23,6 +23,7 @@ import { import * as FileSystem from 'expo-file-system'; import { Topic } from './Topic'; import {openFileInSources} from "./offline"; +import {trackEvent} from "./Analytics"; Sefaria = { @@ -1022,7 +1023,7 @@ Sefaria = { }, event: function(event, params) { - analytics().logEvent(event, params) + trackEvent(event, params); }, pageview: function(page, customDimensions, contentGroups) {