@buoy-gg/highlight-updates 3.0.1 → 3.0.2

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.
Files changed (83) hide show
  1. package/lib/commonjs/highlight-updates/HighlightUpdatesOverlay.js +285 -1
  2. package/lib/commonjs/highlight-updates/components/HighlightFilterView.js +1371 -1
  3. package/lib/commonjs/highlight-updates/components/HighlightUpdatesModal.js +564 -1
  4. package/lib/commonjs/highlight-updates/components/IdentifierBadge.js +267 -1
  5. package/lib/commonjs/highlight-updates/components/IsolatedRenderList.js +178 -1
  6. package/lib/commonjs/highlight-updates/components/ModalHeaderContent.js +309 -1
  7. package/lib/commonjs/highlight-updates/components/RenderCauseBadge.js +500 -1
  8. package/lib/commonjs/highlight-updates/components/RenderDetailView.js +803 -1
  9. package/lib/commonjs/highlight-updates/components/RenderHistoryViewer.js +894 -1
  10. package/lib/commonjs/highlight-updates/components/RenderListItem.js +220 -1
  11. package/lib/commonjs/highlight-updates/components/RendersCopySettingsView.js +562 -1
  12. package/lib/commonjs/highlight-updates/components/StatsDisplay.js +70 -1
  13. package/lib/commonjs/highlight-updates/components/index.js +97 -1
  14. package/lib/commonjs/highlight-updates/types/copySettings.js +107 -1
  15. package/lib/commonjs/highlight-updates/utils/HighlightUpdatesController.js +1819 -1
  16. package/lib/commonjs/highlight-updates/utils/PerformanceLogger.js +359 -1
  17. package/lib/commonjs/highlight-updates/utils/ProfilerInterceptor.js +371 -1
  18. package/lib/commonjs/highlight-updates/utils/RenderCauseDetector.js +1828 -1
  19. package/lib/commonjs/highlight-updates/utils/RenderTracker.js +919 -1
  20. package/lib/commonjs/highlight-updates/utils/ViewTypeMapper.js +264 -1
  21. package/lib/commonjs/highlight-updates/utils/copySettingsStorage.js +49 -1
  22. package/lib/commonjs/highlight-updates/utils/renderExportFormatter.js +58 -1
  23. package/lib/commonjs/highlight-updates/utils/rendersExportFormatter.js +485 -1
  24. package/lib/commonjs/index.js +320 -1
  25. package/lib/commonjs/preset.js +278 -1
  26. package/lib/commonjs/sync/highlightUpdatesSyncAdapter.js +83 -0
  27. package/lib/module/highlight-updates/HighlightUpdatesOverlay.js +278 -1
  28. package/lib/module/highlight-updates/components/HighlightFilterView.js +1365 -1
  29. package/lib/module/highlight-updates/components/HighlightUpdatesModal.js +558 -1
  30. package/lib/module/highlight-updates/components/IdentifierBadge.js +259 -1
  31. package/lib/module/highlight-updates/components/IsolatedRenderList.js +174 -1
  32. package/lib/module/highlight-updates/components/ModalHeaderContent.js +304 -1
  33. package/lib/module/highlight-updates/components/RenderCauseBadge.js +491 -1
  34. package/lib/module/highlight-updates/components/RenderDetailView.js +797 -1
  35. package/lib/module/highlight-updates/components/RenderHistoryViewer.js +888 -1
  36. package/lib/module/highlight-updates/components/RenderListItem.js +215 -1
  37. package/lib/module/highlight-updates/components/RendersCopySettingsView.js +558 -1
  38. package/lib/module/highlight-updates/components/StatsDisplay.js +67 -1
  39. package/lib/module/highlight-updates/components/index.js +16 -1
  40. package/lib/module/highlight-updates/types/copySettings.js +102 -1
  41. package/lib/module/highlight-updates/utils/HighlightUpdatesController.js +1815 -1
  42. package/lib/module/highlight-updates/utils/PerformanceLogger.js +353 -1
  43. package/lib/module/highlight-updates/utils/ProfilerInterceptor.js +358 -1
  44. package/lib/module/highlight-updates/utils/RenderCauseDetector.js +1818 -1
  45. package/lib/module/highlight-updates/utils/RenderTracker.js +916 -1
  46. package/lib/module/highlight-updates/utils/ViewTypeMapper.js +255 -1
  47. package/lib/module/highlight-updates/utils/copySettingsStorage.js +43 -1
  48. package/lib/module/highlight-updates/utils/renderExportFormatter.js +54 -1
  49. package/lib/module/highlight-updates/utils/rendersExportFormatter.js +478 -1
  50. package/lib/module/index.js +74 -1
  51. package/lib/module/preset.js +272 -1
  52. package/lib/module/sync/highlightUpdatesSyncAdapter.js +78 -0
  53. package/lib/typescript/highlight-updates/HighlightUpdatesOverlay.d.ts.map +1 -0
  54. package/lib/typescript/highlight-updates/components/HighlightFilterView.d.ts.map +1 -0
  55. package/lib/typescript/highlight-updates/components/HighlightUpdatesModal.d.ts.map +1 -0
  56. package/lib/typescript/highlight-updates/components/IdentifierBadge.d.ts.map +1 -0
  57. package/lib/typescript/highlight-updates/components/IsolatedRenderList.d.ts.map +1 -0
  58. package/lib/typescript/highlight-updates/components/ModalHeaderContent.d.ts.map +1 -0
  59. package/lib/typescript/highlight-updates/components/RenderCauseBadge.d.ts.map +1 -0
  60. package/lib/typescript/highlight-updates/components/RenderDetailView.d.ts.map +1 -0
  61. package/lib/typescript/highlight-updates/components/RenderHistoryViewer.d.ts.map +1 -0
  62. package/lib/typescript/highlight-updates/components/RenderListItem.d.ts.map +1 -0
  63. package/lib/typescript/highlight-updates/components/RendersCopySettingsView.d.ts.map +1 -0
  64. package/lib/typescript/highlight-updates/components/StatsDisplay.d.ts.map +1 -0
  65. package/lib/typescript/highlight-updates/components/index.d.ts.map +1 -0
  66. package/lib/typescript/highlight-updates/types/copySettings.d.ts.map +1 -0
  67. package/lib/typescript/highlight-updates/utils/HighlightUpdatesController.d.ts +90 -0
  68. package/lib/typescript/highlight-updates/utils/HighlightUpdatesController.d.ts.map +1 -0
  69. package/lib/typescript/highlight-updates/utils/PerformanceLogger.d.ts.map +1 -0
  70. package/lib/typescript/highlight-updates/utils/ProfilerInterceptor.d.ts.map +1 -0
  71. package/lib/typescript/highlight-updates/utils/RenderCauseDetector.d.ts.map +1 -0
  72. package/lib/typescript/highlight-updates/utils/RenderTracker.d.ts +10 -0
  73. package/lib/typescript/highlight-updates/utils/RenderTracker.d.ts.map +1 -0
  74. package/lib/typescript/highlight-updates/utils/ViewTypeMapper.d.ts.map +1 -0
  75. package/lib/typescript/highlight-updates/utils/copySettingsStorage.d.ts.map +1 -0
  76. package/lib/typescript/highlight-updates/utils/renderExportFormatter.d.ts.map +1 -0
  77. package/lib/typescript/highlight-updates/utils/rendersExportFormatter.d.ts.map +1 -0
  78. package/lib/typescript/index.d.ts +1 -0
  79. package/lib/typescript/index.d.ts.map +1 -0
  80. package/lib/typescript/preset.d.ts.map +1 -0
  81. package/lib/typescript/sync/highlightUpdatesSyncAdapter.d.ts +36 -0
  82. package/lib/typescript/sync/highlightUpdatesSyncAdapter.d.ts.map +1 -0
  83. package/package.json +7 -7
@@ -1 +1,916 @@
1
- "use strict";import{getComponentDisplayName}from"./ViewTypeMapper";import{PerformanceLogger}from"./PerformanceLogger";const MAX_TRACKED_COMPONENTS=200,DEFAULT_BATCH_SIZE=150;class RenderTrackerSingleton{renders=new Map;listeners=new Set;stateListeners=new Set;settingsListeners=new Set;filterListeners=new Set;renderEventCallbacks=new Set;isTracking=!1;isPaused=!1;settings={batchSize:150,showRenderCount:!0,performanceLogging:!1,trackRenderCauses:!1,enableRenderHistory:!0,maxRenderHistoryPerComponent:20,capturePropsOnRender:!1,captureStateOnRender:!1,debugLogLevel:"off",excludeDevTools:!0};isBatchMode=!1;batchDirty=!1;filters={includeTestID:new Set,includeNativeID:new Set,includeViewType:new Set,includeComponent:new Set,excludeTestID:new Set,excludeNativeID:new Set,excludeViewType:new Set,excludeComponent:new Set,includePatterns:[],excludePatterns:[]};trackRender(e){if(this.isPaused)return;const t=String(e.nativeTag),s=Date.now(),i=this.renders.get(t);if(i){if(i.renderCount=e.count,i.lastRenderTime=s,i.color=e.color,e.measurements&&(i.measurements=e.measurements),e.testID&&!i.testID&&(i.testID=e.testID),e.nativeID&&!i.nativeID&&(i.nativeID=e.nativeID),e.accessibilityLabel&&!i.accessibilityLabel&&(i.accessibilityLabel=e.accessibilityLabel),e.componentName&&!i.componentName&&(i.componentName=e.componentName),e.renderCause&&(i.lastRenderCause=e.renderCause),this.settings.enableRenderHistory){const t=e.renderCause||{type:"unknown",timestamp:s};this.addRenderEvent(i,t,e.capturedProps,e.capturedState)}}else{const i={id:t,nativeTag:e.nativeTag,viewType:e.viewType,displayName:getComponentDisplayName(e.viewType),testID:e.testID,nativeID:e.nativeID,accessibilityLabel:e.accessibilityLabel,componentName:e.componentName,renderCount:e.count,firstRenderTime:s,lastRenderTime:s,measurements:e.measurements,color:e.color,lastRenderCause:e.renderCause};if(this.settings.enableRenderHistory){i.renderHistory=[];const t=e.renderCause||{type:1===e.count?"mount":"unknown",timestamp:s};this.addRenderEvent(i,t,e.capturedProps,e.capturedState)}if(this.renders.set(t,i),this.renders.size>200){const e=Array.from(this.renders.values()).sort((e,t)=>e.lastRenderTime-t.lastRenderTime).slice(0,this.renders.size-200);for(const t of e)this.renders.delete(t.id)}}this.isBatchMode?this.batchDirty=!0:this.notifyListeners()}addRenderEvent(e,t,s,i){e.renderHistory||(e.renderHistory=[]);const n={id:`${e.nativeTag}-${t.timestamp}`,timestamp:t.timestamp,cause:t,renderNumber:e.renderCount,capturedProps:this.settings.capturePropsOnRender?s:void 0,capturedState:this.settings.captureStateOnRender?i:void 0};e.renderHistory.push(n);const r=Math.max(5,Math.min(50,this.settings.maxRenderHistoryPerComponent));e.renderHistory.length>r&&(e.renderHistory=e.renderHistory.slice(-r)),this.notifyRenderEventCallbacks(n,e)}notifyRenderEventCallbacks(e,t){this.renderEventCallbacks.forEach(s=>{try{s(e,t)}catch{}})}emitRenderEvent(e){if(0===this.renderEventCallbacks.size)return;const t=Date.now(),s=e.renderCause||{type:"unknown",timestamp:t},i={id:String(e.nativeTag),nativeTag:e.nativeTag,viewType:e.viewType,displayName:getComponentDisplayName(e.viewType),testID:e.testID,nativeID:e.nativeID,accessibilityLabel:e.accessibilityLabel,componentName:e.componentName,renderCount:e.count,firstRenderTime:t,lastRenderTime:t,measurements:e.measurements,color:e.color,lastRenderCause:s,renderHistory:[]},n={id:`${e.nativeTag}-${t}-${e.count}`,timestamp:t,renderNumber:e.count,cause:s,capturedProps:void 0,capturedState:void 0};this.notifyRenderEventCallbacks(n,i)}startBatch(){this.isBatchMode=!0,this.batchDirty=!1}endBatch(){this.isBatchMode=!1,this.batchDirty&&(this.batchDirty=!1,this.notifyListeners())}getRenders(){return Array.from(this.renders.values()).map(e=>({...e}))}passesFilters(e){if(this.filters.includePatterns.length>0&&!this.matchesAnyPatternForInfo(e,this.filters.includePatterns))return!1;if(this.filters.excludePatterns.length>0&&this.matchesAnyPatternForInfo(e,this.filters.excludePatterns))return!1;if(this.filters.includeViewType.size>0){const t=e.viewType||"",s=e.displayName||"";if(!this.matchesPattern(t,this.filters.includeViewType)&&!this.matchesPattern(s,this.filters.includeViewType))return!1}if(this.filters.includeTestID.size>0&&(!e.testID||!this.matchesPattern(e.testID,this.filters.includeTestID)))return!1;if(this.filters.includeNativeID.size>0&&(!e.nativeID||!this.matchesPattern(e.nativeID,this.filters.includeNativeID)))return!1;if(this.filters.includeComponent.size>0&&(!e.componentName||!this.matchesPattern(e.componentName,this.filters.includeComponent)))return!1;if(this.filters.excludeViewType.size>0){const t=e.viewType||"",s=e.displayName||"";if(this.matchesPattern(t,this.filters.excludeViewType)||this.matchesPattern(s,this.filters.excludeViewType))return!1}return!(this.filters.excludeTestID.size>0&&e.testID&&this.matchesPattern(e.testID,this.filters.excludeTestID)||this.filters.excludeNativeID.size>0&&e.nativeID&&this.matchesPattern(e.nativeID,this.filters.excludeNativeID)||this.filters.excludeComponent.size>0&&e.componentName&&this.matchesPattern(e.componentName,this.filters.excludeComponent))}matchesAnyPatternForInfo(e,t){for(const s of t){const t=s.value.toLowerCase();switch(s.type){case"any":if(e.viewType.toLowerCase().includes(t)||e.displayName?.toLowerCase().includes(t)||e.testID?.toLowerCase().includes(t)||e.nativeID?.toLowerCase().includes(t)||e.componentName?.toLowerCase().includes(t)||e.accessibilityLabel?.toLowerCase().includes(t))return!0;break;case"viewType":if(e.viewType.toLowerCase().includes(t)||e.displayName?.toLowerCase().includes(t))return!0;break;case"testID":if(e.testID?.toLowerCase().includes(t))return!0;break;case"nativeID":if(e.nativeID?.toLowerCase().includes(t))return!0;break;case"component":if(e.componentName?.toLowerCase().includes(t))return!0;break;case"accessibilityLabel":if(e.accessibilityLabel?.toLowerCase().includes(t))return!0}}return!1}hasActiveFilters(){return this.filters.includePatterns.length>0||this.filters.excludePatterns.length>0||this.filters.includeViewType.size>0||this.filters.includeTestID.size>0||this.filters.includeNativeID.size>0||this.filters.includeComponent.size>0||this.filters.excludeViewType.size>0||this.filters.excludeTestID.size>0||this.filters.excludeNativeID.size>0||this.filters.excludeComponent.size>0||void 0!==this.filters.minRenderCount||void 0!==this.filters.maxRenderCount}getFilteredRenders(e=""){let t=this.getRenders();if(e.trim()){const s=e.toLowerCase();t=t.filter(e=>e.viewType.toLowerCase().includes(s)||e.displayName.toLowerCase().includes(s)||e.testID?.toLowerCase().includes(s)||e.nativeID?.toLowerCase().includes(s)||e.accessibilityLabel?.toLowerCase().includes(s)||e.componentName?.toLowerCase().includes(s)||String(e.nativeTag).includes(s))}return t=t.filter(e=>this.passesFilters(e)),void 0!==this.filters.minRenderCount&&(t=t.filter(e=>e.renderCount>=this.filters.minRenderCount)),void 0!==this.filters.maxRenderCount&&(t=t.filter(e=>e.renderCount<=this.filters.maxRenderCount)),t.sort((e,t)=>t.lastRenderTime-e.lastRenderTime)}matchesPattern(e,t){const s=e.toLowerCase();for(const e of t){const t=e.toLowerCase();if(s.includes(t))return!0}return!1}getRender(e){return this.renders.get(e)}clear(){this.renders.clear(),this.notifyListeners()}resetRenderCount(e){const t=this.renders.get(e);t&&(t.renderCount=0,this.notifyListeners())}start(){this.isTracking=!0,this.isPaused=!1,this.notifyStateListeners()}stop(){this.isTracking=!1,this.isPaused=!1,this.notifyStateListeners()}pause(){this.isPaused=!0,this.notifyStateListeners()}resume(){this.isPaused=!1,this.notifyStateListeners()}togglePause(){this.isPaused=!this.isPaused,this.notifyStateListeners()}getState(){return{isTracking:this.isTracking,isPaused:this.isPaused}}getFilters(){return this.filters}setFilters(e){this.filters={...this.filters,...e},this.notifyListeners(),this.notifyFilterListeners()}addIncludePattern(e,t){const s=`include${e.charAt(0).toUpperCase()+e.slice(1)}`;this.filters[s].add(t),this.notifyListeners()}removeIncludePattern(e,t){const s=`include${e.charAt(0).toUpperCase()+e.slice(1)}`;this.filters[s].delete(t),this.notifyListeners()}addExcludePattern(e,t){const s=`exclude${e.charAt(0).toUpperCase()+e.slice(1)}`;this.filters[s].add(t),this.notifyListeners()}removeExcludePattern(e,t){const s=`exclude${e.charAt(0).toUpperCase()+e.slice(1)}`;this.filters[s].delete(t),this.notifyListeners()}clearFilters(){this.filters={includeTestID:new Set,includeNativeID:new Set,includeViewType:new Set,includeComponent:new Set,excludeTestID:new Set,excludeNativeID:new Set,excludeViewType:new Set,excludeComponent:new Set,includePatterns:[],excludePatterns:[],minRenderCount:void 0,maxRenderCount:void 0},this.notifyListeners()}getAvailableProps(){const e=new Set,t=new Set,s=new Set,i=new Set,n=new Set;for(const r of this.renders.values())r.viewType&&e.add(r.viewType),r.testID&&t.add(r.testID),r.nativeID&&s.add(r.nativeID),r.componentName&&i.add(r.componentName),r.accessibilityLabel&&n.add(r.accessibilityLabel);return{viewTypes:Array.from(e).sort(),testIDs:Array.from(t).sort(),nativeIDs:Array.from(s).sort(),componentNames:Array.from(i).sort(),accessibilityLabels:Array.from(n).sort()}}getStats(){let e=0;for(const t of this.renders.values())e+=t.renderCount;return{totalComponents:this.renders.size,totalRenders:e}}subscribe(e){return this.listeners.add(e),e(this.getRenders()),()=>{this.listeners.delete(e)}}subscribeToState(e){return this.stateListeners.add(e),e(this.getState()),()=>{this.stateListeners.delete(e)}}subscribeToSettings(e){return this.settingsListeners.add(e),e(this.settings),()=>{this.settingsListeners.delete(e)}}subscribeToFilters(e){return this.filterListeners.add(e),e(this.filters),()=>{this.filterListeners.delete(e)}}onRenderEvent(e){return this.renderEventCallbacks.add(e),()=>{this.renderEventCallbacks.delete(e)}}getRenderEventCallbackCount(){return this.renderEventCallbacks.size}shouldShowRender(e){const t=this.filters;return!(t.includePatterns.length>0&&!this.matchesAnyPatternForInfo(e,t.includePatterns))&&(!(t.excludePatterns.length>0&&this.matchesAnyPatternForInfo(e,t.excludePatterns))&&(!((t.includeViewType.size>0||t.includeTestID.size>0||t.includeNativeID.size>0||t.includeComponent.size>0)&&!(t.includeViewType.size>0&&t.includeViewType.has(e.viewType)||t.includeTestID.size>0&&e.testID&&t.includeTestID.has(e.testID)||t.includeNativeID.size>0&&e.nativeID&&t.includeNativeID.has(e.nativeID)||t.includeComponent.size>0&&e.componentName&&t.includeComponent.has(e.componentName)))&&!(t.excludeViewType.has(e.viewType)||e.testID&&t.excludeTestID.has(e.testID)||e.nativeID&&t.excludeNativeID.has(e.nativeID)||e.componentName&&t.excludeComponent.has(e.componentName))))}getSettings(){return{...this.settings}}setSettings(e){void 0!==e.batchSize&&(e.batchSize=Math.max(10,Math.min(500,e.batchSize))),void 0!==e.maxRenderHistoryPerComponent&&(e.maxRenderHistoryPerComponent=Math.max(5,Math.min(50,e.maxRenderHistoryPerComponent))),e.enableRenderHistory&&!this.settings.trackRenderCauses&&(e.trackRenderCauses=!0),!1===e.trackRenderCauses&&this.settings.enableRenderHistory&&(e.enableRenderHistory=!1),this.settings={...this.settings,...e},void 0!==e.performanceLogging&&PerformanceLogger.setEnabled(e.performanceLogging),this.notifySettingsListeners()}clearAllRenderHistory(){for(const e of this.renders.values())e.renderHistory=[];this.notifyListeners()}clearRenderHistory(e){const t=this.renders.get(e);t&&(t.renderHistory=[],this.notifyListeners())}getRenderHistoryStats(){let e=0,t=0;for(const s of this.renders.values())s.renderHistory&&s.renderHistory.length>0&&(e+=s.renderHistory.length,t++);return{totalEvents:e,componentsWithHistory:t,averageEventsPerComponent:t>0?e/t:0}}getBatchSize(){return this.settings.batchSize}setBatchSize(e){this.setSettings({batchSize:e})}notifyListeners(){const e=this.getRenders();for(const t of this.listeners)try{t(e)}catch(e){console.error("[RenderTracker] Error in listener:",e)}}notifyStateListeners(){const e=this.getState();for(const t of this.stateListeners)try{t(e)}catch(e){console.error("[RenderTracker] Error in state listener:",e)}}notifySettingsListeners(){const e=this.getSettings();for(const t of this.settingsListeners)try{t(e)}catch(e){console.error("[RenderTracker] Error in settings listener:",e)}}notifyFilterListeners(){const e=this.getFilters();for(const t of this.filterListeners)try{t(e)}catch(e){console.error("[RenderTracker] Error in filter listener:",e)}}}export const RenderTracker=new RenderTrackerSingleton;export default RenderTracker;
1
+ /**
2
+ * RenderTracker
3
+ *
4
+ * Singleton that tracks component render history for the Highlight Updates modal.
5
+ * Stores information about each tracked component including render counts,
6
+ * timestamps, and identifying props (testID, nativeID, etc.)
7
+ */
8
+
9
+ "use strict";
10
+
11
+ import { getComponentDisplayName } from "./ViewTypeMapper";
12
+ import { PerformanceLogger } from "./PerformanceLogger";
13
+
14
+ /**
15
+ * Debug logging levels for render cause detection.
16
+ * Controls verbosity of console output for debugging "Why Did You Render" feature.
17
+ *
18
+ * - "off": No debug logging (default, best for production)
19
+ * - "minimal": Only log state/hook value changes (e.g., "useState: 3334 → 3335")
20
+ * - "verbose": Log component info + cause + value changes
21
+ * - "all": Full fiber dump with everything (native fiber, component fiber, hooks, batch context)
22
+ */
23
+
24
+ // Render cause types - why did a component render?
25
+
26
+ // Could not determine
27
+
28
+ // Component-level cause - why did the React component re-render?
29
+
30
+ // Could not determine
31
+
32
+ /**
33
+ * Represents a change in a single hook's state
34
+ * Used to show meaningful before/after values for debugging
35
+ */
36
+
37
+ /**
38
+ * A single render event in the history
39
+ * Captures everything about one render occurrence
40
+ */
41
+
42
+ /** Callback for individual render events (for unified events integration) */
43
+
44
+ // Maximum number of tracked components to prevent memory issues
45
+ const MAX_TRACKED_COMPONENTS = 200;
46
+
47
+ // Default batch size for highlight rendering
48
+ const DEFAULT_BATCH_SIZE = 150;
49
+ class RenderTrackerSingleton {
50
+ renders = new Map();
51
+ listeners = new Set();
52
+ stateListeners = new Set();
53
+ settingsListeners = new Set();
54
+ filterListeners = new Set();
55
+ /** Callbacks for individual render events (unified events integration) */
56
+ renderEventCallbacks = new Set();
57
+ isTracking = false;
58
+ isPaused = false;
59
+ settings = {
60
+ batchSize: DEFAULT_BATCH_SIZE,
61
+ showRenderCount: true,
62
+ performanceLogging: false,
63
+ trackRenderCauses: false,
64
+ // History settings (always enabled)
65
+ enableRenderHistory: true,
66
+ maxRenderHistoryPerComponent: 20,
67
+ capturePropsOnRender: false,
68
+ captureStateOnRender: false,
69
+ // Debug settings
70
+ debugLogLevel: "off",
71
+ // Dev tools visibility - exclude by default
72
+ excludeDevTools: true
73
+ };
74
+
75
+ // Batch mode: defer notifyListeners until endBatch is called
76
+ isBatchMode = false;
77
+ batchDirty = false;
78
+ filters = {
79
+ includeTestID: new Set(),
80
+ includeNativeID: new Set(),
81
+ includeViewType: new Set(),
82
+ includeComponent: new Set(),
83
+ excludeTestID: new Set(),
84
+ excludeNativeID: new Set(),
85
+ excludeViewType: new Set(),
86
+ excludeComponent: new Set(),
87
+ includePatterns: [],
88
+ excludePatterns: []
89
+ };
90
+
91
+ /**
92
+ * Track a component render
93
+ */
94
+ trackRender(data) {
95
+ if (this.isPaused) return;
96
+ const id = String(data.nativeTag);
97
+ const now = Date.now();
98
+ const existing = this.renders.get(id);
99
+ if (existing) {
100
+ // Mutate in place to keep Map entry correct
101
+ existing.renderCount = data.count;
102
+ existing.lastRenderTime = now;
103
+ existing.color = data.color;
104
+ if (data.measurements) {
105
+ existing.measurements = data.measurements;
106
+ }
107
+ // Update props if they weren't set before
108
+ if (data.testID && !existing.testID) existing.testID = data.testID;
109
+ if (data.nativeID && !existing.nativeID) existing.nativeID = data.nativeID;
110
+ if (data.accessibilityLabel && !existing.accessibilityLabel) {
111
+ existing.accessibilityLabel = data.accessibilityLabel;
112
+ }
113
+ if (data.componentName && !existing.componentName) {
114
+ existing.componentName = data.componentName;
115
+ }
116
+ // Update render cause if provided
117
+ if (data.renderCause) {
118
+ existing.lastRenderCause = data.renderCause;
119
+ }
120
+
121
+ // Add to render history if enabled (create minimal cause if not provided)
122
+ if (this.settings.enableRenderHistory) {
123
+ const cause = data.renderCause || {
124
+ type: "unknown",
125
+ timestamp: now
126
+ };
127
+ this.addRenderEvent(existing, cause, data.capturedProps, data.capturedState);
128
+ }
129
+ } else {
130
+ // Add new render
131
+ const newRender = {
132
+ id,
133
+ nativeTag: data.nativeTag,
134
+ viewType: data.viewType,
135
+ displayName: getComponentDisplayName(data.viewType),
136
+ testID: data.testID,
137
+ nativeID: data.nativeID,
138
+ accessibilityLabel: data.accessibilityLabel,
139
+ componentName: data.componentName,
140
+ renderCount: data.count,
141
+ firstRenderTime: now,
142
+ lastRenderTime: now,
143
+ measurements: data.measurements,
144
+ color: data.color,
145
+ lastRenderCause: data.renderCause
146
+ };
147
+
148
+ // Initialize render history if enabled (create minimal cause if not provided)
149
+ if (this.settings.enableRenderHistory) {
150
+ newRender.renderHistory = [];
151
+ const cause = data.renderCause || {
152
+ type: data.count === 1 ? "mount" : "unknown",
153
+ timestamp: now
154
+ };
155
+ this.addRenderEvent(newRender, cause, data.capturedProps, data.capturedState);
156
+ }
157
+ this.renders.set(id, newRender);
158
+
159
+ // Enforce max limit - remove oldest renders
160
+ if (this.renders.size > MAX_TRACKED_COMPONENTS) {
161
+ const sorted = Array.from(this.renders.values()).sort((a, b) => a.lastRenderTime - b.lastRenderTime);
162
+ const toRemove = sorted.slice(0, this.renders.size - MAX_TRACKED_COMPONENTS);
163
+ for (const render of toRemove) {
164
+ this.renders.delete(render.id);
165
+ }
166
+ }
167
+ }
168
+
169
+ // In batch mode, defer notification until endBatch()
170
+ if (this.isBatchMode) {
171
+ this.batchDirty = true;
172
+ } else {
173
+ this.notifyListeners();
174
+ }
175
+ }
176
+
177
+ /**
178
+ * Add a render event to a component's history (circular buffer)
179
+ */
180
+ addRenderEvent(render, cause, capturedProps, capturedState) {
181
+ // Initialize history array if needed
182
+ if (!render.renderHistory) {
183
+ render.renderHistory = [];
184
+ }
185
+ const event = {
186
+ id: `${render.nativeTag}-${cause.timestamp}`,
187
+ timestamp: cause.timestamp,
188
+ cause,
189
+ renderNumber: render.renderCount,
190
+ // Only include captured data if settings allow and data is provided
191
+ capturedProps: this.settings.capturePropsOnRender ? capturedProps : undefined,
192
+ capturedState: this.settings.captureStateOnRender ? capturedState : undefined
193
+ };
194
+ render.renderHistory.push(event);
195
+
196
+ // Enforce max history size (circular buffer behavior)
197
+ const maxHistory = Math.max(5, Math.min(50, this.settings.maxRenderHistoryPerComponent));
198
+ if (render.renderHistory.length > maxHistory) {
199
+ // Remove oldest events
200
+ render.renderHistory = render.renderHistory.slice(-maxHistory);
201
+ }
202
+
203
+ // Notify render event callbacks (for unified events integration)
204
+ this.notifyRenderEventCallbacks(event, render);
205
+ }
206
+
207
+ /**
208
+ * Notify all render event callbacks
209
+ */
210
+ notifyRenderEventCallbacks(event, render) {
211
+ this.renderEventCallbacks.forEach(callback => {
212
+ try {
213
+ callback(event, render);
214
+ } catch {
215
+ // Silently ignore callback errors
216
+ }
217
+ });
218
+ }
219
+
220
+ /**
221
+ * Emit a render event to callbacks only (without storing in the render list).
222
+ * Used by Events tool when visual highlights are off - we want events but not list updates.
223
+ */
224
+ emitRenderEvent(data) {
225
+ if (this.renderEventCallbacks.size === 0) return;
226
+ const now = Date.now();
227
+ const cause = data.renderCause || {
228
+ type: "unknown",
229
+ timestamp: now
230
+ };
231
+
232
+ // Create a minimal TrackedRender for the event (not stored)
233
+ const render = {
234
+ id: String(data.nativeTag),
235
+ nativeTag: data.nativeTag,
236
+ viewType: data.viewType,
237
+ displayName: getComponentDisplayName(data.viewType),
238
+ testID: data.testID,
239
+ nativeID: data.nativeID,
240
+ accessibilityLabel: data.accessibilityLabel,
241
+ componentName: data.componentName,
242
+ renderCount: data.count,
243
+ firstRenderTime: now,
244
+ lastRenderTime: now,
245
+ measurements: data.measurements,
246
+ color: data.color,
247
+ lastRenderCause: cause,
248
+ renderHistory: []
249
+ };
250
+
251
+ // Create the event
252
+ const event = {
253
+ id: `${data.nativeTag}-${now}-${data.count}`,
254
+ timestamp: now,
255
+ renderNumber: data.count,
256
+ cause,
257
+ capturedProps: undefined,
258
+ capturedState: undefined
259
+ };
260
+
261
+ // Notify callbacks only (no storage, no list listeners)
262
+ this.notifyRenderEventCallbacks(event, render);
263
+ }
264
+
265
+ /**
266
+ * Start batch mode - defers listener notifications until endBatch() is called.
267
+ * Use this when tracking multiple renders in a loop to avoid O(n²) notifications.
268
+ */
269
+ startBatch() {
270
+ this.isBatchMode = true;
271
+ this.batchDirty = false;
272
+ }
273
+
274
+ /**
275
+ * End batch mode and notify listeners if any renders were tracked.
276
+ */
277
+ endBatch() {
278
+ this.isBatchMode = false;
279
+ if (this.batchDirty) {
280
+ this.batchDirty = false;
281
+ this.notifyListeners();
282
+ }
283
+ }
284
+
285
+ /**
286
+ * Get all tracked renders
287
+ * Creates new object copies to trigger React.memo re-renders
288
+ */
289
+ getRenders() {
290
+ return Array.from(this.renders.values()).map(r => ({
291
+ ...r
292
+ }));
293
+ }
294
+
295
+ /**
296
+ * Check if a component passes all active filters.
297
+ * Used by both the modal list and the overlay to filter components.
298
+ *
299
+ * @param info Component info to check (can be TrackedRender or extracted component info)
300
+ * @returns true if the component passes all filters, false if it should be hidden
301
+ */
302
+ passesFilters(info) {
303
+ // Check new unified include patterns (if any are set, must match at least one)
304
+ if (this.filters.includePatterns.length > 0) {
305
+ const matchesInclude = this.matchesAnyPatternForInfo(info, this.filters.includePatterns);
306
+ if (!matchesInclude) return false;
307
+ }
308
+
309
+ // Check new unified exclude patterns
310
+ if (this.filters.excludePatterns.length > 0) {
311
+ const matchesExclude = this.matchesAnyPatternForInfo(info, this.filters.excludePatterns);
312
+ if (matchesExclude) return false;
313
+ }
314
+
315
+ // Legacy filter support (for backwards compatibility)
316
+ // Include filters: if any are set, component must match at least one
317
+ if (this.filters.includeViewType.size > 0) {
318
+ const viewType = info.viewType || '';
319
+ const displayName = info.displayName || '';
320
+ if (!this.matchesPattern(viewType, this.filters.includeViewType) && !this.matchesPattern(displayName, this.filters.includeViewType)) {
321
+ return false;
322
+ }
323
+ }
324
+ if (this.filters.includeTestID.size > 0) {
325
+ if (!info.testID || !this.matchesPattern(info.testID, this.filters.includeTestID)) {
326
+ return false;
327
+ }
328
+ }
329
+ if (this.filters.includeNativeID.size > 0) {
330
+ if (!info.nativeID || !this.matchesPattern(info.nativeID, this.filters.includeNativeID)) {
331
+ return false;
332
+ }
333
+ }
334
+ if (this.filters.includeComponent.size > 0) {
335
+ if (!info.componentName || !this.matchesPattern(info.componentName, this.filters.includeComponent)) {
336
+ return false;
337
+ }
338
+ }
339
+
340
+ // Exclude filters: if component matches any, it should be hidden
341
+ if (this.filters.excludeViewType.size > 0) {
342
+ const viewType = info.viewType || '';
343
+ const displayName = info.displayName || '';
344
+ if (this.matchesPattern(viewType, this.filters.excludeViewType) || this.matchesPattern(displayName, this.filters.excludeViewType)) {
345
+ return false;
346
+ }
347
+ }
348
+ if (this.filters.excludeTestID.size > 0) {
349
+ if (info.testID && this.matchesPattern(info.testID, this.filters.excludeTestID)) {
350
+ return false;
351
+ }
352
+ }
353
+ if (this.filters.excludeNativeID.size > 0) {
354
+ if (info.nativeID && this.matchesPattern(info.nativeID, this.filters.excludeNativeID)) {
355
+ return false;
356
+ }
357
+ }
358
+ if (this.filters.excludeComponent.size > 0) {
359
+ if (info.componentName && this.matchesPattern(info.componentName, this.filters.excludeComponent)) {
360
+ return false;
361
+ }
362
+ }
363
+ return true;
364
+ }
365
+
366
+ /**
367
+ * Check if component info matches any of the given patterns
368
+ */
369
+ matchesAnyPatternForInfo(info, patterns) {
370
+ for (const pattern of patterns) {
371
+ const lowerValue = pattern.value.toLowerCase();
372
+ switch (pattern.type) {
373
+ case "any":
374
+ // Match against all fields
375
+ if (info.viewType.toLowerCase().includes(lowerValue) || info.displayName?.toLowerCase().includes(lowerValue) || info.testID?.toLowerCase().includes(lowerValue) || info.nativeID?.toLowerCase().includes(lowerValue) || info.componentName?.toLowerCase().includes(lowerValue) || info.accessibilityLabel?.toLowerCase().includes(lowerValue)) {
376
+ return true;
377
+ }
378
+ break;
379
+ case "viewType":
380
+ if (info.viewType.toLowerCase().includes(lowerValue) || info.displayName?.toLowerCase().includes(lowerValue)) {
381
+ return true;
382
+ }
383
+ break;
384
+ case "testID":
385
+ if (info.testID?.toLowerCase().includes(lowerValue)) {
386
+ return true;
387
+ }
388
+ break;
389
+ case "nativeID":
390
+ if (info.nativeID?.toLowerCase().includes(lowerValue)) {
391
+ return true;
392
+ }
393
+ break;
394
+ case "component":
395
+ if (info.componentName?.toLowerCase().includes(lowerValue)) {
396
+ return true;
397
+ }
398
+ break;
399
+ case "accessibilityLabel":
400
+ if (info.accessibilityLabel?.toLowerCase().includes(lowerValue)) {
401
+ return true;
402
+ }
403
+ break;
404
+ }
405
+ }
406
+ return false;
407
+ }
408
+
409
+ /**
410
+ * Check if any filters are currently active
411
+ */
412
+ hasActiveFilters() {
413
+ return this.filters.includePatterns.length > 0 || this.filters.excludePatterns.length > 0 || this.filters.includeViewType.size > 0 || this.filters.includeTestID.size > 0 || this.filters.includeNativeID.size > 0 || this.filters.includeComponent.size > 0 || this.filters.excludeViewType.size > 0 || this.filters.excludeTestID.size > 0 || this.filters.excludeNativeID.size > 0 || this.filters.excludeComponent.size > 0 || this.filters.minRenderCount !== undefined || this.filters.maxRenderCount !== undefined;
414
+ }
415
+
416
+ /**
417
+ * Get filtered renders based on current filter config
418
+ */
419
+ getFilteredRenders(searchText = "") {
420
+ let renders = this.getRenders();
421
+
422
+ // Apply search filter
423
+ if (searchText.trim()) {
424
+ const search = searchText.toLowerCase();
425
+ renders = renders.filter(r => {
426
+ return r.viewType.toLowerCase().includes(search) || r.displayName.toLowerCase().includes(search) || r.testID?.toLowerCase().includes(search) || r.nativeID?.toLowerCase().includes(search) || r.accessibilityLabel?.toLowerCase().includes(search) || r.componentName?.toLowerCase().includes(search) || String(r.nativeTag).includes(search);
427
+ });
428
+ }
429
+
430
+ // Apply filters using the shared passesFilters method
431
+ renders = renders.filter(r => this.passesFilters(r));
432
+
433
+ // Apply render count range filter
434
+ if (this.filters.minRenderCount !== undefined) {
435
+ renders = renders.filter(r => r.renderCount >= this.filters.minRenderCount);
436
+ }
437
+ if (this.filters.maxRenderCount !== undefined) {
438
+ renders = renders.filter(r => r.renderCount <= this.filters.maxRenderCount);
439
+ }
440
+
441
+ // Sort by last render time (most recent first)
442
+ return renders.sort((a, b) => b.lastRenderTime - a.lastRenderTime);
443
+ }
444
+
445
+ /**
446
+ * Check if a value matches any pattern in the set
447
+ */
448
+ matchesPattern(value, patterns) {
449
+ const lowerValue = value.toLowerCase();
450
+ for (const pattern of patterns) {
451
+ const lowerPattern = pattern.toLowerCase();
452
+ if (lowerValue.includes(lowerPattern)) {
453
+ return true;
454
+ }
455
+ }
456
+ return false;
457
+ }
458
+
459
+ /**
460
+ * Get a single render by id
461
+ */
462
+ getRender(id) {
463
+ return this.renders.get(id);
464
+ }
465
+
466
+ /**
467
+ * Clear all tracked renders
468
+ */
469
+ clear() {
470
+ this.renders.clear();
471
+ this.notifyListeners();
472
+ }
473
+
474
+ /**
475
+ * Reset render count for a specific component
476
+ */
477
+ resetRenderCount(id) {
478
+ const render = this.renders.get(id);
479
+ if (render) {
480
+ render.renderCount = 0;
481
+ this.notifyListeners();
482
+ }
483
+ }
484
+
485
+ /**
486
+ * Start tracking
487
+ */
488
+ start() {
489
+ this.isTracking = true;
490
+ this.isPaused = false;
491
+ this.notifyStateListeners();
492
+ }
493
+
494
+ /**
495
+ * Stop tracking
496
+ */
497
+ stop() {
498
+ this.isTracking = false;
499
+ this.isPaused = false;
500
+ this.notifyStateListeners();
501
+ }
502
+
503
+ /**
504
+ * Pause tracking (keeps state but stops adding new renders)
505
+ */
506
+ pause() {
507
+ this.isPaused = true;
508
+ this.notifyStateListeners();
509
+ }
510
+
511
+ /**
512
+ * Resume tracking
513
+ */
514
+ resume() {
515
+ this.isPaused = false;
516
+ this.notifyStateListeners();
517
+ }
518
+
519
+ /**
520
+ * Toggle pause state
521
+ */
522
+ togglePause() {
523
+ this.isPaused = !this.isPaused;
524
+ this.notifyStateListeners();
525
+ }
526
+
527
+ /**
528
+ * Get current tracking state
529
+ */
530
+ getState() {
531
+ return {
532
+ isTracking: this.isTracking,
533
+ isPaused: this.isPaused
534
+ };
535
+ }
536
+
537
+ /**
538
+ * Replace all tracked renders (and optionally tracking state) from a
539
+ * synced device snapshot. Remote mirror mode only (desktop dashboard) —
540
+ * nothing tracks locally there, so this is the sole data source. Filters
541
+ * and search stay local: they apply at read time via getFilteredRenders.
542
+ */
543
+ replaceRenders(renders, state) {
544
+ this.renders = new Map(renders.map(render => [render.id, render]));
545
+ if (state && (this.isTracking !== state.isTracking || this.isPaused !== state.isPaused)) {
546
+ this.isTracking = state.isTracking;
547
+ this.isPaused = state.isPaused;
548
+ this.notifyStateListeners();
549
+ }
550
+ this.notifyListeners();
551
+ }
552
+
553
+ /**
554
+ * Get filter config
555
+ */
556
+ getFilters() {
557
+ return this.filters;
558
+ }
559
+
560
+ /**
561
+ * Update filter config
562
+ */
563
+ setFilters(filters) {
564
+ this.filters = {
565
+ ...this.filters,
566
+ ...filters
567
+ };
568
+ this.notifyListeners();
569
+ this.notifyFilterListeners();
570
+ }
571
+
572
+ /**
573
+ * Add an include pattern
574
+ */
575
+ addIncludePattern(type, pattern) {
576
+ const key = `include${type.charAt(0).toUpperCase() + type.slice(1)}`;
577
+ this.filters[key].add(pattern);
578
+ this.notifyListeners();
579
+ }
580
+
581
+ /**
582
+ * Remove an include pattern
583
+ */
584
+ removeIncludePattern(type, pattern) {
585
+ const key = `include${type.charAt(0).toUpperCase() + type.slice(1)}`;
586
+ this.filters[key].delete(pattern);
587
+ this.notifyListeners();
588
+ }
589
+
590
+ /**
591
+ * Add an exclude pattern
592
+ */
593
+ addExcludePattern(type, pattern) {
594
+ const key = `exclude${type.charAt(0).toUpperCase() + type.slice(1)}`;
595
+ this.filters[key].add(pattern);
596
+ this.notifyListeners();
597
+ }
598
+
599
+ /**
600
+ * Remove an exclude pattern
601
+ */
602
+ removeExcludePattern(type, pattern) {
603
+ const key = `exclude${type.charAt(0).toUpperCase() + type.slice(1)}`;
604
+ this.filters[key].delete(pattern);
605
+ this.notifyListeners();
606
+ }
607
+
608
+ /**
609
+ * Clear all filters
610
+ */
611
+ clearFilters() {
612
+ this.filters = {
613
+ includeTestID: new Set(),
614
+ includeNativeID: new Set(),
615
+ includeViewType: new Set(),
616
+ includeComponent: new Set(),
617
+ excludeTestID: new Set(),
618
+ excludeNativeID: new Set(),
619
+ excludeViewType: new Set(),
620
+ excludeComponent: new Set(),
621
+ includePatterns: [],
622
+ excludePatterns: [],
623
+ minRenderCount: undefined,
624
+ maxRenderCount: undefined
625
+ };
626
+ this.notifyListeners();
627
+ }
628
+
629
+ /**
630
+ * Get available prop values from tracked renders
631
+ */
632
+ getAvailableProps() {
633
+ const viewTypes = new Set();
634
+ const testIDs = new Set();
635
+ const nativeIDs = new Set();
636
+ const componentNames = new Set();
637
+ const accessibilityLabels = new Set();
638
+ for (const render of this.renders.values()) {
639
+ if (render.viewType) viewTypes.add(render.viewType);
640
+ if (render.testID) testIDs.add(render.testID);
641
+ if (render.nativeID) nativeIDs.add(render.nativeID);
642
+ if (render.componentName) componentNames.add(render.componentName);
643
+ if (render.accessibilityLabel) accessibilityLabels.add(render.accessibilityLabel);
644
+ }
645
+ return {
646
+ viewTypes: Array.from(viewTypes).sort(),
647
+ testIDs: Array.from(testIDs).sort(),
648
+ nativeIDs: Array.from(nativeIDs).sort(),
649
+ componentNames: Array.from(componentNames).sort(),
650
+ accessibilityLabels: Array.from(accessibilityLabels).sort()
651
+ };
652
+ }
653
+
654
+ /**
655
+ * Get summary stats
656
+ */
657
+ getStats() {
658
+ let totalRenders = 0;
659
+ for (const render of this.renders.values()) {
660
+ totalRenders += render.renderCount;
661
+ }
662
+ return {
663
+ totalComponents: this.renders.size,
664
+ totalRenders
665
+ };
666
+ }
667
+
668
+ /**
669
+ * Subscribe to render updates
670
+ */
671
+ subscribe(listener) {
672
+ this.listeners.add(listener);
673
+ // Immediately notify with current state
674
+ listener(this.getRenders());
675
+ return () => {
676
+ this.listeners.delete(listener);
677
+ };
678
+ }
679
+
680
+ /**
681
+ * Subscribe to state changes (tracking/paused)
682
+ */
683
+ subscribeToState(listener) {
684
+ this.stateListeners.add(listener);
685
+ // Immediately notify with current state
686
+ listener(this.getState());
687
+ return () => {
688
+ this.stateListeners.delete(listener);
689
+ };
690
+ }
691
+
692
+ /**
693
+ * Subscribe to settings changes
694
+ */
695
+ subscribeToSettings(listener) {
696
+ this.settingsListeners.add(listener);
697
+ // Immediately notify with current settings
698
+ listener(this.settings);
699
+ return () => {
700
+ this.settingsListeners.delete(listener);
701
+ };
702
+ }
703
+
704
+ /**
705
+ * Subscribe to filter changes
706
+ */
707
+ subscribeToFilters(listener) {
708
+ this.filterListeners.add(listener);
709
+ // Immediately notify with current filters
710
+ listener(this.filters);
711
+ return () => {
712
+ this.filterListeners.delete(listener);
713
+ };
714
+ }
715
+
716
+ /**
717
+ * Subscribe to individual render events as they occur.
718
+ * Used by unified events system to integrate render events into the timeline.
719
+ *
720
+ * @param callback - Called for each render event with the event and component info
721
+ * @returns Unsubscribe function
722
+ */
723
+ onRenderEvent(callback) {
724
+ this.renderEventCallbacks.add(callback);
725
+ return () => {
726
+ this.renderEventCallbacks.delete(callback);
727
+ };
728
+ }
729
+
730
+ /**
731
+ * Get the number of render event callbacks (for debugging)
732
+ */
733
+ getRenderEventCallbackCount() {
734
+ return this.renderEventCallbacks.size;
735
+ }
736
+
737
+ /**
738
+ * Check if a render should be visible based on current filters
739
+ * Used by the overlay to filter frozen highlights
740
+ */
741
+ shouldShowRender(render) {
742
+ const filters = this.filters;
743
+
744
+ // Check new pattern-based filters first
745
+ if (filters.includePatterns.length > 0) {
746
+ if (!this.matchesAnyPatternForInfo(render, filters.includePatterns)) {
747
+ return false;
748
+ }
749
+ }
750
+ if (filters.excludePatterns.length > 0) {
751
+ if (this.matchesAnyPatternForInfo(render, filters.excludePatterns)) {
752
+ return false;
753
+ }
754
+ }
755
+
756
+ // Legacy set-based filters
757
+ const hasLegacyIncludeFilters = filters.includeViewType.size > 0 || filters.includeTestID.size > 0 || filters.includeNativeID.size > 0 || filters.includeComponent.size > 0;
758
+ if (hasLegacyIncludeFilters) {
759
+ const matchesInclude = filters.includeViewType.size > 0 && filters.includeViewType.has(render.viewType) || filters.includeTestID.size > 0 && render.testID && filters.includeTestID.has(render.testID) || filters.includeNativeID.size > 0 && render.nativeID && filters.includeNativeID.has(render.nativeID) || filters.includeComponent.size > 0 && render.componentName && filters.includeComponent.has(render.componentName);
760
+ if (!matchesInclude) {
761
+ return false;
762
+ }
763
+ }
764
+
765
+ // Check exclude filters
766
+ if (filters.excludeViewType.has(render.viewType)) return false;
767
+ if (render.testID && filters.excludeTestID.has(render.testID)) return false;
768
+ if (render.nativeID && filters.excludeNativeID.has(render.nativeID)) return false;
769
+ if (render.componentName && filters.excludeComponent.has(render.componentName)) return false;
770
+ return true;
771
+ }
772
+
773
+ /**
774
+ * Get current settings
775
+ */
776
+ getSettings() {
777
+ return {
778
+ ...this.settings
779
+ };
780
+ }
781
+
782
+ /**
783
+ * Update settings
784
+ */
785
+ setSettings(newSettings) {
786
+ // Validate batchSize
787
+ if (newSettings.batchSize !== undefined) {
788
+ newSettings.batchSize = Math.max(10, Math.min(500, newSettings.batchSize));
789
+ }
790
+
791
+ // Validate maxRenderHistoryPerComponent
792
+ if (newSettings.maxRenderHistoryPerComponent !== undefined) {
793
+ newSettings.maxRenderHistoryPerComponent = Math.max(5, Math.min(50, newSettings.maxRenderHistoryPerComponent));
794
+ }
795
+
796
+ // If enabling render history, also enable trackRenderCauses
797
+ if (newSettings.enableRenderHistory && !this.settings.trackRenderCauses) {
798
+ newSettings.trackRenderCauses = true;
799
+ }
800
+
801
+ // If disabling trackRenderCauses, also disable render history
802
+ if (newSettings.trackRenderCauses === false && this.settings.enableRenderHistory) {
803
+ newSettings.enableRenderHistory = false;
804
+ }
805
+ this.settings = {
806
+ ...this.settings,
807
+ ...newSettings
808
+ };
809
+
810
+ // Sync performance logging with PerformanceLogger
811
+ if (newSettings.performanceLogging !== undefined) {
812
+ PerformanceLogger.setEnabled(newSettings.performanceLogging);
813
+ }
814
+ this.notifySettingsListeners();
815
+ }
816
+
817
+ /**
818
+ * Clear render history for all components
819
+ */
820
+ clearAllRenderHistory() {
821
+ for (const render of this.renders.values()) {
822
+ render.renderHistory = [];
823
+ }
824
+ this.notifyListeners();
825
+ }
826
+
827
+ /**
828
+ * Clear render history for a specific component
829
+ */
830
+ clearRenderHistory(id) {
831
+ const render = this.renders.get(id);
832
+ if (render) {
833
+ render.renderHistory = [];
834
+ this.notifyListeners();
835
+ }
836
+ }
837
+
838
+ /**
839
+ * Get render history stats for debugging
840
+ */
841
+ getRenderHistoryStats() {
842
+ let totalEvents = 0;
843
+ let componentsWithHistory = 0;
844
+ for (const render of this.renders.values()) {
845
+ if (render.renderHistory && render.renderHistory.length > 0) {
846
+ totalEvents += render.renderHistory.length;
847
+ componentsWithHistory++;
848
+ }
849
+ }
850
+ return {
851
+ totalEvents,
852
+ componentsWithHistory,
853
+ averageEventsPerComponent: componentsWithHistory > 0 ? totalEvents / componentsWithHistory : 0
854
+ };
855
+ }
856
+
857
+ /**
858
+ * Get batch size (convenience method)
859
+ */
860
+ getBatchSize() {
861
+ return this.settings.batchSize;
862
+ }
863
+
864
+ /**
865
+ * Set batch size (convenience method)
866
+ */
867
+ setBatchSize(size) {
868
+ this.setSettings({
869
+ batchSize: size
870
+ });
871
+ }
872
+ notifyListeners() {
873
+ const renders = this.getRenders();
874
+ for (const listener of this.listeners) {
875
+ try {
876
+ listener(renders);
877
+ } catch (error) {
878
+ console.error("[RenderTracker] Error in listener:", error);
879
+ }
880
+ }
881
+ }
882
+ notifyStateListeners() {
883
+ const state = this.getState();
884
+ for (const listener of this.stateListeners) {
885
+ try {
886
+ listener(state);
887
+ } catch (error) {
888
+ console.error("[RenderTracker] Error in state listener:", error);
889
+ }
890
+ }
891
+ }
892
+ notifySettingsListeners() {
893
+ const settings = this.getSettings();
894
+ for (const listener of this.settingsListeners) {
895
+ try {
896
+ listener(settings);
897
+ } catch (error) {
898
+ console.error("[RenderTracker] Error in settings listener:", error);
899
+ }
900
+ }
901
+ }
902
+ notifyFilterListeners() {
903
+ const filters = this.getFilters();
904
+ for (const listener of this.filterListeners) {
905
+ try {
906
+ listener(filters);
907
+ } catch (error) {
908
+ console.error("[RenderTracker] Error in filter listener:", error);
909
+ }
910
+ }
911
+ }
912
+ }
913
+
914
+ // Export singleton instance
915
+ export const RenderTracker = new RenderTrackerSingleton();
916
+ export default RenderTracker;