@applicaster/zapp-react-native-utils 13.0.16-rc.0 → 13.0.17-alpha.1002505828

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.
@@ -23,19 +23,6 @@ export const useAccessibilityManager = (
23
23
  }
24
24
  }, [pluginConfiguration, accessibilityManager]);
25
25
 
26
- useEffect(() => {
27
- const subscription = accessibilityManager.getStateAsObservable().subscribe({
28
- next: () => {
29
- // TODO: handle accessibility states
30
- // screenReaderEnabled: false
31
- // reduceMotionEnabled: false
32
- // boldTextEnabled: false
33
- },
34
- });
35
-
36
- return () => subscription.unsubscribe();
37
- }, [accessibilityManager]);
38
-
39
26
  return accessibilityManager;
40
27
  };
41
28
 
@@ -72,3 +59,23 @@ export const useAnnouncementActive = (
72
59
 
73
60
  return isActive;
74
61
  };
62
+
63
+ export const useAccessibilityState = (
64
+ pluginConfiguration: Record<string, any> = {}
65
+ ) => {
66
+ const accessibilityManager = useAccessibilityManager(pluginConfiguration);
67
+
68
+ const [state, setState] = useState<AccessibilityState>(
69
+ accessibilityManager.getState()
70
+ );
71
+
72
+ useEffect(() => {
73
+ const subscription = accessibilityManager
74
+ .getStateAsObservable()
75
+ .subscribe(setState);
76
+
77
+ return () => subscription.unsubscribe();
78
+ }, [accessibilityManager]);
79
+
80
+ return state;
81
+ };
@@ -36,7 +36,20 @@ export class AccessibilityManager {
36
36
  false
37
37
  );
38
38
 
39
- private constructor() {}
39
+ private constructor() {
40
+ this.ttsManager
41
+ .getScreenReaderEnabledAsObservable()
42
+ .subscribe((enabled) => {
43
+ const state = this.state$.getValue();
44
+
45
+ if (state.screenReaderEnabled !== enabled) {
46
+ this.state$.next({
47
+ ...state,
48
+ screenReaderEnabled: enabled,
49
+ });
50
+ }
51
+ });
52
+ }
40
53
 
41
54
  public static getInstance(): AccessibilityManager {
42
55
  if (!AccessibilityManager._instance) {
@@ -92,8 +105,15 @@ export class AccessibilityManager {
92
105
  /**
93
106
  * Adds a heading to the queue, headings will be read before the next text
94
107
  * Each heading will be read once and removed from the queue
108
+ * Does nothing if screen reader is not enabled
95
109
  */
96
110
  public addHeading(heading: string) {
111
+ const state = this.state$.getValue();
112
+
113
+ if (!state.screenReaderEnabled) {
114
+ return;
115
+ }
116
+
97
117
  if (!this.pendingFocusId) {
98
118
  this.pendingFocusId = Date.now().toString();
99
119
  }
@@ -108,6 +128,7 @@ export class AccessibilityManager {
108
128
  *
109
129
  * Implements a delay mechanism to reduce noise during rapid navigation.
110
130
  * Only the most recent announcement will be read after the delay period.
131
+ * Does nothing if screen reader is not enabled
111
132
  */
112
133
  public readText({
113
134
  text,
@@ -116,6 +137,12 @@ export class AccessibilityManager {
116
137
  text: string;
117
138
  keyOfLocalizedText?: string;
118
139
  }) {
140
+ const state = this.state$.getValue();
141
+
142
+ if (!state.screenReaderEnabled) {
143
+ return;
144
+ }
145
+
119
146
  let textToRead = text;
120
147
 
121
148
  if (keyOfLocalizedText) {
@@ -12,13 +12,44 @@ export function calculateReadingTime(
12
12
  minimumPause: number = 500,
13
13
  announcementDelay: number = 700
14
14
  ): number {
15
- const words = text
16
- .trim()
15
+ const trimmed = text.trim();
16
+
17
+ // Count words (split on whitespace and punctuation, keep alnum boundaries)
18
+ const words = trimmed
17
19
  .split(/(?<=\d)(?=[a-zA-Z])|(?<=[a-zA-Z])(?=\d)|[^\w\s]+|\s+/)
18
20
  .filter(Boolean).length;
19
21
 
20
- return (
21
- Math.max(minimumPause, (words / wordsPerMinute) * 60 * 1000) +
22
- announcementDelay
22
+ // Count spaces - multiple consecutive spaces add extra pause time
23
+ const spaceMatches: string[] = trimmed.match(/\s+/g) || [];
24
+
25
+ const totalSpaces = spaceMatches.reduce(
26
+ (sum: number, match: string) => sum + match.length,
27
+ 0
23
28
  );
29
+
30
+ const extraSpaces = Math.max(0, totalSpaces - (words - 1)); // words-1 is normal spacing
31
+
32
+ // Heuristic: punctuation increases TTS duration beyond word-based WPM.
33
+ // Commas typically introduce short pauses, sentence terminators longer ones.
34
+ const commaCount = (trimmed.match(/,/g) || []).length;
35
+ const semicolonCount = (trimmed.match(/;/g) || []).length;
36
+ const colonCount = (trimmed.match(/:/g) || []).length;
37
+ const dashCount = (trimmed.match(/\u2013|\u2014|-/g) || []).length; // – — -
38
+ const sentenceEndCount = (trimmed.match(/[.!?](?!\d)/g) || []).length;
39
+
40
+ const commaPauseMs = 220; // short prosody pause for ","
41
+ const midPauseMs = 260; // for ";", ":", dashes
42
+ const sentenceEndPauseMs = 420; // for ".", "!", "?"
43
+ const extraSpacePauseMs = 50; // per extra space beyond normal spacing
44
+
45
+ const punctuationPause =
46
+ commaCount * commaPauseMs +
47
+ (semicolonCount + colonCount + dashCount) * midPauseMs +
48
+ sentenceEndCount * sentenceEndPauseMs +
49
+ extraSpaces * extraSpacePauseMs;
50
+
51
+ const baseByWordsMs = (words / wordsPerMinute) * 60 * 1000;
52
+ const estimatedMs = Math.max(minimumPause, baseByWordsMs + punctuationPause);
53
+
54
+ return estimatedMs + announcementDelay;
24
55
  }
@@ -170,7 +170,9 @@ export const getClosedCaptionState = () => {
170
170
  */
171
171
  export class TTSManager {
172
172
  private ttsState$ = new BehaviorSubject<boolean>(false);
173
+ private screenReaderEnabled$ = new BehaviorSubject<boolean>(false);
173
174
  private static ttsManagerInstance: TTSManager;
175
+ private samsungListenerId: number | null = null;
174
176
 
175
177
  private constructor() {
176
178
  this.initialize();
@@ -185,23 +187,116 @@ export class TTSManager {
185
187
  }
186
188
 
187
189
  async initialize() {
188
- if (!isVizioPlatform()) return;
190
+ if (isVizioPlatform()) {
191
+ document.addEventListener(
192
+ "VIZIO_TTS_ENABLED",
193
+ () => {
194
+ log_debug("Vizio screen reader enabled");
195
+ this.screenReaderEnabled$.next(true);
196
+ },
197
+ false
198
+ );
189
199
 
190
- document.addEventListener(
191
- "VIZIO_TTS_ENABLED",
192
- () => {
193
- this.ttsState$.next(true);
194
- },
195
- false
196
- );
200
+ document.addEventListener(
201
+ "VIZIO_TTS_DISABLED",
202
+ () => {
203
+ log_debug("Vizio screen reader disabled");
204
+ this.screenReaderEnabled$.next(false);
205
+ },
206
+ false
207
+ );
208
+ }
197
209
 
198
- document.addEventListener(
199
- "VIZIO_TTS_DISABLED",
200
- () => {
201
- this.ttsState$.next(false);
202
- },
203
- false
204
- );
210
+ if (isLgPlatform() && window.webOS?.service) {
211
+ try {
212
+ // https://webostv.developer.lge.com/develop/references/settings-service
213
+ window.webOS.service.request("luna://com.webos.settingsservice", {
214
+ method: "getSystemSettings",
215
+ parameters: {
216
+ category: "option",
217
+ keys: ["audioGuidance"],
218
+ subscribe: true, // Request a subscription to changes
219
+ },
220
+ onSuccess: (response: any) => {
221
+ const isEnabled = response?.settings?.audioGuidance === "on";
222
+
223
+ log_debug("LG Audio Guidance status changed", {
224
+ isEnabled,
225
+ response,
226
+ });
227
+
228
+ this.screenReaderEnabled$.next(isEnabled);
229
+ },
230
+ onFailure: (error: any) => {
231
+ log_debug("webOS settings subscription failed", { error });
232
+ this.screenReaderEnabled$.next(false);
233
+ },
234
+ });
235
+ } catch (error) {
236
+ log_debug("webOS settings service request error", { error });
237
+ // Fallback to false if the service is not available
238
+ this.screenReaderEnabled$.next(false);
239
+ }
240
+ }
241
+
242
+ if (isSamsungPlatform() && typeof window.webapis !== "undefined") {
243
+ try {
244
+ if (
245
+ window.webapis?.tvinfo &&
246
+ typeof window.webapis.tvinfo.getMenuValue === "function" &&
247
+ typeof window.webapis.tvinfo.addCaptionChangeListener === "function"
248
+ ) {
249
+ // Get initial Voice Guide status
250
+ const initialStatus = window.webapis.tvinfo.getMenuValue(
251
+ window.webapis.tvinfo.TvInfoMenuKey.VOICE_GUIDE_KEY
252
+ );
253
+
254
+ const isEnabled =
255
+ initialStatus === window.webapis.tvinfo.TvInfoMenuValue.ON;
256
+
257
+ log_debug("Samsung Voice Guide initial status", {
258
+ isEnabled,
259
+ initialStatus,
260
+ });
261
+
262
+ this.screenReaderEnabled$.next(isEnabled);
263
+
264
+ // Listen for Voice Guide status changes
265
+ const onChange = () => {
266
+ const currentStatus = window.webapis.tvinfo.getMenuValue(
267
+ window.webapis.tvinfo.TvInfoMenuKey.VOICE_GUIDE_KEY
268
+ );
269
+
270
+ const enabled =
271
+ currentStatus === window.webapis.tvinfo.TvInfoMenuValue.ON;
272
+
273
+ log_debug("Samsung Voice Guide status changed", {
274
+ enabled,
275
+ currentStatus,
276
+ });
277
+
278
+ this.screenReaderEnabled$.next(enabled);
279
+ };
280
+
281
+ this.samsungListenerId =
282
+ window.webapis.tvinfo.addCaptionChangeListener(
283
+ window.webapis.tvinfo.TvInfoMenuKey.VOICE_GUIDE_KEY,
284
+ onChange
285
+ );
286
+
287
+ log_debug("Samsung Voice Guide listener registered", {
288
+ listenerId: this.samsungListenerId,
289
+ });
290
+ } else {
291
+ log_debug("Samsung TvInfo API not available");
292
+ this.screenReaderEnabled$.next(false);
293
+ }
294
+ } catch (error) {
295
+ log_debug("Samsung Voice Guide listener error", { error });
296
+ // Fallback to false if the service is not available
297
+ this.screenReaderEnabled$.next(false);
298
+ }
299
+ }
205
300
  }
206
301
 
207
302
  getCurrentState(): boolean {
@@ -212,6 +307,10 @@ export class TTSManager {
212
307
  return this.ttsState$.asObservable();
213
308
  }
214
309
 
310
+ getScreenReaderEnabledAsObservable() {
311
+ return this.screenReaderEnabled$.asObservable();
312
+ }
313
+
215
314
  readText(text: string) {
216
315
  this.ttsState$.next(true);
217
316
 
@@ -5,6 +5,7 @@ import { createLogger, utilsLogger } from "../../../logger";
5
5
  import { appStore } from "@applicaster/zapp-react-native-redux/AppStore";
6
6
  import {
7
7
  findPluginByIdentifier,
8
+ loadFeedEntry,
8
9
  loadFeedAndPrefetchThumbnailImage,
9
10
  parseTimeToSeconds,
10
11
  retrieveFeedUrl,
@@ -41,6 +42,19 @@ export type ChapterMarkerEvent = {
41
42
  actions: ActionChapter[];
42
43
  };
43
44
 
45
+ export type LiveMetadataEvent = {
46
+ programId: string;
47
+ assetId: string;
48
+ title: string;
49
+ programStartTime: string;
50
+ programEndTime: string;
51
+ };
52
+
53
+ export type LiveMetadataConfig = {
54
+ updateUrl: string;
55
+ updateInterval: number;
56
+ };
57
+
44
58
  export type TitleSummaryEvent = {
45
59
  title: string | number;
46
60
  summary: string | number;
@@ -65,11 +79,13 @@ export type PlayNextState = PlayNextConfig & {
65
79
  export class OverlaysObserver {
66
80
  readonly chapterSubject: BehaviorSubject<ChapterMarkerEvent>;
67
81
  private playNextSubject: BehaviorSubject<PlayNextState>;
82
+ private liveMetadataSubject: BehaviorSubject<LiveMetadataEvent>;
68
83
  private titleSummarySubject: BehaviorSubject<TitleSummaryEvent>;
69
84
  private feedUrl: string;
70
85
  private reloadData: () => void;
71
86
  private updateTitleAndDescription: (data: any) => void;
72
87
  private feedDataInterval: any;
88
+ private liveMetadataUpdateInterval: any;
73
89
  private releasePlayerObserver?: () => void;
74
90
  readonly entry: ZappEntry;
75
91
  private chapterMarkerEvents: ChapterMarkerEvent[];
@@ -81,13 +97,17 @@ export class OverlaysObserver {
81
97
  this.chapterSubject = new BehaviorSubject(null);
82
98
  this.playNextSubject = new BehaviorSubject(null);
83
99
 
100
+ this.player = player;
101
+ this.entry = player.getEntry();
102
+
84
103
  this.titleSummarySubject = new BehaviorSubject<TitleSummaryEvent>({
85
- title: player.getEntry()?.title || "",
86
- summary: player.getEntry()?.summary || "",
104
+ title: this.entry?.title || "",
105
+ summary: this.entry?.summary || "",
87
106
  });
88
107
 
89
- this.entry = player.getEntry();
90
- this.player = player;
108
+ this.liveMetadataSubject = new BehaviorSubject<LiveMetadataEvent>(
109
+ this.entry?.extensions?.liveMetadata || null
110
+ );
91
111
 
92
112
  this.chapterMarkerEvents = this.prepareChapterMarkers();
93
113
  this.releasePlayerObserver = this.subscribeToPlayerEvents();
@@ -95,7 +115,9 @@ export class OverlaysObserver {
95
115
  this.reloadData = () => {};
96
116
  this.updateTitleAndDescription = () => {};
97
117
  this.feedDataInterval = null;
118
+ this.liveMetadataUpdateInterval = null;
98
119
  void this.preparePlayNext();
120
+ void this.prepareLiveMetadata();
99
121
  }
100
122
 
101
123
  private setupFeedDataInterval(interval: number) {
@@ -113,6 +135,13 @@ export class OverlaysObserver {
113
135
  }
114
136
  }
115
137
 
138
+ public clearLiveMetadataUpdateInterval() {
139
+ if (this.liveMetadataUpdateInterval) {
140
+ clearInterval(this.liveMetadataUpdateInterval);
141
+ this.liveMetadataUpdateInterval = null;
142
+ }
143
+ }
144
+
116
145
  public setFeedDataHandlers(
117
146
  feedUrl: string,
118
147
  reloadData: () => void,
@@ -212,6 +241,48 @@ export class OverlaysObserver {
212
241
  }
213
242
  };
214
243
 
244
+ prepareLiveMetadata = async () => {
245
+ if (!this.player?.isLive?.()) {
246
+ log_debug("prepareLiveMetadata: Player is not live. Skipping...");
247
+
248
+ return;
249
+ }
250
+
251
+ const config: LiveMetadataConfig =
252
+ this.entry?.extensions?.liveMetadataConfig;
253
+
254
+ if (!config?.updateUrl || !config?.updateInterval) {
255
+ log_debug(
256
+ "prepareLiveMetadata: Live metadata configuration is not available. Skipping...",
257
+ { config }
258
+ );
259
+
260
+ return;
261
+ }
262
+
263
+ const reloadData = async () => {
264
+ try {
265
+ const entry = await loadFeedEntry(config.updateUrl, this.entry);
266
+
267
+ this.onLiveMetadataUpdated(entry?.extensions?.liveMetadata || null);
268
+ } catch (error) {
269
+ log_error("prepareLiveMetadata: Metadata fetching failed", {
270
+ error,
271
+ config,
272
+ });
273
+ }
274
+ };
275
+
276
+ log_debug("prepareLiveMetadata: Setting up live metadata observer update", {
277
+ config,
278
+ });
279
+
280
+ const interval = Number(config?.updateInterval) || 60;
281
+
282
+ this.clearLiveMetadataUpdateInterval();
283
+ this.liveMetadataUpdateInterval = setInterval(reloadData, interval * 1000);
284
+ };
285
+
215
286
  // TODO: Hack for video end, will be replaced with playlist prev/next in the future
216
287
  getPlayNextEntry = () =>
217
288
  !this.isCanceledByUser ? this.playNextConfig?.entry : null;
@@ -332,9 +403,11 @@ export class OverlaysObserver {
332
403
  onPlayerClose = () => {
333
404
  this.chapterSubject.complete();
334
405
  this.playNextSubject.complete();
406
+ this.liveMetadataSubject.complete();
335
407
  this.titleSummarySubject.complete();
336
408
  this.releasePlayerObserver?.();
337
409
  this.clearFeedDataInterval();
410
+ this.clearLiveMetadataUpdateInterval();
338
411
  this.releasePlayerObserver = null;
339
412
  };
340
413
 
@@ -367,4 +440,21 @@ export class OverlaysObserver {
367
440
  (prev, curr) => prev?.triggerTime === curr?.triggerTime
368
441
  )
369
442
  );
443
+
444
+ private onLiveMetadataUpdated = (liveMetadataEvent: LiveMetadataEvent) => {
445
+ this.liveMetadataSubject.next(liveMetadataEvent);
446
+ };
447
+
448
+ public getLiveMetadataObservable = (): Observable<LiveMetadataEvent> => {
449
+ return this.liveMetadataSubject.pipe(
450
+ distinctUntilChanged(
451
+ (prev, curr) =>
452
+ prev?.programId === curr?.programId && prev?.assetId === curr?.assetId
453
+ )
454
+ );
455
+ };
456
+
457
+ public getLiveMetadataValue = (): LiveMetadataEvent => {
458
+ return this.liveMetadataSubject.value;
459
+ };
370
460
  }
@@ -92,22 +92,26 @@ export const prefetchImage = (playableItem: ZappEntry, config: any = {}) => {
92
92
  }
93
93
  };
94
94
 
95
- export const loadFeedAndPrefetchThumbnailImage = async (
96
- playNextFeedUrl: string,
97
- entry: ZappEntry,
98
- playNextPlugin
99
- ) => {
95
+ /**
96
+ * Loads a feed entry from the given feed URL using the provided Zapp entry as context.
97
+ *
98
+ * @param {string} feedUrl - The URL of the feed to load the entry from.
99
+ * @param {ZappEntry} entry - The Zapp entry to use as context for the request.
100
+ * @returns {Promise<ZappEntry>} A promise that resolves to the loaded Zapp entry.
101
+ * @throws {Error} If the feed loading fails or no entry is found in the response.
102
+ */
103
+ export const loadFeedEntry = async (feedUrl: string, entry: ZappEntry) => {
100
104
  const requestBuilder = new RequestBuilder()
101
105
  .setEntryContext(entry)
102
106
  .setScreenContext({} as ZappRiver)
103
- .setUrl(playNextFeedUrl);
107
+ .setUrl(feedUrl);
104
108
 
105
109
  const responseObject = await requestBuilder.call<ZappEntry>();
106
110
  const responseHelper = new PipesClientResponseHelper(responseObject);
107
111
 
108
112
  if (responseHelper.error) {
109
113
  log_error(
110
- `loadFeedAndPrefetchThumbnailImage: loading failed with error: ${responseHelper.error.message}. Play next observer, will not be executed`,
114
+ `loadFeedEntry: loading failed with error: ${responseHelper.error.message}. Observer will not be executed`,
111
115
  {
112
116
  response: responseHelper.getLogsData(),
113
117
  }
@@ -116,29 +120,37 @@ export const loadFeedAndPrefetchThumbnailImage = async (
116
120
  throw responseHelper.error;
117
121
  } else {
118
122
  log_info(
119
- `loadFeedAndPrefetchThumbnailImage: Play next url was successfully loaded for url: ${playNextFeedUrl}. Prefetching image`,
123
+ `loadFeedEntry: Feed was successfully loaded for url: ${feedUrl}`,
120
124
  responseHelper.getLogsData()
121
125
  );
122
126
 
123
- const playNextEntry = responseHelper.responseData?.entry[0];
127
+ const entry = responseHelper.responseData?.entry[0];
124
128
 
125
- if (!playNextEntry) {
126
- log_error(
127
- "loadFeedAndPrefetchThumbnailImage: Can not retrieve play next entry, feed was loaded but no entry was found",
128
- responseHelper.getLogsData()
129
- );
129
+ if (!entry) {
130
+ const error =
131
+ "loadFeedEntry: Can not retrieve entry, feed was loaded but no entry was found";
130
132
 
131
- throw new Error(
132
- "Can not retrieve play next entry, feed was loaded but no entry was found"
133
- );
134
- }
133
+ log_error(error, responseHelper.getLogsData());
135
134
 
136
- prefetchImage(playNextEntry, playNextPlugin?.configuration);
135
+ throw new Error(error);
136
+ }
137
137
 
138
- return playNextEntry;
138
+ return entry;
139
139
  }
140
140
  };
141
141
 
142
+ export const loadFeedAndPrefetchThumbnailImage = async (
143
+ playNextFeedUrl: string,
144
+ entry: ZappEntry,
145
+ playNextPlugin
146
+ ) => {
147
+ const playNextEntry = await loadFeedEntry(playNextFeedUrl, entry);
148
+
149
+ prefetchImage(playNextEntry, playNextPlugin?.configuration);
150
+
151
+ return playNextEntry;
152
+ };
153
+
142
154
  export const findPluginByIdentifier = (
143
155
  identifier: string,
144
156
  plugins: ZappPlugin[]
@@ -505,3 +505,35 @@ export const getImageContainerMarginStyles = ({ value }: { value: any }) => {
505
505
  marginRight: value("image_margin_right"),
506
506
  };
507
507
  };
508
+
509
+ export const isTextLabel = (key: string): boolean =>
510
+ key.includes("text_label_") && key.endsWith("_switch");
511
+
512
+ export const hasTextLabelsEnabled = (
513
+ configuration: Record<string, any>
514
+ ): boolean => {
515
+ const textLabelsKeys = Object.keys(configuration).filter(isTextLabel);
516
+
517
+ const picked = textLabelsKeys.reduce(
518
+ (acc, key) => {
519
+ acc[key] = configuration[key];
520
+
521
+ return acc;
522
+ },
523
+ {} as Record<string, any>
524
+ );
525
+
526
+ const pickedValues = Object.values(picked);
527
+
528
+ return pickedValues.some((value) => {
529
+ if (typeof value === "boolean") {
530
+ return value === true;
531
+ }
532
+
533
+ if (typeof value === "string") {
534
+ return value !== "" && value.toLowerCase() !== "false";
535
+ }
536
+
537
+ return Boolean(value);
538
+ });
539
+ };
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@applicaster/zapp-react-native-utils",
3
- "version": "13.0.16-rc.0",
3
+ "version": "13.0.17-alpha.1002505828",
4
4
  "description": "Applicaster Zapp React Native utilities package",
5
5
  "main": "index.js",
6
6
  "types": "index.d.ts",
@@ -27,7 +27,7 @@
27
27
  },
28
28
  "homepage": "https://github.com/applicaster/quickbrick#readme",
29
29
  "dependencies": {
30
- "@applicaster/applicaster-types": "13.0.16-rc.0",
30
+ "@applicaster/applicaster-types": "13.0.17-alpha.1002505828",
31
31
  "buffer": "^5.2.1",
32
32
  "camelize": "^1.0.0",
33
33
  "dayjs": "^1.11.10",
@@ -1,14 +1,19 @@
1
1
  import * as React from "react";
2
2
  import { usePlayer } from "@applicaster/zapp-react-native-utils/appUtils/playerManager/usePlayer";
3
- import { useAccessibilityManager } from "@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/hooks";
3
+ import {
4
+ useAccessibilityManager,
5
+ useAccessibilityState,
6
+ } from "@applicaster/zapp-react-native-utils/appUtils/accessibilityManager/hooks";
4
7
  import { PlayerTTS } from "@applicaster/zapp-react-native-utils/playerUtils/PlayerTTS";
5
8
 
6
9
  export const usePlayerTTS = () => {
7
10
  const player = usePlayer();
8
11
  const accessibilityManager = useAccessibilityManager({});
12
+ const accessibilityState = useAccessibilityState();
13
+ const isScreenReaderEnabled = accessibilityState.screenReaderEnabled;
9
14
 
10
15
  React.useEffect(() => {
11
- if (player && accessibilityManager) {
16
+ if (player && accessibilityManager && isScreenReaderEnabled) {
12
17
  const playerTTS = new PlayerTTS(player, accessibilityManager);
13
18
  const unsubscribe = playerTTS.init();
14
19
 
@@ -17,5 +22,5 @@ export const usePlayerTTS = () => {
17
22
  playerTTS.destroy();
18
23
  };
19
24
  }
20
- }, [player, accessibilityManager]);
25
+ }, [player, accessibilityManager, isScreenReaderEnabled]);
21
26
  };